2013-11-16 06:21:12 +08:00
|
|
|
/* RainLoop Webmail (c) RainLoop Team | Licensed under CC BY-NC-SA 3.0 */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @constructor
|
|
|
|
*/
|
|
|
|
function Knoin()
|
|
|
|
{
|
|
|
|
this.sDefaultScreenName = '';
|
|
|
|
this.oScreens = {};
|
|
|
|
this.oBoot = null;
|
|
|
|
this.oCurrentScreen = null;
|
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:58 +08:00
|
|
|
/**
|
|
|
|
* @param {Object} thisObject
|
|
|
|
*/
|
|
|
|
Knoin.constructorEnd = function (thisObject)
|
|
|
|
{
|
|
|
|
if (Utils.isFunc(thisObject['__constructor_end']))
|
|
|
|
{
|
|
|
|
thisObject['__constructor_end'].call(thisObject);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-11-16 06:21:12 +08:00
|
|
|
Knoin.prototype.sDefaultScreenName = '';
|
|
|
|
Knoin.prototype.oScreens = {};
|
|
|
|
Knoin.prototype.oBoot = null;
|
|
|
|
Knoin.prototype.oCurrentScreen = null;
|
|
|
|
|
|
|
|
Knoin.prototype.hideLoading = function ()
|
|
|
|
{
|
|
|
|
$('#rl-loading').hide();
|
|
|
|
};
|
|
|
|
|
|
|
|
Knoin.prototype.routeOff = function ()
|
|
|
|
{
|
|
|
|
hasher.changed.active = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
Knoin.prototype.routeOn = function ()
|
|
|
|
{
|
|
|
|
hasher.changed.active = true;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {Object} oBoot
|
|
|
|
* @return {Knoin}
|
|
|
|
*/
|
|
|
|
Knoin.prototype.setBoot = function (oBoot)
|
|
|
|
{
|
|
|
|
if (Utils.isNormal(oBoot))
|
|
|
|
{
|
|
|
|
this.oBoot = oBoot;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} sScreenName
|
|
|
|
* @return {?Object}
|
|
|
|
*/
|
|
|
|
Knoin.prototype.screen = function (sScreenName)
|
|
|
|
{
|
|
|
|
return ('' !== sScreenName && !Utils.isUnd(this.oScreens[sScreenName])) ? this.oScreens[sScreenName] : null;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {Function} ViewModelClass
|
|
|
|
* @param {Object=} oScreen
|
|
|
|
*/
|
|
|
|
Knoin.prototype.buildViewModel = function (ViewModelClass, oScreen)
|
|
|
|
{
|
|
|
|
if (ViewModelClass && !ViewModelClass.__builded)
|
|
|
|
{
|
|
|
|
var
|
|
|
|
oViewModel = new ViewModelClass(oScreen),
|
|
|
|
sPosition = oViewModel.viewModelPosition(),
|
|
|
|
oViewModelPlace = $('#rl-content #rl-' + sPosition.toLowerCase()),
|
|
|
|
oViewModelDom = null
|
|
|
|
;
|
|
|
|
|
|
|
|
ViewModelClass.__builded = true;
|
|
|
|
ViewModelClass.__vm = oViewModel;
|
|
|
|
oViewModel.data = RL.data();
|
|
|
|
|
|
|
|
oViewModel.viewModelName = ViewModelClass.__name;
|
|
|
|
|
|
|
|
if (oViewModelPlace && 1 === oViewModelPlace.length)
|
|
|
|
{
|
|
|
|
oViewModelDom = $('<div>').addClass('rl-view-model').addClass('RL-' + oViewModel.viewModelTemplate()).hide().attr('data-bind',
|
|
|
|
'template: {name: "' + oViewModel.viewModelTemplate() + '"}, i18nInit: true');
|
|
|
|
|
|
|
|
oViewModelDom.appendTo(oViewModelPlace);
|
|
|
|
oViewModel.viewModelDom = oViewModelDom;
|
|
|
|
ViewModelClass.__dom = oViewModelDom;
|
|
|
|
|
|
|
|
if ('Popups' === sPosition)
|
|
|
|
{
|
|
|
|
oViewModel.cancelCommand = oViewModel.closeCommand = Utils.createCommand(oViewModel, function () {
|
|
|
|
kn.hideScreenPopup(ViewModelClass);
|
|
|
|
});
|
2014-04-13 08:32:07 +08:00
|
|
|
|
|
|
|
oViewModel.modalVisibility.subscribe(function (bValue) {
|
|
|
|
|
|
|
|
var self = this;
|
|
|
|
if (bValue)
|
|
|
|
{
|
|
|
|
this.viewModelDom.show();
|
|
|
|
this.storeAndSetKeyScope();
|
|
|
|
|
|
|
|
RL.popupVisibilityNames.push(this.viewModelName);
|
|
|
|
|
|
|
|
Utils.delegateRun(this, 'onFocus', [], 500);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Utils.delegateRun(this, 'onHide');
|
|
|
|
this.restoreKeyScope();
|
|
|
|
|
|
|
|
RL.popupVisibilityNames.remove(this.viewModelName);
|
|
|
|
|
|
|
|
_.delay(function () {
|
|
|
|
self.viewModelDom.hide();
|
|
|
|
}, 300);
|
|
|
|
}
|
|
|
|
|
|
|
|
}, oViewModel);
|
2013-11-16 06:21:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Plugins.runHook('view-model-pre-build', [ViewModelClass.__name, oViewModel, oViewModelDom]);
|
|
|
|
|
|
|
|
ko.applyBindings(oViewModel, oViewModelDom[0]);
|
2013-12-29 04:42:07 +08:00
|
|
|
Utils.delegateRun(oViewModel, 'onBuild', [oViewModelDom]);
|
2014-03-13 06:29:33 +08:00
|
|
|
if (oViewModel && 'Popups' === sPosition && !oViewModel.bDisabeCloseOnEsc)
|
|
|
|
{
|
|
|
|
oViewModel.registerPopupEscapeKey();
|
|
|
|
}
|
2013-11-16 06:21:12 +08:00
|
|
|
|
|
|
|
Plugins.runHook('view-model-post-build', [ViewModelClass.__name, oViewModel, oViewModelDom]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Utils.log('Cannot find view model position: ' + sPosition);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ViewModelClass ? ViewModelClass.__vm : null;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {Object} oViewModel
|
|
|
|
* @param {Object} oViewModelDom
|
|
|
|
*/
|
|
|
|
Knoin.prototype.applyExternal = function (oViewModel, oViewModelDom)
|
|
|
|
{
|
|
|
|
if (oViewModel && oViewModelDom)
|
|
|
|
{
|
|
|
|
ko.applyBindings(oViewModel, oViewModelDom);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {Function} ViewModelClassToHide
|
|
|
|
*/
|
|
|
|
Knoin.prototype.hideScreenPopup = function (ViewModelClassToHide)
|
|
|
|
{
|
|
|
|
if (ViewModelClassToHide && ViewModelClassToHide.__vm && ViewModelClassToHide.__dom)
|
|
|
|
{
|
|
|
|
ViewModelClassToHide.__vm.modalVisibility(false);
|
2013-12-28 22:11:19 +08:00
|
|
|
Plugins.runHook('view-model-on-hide', [ViewModelClassToHide.__name, ViewModelClassToHide.__vm]);
|
2013-11-16 06:21:12 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {Function} ViewModelClassToShow
|
|
|
|
* @param {Array=} aParameters
|
|
|
|
*/
|
|
|
|
Knoin.prototype.showScreenPopup = function (ViewModelClassToShow, aParameters)
|
|
|
|
{
|
|
|
|
if (ViewModelClassToShow)
|
|
|
|
{
|
|
|
|
this.buildViewModel(ViewModelClassToShow);
|
|
|
|
|
|
|
|
if (ViewModelClassToShow.__vm && ViewModelClassToShow.__dom)
|
|
|
|
{
|
|
|
|
ViewModelClassToShow.__vm.modalVisibility(true);
|
2013-12-29 04:42:07 +08:00
|
|
|
Utils.delegateRun(ViewModelClassToShow.__vm, 'onShow', aParameters || []);
|
2013-11-16 06:21:12 +08:00
|
|
|
Plugins.runHook('view-model-on-show', [ViewModelClassToShow.__name, ViewModelClassToShow.__vm, aParameters || []]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} sScreenName
|
|
|
|
* @param {string} sSubPart
|
|
|
|
*/
|
|
|
|
Knoin.prototype.screenOnRoute = function (sScreenName, sSubPart)
|
|
|
|
{
|
|
|
|
var
|
|
|
|
self = this,
|
|
|
|
oScreen = null,
|
|
|
|
oCross = null
|
|
|
|
;
|
|
|
|
|
|
|
|
if ('' === Utils.pString(sScreenName))
|
|
|
|
{
|
|
|
|
sScreenName = this.sDefaultScreenName;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ('' !== sScreenName)
|
|
|
|
{
|
|
|
|
oScreen = this.screen(sScreenName);
|
|
|
|
if (!oScreen)
|
|
|
|
{
|
|
|
|
oScreen = this.screen(this.sDefaultScreenName);
|
|
|
|
if (oScreen)
|
|
|
|
{
|
|
|
|
sSubPart = sScreenName + '/' + sSubPart;
|
|
|
|
sScreenName = this.sDefaultScreenName;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (oScreen && oScreen.__started)
|
|
|
|
{
|
|
|
|
if (!oScreen.__builded)
|
|
|
|
{
|
|
|
|
oScreen.__builded = true;
|
|
|
|
|
|
|
|
if (Utils.isNonEmptyArray(oScreen.viewModels()))
|
|
|
|
{
|
|
|
|
_.each(oScreen.viewModels(), function (ViewModelClass) {
|
|
|
|
this.buildViewModel(ViewModelClass, oScreen);
|
|
|
|
}, this);
|
|
|
|
}
|
|
|
|
|
2013-12-29 04:42:07 +08:00
|
|
|
Utils.delegateRun(oScreen, 'onBuild');
|
2013-11-16 06:21:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
_.defer(function () {
|
|
|
|
|
|
|
|
// hide screen
|
|
|
|
if (self.oCurrentScreen)
|
|
|
|
{
|
2013-12-29 04:42:07 +08:00
|
|
|
Utils.delegateRun(self.oCurrentScreen, 'onHide');
|
2013-11-16 06:21:12 +08:00
|
|
|
|
|
|
|
if (Utils.isNonEmptyArray(self.oCurrentScreen.viewModels()))
|
|
|
|
{
|
|
|
|
_.each(self.oCurrentScreen.viewModels(), function (ViewModelClass) {
|
|
|
|
|
|
|
|
if (ViewModelClass.__vm && ViewModelClass.__dom &&
|
|
|
|
'Popups' !== ViewModelClass.__vm.viewModelPosition())
|
|
|
|
{
|
|
|
|
ViewModelClass.__dom.hide();
|
|
|
|
ViewModelClass.__vm.viewModelVisibility(false);
|
2013-12-29 04:42:07 +08:00
|
|
|
Utils.delegateRun(ViewModelClass.__vm, 'onHide');
|
2013-11-16 06:21:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// --
|
|
|
|
|
|
|
|
self.oCurrentScreen = oScreen;
|
|
|
|
|
|
|
|
// show screen
|
|
|
|
if (self.oCurrentScreen)
|
|
|
|
{
|
2014-02-07 07:22:39 +08:00
|
|
|
Utils.delegateRun(self.oCurrentScreen, 'onShow');
|
2013-11-16 06:21:12 +08:00
|
|
|
|
2014-02-07 07:22:39 +08:00
|
|
|
Plugins.runHook('screen-on-show', [self.oCurrentScreen.screenName(), self.oCurrentScreen]);
|
2013-11-16 06:21:12 +08:00
|
|
|
|
2014-02-07 07:22:39 +08:00
|
|
|
if (Utils.isNonEmptyArray(self.oCurrentScreen.viewModels()))
|
|
|
|
{
|
|
|
|
_.each(self.oCurrentScreen.viewModels(), function (ViewModelClass) {
|
2013-11-16 06:21:12 +08:00
|
|
|
|
2014-02-07 07:22:39 +08:00
|
|
|
if (ViewModelClass.__vm && ViewModelClass.__dom &&
|
|
|
|
'Popups' !== ViewModelClass.__vm.viewModelPosition())
|
|
|
|
{
|
|
|
|
ViewModelClass.__dom.show();
|
|
|
|
ViewModelClass.__vm.viewModelVisibility(true);
|
|
|
|
Utils.delegateRun(ViewModelClass.__vm, 'onShow');
|
|
|
|
Utils.delegateRun(ViewModelClass.__vm, 'onFocus', [], 200);
|
2013-11-16 06:21:12 +08:00
|
|
|
|
2014-02-07 07:22:39 +08:00
|
|
|
Plugins.runHook('view-model-on-show', [ViewModelClass.__name, ViewModelClass.__vm]);
|
|
|
|
}
|
2013-11-16 06:21:12 +08:00
|
|
|
|
2014-02-07 07:22:39 +08:00
|
|
|
}, self);
|
|
|
|
}
|
2013-11-16 06:21:12 +08:00
|
|
|
}
|
|
|
|
// --
|
|
|
|
|
|
|
|
oCross = oScreen.__cross();
|
|
|
|
if (oCross)
|
|
|
|
{
|
|
|
|
oCross.parse(sSubPart);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {Array} aScreensClasses
|
|
|
|
*/
|
|
|
|
Knoin.prototype.startScreens = function (aScreensClasses)
|
|
|
|
{
|
2013-12-27 09:00:40 +08:00
|
|
|
$('#rl-content').css({
|
|
|
|
'visibility': 'hidden'
|
|
|
|
});
|
|
|
|
|
2013-11-16 06:21:12 +08:00
|
|
|
_.each(aScreensClasses, function (CScreen) {
|
|
|
|
|
|
|
|
var
|
|
|
|
oScreen = new CScreen(),
|
|
|
|
sScreenName = oScreen ? oScreen.screenName() : ''
|
|
|
|
;
|
|
|
|
|
|
|
|
if (oScreen && '' !== sScreenName)
|
|
|
|
{
|
|
|
|
if ('' === this.sDefaultScreenName)
|
|
|
|
{
|
|
|
|
this.sDefaultScreenName = sScreenName;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.oScreens[sScreenName] = oScreen;
|
|
|
|
}
|
|
|
|
|
|
|
|
}, this);
|
|
|
|
|
|
|
|
|
|
|
|
_.each(this.oScreens, function (oScreen) {
|
|
|
|
if (oScreen && !oScreen.__started && oScreen.__start)
|
|
|
|
{
|
|
|
|
oScreen.__started = true;
|
|
|
|
oScreen.__start();
|
|
|
|
|
|
|
|
Plugins.runHook('screen-pre-start', [oScreen.screenName(), oScreen]);
|
2013-12-29 04:42:07 +08:00
|
|
|
Utils.delegateRun(oScreen, 'onStart');
|
2013-11-16 06:21:12 +08:00
|
|
|
Plugins.runHook('screen-post-start', [oScreen.screenName(), oScreen]);
|
|
|
|
}
|
|
|
|
}, this);
|
|
|
|
|
|
|
|
var oCross = crossroads.create();
|
|
|
|
oCross.addRoute(/^([a-zA-Z0-9\-]*)\/?(.*)$/, _.bind(this.screenOnRoute, this));
|
|
|
|
|
|
|
|
hasher.initialized.add(oCross.parse, oCross);
|
|
|
|
hasher.changed.add(oCross.parse, oCross);
|
|
|
|
hasher.init();
|
2013-12-27 09:00:40 +08:00
|
|
|
|
|
|
|
$('#rl-content').css({
|
|
|
|
'visibility': 'visible'
|
|
|
|
});
|
|
|
|
|
|
|
|
_.delay(function () {
|
|
|
|
$html.removeClass('rl-started-trigger').addClass('rl-started');
|
|
|
|
}, 50);
|
2013-11-16 06:21:12 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} sHash
|
|
|
|
* @param {boolean=} bSilence = false
|
|
|
|
* @param {boolean=} bReplace = false
|
|
|
|
*/
|
|
|
|
Knoin.prototype.setHash = function (sHash, bSilence, bReplace)
|
|
|
|
{
|
|
|
|
sHash = '#' === sHash.substr(0, 1) ? sHash.substr(1) : sHash;
|
|
|
|
sHash = '/' === sHash.substr(0, 1) ? sHash.substr(1) : sHash;
|
|
|
|
|
|
|
|
bReplace = Utils.isUnd(bReplace) ? false : !!bReplace;
|
|
|
|
|
|
|
|
if (Utils.isUnd(bSilence) ? false : !!bSilence)
|
|
|
|
{
|
|
|
|
hasher.changed.active = false;
|
|
|
|
hasher[bReplace ? 'replaceHash' : 'setHash'](sHash);
|
|
|
|
hasher.changed.active = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
hasher.changed.active = true;
|
|
|
|
hasher[bReplace ? 'replaceHash' : 'setHash'](sHash);
|
|
|
|
hasher.setHash(sHash);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return {Knoin}
|
|
|
|
*/
|
|
|
|
Knoin.prototype.bootstart = function ()
|
|
|
|
{
|
|
|
|
if (this.oBoot && this.oBoot.bootstart)
|
|
|
|
{
|
|
|
|
this.oBoot.bootstart();
|
|
|
|
}
|
|
|
|
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
kn = new Knoin();
|