summaryrefslogtreecommitdiffstats
path: root/examples/psftoneletters.py
diff options
context:
space:
mode:
Diffstat (limited to 'examples/psftoneletters.py')
-rw-r--r--examples/psftoneletters.py327
1 files changed, 327 insertions, 0 deletions
diff --git a/examples/psftoneletters.py b/examples/psftoneletters.py
new file mode 100644
index 0000000..b6f7b7a
--- /dev/null
+++ b/examples/psftoneletters.py
@@ -0,0 +1,327 @@
+#!/usr/bin/env python3
+from __future__ import unicode_literals
+'''Creates Latin script tone letters (pitch contours)'''
+__url__ = 'https://github.com/silnrsi/pysilfont'
+__copyright__ = 'Copyright (c) 2017 SIL International (https://www.sil.org)'
+__license__ = 'Released under the MIT License (https://opensource.org/licenses/MIT)'
+__author__ = 'Victor Gaultney'
+
+# Usage: psftoneletters ifont ofont
+# Assumption is that the named tone letters already exist in the font,
+# so this script is only to update (rebuild) them. New tone letter spaces
+# in the font can be created with psfbuildcomp.py
+
+# To Do
+# Get parameters from lib.plist org.sil.lcg.toneLetters
+
+# main input, output, and execution handled by pysilfont framework
+from silfont.core import execute
+import silfont.ufo as UFO
+
+from robofab.world import OpenFont
+
+from math import tan, radians, sqrt
+
+suffix = '_toneletters'
+argspec = [
+ ('ifont',{'help': 'Input font file'}, {'type': 'filename'}),
+ ('ofont',{'help': 'Output font file','nargs': '?' }, {'type': 'filename', 'def': "_"+suffix}),
+ ('-l','--log',{'help': 'Log file'}, {'type': 'outfile', 'def': suffix+'log'})]
+
+
+def getParameters(font):
+ global glyphHeight, marginFlatLeft, marginPointLeft, marginFlatRight, marginPointRight, contourWidth, marginDotLeft, marginDotRight, dotSpacing, italicAngle, radius, strokeHeight, strokeDepth, contourGap, fakeBottom, dotRadius, dotBCP, contourGapDot, fakeBottomDot, anchorHeight, anchorOffset
+
+ source = font.lib.getval("org.sil.lcg.toneLetters")
+
+ strokeThickness = int(source["strokeThickness"]) # total width of stroke (ideally an even number)
+ glyphHeight = int(source["glyphHeight"]) # height, including overshoot
+ glyphDepth = int(source["glyphDepth"]) # depth - essentially overshoot (typically negative)
+ marginFlatLeft = int(source["marginFlatLeft"]) # left sidebearing for straight bar
+ marginPointLeft = int(source["marginPointLeft"]) # left sidebearing for endpoints
+ marginFlatRight = int(source["marginFlatRight"]) # left sidebearing for straight bar
+ marginPointRight = int(source["marginPointRight"]) # left sidebearing for endpoints
+ contourWidth = int(source["contourWidth"]) # this is how wide the contour portions are, from the middle
+ # of one end to the other, in the horizontal axis. The actual
+ # bounding box of the contours would then be this plus the
+ # strokeThickness.
+ marginDotLeft = int(source["marginDotLeft"]) # left sidebearing for dots
+ marginDotRight = int(source["marginDotRight"]) # right sidebearing for dots
+ dotSize = int(source["dotSize"]) # the diameter of the dot, normally 150% of the stroke weight
+ # (ideally an even number)
+ dotSpacing = int(source["dotSpacing"]) # the space between the edge of the dot and the
+ # edge of the expanded stroke
+ italicAngle = float(source["italicAngle"]) # angle of italic slant, 0 for upright
+
+ radius = round(strokeThickness / 2)
+ strokeHeight = glyphHeight - radius # for the unexpanded stroke
+ strokeDepth = glyphDepth + radius
+ strokeLength = strokeHeight - strokeDepth
+ contourGap = round(strokeLength / 4) # gap between contour levels
+ fakeBottom = strokeDepth - contourGap # a false 'bottom' for building contours
+
+ dotRadius = round(dotSize / 2) # this gets redefined during nine tone process
+ dotBCP = round((dotSize / 2) * .55) # this gets redefined during nine tone process
+ contourGapDot = round(( (glyphHeight - dotRadius) - (glyphDepth + dotRadius) ) / 4)
+ fakeBottomDot = (glyphDepth + dotRadius) - contourGapDot
+
+ anchorHeight = [ 0 , strokeDepth , (strokeDepth + contourGap) , (strokeDepth + contourGap * 2) , (strokeHeight - contourGap) , strokeHeight ]
+ anchorOffset = 20 # hardcoded for now
+
+# drawing functions
+
+def drawLine(glyph,startX,startY,endX,endY):
+
+ dx = (endX - startX) # dx of original stroke
+ dy = (endY - startY) # dy of original stroke
+ len = sqrt( dx * dx + dy * dy ) # length of original stroke
+ opp = round(dy * (radius / len)) # offsets for on-curve points
+ adj = round(dx * (radius / len))
+ oppOff = round(opp * .55) # offsets for off-curve from on-curve
+ adjOff = round(adj * .55)
+
+ glyph.clearContours()
+
+ pen = glyph.getPen()
+
+ # print startX + opp, startY - adj
+
+ pen.moveTo((startX + opp, startY - adj))
+ pen.lineTo((endX + opp, endY - adj)) # first straight line
+
+ bcp1x = endX + opp + adjOff
+ bcp1y = endY - adj + oppOff
+ bcp2x = endX + adj + oppOff
+ bcp2y = endY + opp - adjOff
+ pen.curveTo((bcp1x, bcp1y), (bcp2x, bcp2y), (endX + adj, endY + opp))
+
+ bcp1x = endX + adj - oppOff
+ bcp1y = endY + opp + adjOff
+ bcp2x = endX - opp + adjOff
+ bcp2y = endY + adj + oppOff
+ pen.curveTo((bcp1x, bcp1y), (bcp2x, bcp2y), (endX - opp, endY + adj))
+
+ pen.lineTo((startX - opp, startY + adj)) # second straight line
+
+ bcp1x = startX - opp - adjOff
+ bcp1y = startY + adj - oppOff
+ bcp2x = startX - adj - oppOff
+ bcp2y = startY - opp + adjOff
+ pen.curveTo((bcp1x, bcp1y), (bcp2x, bcp2y), (startX - adj, startY - opp))
+
+ bcp1x = startX - adj + oppOff
+ bcp1y = startY - opp - adjOff
+ bcp2x = startX + opp - adjOff
+ bcp2y = startY - adj - oppOff
+ pen.curveTo((bcp1x, bcp1y), (bcp2x, bcp2y), (startX + opp, startY - adj))
+ # print startX + opp, startY - adj
+
+ pen.closePath()
+
+
+def drawDot(glyph,dotX,dotY):
+
+ glyph.clearContours()
+
+ pen = glyph.getPen()
+
+ pen.moveTo((dotX, dotY - dotRadius))
+ pen.curveTo((dotX + dotBCP, dotY - dotRadius), (dotX + dotRadius, dotY - dotBCP), (dotX + dotRadius, dotY))
+ pen.curveTo((dotX + dotRadius, dotY + dotBCP), (dotX + dotBCP, dotY + dotRadius), (dotX, dotY + dotRadius))
+ pen.curveTo((dotX - dotBCP, dotY + dotRadius), (dotX - dotRadius, dotY + dotBCP), (dotX - dotRadius, dotY))
+ pen.curveTo((dotX - dotRadius, dotY - dotBCP), (dotX - dotBCP, dotY - dotRadius), (dotX, dotY - dotRadius))
+ pen.closePath()
+
+
+def adjItalX(aiX,aiY):
+ newX = aiX + round(tan(radians(italicAngle)) * aiY)
+ return newX
+
+
+def buildComp(f,g,pieces,ancLevelLeft,ancLevelMidLeft,ancLevelMidRight,ancLevelRight):
+
+ g.clear()
+ g.width = 0
+
+ for p in pieces:
+ g.appendComponent(p, (g.width, 0))
+ g.width += f[p].width
+
+ if ancLevelLeft > 0:
+ anc_nm = "_TL"
+ anc_x = adjItalX(0,anchorHeight[ancLevelLeft])
+ if g.name[0:7] == 'TnStaff':
+ anc_x = anc_x - anchorOffset
+ anc_y = anchorHeight[ancLevelLeft]
+ g.appendAnchor(anc_nm, (anc_x, anc_y))
+
+ if ancLevelMidLeft > 0:
+ anc_nm = "_TL"
+ anc_x = adjItalX(marginPointLeft + radius,anchorHeight[ancLevelMidLeft])
+ anc_y = anchorHeight[ancLevelMidLeft]
+ g.appendAnchor(anc_nm, (anc_x, anc_y))
+
+ if ancLevelMidRight > 0:
+ anc_nm = "TL"
+ anc_x = adjItalX(g.width - marginPointRight - radius,anchorHeight[ancLevelMidRight])
+ anc_y = anchorHeight[ancLevelMidRight]
+ g.appendAnchor(anc_nm, (anc_x, anc_y))
+
+ if ancLevelRight > 0:
+ anc_nm = "TL"
+ anc_x = adjItalX(g.width,anchorHeight[ancLevelRight])
+ if g.name[0:7] == 'TnStaff':
+ anc_x = anc_x + anchorOffset
+ anc_y = anchorHeight[ancLevelRight]
+ g.appendAnchor(anc_nm, (anc_x, anc_y))
+
+
+# updating functions
+
+def updateTLPieces(targetfont):
+
+ f = targetfont
+
+ # set spacer widths
+ f["TnLtrSpcFlatLeft"].width = marginFlatLeft + radius
+ f["TnLtrSpcPointLeft"].width = marginPointLeft + radius - 1 # -1 corrects final sidebearing
+ f["TnLtrSpcFlatRight"].width = marginFlatRight + radius
+ f["TnLtrSpcPointRight"].width = marginPointRight + radius - 1 # -1 corrects final sidebearing
+ f["TnLtrSpcDotLeft"].width = marginDotLeft + dotRadius
+ f["TnLtrSpcDotMiddle"].width = dotRadius + dotSpacing + radius
+ f["TnLtrSpcDotRight"].width = dotRadius + marginDotRight
+
+ # redraw bar
+ g = f["TnLtrBar"]
+ drawLine(g,adjItalX(0,strokeDepth),strokeDepth,adjItalX(0,strokeHeight),strokeHeight)
+ g.width = 0
+
+ # redraw contours
+ namePre = 'TnLtrSeg'
+ for i in range(1,6):
+ for j in range(1,6):
+
+ nameFull = namePre + str(i) + str(j)
+
+ if i == 5: # this deals with round off errors
+ startLevel = strokeHeight
+ else:
+ startLevel = fakeBottom + i * contourGap
+ if j == 5:
+ endLevel = strokeHeight
+ else:
+ endLevel = fakeBottom + j * contourGap
+
+ g = f[nameFull]
+ g.width = contourWidth
+ drawLine(g,adjItalX(1,startLevel),startLevel,adjItalX(contourWidth-1,endLevel),endLevel)
+
+
+ # redraw dots
+ namePre = 'TnLtrDot'
+ for i in range(1,6):
+
+ nameFull = namePre + str(i)
+
+ if i == 5: # this deals with round off errors
+ dotLevel = glyphHeight - dotRadius
+ else:
+ dotLevel = fakeBottomDot + i * contourGapDot
+
+ g = f[nameFull]
+ drawDot(g,adjItalX(0,dotLevel),dotLevel)
+
+
+def rebuildTLComps(targetfont):
+
+ f = targetfont
+
+ # staff right
+ for i in range(1,6):
+ nameFull = 'TnStaffRt' + str(i)
+ buildComp(f,f[nameFull],['TnLtrBar','TnLtrSpcFlatRight'],i,0,0,0)
+
+ # staff right no outline
+ for i in range(1,6):
+ nameFull = 'TnStaffRt' + str(i) + 'no'
+ buildComp(f,f[nameFull],['TnLtrSpcFlatRight'],i,0,0,0)
+
+ # staff left
+ for i in range(1,6):
+ nameFull = 'TnStaffLft' + str(i)
+ buildComp(f,f[nameFull],['TnLtrSpcFlatLeft','TnLtrBar'],0,0,0,i)
+
+ # staff left no outline
+ for i in range(1,6):
+ nameFull = 'TnStaffLft' + str(i) + 'no'
+ buildComp(f,f[nameFull],['TnLtrSpcFlatLeft'],0,0,0,i)
+
+ # contours right
+ for i in range(1,6):
+ for j in range(1,6):
+ nameFull = 'TnContRt' + str(i) + str(j)
+ segment = 'TnLtrSeg' + str(i) + str(j)
+ buildComp(f,f[nameFull],['TnLtrSpcPointLeft',segment],0,i,0,j)
+
+ # contours left
+ for i in range(1,6):
+ for j in range(1,6):
+ nameFull = 'TnContLft' + str(i) + str(j)
+ segment = 'TnLtrSeg' + str(i) + str(j)
+ buildComp(f,f[nameFull],[segment,'TnLtrSpcPointRight'],i,0,j,0)
+
+ # basic tone letters
+ for i in range(1,6):
+ nameFull = 'TnLtr' + str(i)
+ segment = 'TnLtrSeg' + str(i) + str(i)
+ buildComp(f,f[nameFull],['TnLtrSpcPointLeft',segment,'TnLtrBar','TnLtrSpcFlatRight'],0,0,0,0)
+
+ # basic tone letters no outline
+ for i in range(1,6):
+ nameFull = 'TnLtr' + str(i) + 'no'
+ segment = 'TnLtrSeg' + str(i) + str(i)
+ buildComp(f,f[nameFull],['TnLtrSpcPointLeft',segment,'TnLtrSpcFlatRight'],0,i,0,0)
+
+ # left stem tone letters
+ for i in range(1,6):
+ nameFull = 'LftStemTnLtr' + str(i)
+ segment = 'TnLtrSeg' + str(i) + str(i)
+ buildComp(f,f[nameFull],['TnLtrSpcFlatLeft','TnLtrBar',segment,'TnLtrSpcPointRight'],0,0,0,0)
+
+ # left stem tone letters no outline
+ for i in range(1,6):
+ nameFull = 'LftStemTnLtr' + str(i) + 'no'
+ segment = 'TnLtrSeg' + str(i) + str(i)
+ buildComp(f,f[nameFull],['TnLtrSpcFlatLeft',segment,'TnLtrSpcPointRight'],0,0,i,0)
+
+ # dotted tone letters
+ for i in range(1,6):
+ nameFull = 'DotTnLtr' + str(i)
+ dot = 'TnLtrDot' + str(i)
+ buildComp(f,f[nameFull],['TnLtrSpcDotLeft',dot,'TnLtrSpcDotMiddle','TnLtrBar','TnLtrSpcFlatRight'],0,0,0,0)
+
+ # dotted left stem tone letters
+ for i in range(1,6):
+ nameFull = 'DotLftStemTnLtr' + str(i)
+ dot = 'TnLtrDot' + str(i)
+ buildComp(f,f[nameFull],['TnLtrSpcFlatLeft','TnLtrBar','TnLtrSpcDotMiddle',dot,'TnLtrSpcDotRight'],0,0,0,0)
+
+
+def doit(args):
+
+ psffont = UFO.Ufont(args.ifont, params = args.paramsobj)
+ rffont = OpenFont(args.ifont)
+ outfont = args.ofont
+
+ getParameters(psffont)
+
+ updateTLPieces(rffont)
+ rebuildTLComps(rffont)
+
+
+ rffont.save(outfont)
+
+ return
+
+def cmd() : execute(None,doit,argspec)
+if __name__ == "__main__": cmd()