Skip to content

HTTPS clone URL

Subversion checkout URL

You can clone with
or
.
Download ZIP
Browse files

initial commit

  • Loading branch information...
commit 4f75dce3d4290790157fe8400521e760d68a376f 1 parent 87941a7
@pbakaus authored
Showing with 255 additions and 0 deletions.
  1. +54 −0 css/main.css
  2. +113 −0 index.html
  3. +87 −0 src/domvas.js
  4. +1 −0  src/domvas.min.js
View
54 css/main.css
@@ -0,0 +1,54 @@
+body {
+ font-family: Open Sans, sans-serif;
+ font-size: 0.9em;
+ color: #333;
+
+
+ background-image: url();
+}
+
+section {
+ max-width: 710px;
+ margin: 0 auto;
+}
+
+.sub {
+ height: 210px;
+}
+
+.sub > * {
+ float: left;
+}
+
+pre {
+ background: #333;
+ color: #eee;
+ padding: 10px;
+}
+
+canvas {
+ background: #eee;
+}
+
+h1 {
+ border-bottom: 1px solid #fff;
+ clear: both;
+ margin-top: 1em;
+ text-shadow: 1px 1px 0px white;
+}
+
+p {
+ color: #555;
+ line-height: 1.4em;
+}
+
+strong {
+ font-weight: 600;
+}
+
+h1::after {
+ content: '';
+ width: 100%;
+ display: block;
+ border-bottom: 1px solid rgba(30,30,30,0.4);
+}
View
113 index.html
@@ -0,0 +1,113 @@
+<!DOCTYPE html>
+<html>
+ <head>
+ <meta charset="utf-8">
+ <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
+ <title>Domvas!</title>
+
+ <link href='http://fonts.googleapis.com/css?family=Open+Sans:400,600,300' rel='stylesheet' type='text/css'>
+ <link rel="stylesheet" href="css/main.css">
+
+ <style type="text/css">
+
+ #dom {
+ font-family: Arial;
+ border: 5px solid rgba(0,0,0,0.2);
+ border-radius: 10px;
+ width: 260px;
+ padding: 5px 20px;
+ background: #4f8eff;
+
+ text-shadow: 1px 1px 0 #264782;
+ line-height: 1.4em;
+
+ margin: 20px;
+ }
+
+ #dom p {
+ color: rgba(255,255,255,0.8);
+ }
+
+ #dom strong {
+ color: #fff;
+ font-style: italic;
+ }
+
+ </style>
+</head>
+<body>
+
+ <section>
+
+ <h1>Domvas!</h1>
+
+ <p><strong>Domvas implements the missing piece that connects the DOM and Canvas</strong>. It gives to the ability to take arbitrary DOM content and paint it to a Canvas of your choice.</p>
+
+ <p>That dialog that is a pain to layout in Canvas but needs to be animated? Yep. Always wanted to use CSS transforms in Canvas? Yep. I think you get the idea.</p>
+
+ <h1>Demo</h1>
+
+ <div class="sub">
+
+ <div style='font-size:30px' id="dom">
+ <p>EHRMAGERD <strong>DOM IN CANVAS</strong>!</p>
+ </div>
+
+ <canvas id="test" width="350" height="206"></canvas>
+
+ </div>
+
+ <h1>Usage</h1>
+
+ <pre>
+var canvas = document.getElementById("test");
+var context = canvas.getContext('2d');
+
+domvas.toImage(document.getElementById("dom"), function() {
+ context.drawImage(this, 20, 20);
+});
+ </pre>
+
+ <p>Extended documentation:</p>
+
+ <pre>
+domvas.toImage(domElement, readyCallback, width, height, left, top);
+ </pre>
+
+ <h1>How it works</h1>
+
+ <p>Domvas uses a feature of SVG that allows you to embed XHTML content into the SVG – and as you might know, the actual SVG can be used as a data uri, and therefore behaves like a standard image.</p>
+
+ <p>I have written about this technique in 2008 <a href="http://paulbakaus.com/2008/08/19/css-transforms-for-firefox/">when I brought CSS transforms to browsers that did not have them</a>. It took a little more experimentation to transform it into a reusable plugin: HTML content needs to be serialized to XML, and all styles have to be inlined.</p>
+
+ <h1>Caveats</h1>
+
+ <ul>
+ <li><strong>Internet Explorer</strong> is not supported, at it doesn't support the foreignObject tag in SVG.</li>
+ <li>SVG's foreignObject is subject to strong security – meaning any external content will likely fail (i.e. iframes, web fonts)</li>
+ <li>The DOM object is <strong>not linked, but copied</strong> – if you change the style of the DOM object, it will not automatically update in Canvas</li>
+ <li>Content outside the bounding box of the element will be cut of per default if painted to Canvas. Don't worry though, simply pass a more confortable offset to the toImage function (see above)</li>
+ </ul>
+
+ <h1>Credits / License</h1>
+
+ <p>&copy;2012 Paul Bakaus. Licensed under MIT. Reach out on <a href='http://twitter.com/pbakaus'>Twitter</a>!</p>
+
+
+ </section>
+
+ <script src="src/domvas.js"></script>
+ <script>
+
+ var canvas = document.getElementById("test");
+ var context = canvas.getContext('2d');
+
+ domvas.toImage(document.getElementById("dom"), function() {
+ // Look ma, I just converted this element to an image and can now to funky stuff!
+ context.drawImage(this, 20, 20);
+ });
+
+ </script>
+
+</body>
+</html>
View
87 src/domvas.js
@@ -0,0 +1,87 @@
+"use strict";
+
+(function() {
+
+ var supportsCSSText = getComputedStyle(document.body).cssText !== "";
+
+ function copyCSS(elem, origElem, log) {
+
+ var computedStyle = getComputedStyle(origElem);
+
+ if(supportsCSSText) {
+ elem.style.cssText = computedStyle.cssText;
+
+ } else {
+
+ // Really, Firefox?
+ for(var prop in computedStyle) {
+ if(isNaN(parseInt(prop, 10)) && typeof computedStyle[prop] !== 'function' && !(/^(cssText|length|parentRule)$/).test(prop)) {
+ elem.style[prop] = computedStyle[prop];
+ }
+ }
+
+ }
+
+ }
+
+ function inlineStyles(elem, origElem) {
+
+ var children = elem.querySelectorAll('*');
+ var origChildren = origElem.querySelectorAll('*');
+
+ // copy the current style to the clone
+ copyCSS(elem, origElem, 1);
+
+ // collect all nodes within the element, copy the current style to the clone
+ Array.prototype.forEach.call(children, function(child, i) {
+ copyCSS(child, origChildren[i]);
+ });
+
+ // strip margins from the outer element
+ elem.style.margin = elem.style.marginLeft = elem.style.marginTop = elem.style.marginBottom = elem.style.marginRight = '';
+
+ }
+
+ window.domvas = {
+
+ toImage: function(origElem, callback, width, height, left, top) {
+
+ left = (left || 0);
+ top = (top || 0);
+
+ var elem = origElem.cloneNode(true);
+
+ // inline all CSS (ugh..)
+ inlineStyles(elem, origElem);
+
+ // unfortunately, SVG can only eat well formed XHTML
+ elem.setAttribute("xmlns", "http://www.w3.org/1999/xhtml");
+
+ // serialize the DOM node to a String
+ var serialized = new XMLSerializer().serializeToString(elem);
+
+ // Create well formed data URL with our DOM string wrapped in SVG
+ var dataUri = "data:image/svg+xml," +
+ "<svg xmlns='http://www.w3.org/2000/svg' width='" + ((width || origElem.offsetWidth) + left) + "' height='" + ((height || origElem.offsetHeight) + top) + "'>" +
+ "<foreignObject width='100%' height='100%' x='" + left + "' y='" + top + "'>" +
+ serialized +
+ "</foreignObject>" +
+ "</svg>";
+
+ // create new, actual image
+ var img = new Image();
+ img.src = dataUri;
+
+ // when loaded, fire onload callback with actual image node
+ img.onload = function() {
+ if(callback) {
+ callback.call(this, this);
+ }
+ };
+
+ }
+
+ };
+
+})();
+
View
1  src/domvas.min.js
@@ -0,0 +1 @@
+"use strict";(function(){function t(t,n,r){var i=getComputedStyle(n);if(e)t.style.cssText=i.cssText;else for(var s in i)isNaN(parseInt(s,10))&&typeof i[s]!="function"&&!/^(cssText|length|parentRule)$/.test(s)&&(t.style[s]=i[s])}function n(e,n){var r=e.querySelectorAll("*"),i=n.querySelectorAll("*");t(e,n,1),Array.prototype.forEach.call(r,function(e,n){t(e,i[n])}),e.style.margin=e.style.marginLeft=e.style.marginTop=e.style.marginBottom=e.style.marginRight=""}var e=getComputedStyle(document.body).cssText!=="";window.domvas={toImage:function(e,t,r,i,s,o){s=s||0,o=o||0;var u=e.cloneNode(!0);n(u,e),u.setAttribute("xmlns","http://www.w3.org/1999/xhtml");var a=(new XMLSerializer).serializeToString(u),f="data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' width='"+((r||e.offsetWidth)+s)+"' height='"+((i||e.offsetHeight)+o)+"'>"+"<foreignObject width='100%' height='100%' x='"+s+"' y='"+o+"'>"+a+"</foreignObject>"+"</svg>",l=new Image;l.src=f,l.onload=function(){t&&t.call(this,this)}}}})()
Please sign in to comment.
Something went wrong with that request. Please try again.