eeDepart Architecture overview
Core libraries
- jQuery
- Underscore
- Backbone
- Marionette
- Require
Core custom libraries
- Backbone.Components
- Backbone.EventsHub
- Backbone.Services
- Backbone.Resources
- Backbone.AjaxCommands
Repository structure
Core principles
- Modularity
- Extensibility
- Scalability
- Separation of concerns
- Code re-use
- Code quality increase
Modularity
Achieved by creating Modules and Components. Modules and Components are self contained parts of functionality that don't depend on anything else.
Modules encapsulate a single route - every route is added a module, and withing that module logic is implemented by utilizing Components.
Extensibility
Modules and Components in eeDepart are completely override-able. They can either be slightly modified in the custom project or behavior can be completely different ( like for SWR )
Extensibility is achieved by using require.js and a plugin we modified for this purpose. Exact details are in the documentation
Override process is very straightforward and when done once becomes very familiar.
Scalability
Scalability in this sense means that the project can have as many or as little modules and components as necessary. Since every module and every component are completely separate, and usually don't depend on each other, it's perfectly safe to add or remove modules from the application.
Separation of concerns
Along Modules and Components we have introduced specialized parts of application that serve specific purposes:
* Services - influenced by Anguar.js services
* EventsHub - Flux like dispatcher but with more control
* AjaxCommands - utility to ease preparation of ajax request
* Resources - separate each resource in its own container
Code quality
Since eeDepart serves as a CORE for all other Check-ins that are build on this architecture we go to great lengths that only necessary changes are added and we try to improve the code quality overall. We do this by:
* New branching mechanism - that allows proper versioning
* (Manually) taking care that only smallest set of common functionality gets into the core
* Using code style tools: jshint and jscs
* Writing Unit Tests ( inprogress )
* Frequent reviews of our processes
How does it work
Documentation covers the process pretty well but just to summarize:
1. We load all of the dependencies
2. Using pegasus.js we fetch config and resource data
3. When config is ready we start the application
4. On application start ALL modules and standalone components are instantiated
5. Modules set up route listeners and wait to be activated
6. Once a route is reached the module starts and calls the controller to prepare the screen data and render.
What we solved:
* Routing via modules - it's trivially easy to add a new route / module to the app
* Extendible by default
* Modules / Components system - as long as the same principles are followed
* Screen rendering process - already set up for mustache and Marionette
* Local storage handling by default
* Error handling
* Loading screen handling
* Component shareable across projects by default - again as long as principles are followed.
RetrieveApp.js - set up route action
var routerAPI = {
loadRoute: function() {
var route = eeDepart.getCurrentRoute();
Backbone.EventsHub.useHub( 'stateComponentHub' )
.trigger( 'setActiveScreenState', route, {});
}
};
...
...
/**
* Add the route bindings for this module and pass the callback method.
*/
eeDepart.addRoute( routerAPI, routes );
RetrieveApp.js - add route callbacks
function appStateChanged( stateModel, actionData ) {
if ( stateModel.get( 'currentPage' ) === ee.Constants.RouteEvents.RETRIEVE ||
stateModel.get( 'currentPage' ) === '' ) {
API.loadRetrieve( actionData );
}
}
...
...
/**
* Listen to a 'route state' change. When a route is changed in the
* browser, or when a new state is triggered programatically listen
* for the new state change and react accordingly.
*/
Backbone.EventsHub.useHub( 'stateComponentHub' )
.on( 'eeDepart:stateContainer:stateChanged', appStateChanged, this );
/**
* Add the route bindings for this module and pass the callback method.
*/
RetrieveApp.js - when route is reached fire up the screen loading process
loadRetrieve: function( data ) {
var that = this;
require([ 'replace!ROOT_MODULE/retrieve/main/retrieveController:retrieveController' ],
function( RetrieveController ) {
eeDepart.navigate( ee.Constants.RouteEvents.RETRIEVE, data );
if ( !that.controller ) {
that.controller = new RetrieveController();
}
that.controller.loadContent( data );
}
);
}
http://eedepart/app/ck.fly#ck_retrieve
RetrieveController.js - Instantiate the model and the Layout view.
loadContent : function( data ) {
var isDeeplink = Backbone.EventsHub.useHub( 'stateComponentHub' ).request( 'getStateValue', 'isDeeplink' );
this.model = new RetrieveModel( data );
var dataLoadedCallback = _.bind(function() {
var LayoutView = new RetrieveLayoutView({
model: this.model
});
eeDepart.loadContent( LayoutView, isDeeplink );
// Set deeplink as false, we no longer need that info
Backbone.EventsHub.useHub( 'stateComponentHub' ).trigger( 'setStateValue', 'isDeeplink', false );
}, this );
/**
* Promise - based approach for loading data. A model will resolve a
* promise when the data is ready and the callback will fire rendering
* the Views.
*/
this.model.loadAndPrepareData().then( dataLoadedCallback );
}
Core utilities - Services
define([ 'backboneServices' ],
function() {
'use strict';
return Backbone.Service.extend({
initialize : function() {
this.config = new Backbone.Model();
},
services : {
'setConfig': 'setConfigCallback', // Set the initial config we
// get from PHP
'getConfig': 'getConfigCallback', // Get the whole config object
'getParam': 'getParamCallback' // Get one config param
},
setConfigCallback: function( configData ) {
this.config.set( configData );
},
getConfigCallback: function() {
return this.config;
},
getParamCallback : function( param ) {
return this.config.get( param );
}
});
}
);
Backbone.serviceFactory( 'configService', ConfigService); // Register
var myService = Backbone.serviceProvider( 'logtextService' ); // Fetch
/**
* myService now has all above methods available
*/
Resources
define([ 'backbone', 'backboneResources' ],
function( Backbone ) {
'use strict';
var airportsResource = Backbone.Resource.extend({
fetch: function() {
this.deferred = $.Deferred();
if ( !this.done ) {
_.defer(function( context ) {
var airports = Backbone.serviceProvider( 'configService' )
.getParam( 'airportList' );
context.resource.set( airports );
context.done = true;
context.deferred.resolve();
}, this );
} else {
this.deferred.resolve();
}
return this.deferred;
},
getAirportCodesWithNames: function() {
return _.object( this.resource.map(function( model ) {
return [ model.get( 'airport_code' ), model.get( 'airport_name' ) ];
}) );
}
});
Backbone.resourceFactory( 'airportsResource', airportsResource ); // Register
}
);
Backbone.resourceProvider( 'airportsResource' ).fetch().then( callback ) // Use
Behaviors
define([ 'marionette' ],
function( Marionette ) {
'use strict';
return Marionette.Behavior.extend({
events : {
'change select' : 'saveByValue',
'change [type="text"]' : 'saveByValue',
'click [type="radio"]' : 'saveByValue',
'click [type="checkbox"]': 'saveByChecked'
},
saveByValue : function( e ) {
var canSave = true;
if ( !_.isEmpty( this.options.allowedTargets ) &&
!_.contains( this.options.allowedTargets, e.target.name ) ) {
canSave = false;
}
if ( canSave ) {
this.view.model.set( e.target.name, e.target.value );
}
},
saveByChecked: function( e ) {
this.view.model.set( e.target.name, e.target.checked );
}
});
});
Components
define([ 'components/loadingScreen/loadingScreenView',
'components/loadingScreen/loadingScreenModel',
'text!components/loadingScreen/loadingScreenTemplate.html'
],
function( View, Model ) {
'use strict';
var loadingScreen = Backbone.Components.Component.extend({
name : 'loadingScreenComponent',
view : View,
model : Model,
template : 'text!components/loadingScreen/loadingScreenTemplate.html',
initialize : function() {
var hub = Backbone.EventsHub.registerHub( 'loadingScreen' );
// we initially set that loading screen is not displayed
this.displayed = false;
// and we tell this component what events to listen to
hub.on( 'loadingScreen:showLoader', this.showLoadingScreen, this );
hub.on( 'loadingScreen:hideLoader', this.hideLoadingScreen, this );
Backbone.EventsHub.useHub( 'eeDepart' ).on( 'navigate', this.showLoadingScreen, this );
Backbone.EventsHub.useHub( 'eeDepart' ).on( 'contentLoaded', this.hideLoadingScreen, this );
Backbone.EventsHub.useHub( 'eeDepart' ).on( 'componentsLoaded', this.componentsLoaded, this );
},
componentsLoaded : function() {
this.create().render();
},
showLoadingScreen: function() {
//...
},
hideLoadingScreen: function() {
// ...
}
});
/**
* Initialize the component immediately
*/
new loadingScreen();
});
Two sides of one coin
While here we're mentioning only client side features rest assured that the changes and future plans for Server side processing of eeDepart are equally important for the project long term success.
eeDepart Architecture Overview
By jurza
eeDepart Architecture Overview
- 914