-
Notifications
You must be signed in to change notification settings - Fork 40
/
Copy pathgenerate.py
executable file
·107 lines (78 loc) · 3.08 KB
/
generate.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
#! /usr/bin/python
import pyqrcode
from lxml import etree
import math
import sys
import logging
logging.basicConfig(level=logging.DEBUG if __debug__ else logging.INFO)
block_size = 10
circle_radius = block_size * 4
def distance(p0, p1):
return math.sqrt((p0[0] - p1[0])**2 + (p0[1] - p1[1])**2)
def generateQRImageForUrl(url):
qr_image = pyqrcode.MakeQRImage(url, errorCorrectLevel = pyqrcode.QRErrorCorrectLevel.H, block_in_pixels = 1, border_in_blocks=0)
return qr_image
def getSVGFileContent(filename):
'''
root may be the svg element itself, so search from tree
solution for multiple (or no) namespaces from
http://stackoverflow.com/a/14552559/493161
'''
document = etree.parse(filename)
logging.debug('document: %s', document)
svg = document.xpath('//*[local-name()="svg"]')[0]
logging.debug('svg: %s', svg)
return svg
def touchesBounds(center, x, y, radius, block_size):
scaled_center = center / block_size
dis = distance((scaled_center , scaled_center), (x, y))
rad = radius / block_size
return dis <= rad + 1
if len(sys.argv) < 4:
print "Incorrect args, try:"
print './generate.py ./octocat.svg "http://github.com" ./out.svg'
sys.exit(0)
logoPath = sys.argv[1]
url = sys.argv[2]
outputname = sys.argv[3]
im = generateQRImageForUrl(url);
imageSize = str(im.size[0] * block_size)
# create an SVG XML element (see the SVG specification for attribute details)
doc = etree.Element('svg', width=imageSize, height=imageSize, version='1.1', xmlns='http://www.w3.org/2000/svg')
pix = im.load()
center = im.size[0] * block_size / 2
for xPos in range(0,im.size[0]):
for yPos in range(0, im.size[1]):
color = pix[xPos, yPos]
if color == (0,0,0,255):
withinBounds = not touchesBounds(center, xPos, yPos, circle_radius, block_size)
if (withinBounds):
etree.SubElement(doc, 'rect', x=str(xPos*block_size), y=str(yPos*block_size), width='10', height='10', fill='black')
logo = getSVGFileContent(logoPath)
test = str(logo.get("viewBox"))
Array = []
if (test != "None"):
Array = test.split(" ")
width = float(Array[2])
height = float(Array[3])
else :
width = float(str(logo.get("width")).replace("px", ""))
height = float(str(logo.get("height")).replace("px", ""))
dim = height
if (width > dim):
dim = width
scale = circle_radius * 2.0 / width
scale_str = "scale(" + str(scale) + ")"
xTrans = ((im.size[0] * block_size) - (width * scale)) / 2.0
yTrans = ((im.size[1] * block_size) - (height * scale)) / 2.0
translate = "translate(" + str(xTrans) + " " + str(yTrans) + ")"
logo_scale_container = etree.SubElement(doc, 'g', transform=translate + " " + scale_str)
for element in logo.getchildren():
logo_scale_container.append(element)
# ElementTree 1.2 doesn't write the SVG file header errata, so do that manually
f = open(outputname, 'w')
f.write('<?xml version=\"1.0\" standalone=\"no\"?>\n')
f.write('<!DOCTYPE svg PUBLIC \"-//W3C//DTD SVG 1.1//EN\"\n')
f.write('\"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd\">\n')
f.write(etree.tostring(doc))
f.close()