Freewheeling Apps https://akkartik.name/freewheeling-apps en-us 2025-01-19 http://akkartik.name/post/2025-01-19-devlog 19 Jan 2025 15:49:29 PST http://akkartik.name/post/2025-01-19-devlog
audio/video; 4 minutes; 25MB; scroll down for shorter clips

Transcript

For a while now I've been trying to improve the way I program. I've made some progress. I can edit my programs live, and the tooling that enables it is not much code.

Here is the template I create new apps from. It starts out with something silly: every time I click with my mouse it draws a random rectangle.

30KB

Here is the code for this app.

There's not much right now, but it grows from here and in principle the app continues running as it evolves.

But debugging is still hard. It takes a long time to streamline my thinking about each new problem. I often print information to the terminal, and then struggle to visualize the text my program is printing. And if I draw debug information on the canvas, it competes for space with the app.

So I think I need new infrastructure. There are probably tools I could use that are too hard to build while deep in something else. One such tool is a windowing system. As my app runs it can dump things to other windows, and I can show them separately from my app.

Here's a window where it just prints text like I would otherwise send to the terminal.

Here's a window that plots what I draw to a 2D surface very like the app, except I can add instruments like these axes and dimensions.

180KB

The metaphor for the space is also different. My actions can have different meanings. Where the app uses mouse clicks to add new rectangles, here I can pan around.

Here's another view. This time I have some splits and multiple windows in them.

240KB

Each of these windows is a log I can scroll around in. But the stuff in the log is graphical. This one shows each rectangle drawn separately. This one focuses on just their positions, this one on just their dimensions. This one shows just widths, and this one just heights. All these windows have the same metaphor, and actions have similar meanings. But the data in them is different.

Finally, here's yet another debug view of this app. This time it's showing the sequence of actions in time rather than space, looping back when it reaches the end. I can also adjust the replay speed.

90KB

The code for this windowing system lives with my app, and this is all there is. I have some event handlers to choose from over here, and I can compose them together to create windows here. Then I compose layouts from the windows here. My app will grow downward and to the right, and lower level infrastructure will be easy to ignore above and to the left.

Hopefully this infrastructure will be helpful the next time I dig myself into a hole. It's easier now to create new places to draw debug information. ]]> 2025-01-16 http://akkartik.name/post/2025-01-16-devlog 16 Jan 2025 11:04:03 PST http://akkartik.name/post/2025-01-16-devlog

  • Decker. I'd for some reason assumed it was some npm and/or wasm monstrosity, but it turns out it's nice zero-dependency Javascript in a single html file that you can save and run offline! And that's in addition to the native version that uses SDL just like LÖVE. So in some ways Decker has a better cross-platform story than LÖVE, which is non-trivial to run on iOS. The downside: on a web browser on Android it's even less efficient to run than LÖVE's native app.
  • Rust. I spent some time playing around with a LÖVE-inspired game engine called ggez, and the much more basic wgpu crate. The promise here: Rust seems to be evolving some pretty nice cross-platform tooling, so in time we may end up with a world where it's easy to cross-compile to any platform. Of course, the compile step is not ideal, but it promises to yield much more efficient binaries that might run on lower-end devices like old phones. The eco-system is not quite there, and the npm-like dependency explosions are rough. But I want to try to look outside my comfort zone. It's possible the way forward isn't unique-snowflake minority platforms like LÖVE or Decker, but just to use a majority platform with better taste, taking the time to understand and curate the landscape of dependencies.

    Anyways, here's a little program I made to try to stretch Decker to more of the sort of procedural graphics I tend to gravitate towards on LÖVE:

    This is dancing letters, a fixed piece of text except we're constantly switching the case of each letter at random.

    Here's the code, to give you a flavor for what Decker's quite elegant mix of Lua and APL looks like:

    
    local s: "abcdef"  # put in whatever text you want
    
    on view do
      if ! 5%sys.frame
        me.clear[]
        local y:each c in s random["%u","%l"] format c end
        local margin:15
        me.text[y margin,margin,me.size-margin*2]
      end
    end
    
    And here's that code along with the surrounding card (you need a canvas widget to be present just so) in a less readable form that you can copy and paste into a deck of your own:
    
    %%CRD0{"c":{"name":"home","script":"on view do\n \nend","widgets":{"canvas":{"type":"canvas","size":[300,200],"pos":[48,51],"animated":1,"volatile":1,"script":"local s: \"Call me Ishmael.  Some years ago--never mind how long precisely--having little or no money in my purse, and nothing particular to interest me on shore, I thought I would sail about a little and see the watery part of the world.  It is a way I have of driving off the spleen and regulating the circulation.  Whenever I find myself growing grim about the mouth; whenever it is a damp, drizzly November in my soul; whenever I find myself involuntarily pausing before coffin warehouses, and bringing up the rear of every funeral I meet; and especially whenever my hypos get such an upper hand of me, that it requires a strong moral principle to prevent me from deliberately stepping into the street, and methodically knocking people's hats off--then, I account it high time to get to sea as soon as I can.\"\n\non view do\n if ! 5%sys.frame\n  me.clear[]\n  local y:each c in s random[\"%u\",\"%l\"] format c end\n  local margin:15\n  me.text[y (margin,margin,me.size-margin*2)]\n end\nend","border":1,"scale":1}}},"d":{}}
    
    ]]> 2024-12-29 http://akkartik.name/post/2024-12-29-devlog 29 Dec 2024 15:18:18 PST http://akkartik.name/post/2024-12-29-devlog Jack Rusher and others about Emacs Nature [1, 2] and playing with Seymour by Alessandro Warth, I'm getting interested in building..

    An environment for visualizing programs

    (Not to be confused with visual programs, or visualization more generally.)

    • Start with a tiling window manager for managing named graphical canvas "buffers", using Emacs operations like split and resize.
    • Each buffer exposes a coordinate space of its choosing, listens for messages and positions objects in the space in response to messages.
    • Buffers can send messages to other buffers.

    Examples of coordinate spaces:

    • Graphical game engines use the obvious 2D/3D cartesian systems. You position stuff using (x, y) or (x, y, z). You could also imagine polar or other coordinate systems that are studied in geometry.
    • The HTML DOM is a space where positions can be specified using CSS selectors or XSLT.
    • You can imagine a text editor operating in a coordinate space as well. Emacs seems to use a 1D coordinate, just character count from start of buffer. My stuff so far uses 2D: (line index, UTF-8 codepoint index within line)

    Some examples of messages, to show the sorts of use cases this framework might unlock:

    • In a text editor, the cursor tracks a position, and keyboard and mouse send messages to move the cursor or insert objects (characters or longer text) at the cursor.
    • You can imagine print statements as a message from the "code" coordinate space to a different, 0-D (append-only so there's no notion of coordinate) space.
    • Terminal buffers in Emacs take the mostly 0D space of a terminal and augment it with a cursor. When you scroll up to an earlier command and hit a hotkey, the buffer sends a message to itself with the text around the cursor. The message is received at the bottom of the buffer.
    • Emacs Slime and other IDEs support keyboard shortcuts to send text from the current buffer to a REPL in some other buffer.
    • Ronin and Sketch-n-Sketch support bidirectional messages between two spaces with very different coordinate systems.
    • Live programming systems often show the results of a statement to its right. Examples: alv by S-ol Bekic, Bret Victor (of course), Seymour as above. These too can be seen as a reflexive message from a space to itself. In addition, the message contains an implicit coordinate: the current line.
    • Glamorous Toolkit, Lisp Machines and other Smalltalk systems do a lot of stuff like this. I think all of this can be cast in terms of buffers, coordinate spaces and messages, though you can imagine them as a single, very complex coordinate space like the HTML DOM, or many simple spaces with different possible coordinate systems. For example, in any of them you can create a new "log" space that you can append graphical objects to. Maybe even self-contained interactive graphical widgets.

    The major question for me now is: how do you configure a buffer? You need some concise way to specify the space (perhaps just by naming from a small menu of options), handlers that listen for messages (e.g. keypress or mousepress), handlers for sending messages (e.g. widgets on the space that perform tasks when you interact with them), and generic handlers for sending messages to other buffers (e.g. print; here I'm imagining it to send a message from some arbitrary process, through say a socket, back into the environment, with enough information to route it to the appropriate buffer accompanied by a reasonable coordinate) ]]> 2024-11-07 http://akkartik.name/post/2024-11-07-devlog 07 Nov 2024 16:02:48 PST http://akkartik.name/post/2024-11-07-devlog for hopefully the final time. Here's a static html page you can download and save locally to roughly compare times in different timezones (just hours; you're on your own for minutes).

    The way I'm naïvely imagining using this:

    • Scroll to the timezone you know the time in.
    • Click on the nearest hour.
    • It'll highlight that column all over the page.
    • Scroll or find the timezone you care about.
      • If you're not in a whole-number timezone (Hello India), you'll need to do some additional mental arithmetic by comparing nearby rows.

    That's it. Since it's almost entirely static, you can always be sure that you're seeing the same thing on this page as anyone else.

    Unfortunately you need to know if you're in daylight savings time or not, something that is often beyond me. I'm not sure what to do about that without reintroducing dynamism that takes the current computer's time into account. Then I again end up wondering if others are seeing what I'm seeing.

    There are a few abbreviations for America, Europe and Australia in both pages. You can see *ST and *DT on either page, which might help if you're not observing daylight time yet, but someone else is. There's a tension here between trying not to be overwhelming and emphasizing the Western or Northern hemisphere. My thinking is to only add codes for longitudes with lots of cities or with daylight savings time. Hopefully people in Bhutan or Nigeria or the Chatham Islands won't hold it against me.

    Inspired by Bret Victor, but of course the inevitable mistakes are all mine. ]]> 2024-10-26 http://akkartik.name/post/2024-10-26-devlog 26 Oct 2024 10:20:27 PDT http://akkartik.name/post/2024-10-26-devlog back into teaching kids programming 1:1. Of course, this time using Lua, LÖVE and Carousel. After a couple of months, it occurred to me to collect all my little impromptu puzzles and exercises into a single app anyone can go through on their own schedule.

    Carousel Cards (LÖVE app, really just a zip file containing source code, 169KB)

    
    Lua Carousel showing 6 lines of text:

    What does this program do?
    If you don't know, guess!
    Then hit the 'run' button and check your answer.
    There are no penalties for answering wrong.

    print(3+4)

    Nowhere near done yet. But it has 50 112 little "levels", each taking between a few seconds and a minute. A full game/curriculum might need 2000 levels or something.

    Inspirations:

    ]]>
    2024-10-20 http://akkartik.name/post/2024-10-20-devlog 21 Oct 2024 00:28:43 PDT http://akkartik.name/post/2024-10-20-devlog

    • Inserting/deleting text before a range moves it.
    • Inserting/deleting text after a range leaves it unchanged.
    • Inserting/deleting text within a range grows/shrinks it.
    • Deleting text at a boundary shrinks the range, and deletes it when it becomes empty.
    • Inserting text at boundaries can't disambiguate whether I want the text within or outside the boundaries. But I can grab some handles on the range to adjust.

    The final complexity cost was 200 lines but it was a non-linear path getting there. I started out with the notion of pivots from my doodle app. There, pivots live inside the data structure for a single line of text. Since ranges now need two pivots that could be on different lines, I had to move them out. I started tracking pivots in a separate data structure, maintaining bidirectional mappings between pivots and their locations, and then tracking ranges as combinations of pivots. This quickly blew up the implementation to 500 lines, and I was juggling 27 manual tests of which half were failing.

    The next day I started from scratch and threw out the notion of pivots entirely. Now I just maintain 2 locations directly inside each range, and linearly scan through them all for any book-keeping. The implementation dropped to 200 lines and the tests passed fairly quickly.

    Earlier this year I threw out an implementation after suffering with it for 2+ years. It feels like I'm getting the hang of this programming thing that I threw out an implementation now after just 2 days. I'm setting a higher bar for elegance. At the same time, it's interesting that my instinct remains as poor as ever for the right approach in even a slightly new problem. Here I spent a lot of time trying to squeeze my ranges into lines so that deleting a line would transparently delete ranges within it. But I blindly assumed a fully normalized design with a first-class notion of a pivot must be a good idea.

    As I linked to in the previous post, this app was inspired by this comment by Alex Kladov about how Emacs maps ranges of text to attributes. ]]> 2024-10-18 http://akkartik.name/post/2024-10-18-devlog 18 Oct 2024 10:19:02 PDT http://akkartik.name/post/2024-10-18-devlog Potluck and the Emacs model of text augmented with attributes for ranges. Potluck describes 3 mechanisms:

    1. extracting data using (regex) patterns
    2. performing computations on that data
    3. showing results of computation as annotations

    My notebook app does simple variants of 2 and 3, and replaces 1 with explicit in-document markup.

    Now I'm playing with another approach to 1. I already have the idea of pivots from my doodle app. Putting two of those pivots together should yield a range that adjusts in intuitive ways in the presence of edits. An example may be a WYSIWYG UI for adding a hyperlink to some text:

    • Inserting/deleting text before a range moves it.
    • Inserting/deleting text after a range leaves it unchanged.
    • Inserting/deleting text within a range grows/shrinks it.
    • Deleting text at a boundary shrinks the range, and only deletes the attached attributes if the range becomes empty. This makes ranges more robust to deletion than my doodles which attached to a single pivot.
    • Inserting text at boundaries can't always do what you want. I imagine it'd be nice to have handles that you can drag to adjust a range.
    ]]>
    2024-10-10 http://akkartik.name/post/2024-10-10-devlog 10 Oct 2024 08:57:33 PDT http://akkartik.name/post/2024-10-10-devlog notebook.love, and trigger either selectively based on what blanks I fill in.

    The fine print: to switch directions I have to fill in the right blank, clear the old blank, and then type in something outside the old blank (to indicate I'm not going to type further into the old blank).

    Works better if I clear the old query first, but who can remember that?

    I've decided to just recompute on every keypress and mouse click. It seemed unnecessary, but now I see that there's some benefit from the inefficiency. ]]> 2024-10-07 http://akkartik.name/post/2024-10-07-devlog 07 Oct 2024 17:33:44 PDT http://akkartik.name/post/2024-10-07-devlog Ways I've previously seen text coexist with a canvas:

    I'm instead using the game engine idea of a pivot. Any time I toggle into doodle mode I have to first pick a pivot from one of the characters on screen. All my drawings are relative to that pivot, and edits to text maintain pivots alongside.

    Displacements to the pivot are preserved in font-derived units, so it looks "reasonable" as you resize the font.

    Deleting a character deletes all drawings pivoted on it. (But there's undo.)

    This took 200 lines, so not too much though it was more than I'd initially expected.

    Inspired by this thread. ]]> 2024-10-06 http://akkartik.name/post/2024-10-06-devlog 06 Oct 2024 16:17:35 PDT http://akkartik.name/post/2024-10-06-devlog the notebook fills blanks

    Blanks can be filled in either with the results of computation or with what a person typed in. Now we indicate such conflicts in two ways.

    • Blanks filled in from computations get a distinct look (the cyan background), separate from both hand-written (black on white) and computed, non-editable text (cyan on white)
    • If I manually edit a blank, its background changes, and any code that was overridden doesn't execute. Here are a couple of examples:


    But perhaps the color choice is too strong:

    ]]>