rodamora
1/13/2016 - 5:34 PM

Sample implementation of a decorator for angular2 using cerebral

Sample implementation of a decorator for angular2 using cerebral

/**
 *  Angular2 decorator for Cerebral
 *  @param paths the paths that will be retrieved from the state
 *  1. It holds its own state object based on the paths passed in the decorator
 *  2. When the controller triggers a "change" event, the decorator grabs the same state again from the controller and compares it with the existing state
 *  3. If the state has changed, the state property is updated
 *  4. If not, it is left alone
 */
let controller: Cerebral;
let callbacks: Function[] = [];
let listener: boolean = false;

export function register(controller) {
  controller = controller;
}

export function Cerebral(paths?: Paths) {
  return function<TFunction extends Function>(target: TFunction): TFunction {
    let component = target.prototype;

    component._update = () => {
      let paths: Paths = component._paths ? component._paths : {};
      Object.keys(paths).forEach((key) => {
        let val: any = controller.get(paths[key]);
        if (component.state[key] !== val) {
          component.state[key] = val;
        }
      });
    };

    component._setup = () => {
      component.state = {};
      component.signals = controller.getSignals();
      component._paths = paths ? paths : {};
      component._listener = false;

      if (Object.keys(component._paths).length) {
        callbacks.push(component._update);
      }

      if (!listener) {
        listener = true;
        controller.on('change', function () {
          callbacks.forEach(cb => {
            cb();
          });
        });
      }
      
      component._update();
    };

    component._detach = () => {
      if (Object.keys(component._paths).length) {
        callbacks.splice(callbacks.indexOf(component._update), 1);
      }
    };

    if ('ngOnInit' in component) {
      let originalOnInit = component.ngOnInit;
      let newOnInit = (old => {
        function extendsInit() {
          component._setup();
          old.apply(component);
        }

        return extendsInit;
      })(originalOnInit);
      component.ngOnInit = newOnInit;
    } else {
      component.ngOnInit = component._setup;
    }

    if ('ngOnDestroy' in component) {
      let originalOnDestroy = component.ngOnDestroy;
      let newOnDestroy = (old => {
        function extendsDestroy() {
          component._detach();
          old.apply(component);
        }

        return extendsDestroy;
      })(originalOnDestroy);
      component.ngOnDestroy = newOnDestroy;
    } else {
      component.ngOnDestroy = component._detach;
    }
    return target;
  }
}

/**
 * Bootstraping the angular2 app
 */ 
 import controller from 'cerebral-controller';
 import {register} from 'cerebral-view';
 
bootstrap(TodoComponent)
  .then(() => {
    register(controller);
  })

/**
 * Example of using the decorator on a component
 * The component will have the following properties:
 * @property signals - the signals from the controller
 * @property state - an object containing the state passed from the decorator
 */ 
 
 import {Cerebral} from 'cerebral-view';
 
@Component({
  selector: 'my-component'
})
@Cerebral({
  todos: ['todos'],
  someOtherState: ['some','other','state']
})
export class TodoComponent {
    /**
   * Button event to add a new todo.
   * @param event
   */
  onButtonClick(event): void {
    this.signals.addTodo();
  }
}

/** View **/
<button (click)="onButtonClick($event)">Add Todo</button>
<div *ngFor="#todo of state.todos">
  <div>{{ todo.text }}</div>
</div>