Skip to content

Commit

Permalink
initial commit
Browse files Browse the repository at this point in the history
  • Loading branch information
pbakaus committed Aug 30, 2012
1 parent 87941a7 commit 4f75dce
Show file tree
Hide file tree
Showing 4 changed files with 255 additions and 0 deletions.
54 changes: 54 additions & 0 deletions css/main.css
Original file line number Diff line number Diff line change
@@ -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);
}
113 changes: 113 additions & 0 deletions index.html
Original file line number Diff line number Diff line change
@@ -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>
87 changes: 87 additions & 0 deletions src/domvas.js
Original file line number Diff line number Diff line change
@@ -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);
}
};

}

};

})();

1 change: 1 addition & 0 deletions src/domvas.min.js

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

0 comments on commit 4f75dce

Please sign in to comment.