Skip to content

Latest commit

 

History

History
61 lines (52 loc) · 1.56 KB

state-pattern.md

File metadata and controls

61 lines (52 loc) · 1.56 KB

The State Pattern

First, if you are not familiar with this pattern, I recommend you read this excellent article.

Our Master-Class types include an M.State type. That type helps us turn objects into concurrent, hierarchical state machines.

The M.State type basically merges object types together

let MyClass = M({
  state: M.State([
    {
      delegate: M({n: 3}), // (1) first object type
      subState: [{
        delegate: M({m: 5}) // (1.1) first object type child
      }]
    },
    {
      delegate: M({s: String}) // (2) second object type
    }
  ])
});
let myInstance = new MyClass();
assert.deepEqual(myInstance.state, {n: 3, m: 5, s: ''});

but every merge can be conditional using the when option

MyClass = M({
  flag1: true,
  state: M.State([
    {
      when() {return this.flag2;},
      delegate: M({n: 3}), // (1) first object type
      subState: [{
        when() {return this.root.flag1;},
        delegate: M({m: 5}) // (1.1) first object type child
      }]
    },
    {
      delegate: M({ // (2) second object type
        flag2: Boolean,
        s: String
      })
    }
  ])
});
myInstance = new MyClass();
assert.deepEqual(myInstance.state, {flag2: false, s: ''});

Note that myInstance.state.m property is not included, even though this.root.flag1 is true. This is because its parent state is blocked on this.flag2. Now, if we just assign myInstance.state.flag2 = true, we'll get

assert.deepEqual(
  myInstance.state, 
  {flag2: true, s: '', n: 3, m: 5}
);