Skip to content

AdaBeat/purescript-gojs

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

27 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

GoJS bindings for PureScript

This library contains "low-level" bindings for methods, properties, constructors, and static methods for the GoJS framework for building interactive diagrams.

Module structure

The module structure tries to follow the grouping in the API section of the GoJS docs, with the exception of GraphObject, which follows the class structure. The only reason for this is that the GraphObject class structure is intricate enough to be in the top-level.

Design principles

This library is still experimental so much of the below is subject to change.

Class structure

This library is kept as simple as possible. It uses a rudimentary class structure to model JavaScript classes: for example, in GoJS, there's the following class hierachy:

             ┌---- Shape 
 (abstract)  ├---- TextBlock
GraphObject -┼---- Picture                ┌---- Adornment
             ├---- Placeholder            ├---- Link
             └---- Panel -------- Part ---┴---- Node ------ Group 

In this library, this is modeled with typeclasses IsGraphObject, IsPanel etc. GraphObject being an abstract class simply means it doesn't have a corresponding opaque foreign constructor newGraphObject, unlike Shape_, Panel_ and Part_ etc.

Return types

Collections

Some methods in GoJS return collections of GraphObjects. One can have an array that looks like this, for instance:

myArr: Part[] = [new Part(), new Node(), new Group()]

Collections in PureScript are almost always homogeneous, which means every element of the collection has to be of the same type. There's several ways to encode this in PureScript, like arrays of sum types, existential types, or concrete types that can be converted to their "children". We go with the last option here. For example, if you want to call a node method in a particular element of a collection of Parts, you could do this:

...
let setOfParts = diag # _selection
case fromPart <=< setFirst $ setOfParts of
  Just node -> useNode node -- assuming useNode :: Node_ -> Effect Unit
  Nothing -> pure unit
...

Note that if the useNode function above was polymorphic in its input, we'd need to explicitly specify the type we're converting from with fromPart @Node_, otherwise we'd get an ambiguous type error, which brings us to the next section.

Polymorphic return types

Many methods return classes. In this case we do allow the caller to determine the output, by having the output type as a visible type variable. This is more or less a shorthand for calling fromPart as above, by applying the type directly. This is often needed if the output of such a function is not used by a monomorphic function. For example, consider the following code:

myNode <- someLink # _fromNode -- _fromNode :: forall @n. IsNode n => Link_ -> Maybe n
case myNode of
  Just n -> n # findColumnForLocalX_ 10.0 -- findColumnForLocalX_ :: forall p. IsPanel p => Number -> p -> Effect Number
  Nothing -> pure 20.0

The PureScript compiler can't know what the type of n should be. All it knows is that it implements the IsNode class - it is ambiguous, and so will refuse to compile this code. The solution is to type apply the output of _fromNode, like so:

myNode <- someLink # _fromNode @Group_ -- _fromNode :: Link_ -> Maybe Group_
case myNode of
  Just n -> n # findColumnForLocalX_ 10.0 -- findColumnForLocalX_ :: Number -> Group_ -> Effect Number
  Nothing -> pure 20.0

Getting and setting properties

A couple concerns with the current design:

  1. Pure getters - not referentially transparent!

An important aspect of this library is that getting properties of opaque types is not referentially transparent!

let angle = someLink # _midAngle -- say this returns 10.0
setUnsafe someLink {midAngle: 1.0}
let angle2 = someLink # _midAngle -- this will return 1.0!

This is a big break from pure FP principles, and the reason it was done this way is so that composition of getters was 1. cleaner and 2. more performant, since impure getters would imply additional function calls - an effectful computation, at the JS level, is implemented as a closure. This decision makes the code look a lot like TypeScript.

  1. Setting properties is not type-safe

This was done in order to allow deeply nested effectful setting of properties in opaque types, for example:

setUnsafe someDiagram {"toolManager.relinkingTool.temporaryLink": someLink}

An alternative to this is use a lens-based approach (so replace getters with lenses that operate on newtyped records) and have setUnsafe operate on the same lenses, except do it effectfully; so the code above would look like this:

setUnsafe someDiagram (_toolManager <<< _relinkingTool <<< _temporaryLink) someLink

This will probably be implemented in the future, as lenses also solve another problem, which is that different classes often have properties that are named the same, but with the current approach, there's no way to make the inputs polymorphic and type-safe; with lenses, they can be expressed on newtyped records that have a field named the right thing.

TODO:

  • Implement GoJS's IncrementalData
  • Support optional arguments to methods - currently most optional methods are simply required by the bindings
  • Add documentation
  • Add rest of missing Prototype methods
  • Address peppered TODOs

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Packages

No packages published