11ae29591efbb29492ce05378909ccf4028d7c1eeBehdad Esfahbodfrom __future__ import print_function, division, absolute_import
230e691edd056ba22fa8970280e986747817bec3dBehdad Esfahbodfrom fontTools.misc.py23 import *
37842e56b97ce677b83bdab09cda48bc2d89ac75aJustfrom fontTools import ttLib
47842e56b97ce677b83bdab09cda48bc2d89ac75aJustfrom fontTools.misc.textTools import safeEval
5f8fd4777d273836a1222b72f6761cb6fdf9ec87aJustfrom fontTools.ttLib.tables.DefaultTable import DefaultTable
64bb05c6ad4b112de40642d77663aa87d145f863fjvrimport os
77842e56b97ce677b83bdab09cda48bc2d89ac75aJust
87842e56b97ce677b83bdab09cda48bc2d89ac75aJust
9ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvrclass TTXParseError(Exception): pass
107842e56b97ce677b83bdab09cda48bc2d89ac75aJust
119b7b8e0f5b5f968acb95f8c3a484ea8d42924d65jvrBUFSIZE = 0x4000
129b7b8e0f5b5f968acb95f8c3a484ea8d42924d65jvr
137842e56b97ce677b83bdab09cda48bc2d89ac75aJust
14e388db566b9ba42669c7e353db4293cf27bc2a5bBehdad Esfahbodclass XMLReader(object):
157842e56b97ce677b83bdab09cda48bc2d89ac75aJust
163ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod	def __init__(self, fileName, ttFont, progress=None, quiet=False):
177842e56b97ce677b83bdab09cda48bc2d89ac75aJust		self.ttFont = ttFont
184bb05c6ad4b112de40642d77663aa87d145f863fjvr		self.fileName = fileName
197842e56b97ce677b83bdab09cda48bc2d89ac75aJust		self.progress = progress
2085af40e7d8c4f4f4a3df3562a5ec9e5731cfe43fDave Crossland		self.quiet = quiet
217842e56b97ce677b83bdab09cda48bc2d89ac75aJust		self.root = None
22ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr		self.contentStack = []
23ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr		self.stackSize = 0
24ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr
253ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod	def read(self):
263ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod		if self.progress:
273ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod			import stat
28153ec402094adbea673e914385b87f1d99191d0bBehdad Esfahbod			self.progress.set(0, os.stat(self.fileName)[stat.ST_SIZE] // 100 or 1)
29cc43f3a15c1012bc593b06eca06b694ab5d59dddBehdad Esfahbod		file = open(self.fileName)
30cc43f3a15c1012bc593b06eca06b694ab5d59dddBehdad Esfahbod		self._parseFile(file)
3156bfea43176d03cede26f8de51638c4ab9f4bd64jvr		file.close()
3256bfea43176d03cede26f8de51638c4ab9f4bd64jvr
333ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod	def _parseFile(self, file):
34e493669d196f292f24d4c09ce340c3e67749bce5jvr		from xml.parsers.expat import ParserCreate
354b3df49b5b70b3618e91e3ab954cedca773831d6Behdad Esfahbod		parser = ParserCreate()
363ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod		parser.StartElementHandler = self._startElementHandler
373ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod		parser.EndElementHandler = self._endElementHandler
383ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod		parser.CharacterDataHandler = self._characterDataHandler
39e493669d196f292f24d4c09ce340c3e67749bce5jvr
409b7b8e0f5b5f968acb95f8c3a484ea8d42924d65jvr		pos = 0
41ac1b4359467ca3deab03186a15eae1d55eb35567Behdad Esfahbod		while True:
429b7b8e0f5b5f968acb95f8c3a484ea8d42924d65jvr			chunk = file.read(BUFSIZE)
439b7b8e0f5b5f968acb95f8c3a484ea8d42924d65jvr			if not chunk:
449b7b8e0f5b5f968acb95f8c3a484ea8d42924d65jvr				parser.Parse(chunk, 1)
459b7b8e0f5b5f968acb95f8c3a484ea8d42924d65jvr				break
469b7b8e0f5b5f968acb95f8c3a484ea8d42924d65jvr			pos = pos + len(chunk)
479b7b8e0f5b5f968acb95f8c3a484ea8d42924d65jvr			if self.progress:
4832c10eecffb4923e0721c395e4b80fb732543f18Behdad Esfahbod				self.progress.set(pos // 100)
499b7b8e0f5b5f968acb95f8c3a484ea8d42924d65jvr			parser.Parse(chunk, 0)
50ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr
513ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod	def _startElementHandler(self, name, attrs):
52ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr		stackSize = self.stackSize
53e82b4d5658a7a38cf92b4b9392861257a2cfbb36jvr		self.stackSize = stackSize + 1
54f8fd4777d273836a1222b72f6761cb6fdf9ec87aJust		if not stackSize:
55180ace6a5ff1399ec53bc696e8bef7cce6eef39aBehdad Esfahbod			if name != "ttFont":
56cd5aad92f23737ff93a110d5c73d624658a28da8Behdad Esfahbod				raise TTXParseError("illegal root tag: %s" % name)
574bb05c6ad4b112de40642d77663aa87d145f863fjvr			sfntVersion = attrs.get("sfntVersion")
584bb05c6ad4b112de40642d77663aa87d145f863fjvr			if sfntVersion is not None:
59180ace6a5ff1399ec53bc696e8bef7cce6eef39aBehdad Esfahbod				if len(sfntVersion) != 4:
604bb05c6ad4b112de40642d77663aa87d145f863fjvr					sfntVersion = safeEval('"' + sfntVersion + '"')
614bb05c6ad4b112de40642d77663aa87d145f863fjvr				self.ttFont.sfntVersion = sfntVersion
62ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr			self.contentStack.append([])
63f8fd4777d273836a1222b72f6761cb6fdf9ec87aJust		elif stackSize == 1:
644bb05c6ad4b112de40642d77663aa87d145f863fjvr			subFile = attrs.get("src")
654bb05c6ad4b112de40642d77663aa87d145f863fjvr			if subFile is not None:
664bb05c6ad4b112de40642d77663aa87d145f863fjvr				subFile = os.path.join(os.path.dirname(self.fileName), subFile)
673ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod				subReader = XMLReader(subFile, self.ttFont, self.progress, self.quiet)
683ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod				subReader.read()
694bb05c6ad4b112de40642d77663aa87d145f863fjvr				self.contentStack.append([])
704bb05c6ad4b112de40642d77663aa87d145f863fjvr				return
710011bb691018c7feed1f8c3554d88b9ae096e127jvr			tag = ttLib.xmlToTag(name)
720011bb691018c7feed1f8c3554d88b9ae096e127jvr			msg = "Parsing '%s' table..." % tag
737842e56b97ce677b83bdab09cda48bc2d89ac75aJust			if self.progress:
747842e56b97ce677b83bdab09cda48bc2d89ac75aJust				self.progress.setlabel(msg)
757842e56b97ce677b83bdab09cda48bc2d89ac75aJust			elif self.ttFont.verbose:
767842e56b97ce677b83bdab09cda48bc2d89ac75aJust				ttLib.debugmsg(msg)
777842e56b97ce677b83bdab09cda48bc2d89ac75aJust			else:
7885af40e7d8c4f4f4a3df3562a5ec9e5731cfe43fDave Crossland				if not self.quiet:
793ec6a258238b6068e4eef3fe579f1f5c0a06bbbaBehdad Esfahbod					print(msg)
800011bb691018c7feed1f8c3554d88b9ae096e127jvr			if tag == "GlyphOrder":
810011bb691018c7feed1f8c3554d88b9ae096e127jvr				tableClass = ttLib.GlyphOrder
82e3d7d71b7a9b1c4afae1110b27cf3254c41acfb6Behdad Esfahbod			elif "ERROR" in attrs or ('raw' in attrs and safeEval(attrs['raw'])):
83f8fd4777d273836a1222b72f6761cb6fdf9ec87aJust				tableClass = DefaultTable
84f8fd4777d273836a1222b72f6761cb6fdf9ec87aJust			else:
85f8fd4777d273836a1222b72f6761cb6fdf9ec87aJust				tableClass = ttLib.getTableClass(tag)
86f8fd4777d273836a1222b72f6761cb6fdf9ec87aJust				if tableClass is None:
87f8fd4777d273836a1222b72f6761cb6fdf9ec87aJust					tableClass = DefaultTable
88bc5e1cb195c0bfa1c8e7507326d5a9ad05aecb4bBehdad Esfahbod			if tag == 'loca' and tag in self.ttFont:
89d57c4346e5725bbfe0e1b13067b292c3faaaeb8ejvr				# Special-case the 'loca' table as we need the
90e82b4d5658a7a38cf92b4b9392861257a2cfbb36jvr				#    original if the 'glyf' table isn't recompiled.
91ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr				self.currentTable = self.ttFont[tag]
921670f7af0edf5af2447f9fb157febe8cfb44b629Just			else:
93ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr				self.currentTable = tableClass(tag)
94ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr				self.ttFont[tag] = self.currentTable
95ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr			self.contentStack.append([])
96f8fd4777d273836a1222b72f6761cb6fdf9ec87aJust		elif stackSize == 2:
97ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr			self.contentStack.append([])
98ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr			self.root = (name, attrs, self.contentStack[-1])
997842e56b97ce677b83bdab09cda48bc2d89ac75aJust		else:
100153ec402094adbea673e914385b87f1d99191d0bBehdad Esfahbod			l = []
101153ec402094adbea673e914385b87f1d99191d0bBehdad Esfahbod			self.contentStack[-1].append((name, attrs, l))
102153ec402094adbea673e914385b87f1d99191d0bBehdad Esfahbod			self.contentStack.append(l)
1037842e56b97ce677b83bdab09cda48bc2d89ac75aJust
1043ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod	def _characterDataHandler(self, data):
105ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr		if self.stackSize > 1:
106ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr			self.contentStack[-1].append(data)
1077842e56b97ce677b83bdab09cda48bc2d89ac75aJust
1083ebfea491ea795217e0f9649c23ee81df769a250Behdad Esfahbod	def _endElementHandler(self, name):
109ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr		self.stackSize = self.stackSize - 1
110ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr		del self.contentStack[-1]
111ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr		if self.stackSize == 1:
1127842e56b97ce677b83bdab09cda48bc2d89ac75aJust			self.root = None
113ca4c45681ef2ea9290c6845f8bf61dff281fc5c4jvr		elif self.stackSize == 2:
1143a9fd301808f5a8991ca9ac44028d1ecb22d307fBehdad Esfahbod			name, attrs, content = self.root
1153a9fd301808f5a8991ca9ac44028d1ecb22d307fBehdad Esfahbod			self.currentTable.fromXML(name, attrs, content, self.ttFont)
1167842e56b97ce677b83bdab09cda48bc2d89ac75aJust			self.root = None
1177842e56b97ce677b83bdab09cda48bc2d89ac75aJust
1187842e56b97ce677b83bdab09cda48bc2d89ac75aJust
119e388db566b9ba42669c7e353db4293cf27bc2a5bBehdad Esfahbodclass ProgressPrinter(object):
1207842e56b97ce677b83bdab09cda48bc2d89ac75aJust
1217842e56b97ce677b83bdab09cda48bc2d89ac75aJust	def __init__(self, title, maxval=100):
1223ec6a258238b6068e4eef3fe579f1f5c0a06bbbaBehdad Esfahbod		print(title)
1237842e56b97ce677b83bdab09cda48bc2d89ac75aJust
1247842e56b97ce677b83bdab09cda48bc2d89ac75aJust	def set(self, val, maxval=None):
1257842e56b97ce677b83bdab09cda48bc2d89ac75aJust		pass
1267842e56b97ce677b83bdab09cda48bc2d89ac75aJust
1277842e56b97ce677b83bdab09cda48bc2d89ac75aJust	def increment(self, val=1):
1287842e56b97ce677b83bdab09cda48bc2d89ac75aJust		pass
1297842e56b97ce677b83bdab09cda48bc2d89ac75aJust
1306ab979cacaa9c15666a526e05c669b7f87bb6de9jvr	def setLabel(self, text):
1313ec6a258238b6068e4eef3fe579f1f5c0a06bbbaBehdad Esfahbod		print(text)
1327842e56b97ce677b83bdab09cda48bc2d89ac75aJust
133