Mercurial > MadButterfly
diff nodejs/svg.js @ 759:ae1ae29348d1
Add origin calculation support
author | wycc |
---|---|
date | Sat, 28 Aug 2010 22:14:38 +0800 |
parents | 199bac90b90a |
children | 77b561bb7929 |
line wrap: on
line diff
--- a/nodejs/svg.js Sat Aug 28 17:31:59 2010 +0800 +++ b/nodejs/svg.js Sat Aug 28 22:14:38 2010 +0800 @@ -19,6 +19,7 @@ var nodes = doc.root().childNodes(); var coord = mb_rt.coord_new(root); var k; + var accu=[1,0,0,0,1,0]; this.mb_rt = mb_rt; this.stop_ref={}; this.gradients={}; @@ -28,7 +29,7 @@ if (n == "defs") { this.parseDefs(root,nodes[k]); } else if (n == "g") { - this.parseGroup(root,'root_coord',nodes[k]); + this.parseGroup(accu,root,'root_coord',nodes[k]); } } } @@ -167,6 +168,62 @@ return paint; } +function guessPathBoundingBox(coord,d) +{ + var items = d.split(' '); + var len = items.length; + var pair; + var i; + var minx,miny; + + minx = 10000; + miny = 10000; + + for(i=0;i<len;i++) { + var type = items[i].toLowerCase(); + x = minx;y = miny; + switch(type) { + case 'm': + case 'l': + case 'a': + case 'x': + pair = items[i+1].split(','); + if (pair.length==2) { + x = parseFloat(pair[0]); + y = parseFloat(pair[1]); + i++; + } else { + x = parseFloat(items[i+1]); + y = parseFloat(items[i+2]); + i+=2; + } + break; + case 'q': + // Implement this latter + break; + case 'c': + // Implement this latter + break; + case 's': + // Implement this latter + break; + case 'h': + x = parseFloat(items[i+1]); + break; + case 'v': + y = parseFloat(items[i+1]); + break; + } + if (x < minx) minx = x; + if (y < miny) miny = y; + } + if (coord.center.x > minx) + coord.center.x = minx; + if (coord.center.y > miny) + coord.center.y = miny; +} + + loadSVG.prototype.parsePath=function(coord,id, n) { var d = n.attr('d').value(); @@ -179,6 +236,7 @@ var stroke_color; var black_paint; + guessPathBoundingBox(coord,d); if(style != null) { var items = style.value().split(';'); var alpha; @@ -209,6 +267,7 @@ if(fill_color != "none") { paint = this._prepare_paint_color(fill_color, fill_alpha); paint.fill(path); + sys.puts("paint path with "+fill_color+" "+fill_alpha); } } else { black_paint.fill(path); @@ -217,6 +276,7 @@ if(stroke_color != "none") { paint = this._prepare_paint_color(stroke_color, stroke_alpha); paint.stroke(path); + sys.puts("stroke path"); } } else { black_paint.stroke(path); @@ -232,6 +292,13 @@ var y = getInteger(n,'y'); var tcoord = this.mb_rt.coord_new(coord); var style = new Object(); + + if (n.attr('x')) + if (coord.center.x > x) + coord.center.x = x; + if (n.attr('y')) + if (coord.center.y > y) + coord.center.y = y; style.fs = 20; style.family = 'courier'; parseTextStyle(style,n); @@ -248,6 +315,23 @@ make_mbnames(this.mb_rt, n, tcoord); } + +function multiply(s,d) { + var m=[]; + m[0] = s[0]*d[0]+s[1]*d[3]; + m[1] = s[0]*d[1]+s[1]*d[4]; + m[2] = s[0]*d[2]+s[1]*d[5]+s[2]; + m[3] = s[3]*d[0]+s[4]*d[3]; + m[4] = s[3]*d[1]+s[4]*d[4]; + m[5] = s[3]*d[2]+s[4]*d[5]+s[5]; + s[0] = m[0]; + s[1] = m[1]; + s[2] = m[2]; + s[3] = m[3]; + s[4] = m[4]; + s[5] = m[5]; +} + function parseTransform(coord, s) { var off = s.indexOf('translate'); @@ -267,37 +351,53 @@ var x,y; x = parseFloat(f[0]); y = parseFloat(f[1]); - coord[0] = 1; - coord[1] = 0; - coord[2] = x; - coord[3] = 0; - coord[4] = 1; - coord[5] = y; + coord[2] += x; + coord[5] += y; } off = s.indexOf('matrix'); if (off != -1) { var end = s.indexOf(')'); var m = s.substring(7,end); var fields = m.split(','); - coord[0] = parseFloat(fields[0]); - coord[1] = parseFloat(fields[2]); - coord[2] = parseFloat(fields[4]); - coord[3] = parseFloat(fields[1]); - coord[4] = parseFloat(fields[3]); - coord[5] = parseFloat(fields[5]); + var newm=[]; + newm[0] = parseFloat(fields[0]); + newm[1] = parseFloat(fields[2]); + newm[2] = parseFloat(fields[4]); + newm[3] = parseFloat(fields[1]); + newm[4] = parseFloat(fields[3]); + newm[5] = parseFloat(fields[5]); + multiply(coord,newm); } } -loadSVG.prototype.parseRect=function(coord, id, n) +loadSVG.prototype.parseRect=function(accu_matrix,coord, id, n) { + sys.puts("rect"); var x = getInteger(n,'x'); var y = getInteger(n,'y'); + var rx,ry; var w = getInteger(n,'width'); var h = getInteger(n,'height'); + var trans = n.attr('transform'); var paint; var style = n.attr('style'); + if (trans) { + parseTransform(coord,trans.value()); + var m = [1,0,0,0,1,0]; + parseTransform(m,trans.value()); + rx = x+m[2]; + ry = y+m[5]; + } + + if (coord.center.x > rx) + coord.center.x = rx; + if (coord.center.y > ry) + coord.center.y = ry; + coord.center.x += accu_matrix[2]; + coord.center.y += accu_matrix[5]; + if (style==null) { paint = this.mb_rt.paint_color_new(0,0,0,0.1); } else { @@ -337,16 +437,23 @@ coord.add_shape(rect); } -loadSVG.prototype.parseGroup=function(root, group_id, n) +loadSVG.prototype.parseGroup=function(accu_matrix,root, group_id, n) { var k; var nodes = n.childNodes(); var coord = this.mb_rt.coord_new(root); // Parse the transform and style here var trans = n.attr('transform'); + var accu=[1,0,0,0,1,0]; + coord.center= new Object(); + coord.center.x = 10000; + coord.center.y = 10000; if (trans!=null) { parseTransform(coord, trans.value()); - } + } + accu[2] = accu_matrix[2]+coord[2]; + accu[5] = accu_matrix[5]+coord[5]; + for(k in nodes) { var c = nodes[k].name(); @@ -356,18 +463,17 @@ id = attr.value(); } if (c == "g") { - this.parseGroup(coord, id, nodes[k]); + this.parseGroup(accu,coord, id, nodes[k]); } else if (c == "path") { this.parsePath(coord, id, nodes[k]); } else if (c == "text") { this.parseText(coord, id, nodes[k]); } else if (c == "rect") { - this.parseRect(coord, id, nodes[k]); + this.parseRect(accu_matrix,coord, id, nodes[k]); } else if (c == "image") { this.parseImage(coord, id, nodes[k]); } } - make_mbnames(this.mb_rt, n, coord); } @@ -383,23 +489,26 @@ } else { return; } - sys.puts("Load image "+ref); var w; var h; var x,y; w = n.attr("width"); if (w == null) return; - w = parseInt(w.value()); + w = parseFloat(w.value()); h = n.attr("height"); if (h == null) return; - h = parseInt(h.value()); + h = parseFloat(h.value()); x = n.attr("x"); if (x == null) return; - x = parseInt(x.value()); + x = parseFloat(x.value()); y = n.attr("y"); if (y == null) return; - y = parseInt(y.value()); + y = parseFloat(y.value()); + if (coord.center.x > x) + coord.center.x = x; + if (coord.center.y > y) + coord.center.y = y; var img = this.mb_rt.image_new(x,y,w,h); var img_data = ldr.load(ref); var paint = this.mb_rt.paint_image_new(img_data);