RoundingWellOS/marionette.toolkit

View on GitHub
src/app.js

Summary

Maintainability
A
2 hrs
Test Coverage
import _ from 'underscore';
import { Application } from 'backbone.marionette';
import StateMixin from './mixins/state';
import ChildAppsMixin from './mixins/child-apps';
import EventListenersMixin from './mixins/event-listeners';
import ViewEventsMixin from './mixins/view-events';

const ClassOptions = [
  'startWithParent',
  'restartWithParent',
  'stopWithParent',
  'startAfterInitialized',
  'preventDestroy',
  'StateModel',
  'stateEvents',
  'viewEventPrefix',
  'viewEvents',
  'viewTriggers'
];

/**
 * Marionette.Application with an `initialize` / `start` / `stop` / `destroy` lifecycle.
 *
 * @public
 * @class App
 * @memberOf Toolkit
 * @memberOf Marionette
 */
const App = Application.extend({

  /**
   * Internal flag indiciate when `App` has started but has not yet stopped.
   *
   * @private
   * @type {Boolean}
   * @default false
   */
  _isRunning: false,

  /**
   * Internal flag indiciate when `App` is in the process of stopping then starting.
   *
   * @private
   * @type {Boolean}
   * @default false
   */
  _isRestarting: false,

  /**
   * Set to true if a parent `App` should not be able to destroy this `App`.
   *
   * @type {Boolean|Function}
   * @default false
   */
  preventDestroy: false,

  /**
   * Set to true if `App` should be started after it is initialized.
   *
   * @type {Boolean|Function}
   * @default false
   */
  startAfterInitialized: false,

  /**
   * Set to true if `App` should be started after its parent starts.
   *
   * @type {Boolean|Function}
   * @default false
   */
  startWithParent: false,

  /**
   * Set to false if `App` should not stop after its parent stops.
   *
   * @type {Boolean|Function}
   * @default true
   */
  stopWithParent: true,


  /**
   * Set this to determine if a parent `App` should maintain the child's
   * lifecycle during a restart.
   *
   * @type {Boolean|Function}
   * @default null
   */
  restartWithParent: null,

  /**
   * @public
   * @constructs App
   * @param {Object} [options] - Settings for the App.
   * @param {Boolean} [options.startWithParent]
   * @param {Boolean} [options.restartWithParent]
   * @param {Boolean} [options.stopWithParent]
   * @param {Boolean} [options.startAfterInitialized]
   * @param {Boolean} [options.preventDestroy]
   * @param {Object} [options.state] - Attributes to set on the state model.
   */
  constructor(options = {}) {
    this.mergeOptions(options, ClassOptions);

    this.options = _.extend({}, _.result(this, 'options'), options);

    // ChildAppsMixin
    this._initChildApps(options);

    Application.call(this, options);

    if (_.result(this, 'startAfterInitialized')) {
      this.start(options);
    }
  },

  /**
   * Internal helper to verify if `App` has been destroyed
   *
   * @private
   * @method _ensureAppIsIntact
   * @memberOf App
   * @throws AppDestroyedError - Thrown if `App` has already been destroyed
   */
  _ensureAppIsIntact() {
    if (this._isDestroyed) {
      throw new Error('App has already been destroyed and cannot be used.');
    }
  },

  /**
   * Gets the value of internal `_isRunning` flag
   *
   * @public
   * @method isRunning
   * @memberOf App
   * @returns {Boolean}
   */
  isRunning() {
    return this._isRunning;
  },

  /**
   * Gets the value of internal `_isRestarting` flag
   *
   * @public
   * @method isRestarting
   * @memberOf App
   * @returns {Boolean}
   */
  isRestarting() {
    return this._isRestarting;
  },

  /**
   * Sets the app lifecycle to running.
   *
   * @public
   * @method start
   * @memberOf App
   * @param {Object} [options] - Settings for the App passed through to events
   * @event App#before:start - passes options
   * @returns {App}
   */
  start(options = {}) {
    this._ensureAppIsIntact();

    if (this._isRunning) {
      return this;
    }

    if (options.region) {
      this.setRegion(options.region);
    }

    if (options.view) {
      this.setView(options.view);
    }

    // StateMixin
    this._initState(options);

    // ViewEventMixin
    this._buildEventProxies();

    this.triggerMethod('before:start', options);

    this._isRunning = true;

    this._bindRunningEvents();

    this.triggerStart(options);

    return this;
  },

  /**
   * Sets up region, view, and state events.
   * To only be called after `isRunning` is true
   *
   * @private
   * @method _bindRunningEvents
   * @memberOf App
   */
  _bindRunningEvents() {
    if (this._region) {
      this._regionEventMonitor();
    }

    if (this._view) {
      this._proxyViewEvents(this._view);
    }

    // StateMixin
    this.delegateStateEvents();
  },

  /**
   * Sets the app lifecycle to not running
   * then sets the app lifecycle to running with ending state
   *
   * @public
   * @method restart
   * @memberOf App
   * @returns {App}
   */
  restart(options) {
    const state = this.getState().attributes;

    this._isRestarting = true;
    this.stop().start(_.extend({ state }, options));
    this._isRestarting = false;

    return this;
  },

  /**
   * Starts children and triggers start event
   * For calling within `triggerStart`
   *
   * @public
   * @method finallyStart
   * @memberOf App
   * @event App#start - passes any arguments
   * @returns
   */
  finallyStart() {
    this._startChildApps();
    this.triggerMethod('start', ...arguments);
  },

  /**
   * Triggers start event via finallyStart.
   * Override to introduce async start
   *
   * @public
   * @method triggerStart
   * @memberOf App
   * @param {Object} [options] - Settings for the App passed through to events
   * @returns
   */
  triggerStart(options) {
    this.finallyStart(options);
  },

  /**
   * Sets the app lifecycle to not running.
   * Removes any listeners added during the running state
   *
   * @public
   * @method stop
   * @memberOf App
   * @param {Object} [options] - Settings for the App passed through to events
   * @event App#before:stop - passes options
   * @event App#stop - passes options
   * @returns {App}
   */
  stop(options) {
    if (!this._isRunning) {
      return this;
    }

    this.triggerMethod('before:stop', options);

    this._stopChildApps();

    this._isRunning = false;

    this.triggerMethod('stop', options);

    // Running events are cleaned up after stop so that
    // `stop` event handlers still fire
    this._stopRunningListeners();
    this._stopRunningEvents();

    return this;
  },

  /**
   * Stops the `App` and sets it destroyed.
   *
   * @public
   * @method destroy
   * @memberOf App
   */
  destroy() {
    if (this._isDestroyed) {
      return this;
    }

    this.stop();

    this._removeView();

    this._destroyChildApps();

    Application.prototype.destroy.apply(this, arguments);

    return this;
  },

  /**
   * Set the Application's Region
   *
   * @public
   * @method setRegion
   * @memberOf App
   * @param {Region} [region] - Region to use with the app
   * @returns {Region}
   */
  setRegion(region) {
    if (this._region) {
      this.stopListening(this._region);
    }

    this._region = region;

    if (region.currentView) {
      this.setView(region.currentView);
    }

    if (this._isRunning) {
      this._regionEventMonitor();
    }

    return region;
  },

  /**
   * Monitors the apps region before:show event so the region's view
   * is available to the app
   *
   * @private
   * @method _regionEventMonitor
   * @memberOf App
   */
  _regionEventMonitor() {
    this.listenTo(this._region, {
      'before:show': this._onBeforeShow,
      'empty': this._onEmpty
    });
  },

  /**
   * Region monitor handler which sets the app's view to the region's view
   *
   * @private
   * @method _onBeforeShow
   * @memberOf App
   */
  _onBeforeShow(region, view) {
    this.setView(view);
  },

  /**
   * Region monitor handler which empties the region's view
   *
   * @private
   * @method _onEmpty
   * @memberOf App
   */
  _onEmpty(region, view) {
    if (view !== this._view) {return;}

    this._removeView();
  },

  /**
   * Region monitor handler which deletes the region's view and listeners to view
   *
   * @private
   * @method _removeView
   * @memberOf App
   */
  _removeView() {
    if (this._view) {
      this.stopListening(this._view);
      delete this._view;
    }
  },

  /**
   * Get the Application's Region or
   * Get a region from the Application's View
   *
   * @public
   * @method getRegion
   * @memberOf App
   * @param {String} [regionName] - Optional regionName to get from the view
   * @returns {Region}
   */
  getRegion(regionName) {
    if (!regionName) {
      return this._region;
    }

    return this.getView().getRegion(regionName);
  },

  /**
   * Set the Application's View
   *
   * @public
   * @method setView
   * @memberOf App
   * @param {View} [view] - View to use with the app
   * @returns {View}
   */
  setView(view) {
    if (this._view === view) {
      return view;
    }

    if (this._view) {
      this.stopListening(this._view);
    }

    this._view = view;

    // ViewEventsMixin
    if (this._isRunning) {
      this._proxyViewEvents(view);
    }

    // Internal non-running listener
    this._listenTo(this._view, 'destroy', this._removeView);

    return view;
  },

  /**
   * Get the Application's View
   *
   * @public
   * @method getView
   * @memberOf App
   * @returns {View}
   */
  getView() {
    return this._view || this._region && this._region.currentView;
  },

  /**
   * Shows a view in the Application's region
   *
   * @public
   * @method showView
   * @param {View} view - Child view instance defaults to App's view
   * @param {...args} Additional args that get passed along
   * @returns {View}
   */
  showView(view = this._view, ...args) {
    const region = this.getRegion();

    region.show(view, ...args);

    if (!this.isRunning()) {this.setView(region.currentView);}

    return view;
  },

  /**
   * Shows a view in the region of the app's view
   *
   * @public
   * @method showChildView
   * @param {String} regionName - Name of region to show in
   * @param {View} view - Child view instance
   * @param {...args} Additional args that get passed along
   * @returns {View} - Child view instance
   */
  showChildView(regionName, view, ...args) {
    this.getView().showChildView(regionName, view, ...args);

    return view;
  },

  /**
   * Returns view from the App view by region name.
   *
   * @public
   * @method getChildView
   * @param {String} regionName - Name of region to get view from
   * @returns {View}
   */
  getChildView(regionName) {
    return this.getView().getChildView(regionName);
  }
});

_.extend(App.prototype, StateMixin, ChildAppsMixin, EventListenersMixin, ViewEventsMixin);

export default App;