Forum Replies Created
-
AuthorPosts
-
> UFOCentral also relies on this technique, no?
No. That’s a symptom of FL’s interface problem, not a feature of the script.
I thought of a way that this could be done in the font and glyph objects. Here is the font object version:
def update(self): document = self.document for windowController in document.windowControllers(): window = windowController.window() contentView = window.contentView() contentView.displayIfNeeded()
(Not tested, but it should work. I think. Maybe.)
Is this a good idea? I don’t know. It could slow down lots of scripts for the sake of one script.
I can’t answer for RoboFont, but I can explain what is happening at the lower levels in defcon and defconAppKit. I can also give a little background on the history of the update method.
First, the background: the update method in FontLab is purely a UI refresher. When you make a change to the data, the data is changed but the UI is not. For many scripters, this creates a pretty big problem because if update isn’t called, the UI is out of sync with the data. That’s bad. If you look at the RoboFab docs and examples, you’ll see lots of things like “# don’t forget to call update!” Those are there because we had lots of complaints. The update methods weren’t part of RoboFog’s scripting API and that’s what we were trying to mimic. Almost all of the scripters were familiar with that API (and things happening there automatically) and that made the update calls quite foreign and a source of many complaints. We spent a lot of time trying to make the updates happen automatically, but we never got any responses to our requests to FontLab for information about the optimal times to refresh. For a very short period, we had update happening automatically whenever any data was changed. This was known as the OMG THIS IS SO SLOW version of RoboFab.
To make the RoboFab API consistent, we implemented the update method in the base objects as:
def update(self): pass
It has been that way in NoneLab for a long time.
When I wrote defcon, I wanted to do things in a modern way that allowed for code to be much more flexible and much faster. defcon uses the observer pattern (concept: http://en.wikipedia.org/wiki/Observer_pattern implementation: http://code.typesupply.com/browser/packages/defcon/trunk/Lib/defcon/tools/notifications.py) for keeping data and representations in sync in an abstract way. When you make a change to a defcon object, the object posts notifications through a font level notification dispatcher. Other objects that have subscribed to (object, notification) pairs will receive the notifications. It makes everything very lightweight and very fast. The various defconAppKit (an extension of vanilla) views use the notification system to be alerted of font object changes. When particular notifications are posted, defconAppKit notices and schedules things to be redrawn the next time Cocoa gives the views permission to redraw. Cocoa does this following its own rules. There are ways to hack into Cocoa to force it to redraw at a precise time, but those are frowned upon. I think this has more to do with the general drawing system that OS X uses than it is that Apple thinks that their code is much faster than everyone else.
It’s interesting that you say that this need to update the UI immediately is so critical. In all my years of working on and with RoboFab in FontLab I’ve never seen a script that needed to update the way that you describe. Do you have a lot of these kinds of scripts?
I never said that foundries should tell their customers to wait for a Word update. What I said is that it’s unlikely that I’ll add a wpfNameHack flag to ufo2fdk. Adding flags for legacy application behavior is going to be a bottomless pit. Should I add a flag for generating contextual lookups in reversed order so that they work in InDesign 1 or the Quark betas? Should I add a flag for always inserting a Greek codepage bit into the Unicode ranges so that fonts with CE characters don’t move to the bottom of the Carbon font menu in Mac OS 10.3? Should I add a flag that limits fonts to 256 kerning pairs because some (still active!) newspaper imposition software crashes with 257 or more pairs? It goes on and on. Foundries should make fonts in whatever way they see fit. That’s the beauty of RoboFont. You can generate and then apply your own special recipe to your font with a simple script:
from fontTools.ttLib import TTFont
font = CurrentFont()
outputPath = “whatever.otf”
font.generate(…)
otf = TTFont(outputPath)
nameTable = otf[“name”]
# go from hereAs far as knocking the OT spec goes, yeah, I used to feel like there was no spec/standard. After having spent a lot of time reading and rereading the OT spec (I can recite parts of it from memory at this point) and having been part of a spec process, I don’t have much of a problem with the spec. It’s a little fuzzy here and there; it has lots of legacy bits and fields; it has things that aren’t used. But, it could be a lot worse. The bugs and quirks that are out there are almost always not the fault of the spec. They are the fault of the people who misinterpreted the spec. This isn’t just limited to application developers. It is very, very frequently the fault of font developers. I’ve had engineers from many major software companies tell me essentially the same thing: “The spec is a minor problem. The bigger problem is that font developers don’t follow the spec and we have to clean up the mess.”
We’re way off topic… :-)
If I’m understanding correctly, you are trying to hack around the Windows Presentation Foundation font menu behavior. (I don’t know about the Quark issue. I haven’t heard a complaint about Quark vs. fonts in many years.) The WPF behavior has some quirks, but making fonts that are non-conformant with the spec is generally a bad idea. It may work around a quirk in a three year old piece of software, but new software three years, nine years, twenty-seven years from now? There’s no way of know how bogus fonts will be handled. But, if that’s what you want to do, go for it.
As far as building this “make broken fonts” pref goes, that goes down a level to the core OTF generating code. RoboFont uses my ufo2fdk package for that. I suppose it would be possible to make a setting there that would make fonts with a broken name table. I doubt very strongly that I will do this though for a variety of reasons. Adding tons of prefs that hack bits here and there leads to a confusing API/interface, unpredictable results for 99.9% of users and frustration all around.
I use a variation of ufo2fdk for my own mastering pipeline (used by myself and a few large foundries). The only difference is that the outline source generator is based on FontLab’s PFA generator, but I’ll probably switch this in the future to use ufo2fdk completely. That said, I’m not 100% happy with the output of the FDK, so I have a post-generation processor that makes some tweaks to the final files. (These are cosmetic tweaks—nothing that makes them non-conformant with the spec.) FontTools makes this very easy.
I think that the best way for you to do what you want is to make a script that generates the font, opens it with FontTools and applies your naming hack. It’s probably 10-20 lines of fairly simple code.
Yeah, this is a really hard thing to get right. I’ve done a lot of work on it and maybe you could add to it:
http://code.typesupply.com/browser/packages/feaTools2/trunk/Lib/feaTools2
from feaTools2 import decompileBinaryToFeaSyntax
path = “/Users/you//blah/blah/Blah-Blah.otf”
print decompileBinaryToFeaSyntax(path)GPOS lookup types aren’t supported, the GSUB lookup types are limited to what I needed at the time, it needs more robust testing, etc. But, I was pretty far along and the results were very promising.
-
AuthorPosts