butterick

Forum Replies Created

Viewing 12 posts - 1 through 12 (of 12 total)
  • Author
    Posts
  • in reply to: mojo.robofont vs. robofab.world #3651

    butterick
    Participant

    And just to be clear about the “Robofab extras”:

    http://doc.robofont.com/api/robofab-extras/

    1) These are exclusive to the Robofont environment.

    2) They are not part of any Robofont-native module, for instance mojo.robofont.

    3) They are only available by importing robofab.

    4) But though they are robofab-based, they will not work in environments other than Robofont.

    in reply to: observer callback format #3650

    butterick
    Participant

    Why is “info” a dict instead of an instance of a reall class defined in mojo.events?

    I ask about source code because __doc__ seems to be empty for all the mojo.* modules and classes, and the API documentation is missing a lot. I’m looking for a way to supplement my understanding of the API without asking you a hundred questions.

    in reply to: equivalent of robofab.RFont.update() #3626

    butterick
    Participant

    ” 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.”

    UFOCentral also relies on this technique, no? When I used it the other day, the glyphs and layers flashed before my eyes as I exported them. (Though I understand that UFOCentral is superfluous inside of Robofont.)

    The macro point is that stdout/stderr is useful for debugging only when you can describe the status accurately in text. For instance, plenty of times in FL I’ve used a pointPen to copy an outline from one font to another, only to find out it’s been turned into crumpled space junk along the way. But as far as Robofab or FL was concerned, the pen operation reported no errors.

    Therefore, to rely on stdout/stderr only, you’d have to add in an extra step where you programmatically compare the expected and actual result, so the status can be reported accurately (“oops, you got space junk”). Sometimes that’s easy to do (e.g., making a straight duplicate) but sometimes not (where there’s a transform happening in between).

    Either way, you’re adding new debugging code, which a) will slow your script (no big deal) and b) may itself have bugs in it (very big deal). For complicated pen operations, your debugging code would have to essentially reimplement the pen. Or you’d need to do something ritzy like apply your pen operation into the actual target glyph, and then also a buffer glyph, and compare the results. But it creates a spiraling need to debug the debugger. So what is the advantage of visual debugging? Like stdout and stderr, it works without creating collateral complications.

    Anyways, the vagaries of FL need not be hashed over here. My FL scripts all work fine — I’ve already absorbed those slings & arrows. When I get to the point where Robofont’s update() is causing a more-than-theoretical problem, I will report back. In text.

    in reply to: equivalent of robofab.RFont.update() #3619

    butterick
    Participant

    Thanks Ben. You impliedly make a good case that my goal of having FL & RF compatible scripts is impractical. There is something to be said for just rebuilding Robofont-optimized versions.

    As for update() in the middle of a script, my idiomatic use for this technique in Fontlab was when I was working with Robofab pens, specifically scripts that attempted to do pen operations on hundreds of glyphs at a pull. Of all Robofab operations, I found that pens were the most likely to awaken the FL crash monster.

    The traditional behavior of FL when it encounters a script error is to hang. So relying on a UI update at the end of the script to show me status didn’t work, because it never got there. So I got in the habit of calling update() during the loops. This was slower, but when FL would hang, I could at least see how far it had gotten, giving some clues about why the script was failing, and whether the error was due to my own abuse of Robofab (which I could fix) or Fontlab’s arbitrary refusal to continue (which I could not).

    Clearly, Robofont has a better work ethic about finishing scripts.

    in reply to: equivalent of robofab.RFont.update() #3613

    butterick
    Participant

    GAAAH

    (How do I include a pre tag in a comment? Apparently it’s something different than typing greater-than + pre + less-than)

    in reply to: equivalent of robofab.RFont.update() #3612

    butterick
    Participant

    (How do I include a tag in a comment? Apparently it’s something different than typing )

    in reply to: equivalent of robofab.RFont.update() #3611

    butterick
    Participant

    Fair enough. At the moment, I don’t have another script in my back pocket that illustrates the need for incremental UI updates. If I come across one, I will post it. Like I say, mostly I have used the feature for status reporting and debugging.

    As for update(), my only remaining question is when it actually makes a difference in a Robofont script. For instance, this code snippet (adapted from the EVB script) works the same way with or without the calls to update().

    from robofab.world import CurrentFont
    
    f = CurrentFont()
    
    for k in ["A", "B", "C", "D"]:
            f[k].mark = 100
            f.update()
    
    
    
    
    from robofab.world import CurrentFont
    
    f = CurrentFont()
    
    for k in ["A", "B", "C", "D"]:
            f[k].mark = 100
    
    in reply to: equivalent of robofab.RFont.update() #3609

    butterick
    Participant

    My understanding of why RFont.update() exists & how to use it was gleaned entirely from the Robofab docs & examples. Some Robofabbers may use it more, some less. But it is part of the standard Robofab-in-Fontlab scripting idiom. If you followed the Robofab docs, your scripts rely on it.

    (Tal, since you ask, one example of a script that “needs to update the UI immediately” is EVB’s accent builder example on the Robofab site. (http://robofab.com/howto/buildingaccents.html) And I try to learn from the master.)

    As a matter of policy, having update() do nothing in NoneLab makes logical sense because there is no UI to update.

    Likewise, my comment is also one of policy: now that Robofab is in another UI environment, how should update() behave? Does it matter that it behaves differently than it did in the previous UI environment?

    Ben says no, because update() was designed to accommodate FL and I can test for FL in the script. A fair point, but there’s another dimension to the problem. The update() command gives Robofab-in-Fontlab a nice modular division between the Robofab layer and the Fontlab UI layer. Everything gets encapsulated into the update() command. I don’t have to worry about the specifics. Maybe this was an unintentional result, but to the extent that one of Robofab’s goals is to insulate me from the FL object model, I always thought it was deliberate.

    Above, Frederik showed me how to work with view objects and view.display() to do my updates. That’s fine as far as it goes, but it lacks modularity. I’ve got a script that would be happy if it could just be in the business of manipulating Robofab objects. Now it has to get into the business of manipulating Robofont UI objects too.

    As a scripter, it’s not that I mind manipulating UI objects. It’s just that ideally, I’d like to avoid mixing UI stuff with data stuff, because it won’t scale. For instance, what happens when folks start releasing scripts with new UI widgets? Now my data-manipulation script has to get updated to work with those widgets too. (Tal, perhaps your proposed approach is forward-compatible in that way.)

    Nor am I saying that Robofont or defcon does it “wrong”.

    Again, it’s a policy question of how to expose the Robofont UI to scripts, whether to be consistent with FL, or whether to invent a 3rd idiom (i.e., require Robofab-in-Robofont scripters to take more of an interest in Robofont UI objects).

    in reply to: equivalent of robofab.RFont.update() #3604

    butterick
    Participant

    I see what you mean Frederik, but I still think this is a flaw, for two reasons.

    1) The discrepancy between the behavior of update() in Fontlab vs. update() in Robofont reduces the portability of Robofab scripts and makes it harder to create modules that can be shared between environments. Robofab becomes incompatible with itself.

    If Robofab-in-Robofont has some extras compared to Robofab-in-Fontlab, that’s OK, because can ignore those if I need to. But I wouldn’t expect major subtractions either. RFont.update() is critical and frequently used in Robofab-in-Fontlab. Once I add extra code to make update() work in Robofont the way it did in Fontlab, then I can’t use that script in Fontlab anymore.

    2) More broadly, an environment like Robofont that’s built with a focus on scripting should always defer to the scripter. For instance, if I’m making valid update() requests in the script, why should Cocoa “ignore” them? OK, Cocoa thinks it will be slow. But why does Cocoa get to have an opinion? If it runs slow, that’s my problem. Maybe I have my reasons for wanting it that way. It should be WYSIWYG — What You Script Is What You Get. There should never be situations where lines of script are getting silently ignored.

    (BTW, the animation is very exciting of course, but update() has a real practical function, which is to give visual feedback during multi-step operations. It is an excellent debugging and status-reporting tool.)

    in reply to: how to make revert go crazy #3600

    butterick
    Participant

    Or maybe just simplify it to an all-or-nothing operation. That’s how it’s usually implemented. It’s a useful function to have in the file menu, because it’s a common need, and it saves some clicking and mousing.

    in reply to: equivalent of robofab.RFont.update() #3599

    butterick
    Participant

    OK, then here’s an example illustrating my confusion:

    import robofab.world
    
    f = robofab.world.CurrentFont()
    g = f["A"]
    
    for n in range(0,54):
        g.rotate(10)
        f.update()
    

    I expect this code to rotate the A glyph 540 degrees in 10-degree increments, and that each step would be visible in the UI (because I’m calling update() after each rotate().)

    Instead I only see the final state (a net rotation of 180 degrees). In other words, the code behaves as if f.update() doesn’t exist, and then the UI updates itself after the script exits.

    So is there a flaw in f.update(), or a flaw in my understanding of what it does?

    in reply to: how to make revert go crazy #3590

    butterick
    Participant

    The problem also happens in reverse:

    1) Remove a glyph
    2) Attempt to revert all glyphs

    This time, because the glyph is gone, it won’t show up in the list of possible glyphs to revert, so you can’t get it back without closing & reopening the file.

Viewing 12 posts - 1 through 12 (of 12 total)