mirror of
https://github.com/the-djmaze/snappymail.git
synced 2025-01-23 15:18:00 +08:00
469 lines
12 KiB
JavaScript
469 lines
12 KiB
JavaScript
import ko from 'ko';
|
|
|
|
import { $htmlCL, VIEW_MODELS } from 'Common/Globals';
|
|
|
|
let currentScreen = null,
|
|
defaultScreenName = '',
|
|
popupVisibilityNames = [];
|
|
|
|
const SCREENS = {},
|
|
isNonEmptyArray = Array.isNotEmpty,
|
|
autofocus = dom => {
|
|
// if (!rl.settings.app('mobile')) {
|
|
const af = dom.querySelector('[autofocus]');
|
|
af && af.focus();
|
|
};
|
|
|
|
export const popupVisibility = ko.computed(() => 0 < popupVisibilityNames.length);
|
|
|
|
export const ViewType = {
|
|
Popup: 'Popups',
|
|
Left: 'Left',
|
|
Right: 'Right',
|
|
Center: 'Center'
|
|
};
|
|
|
|
/**
|
|
* @param {Function} fExecute
|
|
* @param {(Function|boolean|null)=} fCanExecute = true
|
|
* @returns {Function}
|
|
*/
|
|
export function createCommand(fExecute, fCanExecute = true) {
|
|
let fResult = null;
|
|
|
|
fResult = fExecute
|
|
? (...args) => {
|
|
if (fResult && fResult.canExecute && fResult.canExecute()) {
|
|
fExecute.apply(null, args);
|
|
}
|
|
return false;
|
|
} : ()=>{};
|
|
fResult.enabled = ko.observable(true);
|
|
fResult.isCommand = true;
|
|
|
|
if (typeof fCanExecute === 'function') {
|
|
fResult.canExecute = ko.computed(() => fResult && fResult.enabled() && fCanExecute.call(null));
|
|
} else {
|
|
fResult.canExecute = ko.computed(() => fResult && fResult.enabled() && !!fCanExecute);
|
|
}
|
|
|
|
return fResult;
|
|
}
|
|
|
|
/**
|
|
* @param {Function} SettingsViewModelClass
|
|
* @param {string} template
|
|
* @param {string} labelName
|
|
* @param {string} route
|
|
* @param {boolean=} isDefault = false
|
|
* @returns {void}
|
|
*/
|
|
export function addSettingsViewModel(SettingsViewModelClass, template, labelName, route, isDefault = false) {
|
|
SettingsViewModelClass.__rlSettingsData = {
|
|
Label: labelName,
|
|
Template: template,
|
|
Route: route,
|
|
IsDefault: !!isDefault
|
|
};
|
|
|
|
VIEW_MODELS.settings.push(SettingsViewModelClass);
|
|
}
|
|
|
|
/**
|
|
* @param {string} screenName
|
|
* @returns {?Object}
|
|
*/
|
|
export function screen(screenName) {
|
|
return screenName && null != SCREENS[screenName] ? SCREENS[screenName] : null;
|
|
}
|
|
|
|
/**
|
|
* @param {Function} ViewModelClassToShow
|
|
* @returns {Function|null}
|
|
*/
|
|
function getScreenPopup(PopuViewModelClass) {
|
|
return PopuViewModelClass ? PopuViewModelClass.default || PopuViewModelClass : null;
|
|
}
|
|
|
|
/**
|
|
* @param {Function} ViewModelClassToHide
|
|
* @returns {void}
|
|
*/
|
|
export function hideScreenPopup(ViewModelClassToHide) {
|
|
const ModalView = getScreenPopup(ViewModelClassToHide);
|
|
if (ModalView && ModalView.__vm && ModalView.__dom) {
|
|
ModalView.__vm.modalVisibility(false);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {Function} ViewModelClass
|
|
* @param {Object=} vmScreen
|
|
* @returns {*}
|
|
*/
|
|
function buildViewModel(ViewModelClass, vmScreen) {
|
|
if (ViewModelClass && !ViewModelClass.__builded) {
|
|
let vmDom = null;
|
|
const vm = new ViewModelClass(vmScreen),
|
|
position = ViewModelClass.__type || '',
|
|
vmPlace = position ? document.querySelector('#rl-content #rl-' + position.toLowerCase()) : null;
|
|
|
|
ViewModelClass.__builded = true;
|
|
ViewModelClass.__vm = vm;
|
|
|
|
vm.viewModelName = ViewModelClass.__name;
|
|
vm.viewModelNames = ViewModelClass.__names;
|
|
vm.viewModelTemplateID = ViewModelClass.__templateID;
|
|
vm.viewModelPosition = ViewModelClass.__type;
|
|
|
|
if (vmPlace) {
|
|
vmDom = Element.fromHTML('<div class="rl-view-model RL-' + vm.viewModelTemplateID + '" hidden=""></div>');
|
|
vmPlace.append(vmDom);
|
|
|
|
vm.viewModelDom = vmDom;
|
|
ViewModelClass.__dom = vmDom;
|
|
|
|
if (ViewType.Popup === position) {
|
|
vm.cancelCommand = vm.closeCommand = createCommand(() => {
|
|
hideScreenPopup(ViewModelClass);
|
|
});
|
|
|
|
// show/hide popup/modal
|
|
const endShowHide = e => {
|
|
if (e.target === vmDom) {
|
|
if (vmDom.classList.contains('show')) {
|
|
autofocus(vmDom);
|
|
vm.onShowWithDelay && vm.onShowWithDelay();
|
|
} else {
|
|
vmDom.hidden = true;
|
|
vm.onHideWithDelay && vm.onHideWithDelay();
|
|
}
|
|
}
|
|
};
|
|
|
|
vm.modalVisibility.subscribe(value => {
|
|
if (value) {
|
|
vmDom.style.zIndex = 3000 + popupVisibilityNames.length + 10;
|
|
vmDom.hidden = false;
|
|
vm.storeAndSetKeyScope();
|
|
popupVisibilityNames.push(vm.viewModelName);
|
|
requestAnimationFrame(() => { // wait just before the next paint
|
|
vmDom.offsetHeight; // force a reflow
|
|
vmDom.classList.add('show'); // trigger the transitions
|
|
});
|
|
} else {
|
|
vm.onHide && vm.onHide();
|
|
vmDom.classList.remove('show');
|
|
vm.restoreKeyScope();
|
|
popupVisibilityNames = popupVisibilityNames.filter(v=>v!==vm.viewModelName);
|
|
}
|
|
vmDom.setAttribute('aria-hidden', !value);
|
|
});
|
|
if ('ontransitionend' in vmDom) {
|
|
vmDom.addEventListener('transitionend', endShowHide);
|
|
} else {
|
|
// For Edge < 79 and mobile browsers
|
|
vm.modalVisibility.subscribe(() => ()=>setTimeout(endShowHide({target:vmDom}), 500));
|
|
}
|
|
}
|
|
|
|
ko.applyBindingAccessorsToNode(
|
|
vmDom,
|
|
{
|
|
i18nInit: true,
|
|
template: () => ({ name: vm.viewModelTemplateID })
|
|
},
|
|
vm
|
|
);
|
|
|
|
vm.onBuild && vm.onBuild(vmDom);
|
|
if (vm && ViewType.Popup === position) {
|
|
vm.registerPopupKeyDown();
|
|
}
|
|
} else {
|
|
console.log('Cannot find view model position: ' + position);
|
|
}
|
|
}
|
|
|
|
return ViewModelClass ? ViewModelClass.__vm : null;
|
|
}
|
|
|
|
/**
|
|
* @param {Function} ViewModelClassToShow
|
|
* @param {Array=} params
|
|
* @returns {void}
|
|
*/
|
|
export function showScreenPopup(ViewModelClassToShow, params = []) {
|
|
const ModalView = getScreenPopup(ViewModelClassToShow);
|
|
if (ModalView) {
|
|
buildViewModel(ModalView);
|
|
|
|
if (ModalView.__vm && ModalView.__dom) {
|
|
params = params || [];
|
|
|
|
ModalView.__vm.onBeforeShow && ModalView.__vm.onBeforeShow(...params);
|
|
|
|
ModalView.__vm.modalVisibility(true);
|
|
|
|
ModalView.__vm.onShow && ModalView.__vm.onShow(...params);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {Function} ViewModelClassToShow
|
|
* @returns {void}
|
|
*/
|
|
export function warmUpScreenPopup(ViewModelClassToShow) {
|
|
const ModalView = getScreenPopup(ViewModelClassToShow);
|
|
if (ModalView) {
|
|
buildViewModel(ModalView);
|
|
|
|
if (ModalView.__vm && ModalView.__dom) {
|
|
ModalView.__vm.onWarmUp && ModalView.__vm.onWarmUp();
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {Function} ViewModelClassToShow
|
|
* @returns {boolean}
|
|
*/
|
|
export function isPopupVisible(ViewModelClassToShow) {
|
|
const ModalView = getScreenPopup(ViewModelClassToShow);
|
|
return ModalView && ModalView.__vm ? ModalView.__vm.modalVisibility() : false;
|
|
}
|
|
|
|
/**
|
|
* @param {string} screenName
|
|
* @param {string} subPart
|
|
* @returns {void}
|
|
*/
|
|
function screenOnRoute(screenName, subPart) {
|
|
let vmScreen = null,
|
|
isSameScreen = false;
|
|
|
|
if (null == screenName || '' == screenName) {
|
|
screenName = defaultScreenName;
|
|
}
|
|
|
|
if (screenName) {
|
|
vmScreen = screen(screenName);
|
|
if (!vmScreen) {
|
|
vmScreen = screen(defaultScreenName);
|
|
if (vmScreen) {
|
|
subPart = screenName + '/' + subPart;
|
|
screenName = defaultScreenName;
|
|
}
|
|
}
|
|
|
|
if (vmScreen && vmScreen.__started) {
|
|
isSameScreen = currentScreen && vmScreen === currentScreen;
|
|
|
|
if (!vmScreen.__builded) {
|
|
vmScreen.__builded = true;
|
|
|
|
if (isNonEmptyArray(vmScreen.viewModels())) {
|
|
vmScreen.viewModels().forEach(ViewModelClass => {
|
|
buildViewModel(ViewModelClass, vmScreen);
|
|
});
|
|
}
|
|
|
|
vmScreen.onBuild && vmScreen.onBuild();
|
|
}
|
|
|
|
setTimeout(() => {
|
|
// hide screen
|
|
if (currentScreen && !isSameScreen) {
|
|
currentScreen.onHide && currentScreen.onHide();
|
|
currentScreen.onHideWithDelay && setTimeout(()=>currentScreen.onHideWithDelay(), 500);
|
|
|
|
if (isNonEmptyArray(currentScreen.viewModels())) {
|
|
currentScreen.viewModels().forEach(ViewModelClass => {
|
|
if (
|
|
ViewModelClass.__vm &&
|
|
ViewModelClass.__dom &&
|
|
ViewType.Popup !== ViewModelClass.__vm.viewModelPosition
|
|
) {
|
|
ViewModelClass.__dom.hidden = true;
|
|
ViewModelClass.__vm.viewModelVisible = false;
|
|
|
|
ViewModelClass.__vm.onHide && ViewModelClass.__vm.onHide();
|
|
ViewModelClass.__vm.onHideWithDelay && setTimeout(()=>ViewModelClass.__vm.onHideWithDelay(), 500);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
// --
|
|
|
|
currentScreen = vmScreen;
|
|
|
|
// show screen
|
|
if (currentScreen && !isSameScreen) {
|
|
currentScreen.onShow && currentScreen.onShow();
|
|
|
|
if (isNonEmptyArray(currentScreen.viewModels())) {
|
|
currentScreen.viewModels().forEach(ViewModelClass => {
|
|
if (
|
|
ViewModelClass.__vm &&
|
|
ViewModelClass.__dom &&
|
|
ViewType.Popup !== ViewModelClass.__vm.viewModelPosition
|
|
) {
|
|
ViewModelClass.__vm.onBeforeShow && ViewModelClass.__vm.onBeforeShow();
|
|
|
|
ViewModelClass.__dom.hidden = false;
|
|
ViewModelClass.__vm.viewModelVisible = true;
|
|
|
|
ViewModelClass.__vm.onShow && ViewModelClass.__vm.onShow();
|
|
|
|
autofocus(ViewModelClass.__dom);
|
|
|
|
ViewModelClass.__vm.onShowWithDelay && setTimeout(()=>ViewModelClass.__vm.onShowWithDelay, 200);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
// --
|
|
|
|
vmScreen && vmScreen.__cross && vmScreen.__cross.parse(subPart);
|
|
}, 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {Array} screensClasses
|
|
* @returns {void}
|
|
*/
|
|
export function startScreens(screensClasses) {
|
|
screensClasses.forEach(CScreen => {
|
|
if (CScreen) {
|
|
const vmScreen = new CScreen(),
|
|
screenName = vmScreen ? vmScreen.screenName() : '';
|
|
|
|
if (vmScreen && screenName) {
|
|
if (!defaultScreenName) {
|
|
defaultScreenName = screenName;
|
|
}
|
|
|
|
SCREENS[screenName] = vmScreen;
|
|
}
|
|
}
|
|
});
|
|
|
|
Object.values(SCREENS).forEach(vmScreen => {
|
|
if (vmScreen && !vmScreen.__started && vmScreen.__start) {
|
|
vmScreen.__started = true;
|
|
vmScreen.__start();
|
|
vmScreen.onStart && vmScreen.onStart();
|
|
}
|
|
});
|
|
|
|
const cross = new Crossroads();
|
|
cross.addRoute(/^([a-zA-Z0-9-]*)\/?(.*)$/, screenOnRoute);
|
|
|
|
hasher.initialized.add(cross.parse, cross);
|
|
hasher.changed.add(cross.parse, cross);
|
|
hasher.init();
|
|
|
|
setTimeout(() => {
|
|
$htmlCL.remove('rl-started-trigger');
|
|
$htmlCL.add('rl-started');
|
|
}, 100);
|
|
setTimeout(() => $htmlCL.add('rl-started-delay'), 200);
|
|
}
|
|
|
|
/**
|
|
* @param {Object} params
|
|
* @returns {Function}
|
|
*/
|
|
function viewDecorator({ name, type, templateID }) {
|
|
return (target) => {
|
|
if (target) {
|
|
if (name) {
|
|
if (Array.isArray(name)) {
|
|
target.__names = name;
|
|
} else {
|
|
target.__names = [name];
|
|
}
|
|
|
|
target.__name = target.__names[0];
|
|
}
|
|
|
|
if (type) {
|
|
target.__type = type;
|
|
}
|
|
|
|
if (templateID) {
|
|
target.__templateID = templateID;
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
/**
|
|
* @param {Object} params
|
|
* @returns {Function}
|
|
*/
|
|
function popupDecorator({ name, templateID }) {
|
|
return viewDecorator({ name, type: ViewType.Popup, templateID });
|
|
}
|
|
|
|
/**
|
|
* @param {Function} canExecute
|
|
* @returns {Function}
|
|
*/
|
|
function commandDecorator(canExecute = true) {
|
|
return (target, key, descriptor) => {
|
|
if (!key || !key.match(/Command$/)) {
|
|
throw new Error(`name "${key}" should end with Command suffix`);
|
|
}
|
|
|
|
const value = descriptor.value || descriptor.initializer(),
|
|
normCanExecute = typeof canExecute === 'function' ? canExecute : () => !!canExecute;
|
|
|
|
descriptor.value = function(...args) {
|
|
if (normCanExecute.call(this, this)) {
|
|
value.apply(this, args);
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
descriptor.value.__realCanExecute = normCanExecute;
|
|
descriptor.value.isCommand = true;
|
|
|
|
return descriptor;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* @param {miced} $items
|
|
* @returns {Function}
|
|
*/
|
|
function settingsMenuKeysHandler(items) {
|
|
return ((event, handler)=>{
|
|
let index = items.length;
|
|
if (event && index) {
|
|
while (index-- && !items[index].matches('.selected'));
|
|
if (handler && 'up' === handler.shortcut) {
|
|
index && --index;
|
|
} else if (index < items.length - 1) {
|
|
++index;
|
|
}
|
|
|
|
const resultHash = items[index].href;
|
|
resultHash && rl.route.setHash(resultHash, false, true);
|
|
}
|
|
}).throttle(200);
|
|
}
|
|
|
|
export {
|
|
commandDecorator,
|
|
commandDecorator as command,
|
|
viewDecorator,
|
|
viewDecorator as view,
|
|
popupDecorator,
|
|
popupDecorator as popup,
|
|
settingsMenuKeysHandler
|
|
};
|