import ko from 'ko'; class Selector { list; listChecked; isListChecked; focusedItem; selectedItem; selectedItemUseCallback = true; iSelectNextHelper = 0; iFocusedNextHelper = 0; oContentScrollable; sItemSelector; sItemSelectedSelector; sItemCheckedSelector; sItemFocusedSelector; sLastUid = ''; oCallbacks = {}; /** * @param {koProperty} koList * @param {koProperty} koSelectedItem * @param {koProperty} koFocusedItem * @param {string} sItemSelector * @param {string} sItemSelectedSelector * @param {string} sItemCheckedSelector * @param {string} sItemFocusedSelector */ constructor( koList, koSelectedItem, koFocusedItem, sItemSelector, sItemSelectedSelector, sItemCheckedSelector, sItemFocusedSelector ) { this.list = koList; this.listChecked = ko.computed(() => this.list().filter(item => item.checked())).extend({ rateLimit: 0 }); this.isListChecked = ko.computed(() => 0 < this.listChecked().length); this.focusedItem = koFocusedItem || ko.observable(null); this.selectedItem = koSelectedItem || ko.observable(null); const itemSelectedThrottle = (item => this.itemSelected(item)).debounce(300); this.listChecked.subscribe((items) => { if (items.length) { if (null === this.selectedItem()) { if (this.selectedItem.valueHasMutated) { this.selectedItem.valueHasMutated(); } } else { this.selectedItem(null); } } else if (this.autoSelect() && this.focusedItem()) { this.selectedItem(this.focusedItem()); } }, this); this.selectedItem.subscribe((item) => { if (item) { if (this.isListChecked()) { this.listChecked().forEach(subItem => subItem.checked(false)); } if (this.selectedItemUseCallback) { itemSelectedThrottle(item); } } else if (this.selectedItemUseCallback) { this.itemSelected(null); } }, this); this.selectedItem = this.selectedItem.extend({ toggleSubscribeProperty: [this, 'selected'] }); this.focusedItem = this.focusedItem.extend({ toggleSubscribeProperty: [null, 'focused'] }); this.sItemSelector = sItemSelector; this.sItemSelectedSelector = sItemSelectedSelector; this.sItemCheckedSelector = sItemCheckedSelector; this.sItemFocusedSelector = sItemFocusedSelector; this.focusedItem.subscribe(item => item && (this.sLastUid = this.getItemUid(item)), this); let aCache = [], aCheckedCache = [], mFocused = null, mSelected = null; this.list.subscribe( items => { if (Array.isArray(items)) { items.forEach(item => { if (item) { const uid = this.getItemUid(item); aCache.push(uid); item.checked() && aCheckedCache.push(uid); if (null === mFocused && item.focused()) { mFocused = uid; } if (null === mSelected && item.selected()) { mSelected = uid; } } }); } }, this, 'beforeChange' ); this.list.subscribe((aItems) => { let temp = null, getNext = false, isNextFocused = mFocused, isChecked = false, isSelected = false, len = 0; const uids = []; this.selectedItemUseCallback = false; this.focusedItem(null); this.selectedItem(null); if (Array.isArray(aItems)) { len = aCheckedCache.length; aItems.forEach(item => { const uid = this.getItemUid(item); uids.push(uid); if (null !== mFocused && mFocused === uid) { this.focusedItem(item); mFocused = null; } if (0 < len && aCheckedCache.includes(uid)) { isChecked = true; item.checked(true); --len; } if (!isChecked && null !== mSelected && mSelected === uid) { isSelected = true; this.selectedItem(item); mSelected = null; } }); this.selectedItemUseCallback = true; if (!isChecked && !isSelected && this.autoSelect()) { if (this.focusedItem()) { this.selectedItem(this.focusedItem()); } else if (aItems.length) { if (null !== isNextFocused) { getNext = false; isNextFocused = aCache.find(sUid => { if (getNext && uids.includes(sUid)) { return sUid; } if (isNextFocused === sUid) { getNext = true; } return false; }); if (isNextFocused) { temp = aItems.find(oItem => isNextFocused === this.getItemUid(oItem)); } } this.selectedItem(temp || null); this.focusedItem(this.selectedItem()); } } if ( (0 !== this.iSelectNextHelper || 0 !== this.iFocusedNextHelper) && aItems.length && !this.focusedItem() ) { temp = null; if (0 !== this.iFocusedNextHelper) { temp = aItems[-1 === this.iFocusedNextHelper ? aItems.length - 1 : 0] || null; } if (!temp && 0 !== this.iSelectNextHelper) { temp = aItems[-1 === this.iSelectNextHelper ? aItems.length - 1 : 0] || null; } if (temp) { if (0 !== this.iSelectNextHelper) { this.selectedItem(temp || null); } this.focusedItem(temp || null); this.scrollToFocused(); setTimeout(this.scrollToFocused, 100); } this.iSelectNextHelper = 0; this.iFocusedNextHelper = 0; } } aCache = []; aCheckedCache = []; mFocused = null; mSelected = null; }); } itemSelected(item) { if (this.isListChecked()) { if (!item) { (this.oCallbacks.onItemSelect || (()=>{}))(item || null); } } else if (item) { (this.oCallbacks.onItemSelect || (()=>{}))(item); } } /** * @param {boolean} forceSelect */ goDown(forceSelect) { this.newSelectPosition('ArrowDown', false, forceSelect); } /** * @param {boolean} forceSelect */ goUp(forceSelect) { this.newSelectPosition('ArrowUp', false, forceSelect); } unselect() { this.selectedItem(null); this.focusedItem(null); } init(contentScrollable, keyScope = 'all') { this.oContentScrollable = contentScrollable; if (contentScrollable) { contentScrollable.addEventListener('click', event => { let el = event.target.closestWithin(this.sItemSelector, contentScrollable); el && this.actionClick(ko.dataFor(el), event); el = event.target.closestWithin(this.sItemCheckedSelector, contentScrollable); if (el) { const item = ko.dataFor(el); if (item) { if (event.shiftKey) { this.actionClick(item, event); } else { this.focusedItem(item); item.checked(!item.checked()); } } } }); shortcuts.add('enter,open', '', keyScope, () => { const focused = this.focusedItem(); if (focused && !focused.selected()) { this.actionClick(focused); return false; } return true; }); shortcuts.add('arrowup', 'meta', keyScope, () => false); shortcuts.add('arrowdown', 'meta', keyScope, () => false); shortcuts.add('arrowup,arrowdown', 'shift', keyScope, event => { this.newSelectPosition(event.key, true); return false; }); shortcuts.add('arrowup,arrowdown,home,end,pageup,pagedown,space', '', keyScope, event => { this.newSelectPosition(event.key, false); return false; }); } } /** * @returns {boolean} */ autoSelect() { return !!(this.oCallbacks.onAutoSelect || (()=>true))(); } /** * @param {Object} oItem * @returns {string} */ getItemUid(item) { let uid = ''; const getItemUidCallback = this.oCallbacks.onItemGetUid || null; if (getItemUidCallback && item) { uid = getItemUidCallback(item); } return uid.toString(); } /** * @param {string} sEventKey * @param {boolean} bShiftKey * @param {boolean=} bForceSelect = false */ newSelectPosition(sEventKey, bShiftKey, bForceSelect) { let isArrow = 'ArrowUp' === sEventKey || 'ArrowDown' === sEventKey, result = null; const pageStep = 10, list = this.list(), listLen = list ? list.length : 0, focused = this.focusedItem(); if (0 < listLen) { if (focused) { if (isArrow) { let i = list.indexOf(focused); if ('ArrowUp' == sEventKey) { i > 0 && (result = list[i-1]); } else if (++i < listLen) { result = list[i]; } if (!result && ' ' !== sEventKey) { (this.oCallbacks.onUpUpOrDownDown || (()=>true))('ArrowUp' === sEventKey); } } else if ('Home' === sEventKey) { result = list[0]; } else if ('End' === sEventKey) { result = list[list.length - 1]; } else if ('PageDown' === sEventKey) { let i = list.indexOf(focused); if (i < listLen - 1) { result = list[Math.min(i + pageStep, listLen - 1)]; } } else if ('PageUp' === sEventKey) { let i = list.indexOf(focused); if (i > 0) { result = list[Math.max(0, i - pageStep)]; } } } else if ( 'ArrowUp' == sEventKey || 'Home' == sEventKey || 'PageUp' == sEventKey ) { result = list[0]; } else if ( 'ArrowDown' === sEventKey || 'End' === sEventKey || 'PageDown' === sEventKey ) { result = list[list.length - 1]; } } if (result) { this.focusedItem(result); if (focused) { if (bShiftKey && isArrow) { focused.checked(!focused.checked()); } else if (' ' === sEventKey) { focused.checked(!focused.checked()); } } if ((this.autoSelect() || !!bForceSelect) && !this.isListChecked() && ' ' !== sEventKey) { this.selectedItem(result); } this.scrollToFocused(); } else if (focused) { if (bShiftKey && isArrow) { focused.checked(!focused.checked()); } else if (' ' === sEventKey) { focused.checked(!focused.checked()); } this.focusedItem(focused); } } /** * @returns {boolean} */ scrollToFocused() { const scrollable = this.oContentScrollable; if (scrollable) { let block, focused = scrollable.querySelector(this.sItemFocusedSelector); if (focused) { const fRect = focused.getBoundingClientRect(), sRect = scrollable.getBoundingClientRect(); if (fRect.top < sRect.top) { block = 'start'; } else if (fRect.bottom > sRect.bottom) { block = 'end'; } block && focused.scrollIntoView(block === 'start'); } else { scrollable.scrollTop = 0; } } } /** * @returns {boolean} */ scrollToTop() { this.oContentScrollable && (this.oContentScrollable.scrollTop = 0); } eventClickFunction(item, event) { let index = 0, length = 0, changeRange = false, isInRange = false, list = [], checked = false, listItem = null, lineUid = ''; const uid = this.getItemUid(item); if (event && event.shiftKey) { if (uid && this.sLastUid && uid !== this.sLastUid) { list = this.list(); checked = item.checked(); for (index = 0, length = list.length; index < length; index++) { listItem = list[index]; lineUid = this.getItemUid(listItem); changeRange = false; if (lineUid === this.sLastUid || lineUid === uid) { changeRange = true; } if (changeRange) { isInRange = !isInRange; } if (isInRange || changeRange) { listItem.checked(checked); } } } } this.sLastUid = uid || ''; } /** * @param {Object} item * @param {Object=} event */ actionClick(item, event = null) { if (item) { let click = true; if (event) { if (event.shiftKey && !(event.ctrlKey || event.metaKey) && !event.altKey) { click = false; if (!this.sLastUid) { this.sLastUid = this.getItemUid(item); } item.checked(!item.checked()); this.eventClickFunction(item, event); this.focusedItem(item); } else if ((event.ctrlKey || event.metaKey) && !event.shiftKey && !event.altKey) { click = false; this.focusedItem(item); if (this.selectedItem() && item !== this.selectedItem()) { this.selectedItem().checked(true); } item.checked(!item.checked()); } } if (click) { this.selectMessageItem(item); } } } on(eventName, callback) { this.oCallbacks[eventName] = callback; } selectMessageItem(messageItem) { this.focusedItem(messageItem); this.selectedItem(messageItem); this.scrollToFocused(); } } export { Selector, Selector as default };