Mercurial > MadButterfly
comparison tools/svg2code.py @ 1067:7b4e80ab671a openvg
merge from default branch
author | Thinker K.F. Li <thinker@codemud.net> |
---|---|
date | Wed, 01 Dec 2010 12:25:56 +0800 |
parents | 586e50f82c1f |
children |
comparison
equal
deleted
inserted
replaced
630:bd18951b51d5 | 1067:7b4e80ab671a |
---|---|
1 #! /usr/bin/env python | 1 #! /usr/bin/env python |
2 # -*- indent-tabs-mode: t; tab-width: 8; c-basic-offset: 4; -*- | |
3 # vim: sw=4:ts=8:sts=4 | |
2 from xml.dom.minidom import parse | 4 from xml.dom.minidom import parse |
3 import sys | 5 import sys |
4 import re | 6 import re |
5 | 7 |
6 svgns='http://www.w3.org/2000/svg' | 8 svgns='http://www.w3.org/2000/svg' |
129 prop_map = node.style_map | 131 prop_map = node.style_map |
130 except: | 132 except: |
131 style_str = node.getAttribute('style') | 133 style_str = node.getAttribute('style') |
132 prop_map = get_style_map(style_str) | 134 prop_map = get_style_map(style_str) |
133 pass | 135 pass |
134 | 136 |
135 if node.hasAttribute('fill-opacity'): | 137 if node.hasAttribute('fill-opacity'): |
136 opacity = float(node.getAttribute('fill-opacity')) | 138 opacity = float(node.getAttribute('fill-opacity')) |
137 elif node.hasAttribute('opacity'): | 139 elif node.hasAttribute('opacity'): |
138 opacity = float(node.getAttribute('opacity')) | 140 opacity = float(node.getAttribute('opacity')) |
139 else: | 141 else: |
232 # This function calculate the ellipse with information from SVG path data. | 234 # This function calculate the ellipse with information from SVG path data. |
233 # | 235 # |
234 # \see calc_center_and_x_aix() | 236 # \see calc_center_and_x_aix() |
235 def _calc_ellipse_of_arc(x0, y0, rx, ry, x_rotate, large, sweep, x, y): | 237 def _calc_ellipse_of_arc(x0, y0, rx, ry, x_rotate, large, sweep, x, y): |
236 import math | 238 import math |
237 | 239 |
238 _sin = math.sin(x_rotate) | 240 _sin = math.sin(x_rotate) |
239 _cos = math.cos(x_rotate) | 241 _cos = math.cos(x_rotate) |
240 | 242 |
241 nrx = x * _cos + y * _sin # Not Rotated X | 243 nrx = x * _cos + y * _sin # Not Rotated X |
242 nry = x * -_sin + y * _cos | 244 nry = x * -_sin + y * _cos |
243 nrx0 = x0 * _cos + y0 * _sin | 245 nrx0 = x0 * _cos + y0 * _sin |
244 nry0 = x0 * -_sin + y0 * _cos | 246 nry0 = x0 * -_sin + y0 * _cos |
245 | 247 |
246 udx = (nrx - nrx0) / 2 / rx # ux - umx | 248 udx = (nrx - nrx0) / 2 / rx # ux - umx |
247 udy = (nry - nry0) / 2 / ry # uy - umy | 249 udy = (nry - nry0) / 2 / ry # uy - umy |
248 umx = (nrx + nrx0) / 2 / rx | 250 umx = (nrx + nrx0) / 2 / rx |
249 umy = (nry + nry0) / 2 / ry | 251 umy = (nry + nry0) / 2 / ry |
250 | 252 |
277 udcy = -udcy | 279 udcy = -udcy |
278 pass | 280 pass |
279 | 281 |
280 nrcx = rx * (udcx + umx) | 282 nrcx = rx * (udcx + umx) |
281 nrcy = ry * (udcy + umy) | 283 nrcy = ry * (udcy + umy) |
282 | 284 |
283 cx = nrcx * _cos - nrcy * _sin | 285 cx = nrcx * _cos - nrcy * _sin |
284 cy = nrcx * _sin + nrcy * _cos | 286 cy = nrcx * _sin + nrcy * _cos |
285 | 287 |
286 return cx, cy | 288 return cx, cy |
287 | 289 |
288 # M x y : Move to (x,y) | 290 # M x y : Move to (x,y) |
289 # Z : close path | 291 # Z : close path |
290 # L x y : lineto (x,y) | 292 # L x y : lineto (x,y) |
311 'T': 2, 't':2, | 313 'T': 2, 't':2, |
312 'A': 7, 'a':7} | 314 'A': 7, 'a':7} |
313 | 315 |
314 def _angle_rotated_ellipse(x, y, rx, ry, x_rotate): | 316 def _angle_rotated_ellipse(x, y, rx, ry, x_rotate): |
315 import math | 317 import math |
316 | 318 |
317 _cos = math.cos(x_rotate) | 319 _cos = math.cos(x_rotate) |
318 _sin = math.sin(x_rotate) | 320 _sin = math.sin(x_rotate) |
319 | 321 |
320 nrx = (x * _cos + y * _sin) / rx | 322 nrx = (x * _cos + y * _sin) / rx |
321 nry = (-x * _sin + y * _cos) / ry | 323 nry = (-x * _sin + y * _cos) / ry |
322 | 324 |
323 xy_tan = nry / nrx | 325 xy_tan = nry / nrx |
324 xy_angle = math.atan(xy_tan) | 326 xy_angle = math.atan(xy_tan) |
325 | 327 |
326 if nrx < 0: | 328 if nrx < 0: |
327 xy_angle = math.pi + xy_angle | 329 xy_angle = math.pi + xy_angle |
328 pass | 330 pass |
329 | 331 |
330 return xy_angle | 332 return xy_angle |
331 | 333 |
332 def rotate(x, y, angle): | 334 def rotate(x, y, angle): |
333 import math | 335 import math |
334 _cos = math.cos(angle) | 336 _cos = math.cos(angle) |
338 return nx, ny | 340 return nx, ny |
339 | 341 |
340 def translate_path_data(data, codefo): | 342 def translate_path_data(data, codefo): |
341 import string | 343 import string |
342 import math | 344 import math |
343 | 345 |
344 temp = data.split() | 346 temp = data.split() |
345 fields=[] | 347 fields=[] |
346 for f in temp: | 348 for f in temp: |
347 for s in f.split(','): | 349 for s in f.split(','): |
348 if s != '': | 350 if s != '': |
383 if cmd == 'v': | 385 if cmd == 'v': |
384 arg = float(f) | 386 arg = float(f) |
385 pnts.append(pnts[-2]) | 387 pnts.append(pnts[-2]) |
386 pnts.append(arg + pnts[-2]) | 388 pnts.append(arg + pnts[-2]) |
387 continue | 389 continue |
388 | 390 |
389 arg = float(f) | 391 arg = float(f) |
390 if (cmd not in 'am') and (cmd in string.lowercase): | 392 if (cmd not in 'am') and (cmd in string.lowercase): |
391 # relative and not arc or moveto | 393 # relative and not arc or moveto |
392 arg = arg + pnts[-2] | 394 arg = arg + pnts[-2] |
393 pass | 395 pass |
422 c2x, c2y = rotate(rx, ry, x_rotate) | 424 c2x, c2y = rotate(rx, ry, x_rotate) |
423 c2x, c2y = c2x + cx, c2y + cy | 425 c2x, c2y = c2x + cx, c2y + cy |
424 | 426 |
425 c3x, c3y = rotate(-rx, ry, x_rotate) | 427 c3x, c3y = rotate(-rx, ry, x_rotate) |
426 c3x, c3y = c3x + cx, c3y + cy | 428 c3x, c3y = c3x + cx, c3y + cy |
427 | 429 |
428 pnts[-7:] = [c0x, c0y, c1x, c1y, c2x, c2y, c3x, c3y, abs_x, abs_y] | 430 pnts[-7:] = [c0x, c0y, c1x, c1y, c2x, c2y, c3x, c3y, abs_x, abs_y] |
429 | 431 |
430 start_angle = _angle_rotated_ellipse(x0 - cx, y0 - cy, | 432 start_angle = _angle_rotated_ellipse(x0 - cx, y0 - cy, |
431 rx, ry, x_rotate) | 433 rx, ry, x_rotate) |
432 stop_angle = _angle_rotated_ellipse(x - cx, y - cy, | 434 stop_angle = _angle_rotated_ellipse(x - cx, y - cy, |
433 rx, ry, x_rotate) | 435 rx, ry, x_rotate) |
434 | 436 |
435 # sweep == 1 for positive-angle direction | 437 # sweep == 1 for positive-angle direction |
436 # sweep == 0 for negative-angle direction | 438 # sweep == 0 for negative-angle direction |
437 if start_angle > stop_angle and sweep: | 439 if start_angle > stop_angle and sweep: |
438 stop_angle = math.pi * 2 + stop_angle | 440 stop_angle = math.pi * 2 + stop_angle |
439 elif start_angle < stop_angle and not sweep: | 441 elif start_angle < stop_angle and not sweep: |
440 start_angle = math.pi * 2 + start_angle | 442 start_angle = math.pi * 2 + start_angle |
441 pass | 443 pass |
442 | 444 |
443 float_args.extend([cx, cy, rx, ry, | 445 float_args.extend([cx, cy, rx, ry, |
444 start_angle, stop_angle, x_rotate]) | 446 start_angle, stop_angle, x_rotate]) |
445 pass | 447 pass |
446 pass | 448 pass |
447 return commands, pnts, float_args | 449 return commands, pnts, float_args |
546 map = translate_font_style(tspan, codefo) | 548 map = translate_font_style(tspan, codefo) |
547 tspan.style_map = map | 549 tspan.style_map = map |
548 pass | 550 pass |
549 if tspan.hasAttribute('x'): | 551 if tspan.hasAttribute('x'): |
550 # Render the tspan as an independent text if the x | 552 # Render the tspan as an independent text if the x |
551 # attribute is defined. All elements inside | 553 # attribute is defined. All elements inside |
552 # the tspan will be ignore by the outter text or tspan elements. | 554 # the tspan will be ignore by the outter text or tspan elements. |
553 # FIXME: We need to apply the style map recursively. | 555 # FIXME: We need to apply the style map recursively. |
554 merge_style(tspan, text) | 556 merge_style(tspan, text) |
555 translate_text(tspan, coord_id, codefo, doc) | 557 translate_text(tspan, coord_id, codefo, doc) |
556 return '' | 558 return '' |
583 else: | 585 else: |
584 font_sz = float(style_map['font-size']) | 586 font_sz = float(style_map['font-size']) |
585 print >> codefo, 'PANGO_SIZE(%d,%d,%d)dnl' % (font_sz*1024,start,end) | 587 print >> codefo, 'PANGO_SIZE(%d,%d,%d)dnl' % (font_sz*1024,start,end) |
586 pass | 588 pass |
587 pass | 589 pass |
588 | 590 |
589 if style_map.has_key('font-style'): | 591 if style_map.has_key('font-style'): |
590 font_style = style_map['font-style'].lower() | 592 font_style = style_map['font-style'].lower() |
591 if font_style == 'normal': | 593 if font_style == 'normal': |
592 print >> codefo, 'PANGO_STYLE(PANGO_STYLE_NORMAL,%d,%d)dnl' % (start,end) | 594 print >> codefo, 'PANGO_STYLE(PANGO_STYLE_NORMAL,%d,%d)dnl' % (start,end) |
593 elif font_style == 'italic': | 595 elif font_style == 'italic': |
672 pass | 674 pass |
673 pass | 675 pass |
674 | 676 |
675 def stext_generate_font_attributes(text, attrs, coord_id, codefo, doc): | 677 def stext_generate_font_attributes(text, attrs, coord_id, codefo, doc): |
676 text_id = _get_id(text) | 678 text_id = _get_id(text) |
677 | 679 |
678 for start, end, node in attrs: | 680 for start, end, node in attrs: |
679 style_map = node.style_map | 681 style_map = node.style_map |
680 | 682 |
681 font_sz = 10 | 683 font_sz = 10 |
682 if style_map.has_key('font-size'): | 684 if style_map.has_key('font-size'): |
683 fsz = style_map['font-size'] | 685 fsz = style_map['font-size'] |
684 if fsz.endswith('px'): | 686 if fsz.endswith('px'): |
685 font_sz = float(fsz[:-2]) | 687 font_sz = float(fsz[:-2]) |
686 else: | 688 else: |
687 font_sz = float(fsz) | 689 font_sz = float(fsz) |
688 pass | 690 pass |
689 pass | 691 pass |
690 | 692 |
691 if style_map.has_key('font-family'): | 693 if style_map.has_key('font-family'): |
692 font_family = style_map['font-family'] | 694 font_family = style_map['font-family'] |
693 else: | 695 else: |
694 font_family = 'serif' | 696 font_family = 'serif' |
695 pass | 697 pass |
696 | 698 |
697 font_slant = 0 | 699 font_slant = 0 |
698 if style_map.has_key('font-style'): | 700 if style_map.has_key('font-style'): |
699 fn_style = style_map['font-style'] | 701 fn_style = style_map['font-style'] |
700 if fn_style == 'normal': | 702 if fn_style == 'normal': |
701 font_slant = 0 | 703 font_slant = 0 |
704 elif fn_style == 'oblique': | 706 elif fn_style == 'oblique': |
705 font_slant = 110 | 707 font_slant = 110 |
706 else: | 708 else: |
707 raise ValueError, '%s is not a valid font-style' % (fn_style) | 709 raise ValueError, '%s is not a valid font-style' % (fn_style) |
708 pass | 710 pass |
709 | 711 |
710 font_weight = 80 | 712 font_weight = 80 |
711 if style_map.has_key('font-weight'): | 713 if style_map.has_key('font-weight'): |
712 fn_weight = style_map['font-weight'] | 714 fn_weight = style_map['font-weight'] |
713 if fn_weight == 'normal': | 715 if fn_weight == 'normal': |
714 font_weight = 80 | 716 font_weight = 80 |
724 font_weight = 70 | 726 font_weight = 70 |
725 else: | 727 else: |
726 font_weight = int(fn_weight) | 728 font_weight = int(fn_weight) |
727 pass | 729 pass |
728 pass | 730 pass |
729 | 731 |
730 print >> codefo, 'STYLE_BLOCK([%s], %d, [%s], %f, %d, %d)dnl' % ( | 732 print >> codefo, 'STYLE_BLOCK([%s], %d, [%s], %f, %d, %d)dnl' % ( |
731 text_id, end - start, font_family, font_sz, | 733 text_id, end - start, font_family, font_sz, |
732 font_slant, font_weight) | 734 font_slant, font_weight) |
733 pass | 735 pass |
734 pass | 736 pass |
735 | 737 |
736 def stext_gen_text(text, coord_id, codefo, doc, txt_strs, attrs): | 738 def stext_gen_text(text, coord_id, codefo, doc, txt_strs, attrs): |
737 if not txt_strs: | 739 if not txt_strs: |
738 return | 740 return |
739 | 741 |
740 text_id = _get_id(text) | 742 text_id = _get_id(text) |
741 x = float(text.getAttribute('x')) | 743 x = float(text.getAttribute('x')) |
742 y = float(text.getAttribute('y')) | 744 y = float(text.getAttribute('y')) |
743 print >> codefo, 'dnl' | 745 print >> codefo, 'dnl' |
744 print >> codefo, \ | 746 print >> codefo, \ |
755 @check_mbname | 757 @check_mbname |
756 def translate_text(text, coord_id, codefo, doc): | 758 def translate_text(text, coord_id, codefo, doc): |
757 coord_id = translate_shape_transform(text, coord_id, codefo) | 759 coord_id = translate_shape_transform(text, coord_id, codefo) |
758 try: | 760 try: |
759 map = text.style_map | 761 map = text.style_map |
760 except: | 762 except: |
761 map = translate_font_style(text, codefo) | 763 map = translate_font_style(text, codefo) |
762 text.style_map = map | 764 text.style_map = map |
763 pass | 765 pass |
764 attrs = [] | 766 attrs = [] |
765 attr = [0, 0, text] | 767 attr = [0, 0, text] |
779 pass | 781 pass |
780 | 782 |
781 @check_mbname | 783 @check_mbname |
782 def translate_image(image, coord_id, codefo, doc): | 784 def translate_image(image, coord_id, codefo, doc): |
783 coord_id = translate_shape_transform(image, coord_id, codefo) | 785 coord_id = translate_shape_transform(image, coord_id, codefo) |
784 | 786 |
785 image_id = _get_id(image) | 787 image_id = _get_id(image) |
786 if not image.hasAttributeNS(xlinkns, 'href'): | 788 if not image.hasAttributeNS(xlinkns, 'href'): |
787 raise ValueError, 'image %s must has a href attribute.' % (image_id) | 789 raise ValueError, 'image %s must has a href attribute.' % (image_id) |
788 href = image.getAttributeNS(xlinkns, 'href') | 790 href = image.getAttributeNS(xlinkns, 'href') |
789 if image.hasAttribute('x'): | 791 if image.hasAttribute('x'): |
813 print >> codefo, 'dnl' | 815 print >> codefo, 'dnl' |
814 print >> codefo, \ | 816 print >> codefo, \ |
815 'ADD_IMAGE([%s], [%s], %f, %f, %f, %f, [%s])dnl' % ( | 817 'ADD_IMAGE([%s], [%s], %f, %f, %f, %f, [%s])dnl' % ( |
816 image_id, href, x, y, width, height, coord_id) | 818 image_id, href, x, y, width, height, coord_id) |
817 pass | 819 pass |
818 | 820 |
819 | 821 |
820 reo_func = re.compile('([a-zA-Z]+)\\([^\\)]*\\)') | 822 reo_func = re.compile('([a-zA-Z]+)\\([^\\)]*\\)') |
821 reo_translate = re.compile('translate\\(([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?)\\)') | 823 reo_translate = re.compile('translate\\(([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?)\\)') |
822 reo_matrix = re.compile('matrix\\(([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?)\\)') | 824 reo_matrix = re.compile('matrix\\(([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?),([-+]?[0-9]+(\\.[0-9]+)?)\\)') |
823 def translate_transform(coord_id, transform, codefo, prefix): | 825 def translate_transform(coord_id, transform, codefo, prefix): |
887 for scene in scenes_node.childNodes: | 889 for scene in scenes_node.childNodes: |
888 if scene.localName != 'scene' or \ | 890 if scene.localName != 'scene' or \ |
889 not scene.hasAttribute('ref') or \ | 891 not scene.hasAttribute('ref') or \ |
890 not scene.hasAttribute('start'): | 892 not scene.hasAttribute('start'): |
891 continue | 893 continue |
892 | 894 |
893 start_str = scene.getAttribute('start') | 895 start_str = scene.getAttribute('start') |
894 start = end = int(start_str) | 896 start = end = int(start_str) |
895 if scene.hasAttribute('end'): | 897 if scene.hasAttribute('end'): |
896 end_str = scene.getAttribute('end') | 898 end_str = scene.getAttribute('end') |
897 end = int(end_str) | 899 end = int(end_str) |
898 pass | 900 pass |
899 ref = scene.getAttribute('ref') | 901 ref = scene.getAttribute('ref') |
900 | 902 |
901 while len(scenes) <= end: | 903 while len(scenes) <= end: |
902 scenes.append([]) | 904 scenes.append([]) |
903 pass | 905 pass |
904 | 906 |
905 for i in range(start, end + 1): | 907 for i in range(start, end + 1): |
906 scenes[i].append(ref) | 908 scenes[i].append(ref) |
907 pass | 909 pass |
908 pass | 910 pass |
909 | 911 |
923 if node.localName == 'svg' and node.namespaceURI == svgns: | 925 if node.localName == 'svg' and node.namespaceURI == svgns: |
924 break; | 926 break; |
925 pass | 927 pass |
926 else: | 928 else: |
927 raise ValueErr, 'no any svg tag node.' | 929 raise ValueErr, 'no any svg tag node.' |
928 | 930 |
929 svg = node | 931 svg = node |
930 for node in svg.childNodes: | 932 for node in svg.childNodes: |
931 if node.localName == 'defs' and node.namespaceURI == svgns: | 933 if node.localName == 'defs' and node.namespaceURI == svgns: |
932 translate_defs(node, codefo, dom) | 934 translate_defs(node, codefo, dom) |
933 elif node.localName == 'metadata' and node.namespaceURI == svgns: | 935 elif node.localName == 'metadata' and node.namespaceURI == svgns: |
944 pass | 946 pass |
945 | 947 |
946 if __name__ == '__main__': | 948 if __name__ == '__main__': |
947 from os import path | 949 from os import path |
948 import optparse | 950 import optparse |
949 | 951 |
950 usage='usage: %prog [options] <SVG file> [<output>]' | 952 usage='usage: %prog [options] <SVG file> [<output>]' |
951 parser = optparse.OptionParser(usage=usage) | 953 parser = optparse.OptionParser(usage=usage) |
952 parser.add_option('-s', '--stext', dest='stext', | 954 parser.add_option('-s', '--stext', dest='stext', |
953 action='store_true', default=False, | 955 action='store_true', default=False, |
954 help='Use sh_stext instead of sh_text'); | 956 help='Use sh_stext instead of sh_text'); |
955 options, args = parser.parse_args() | 957 options, args = parser.parse_args() |
956 | 958 |
957 if len(args) == 2: | 959 if len(args) == 2: |
958 svgfn = args[0] | 960 svgfn = args[0] |
959 codefn = args[1] | 961 codefn = args[1] |
960 elif len(args) == 1: | 962 elif len(args) == 1: |
961 svgfn = args[0] | 963 svgfn = args[0] |
962 codefn = 'out.mb' | 964 codefn = 'out.mb' |
963 else: | 965 else: |
964 parser.print_help() | 966 parser.print_help() |
965 sys.exit(1) | 967 sys.exit(1) |
966 pass | 968 pass |
967 | 969 |
968 struct_name = path.basename(codefn).split('.')[0] | 970 struct_name = path.basename(codefn).split('.')[0] |
969 | 971 |
970 if options.stext: | 972 if options.stext: |
971 gen_text = stext_gen_text | 973 gen_text = stext_gen_text |
972 else: | 974 else: |