Current Path : /var/www/www-root/data/webdav/www.catalog.monolith-realty.ru/bitrix/js/ui/viewer/ |
Current File : /var/www/www-root/data/webdav/www.catalog.monolith-realty.ru/bitrix/js/ui/viewer/ui.viewer.js |
;(function () { 'use strict'; BX.namespace('BX.UI.Viewer'); BX.UI.Viewer.Controller = function(options) { /** * @type {BX.UI.Viewer.Item[]} */ this.items = null; this.currentIndex = null; this.handlers = {}; this.baseContainer = options.baseContainer || document.body; this.setItems(options.items || []); this.isBodyPaddingAdded = null; this.cycleMode = options.hasOwnProperty('cycleMode')? options.cycleMode : true; this.preload = options.hasOwnProperty('preload')? options.preload : 3; this.stretch = options.hasOwnProperty('stretch')? options.stretch : false; this.cachedData = {}; this.optionsByGroup = {}; this.layout = { container: null, content: null, inner: null, itemContainer: null, next: null, prev: null, close: null, error: null, loader: null, loaderContainer: null, loaderText: null, panel: null }; /** * @type {BX.UI.ActionPanel} */ this.actionPanel = new BX.UI.ActionPanel({ darkMode: true, floatMode: false, autoHide: false, showTotalSelectedBlock: false, showResetAllBlock: false, alignItems: 'center', renderTo: function() { return this.getPanelWrapper(); }.bind(this) }); this.eventsAlreadyBinded = false; this.init(); }; BX.UI.Viewer.Controller.prototype = { /** * @param {HTMLElement} node * @return {BX.Promise} */ buildItemListByNode: function (node) { var promise = new BX.Promise(); var nodes = []; if (this.isSeparateItem(node)) { nodes = [node]; } else if(node.dataset.viewerGroupBy) { nodes = [].slice.call(node.ownerDocument.querySelectorAll("[data-viewer][data-viewer-group-by='" + node.dataset.viewerGroupBy + "']")); } else { nodes = [node]; } this.loadExtensions(this.collectExtensionsForItems(nodes)).then(function (){ var items = nodes.map(function(node) { return BX.UI.Viewer.buildItemByNode(node); }); promise.fulfill(items); }.bind(this)); return promise; }, /** * @param {HTMLElement} node * @return {boolean} */ isSeparateItem: function (node) { return node.dataset.viewerSeparateItem; }, shouldProcessSeparateMode: function (items) { return items.length === 1 && items[0].isSeparateItem(); }, shouldRunViewer: function (node) { if (!BX.type.isDomNode(node) || !node.dataset) { return false; } if (!node.dataset.hasOwnProperty('viewer')) { return false; } return true; }, /** * * @param {HTMLElement[]} nodes * @return {Array} */ collectExtensionsForItems: function (nodes) { var extensionSet = new Set(); nodes.forEach(function (node) { if (BX.type.isString(node.dataset.viewerExtension)) { extensionSet.add(node.dataset.viewerExtension); } }); var extensions = []; extensionSet.forEach(function (ext) { if (shouldLoadExtensions(ext)) { extensions.push(ext); } }); return extensions; }, /** * @param {MouseEvent} event * @return {HTMLElement|null} */ extractTargetFromEvent: function (event) { var target = BX.getEventTarget(event); var shouldRunViewer = false; var maxDepth = 8; do { if (this.shouldRunViewer(target)) { shouldRunViewer = true; break; } target = target.parentNode; maxDepth--; } while (maxDepth > 0 && target); return shouldRunViewer? target : null; }, handleDocumentClick: function (event) { var target = this.extractTargetFromEvent(event); if (!target) { return; } if (target.tagName !== 'A' && target.closest('a[target="_blank"]')) { return false; } event.preventDefault(); this.buildItemListByNode(target).then(function(items){ if (items.length === 0) { return; } if (this.shouldProcessSeparateMode(items)) { this.setItems(items).then(function(){ this.openSeparate(0); }.bind(this)); return; } //shortcut for download if ((BX.browser.IsMac() && event.metaKey) || event.ctrlKey) { this.runActionByNode(target, 'download'); } else { this.setItems(items).then(function(){ this.open(this.getIndexByNode(target)); }.bind(this)); } }.bind(this)); }, bindEvents: function () { if (this.eventsAlreadyBinded) { return; } this.eventsAlreadyBinded = true; this.handlers.keyPress = this.handleKeyPress.bind(this); this.handlers.touchStart = this.handleTouchStart.bind(this); this.handlers.touchEnd = this.handleTouchEnd.bind(this); this.handlers.resize = this.adjustViewerHeight.bind(this); this.handlers.showNext = this.showNext.bind(this); this.handlers.showPrev = this.showPrev.bind(this); this.handlers.close = this.close.bind(this); this.handlers.handleClickOnItemContainer = this.handleClickOnItemContainer.bind(this); this.handlers.handleSliderCloseByEsc = this.handleSliderCloseByEsc.bind(this); BX.bind(document, 'keydown', this.handlers.keyPress); BX.bind(window, 'resize', this.handlers.resize); BX.bind(this.getItemContainer(), 'touchstart', this.handlers.touchStart); BX.bind(this.getItemContainer(), 'touchend', this.handlers.touchEnd); BX.bind(this.getItemContainer(), 'click', this.handlers.handleClickOnItemContainer); BX.bind(this.getNextButton(), 'click', this.handlers.showNext); BX.bind(this.getPrevButton(), 'click', this.handlers.showPrev); BX.bind(this.getCloseButton(), 'click', this.handlers.close); BX.addCustomEvent('SidePanel.Slider:onCloseByEsc', this.handlers.handleSliderCloseByEsc); }, handleVisibleControls: function(ev) { if (BX.browser.IsMobile() || BX.hasClass(document.documentElement, 'bx-touch')) { return; } if (this._timerIdReadingMode) { clearTimeout(this._timerIdReadingMode); } if (!this.cursorInPerimeter(ev) || BX.findParent(ev.target, {className: 'ui-viewer-next'}) || BX.findParent(ev.target, {className: 'ui-viewer-prev'})) { this.disableReadingMode(); } else { this._timerIdReadingMode = setTimeout(function () { this.enableReadingMode(); }.bind(this), 2800); } }, enableReadingMode: function(withTimer) { if (BX.browser.IsMobile() || !this.isOnTop()) { return; } if(withTimer) { this._timerIdReadingMode = setTimeout(function() { this.layout.container.classList.add('ui-viewer-reading-mode'); }.bind(this), 5000); return; } this.layout.container.classList.add('ui-viewer-reading-mode'); }, disableReadingMode: function() { if(this._timerIdReadingMode) { clearTimeout(this._timerIdReadingMode); } this.layout.container.classList.remove('ui-viewer-reading-mode'); }, cursorInPerimeter: function(ev) { var offsetVertical = document.body.clientHeight / 100 * 30; var offsetHorizontal = document.body.clientWidth / 100 * 30; offsetHorizontal < 300 ? offsetHorizontal = 300 : null; offsetVertical < 300 ? offsetVertical = 300 : null; if( ev.y < offsetVertical || ev.y > document.body.clientHeight - offsetVertical || ev.x < offsetHorizontal || ev.x > document.body.clientWidth - offsetHorizontal) { return false } return true; }, /** * @param {BX.SidePanel.Event} event */ handleSliderCloseByEsc: function(event) { if (this.isOpen() && (this.getZindex() > event.getSlider().getZindex())) { event.denyAction(); } }, adjustViewport: function () { var viewportNode = document.querySelector('[name="viewport"]'); if (!viewportNode) { return; } this._viewportContent = viewportNode.getAttribute('content'); viewportNode.setAttribute('content', 'width=device-width, user-scalable=no'); }, restoreViewport: function () { var viewportNode = document.querySelector('[name="viewport"]'); if (!this._viewportContent || !viewportNode) { return; } viewportNode.setAttribute('content', this._viewportContent); }, unbindEvents: function() { this.eventsAlreadyBinded = false; BX.unbind(document, 'keydown', this.handlers.keyPress); BX.unbind(window, 'resize', this.handlers.resize); BX.unbind(this.getItemContainer(), 'touchstart', this.handlers.touchStart); BX.unbind(this.getItemContainer(), 'touchend', this.handlers.touchEnd); BX.unbind(this.getItemContainer(), 'click', this.handlers.handleClickOnItemContainer); BX.unbind(this.getNextButton(), 'click', this.handlers.showNext); BX.unbind(this.getPrevButton(), 'click', this.handlers.showPrev); BX.unbind(this.getCloseButton(), 'click', this.handlers.close); }, init: function () {}, openByNode: function (node) { this.buildItemListByNode(node).then(function (items) { if (items.length === 0) { return; } if (this.shouldProcessSeparateMode(items)) { this.setItems(items).then(function(){ this.openSeparate(0); }.bind(this)); return; } this.setItems(items).then(function(){ this.open(this.getIndexByNode(node)); }.bind(this)); }.bind(this)); }, runActionByNode: function (node, actionId, additionalParams) { this.buildItemListByNode(node).then(function (items) { if (items.length === 0) { return; } this.setItems(items).then(function(){ this.runAction(this.getIndexByNode(node), actionId, additionalParams); }.bind(this)); }.bind(this)); }, runAction: function (index, actionId, additionalParams) { var item = this.getItemByIndex(index); var actionToRun = item.getActions().find(function (action) { return action.id === actionId; }); console.log('actionToRun', actionId, actionToRun); if (!BX.type.isFunction(actionToRun.action)) { console.log('action is not a function'); return; } actionToRun.action.call(this, item, additionalParams); }, /** * @returns {number} */ getZindex: function () { var container = this.getViewerContainer(); if (!container.parentNode) { return 0; } var component = BX.ZIndexManager.getComponent(container); return component.getZIndex(); }, /** * @param items * @return {Promise<extensionsCollection>} */ setItems: function (items) { if (!BX.type.isArray(items)) { throw new Error("BX.UI.Viewer.Controller.setItems: 'items' has to be Array."); } BX.onCustomEvent('BX.UI.Viewer.Controller:onSetItems', [this, items]); this.items = items; this.items.forEach(function (item) { item.setController(this); }, this); return this.loadExtensions(this.collectExtensionsForAction(items)); }, /** * * @param {String|String[]} extensions - Extension name * @return {Promise<extensionsCollection>} */ loadExtensions: function (extensions) { return BX.loadExt(extensions); }, /** * * @param {BX.UI.Viewer.Item[]} items * @return {Array} */ collectExtensionsForAction: function (items) { var extensionSet = new Set(); items.forEach(function (item) { var actions = item.getActions() || []; actions.forEach(function (action) { if (!action.extension) { return; } if (!BX.type.isArray(action.extension)) { action.extension = [action.extension]; } action.extension.forEach(function (ext) { extensionSet.add(ext); }); }); }); var extensions = []; extensionSet.forEach(function (ext) { if (shouldLoadExtensions(ext)) { extensions.push(ext); } }); return extensions; }, appendItem: function (item) { if (!(item instanceof BX.UI.Viewer.Item)) { throw new Error("BX.UI.Viewer.Controller.appendItem: 'item' has to be instance of BX.UI.Viewer.Item."); } item.setController(this); this.items.push(item); }, /** * Reloads item in collection items. It means that we replace old item by the one new * which is the copy. * @param {BX.UI.Viewer.Item} item * @param {Object} options */ reloadItem: function (item, options) { options = options || {}; if (!(item instanceof BX.UI.Viewer.Item)) { throw new Error("BX.UI.Viewer.Controller.reloadItem: 'item' has to be instance of BX.UI.Viewer.Item."); } var index = this.items.indexOf(item); if (index === -1) { throw new Error("BX.UI.Viewer.Controller.reloadItem: there is no 'item' in items to reload."); } var newItem = null; if (item.sourceNode) { newItem = BX.UI.Viewer.buildItemByNode(item.sourceNode); } else { newItem = new item.constructor(item.options); } newItem.setController(this); newItem.applyReloadOptions(options); this.items[index] = newItem; }, show: function (index, options) { options = options || {}; if (typeof index === 'undefined') { index = 0; } BX.onCustomEvent('BX.UI.Viewer.Controller:onBeforeShow', [this, index]); var item = this.getItemByIndex(index); if (!item) { return; } this.hideErrorBlock(); this.hideCurrentItem(); this.disableReadingMode(); this.showLoading(); this.currentIndex = index; this.resetActionsInPanelByItem(this.getCurrentItem()); item.load().then(function (loadedItem) { if (this.getCurrentItem() === loadedItem) { console.log('show item'); this.processShowItem(loadedItem); if (options.asFirstToShow) { loadedItem.asFirstToShow(); } } }.bind(this)) .catch(function (reason) { var loadedItem = reason.item; console.log('catch viewer'); BX.onCustomEvent('BX.UI.Viewer.Controller:onItemError', [this, reason, loadedItem]); if (this.getCurrentItem() === loadedItem) { this.processError(reason, loadedItem); } BX.onCustomEvent('BX.UI.Viewer.Controller:onAfterProcessItemError', [this, reason, loadedItem]); }.bind(this)); this.processPreload(this.currentIndex); this.updateControls(); this.lockScroll(); this.adjustViewerHeight(); }, processPreload: function (fromIndex) { if (!this.preload) { return; } var preloadIndex = fromIndex + 1; while(preloadIndex < (this.preload + fromIndex + 1)) { var itemByIndex = this.getItemByIndex(preloadIndex); if (!itemByIndex) { break; } console.log('Trying to preload', preloadIndex); itemByIndex.load(); preloadIndex++; } }, /** * Reloads item and rerun show if item was as current item. * @param {BX.UI.Viewer.Item} item * @param {Object} options */ reload: function (item, options) { var isCurrentVisibleItem = this.getCurrentItem() === item; this.reloadItem(item, options); if (isCurrentVisibleItem) { console.log('reload'); this.show(this.currentIndex); } }, /** * Reloads and show current item. * @param {?Object} options */ reloadCurrentItem: function (options) { this.reload(this.getCurrentItem(), options || {}); }, /** * @param {BX.UI.Viewer.Item} item */ processShowItem: function(item) { this.hideCurrentItem(); this.hideLoading(); var contentWrapper = BX.create('div', { props: { className: 'ui-viewer-inner-content-wrapper' } }); var fragment = document.createDocumentFragment(); fragment.appendChild(item.render()); var title = item.getTitle(); if (title) { fragment.appendChild(BX.create('div', { props: { className: 'ui-viewer-inner-caption' }, children: [ BX.create('span', { text: title }) ] })); } contentWrapper.appendChild(fragment); var classList = this.layout.container.classList; var containerModifiers = item.listContainerModifiers(); if (containerModifiers.length) { classList.add.apply(classList, containerModifiers); } this.layout.itemContainer.appendChild(contentWrapper); item.afterRender(); this.adjustControlsSize(item.getContentWidth()); BX.onCustomEvent('BX.UI.Viewer.Controller:onAfterShow', [this, item]); }, adjustControlsSize: function(contentWidth) { this.getNextButton().style.width = null; this.getPrevButton().style.width = null; this.getNextButton().style.maxWidth = null; this.getPrevButton().style.maxWidth = null; if (contentWidth instanceof BX.Promise) { contentWidth.then(function(width) { var controlWidth = (document.body.offsetWidth - width) / 2; this.getNextButton().style.width = controlWidth + 'px'; this.getPrevButton().style.width = controlWidth + 'px'; this.getNextButton().style.maxWidth = 'none'; this.getPrevButton().style.maxWidth = 'none'; }.bind(this)); } }, /** * @param {Object} reason * @param {BX.UI.Viewer.Item} item */ processError: function(reason, item) { reason = reason || {}; var message = reason.message || null; if (BX.type.isArray(reason.errors) && reason.errors.length) { if (reason.errors[0].code === 1000 && !reason.message) { message = BX.message("JS_UI_VIEWER_ITEM_TRANSFORMATION_ERROR_1").replace('#DOWNLOAD_LINK#', item.getSrc()); } } this.hideCurrentItem(); this.hideLoading(); var contentWrapper = BX.create('div', { props: { className: 'ui-viewer-inner-content-wrapper' } }); var title = item.getTitle(); if (title) { contentWrapper.appendChild(BX.create('div', { props: { className: 'ui-viewer-inner-caption' }, children: [ BX.create('span', { html: title }) ] }) ); } var options = {}; if (message) { options.title = message; } if (reason.description) { options.description = reason.description; } contentWrapper.appendChild(this.getErrorBlock(options, item)); this.layout.itemContainer.appendChild(contentWrapper); }, hideErrorBlock: function() { if (this.layout.error) { BX.remove(this.layout.error); } }, /** * @param {Object} options * @param {?string} [options.viewType='info'] * @param {?string} [options.title] * @param {?string} [options.description] * @param {BX.UI.Viewer.Item} item * @return {null} */ getErrorBlock: function(options, item) { this.hideErrorBlock(); var viewType = BX.prop.getString(options, 'viewType', 'info'); var title = BX.prop.getString(options, 'title', BX.message("JS_UI_VIEWER_ITEM_TRANSFORMATION_ERROR_1").replace('#DOWNLOAD_LINK#', item.getSrc())); var description = BX.prop.getString(options, 'description', BX.message("JS_UI_VIEWER_ITEM_TRANSFORMATION_HINT")); this.layout.error = BX.create('div', { props: { className: 'ui-viewer-error' }, style: { maxWidth: description? '440px' : null }, children: [ BX.create('div', { props: { className: 'ui-viewer-' + viewType + '-title' }, html: title }), BX.create('div', { props: { className: 'ui-viewer-' + viewType + '-text' }, html: description }) ] }); return this.layout.error; }, /** * @param {BX.UI.Viewer.Item} item */ convertItemActionsToPanelItems: function (item) { return item.getActions().map(function(action) { if (action.id === 'download' && action.href) { action.attributes = { target: '_blank' }; } if (!action.href && BX.type.isFunction(action.action)) { var fn = action.action; action.onclick = function(event, panelItem) { fn.call(this, item); }.bind(this); } return action; }, this); }, /** * @param {String} link * @return {boolean} */ isExternalLink: function (link) { var isAbsoluteLink = new RegExp('^([a-z]+://|//)', 'i'); if (!isAbsoluteLink.test(link)) { return false; } if (!BX.getClass('URL')) { return link.indexOf(location.hostname) === -1; } try { return (new URL(link)).hostname !== location.hostname; } catch(e) {} return true; }, /** * @param {BX.UI.Viewer.Item} item */ refineItemActions: function (item) { var defaultActions = { download: { id: 'download', type: 'download', text: BX.message('JS_UI_VIEWER_ITEM_ACTION_DOWNLOAD'), href: item.src, buttonIconClass: 'ui-btn-icon-download' }, edit: { id: 'edit', type: 'edit', text: BX.message('JS_UI_VIEWER_ITEM_ACTION_EDIT'), buttonIconClass: 'ui-btn-icon-edit' }, share: { id: 'share', type: 'share', text: BX.message('JS_UI_VIEWER_ITEM_ACTION_SHARE'), buttonIconClass: 'ui-btn-icon-share' }, print: { id: 'print', type: 'print', text: '', buttonIconClass: 'ui-btn-icon-print ui-btn-disabled' }, info: { id: 'info', type: 'info', text: '', buttonIconClass: 'ui-btn-icon-info ui-action-panel-item-last' }, delete: { id: 'delete', type: 'delete', text: BX.message('JS_UI_VIEWER_ITEM_ACTION_DELETE'), buttonIconClass: 'ui-btn-icon-remove' } }; return item.getNakedActions().map(function(action) { if (defaultActions[action.type]) { action = BX.mergeEx(defaultActions[action.type], action) } if (!action.id) { action.id = action.type; } if (!action.action && action.href) { action.action = function () { window.open(action.href, this.isExternalLink(action.href)? '_blank' : '_self'); }.bind(this); } if (BX.type.isArray(action.items)) { action.items.forEach(function (item) { if (BX.type.isString(item.onclick)) { item.onclick = new Function('event', 'popupItem', item.onclick); } }) } if (BX.type.isString(action.action)) { var params = action.params || {}; var actionString = action.action; action.action = function(item, additionalParams) { try { var fn = eval(actionString); fn.call(this, item, params, additionalParams); } catch(e) { console.log(e); } }.bind(this); } return action; }, this); }, getLoader: function(options) { if (!this.layout.loader) { this.layout.loader = BX.create('div', { props: { className: 'ui-viewer-loader' }, style: { zIndex: -1 }, children: [ this.layout.loaderContainer = BX.create('div', { props: { className: 'ui-viewer-loader-container' } }), this.layout.loaderText = BX.create('div', { props: { className: 'ui-viewer-loader-text' }, text: '' }) ] }); var loader = new BX.Loader({size: 130}); loader.show(this.layout.loaderContainer); } return this.layout.loader; }, getPrevButton: function() { if (!this.layout.prev) { this.layout.prev = BX.create('div', { props: { className: 'ui-viewer-prev' }, events: { mousewheel: function(event) { this.handleMouseWheelOnControlButton(this.layout.prev, event); }.bind(this) } }) } return this.layout.prev; }, getNextButton: function() { if (!this.layout.next) { this.layout.next = BX.create('div', { props: { className: 'ui-viewer-next' }, events: { mousewheel: function(event) { this.handleMouseWheelOnControlButton(this.layout.next, event); }.bind(this) } }); } return this.layout.next; }, handleMouseWheelOnControlButton: function(controlNode, event) { if (this._timeoutIdMouseWheel) { clearTimeout(this._timeoutIdMouseWheel); } controlNode.style.pointerEvents = 'none'; this._timeoutIdMouseWheel = setTimeout(function() { controlNode.style.pointerEvents = null; }, 50); }, getCloseButton: function() { if (!this.layout.close) { this.layout.close = BX.create('div', { props: { className: 'ui-viewer-close' }, html: '<div class="ui-viewer-close-icon"></div>' }); } return this.layout.close; }, isOpen: function () { return this._isOpen; }, addBodyPadding: function() { var padding = window.innerWidth - document.documentElement.clientWidth; if (BX.getClass('BXIM.messenger.popupMessenger') || padding === 0) { return; } document.body.style.paddingRight = padding + 'px'; var imBar = document.getElementById('bx-im-bar'); if(imBar) { var borderColor = 'rgb(238, 242, 244)'; if(document.body.classList.contains('bitrix24-light-theme')) { borderColor = 'rgba(255, 255, 255, .1)'; } if(document.body.classList.contains('bitrix24-dark-theme')) { borderColor = 'rgba(82, 92, 105, .1)'; } imBar.style.borderRight = padding + 'px solid ' + borderColor; } this.isBodyPaddingAdded = true; }, removeBodyPadding: function() { document.body.style.removeProperty('padding-right'); var imBar = document.getElementById('bx-im-bar'); if (imBar) { imBar.style.removeProperty('border-right'); } this.isBodyPaddingAdded = false; }, openSeparate: function(index) { var item = this.getItemByIndex(index); if (!item) { return; } item.load() .then(function (loadedItem) {}.bind(this)) .catch(function (reason) { var loadedItem = reason.item; console.log('catch viewer'); BX.onCustomEvent('BX.UI.Viewer.Controller:onItemError', [this, reason, loadedItem]); if (this.getCurrentItem() === loadedItem) { this.processError(reason, loadedItem); } BX.onCustomEvent('BX.UI.Viewer.Controller:onAfterProcessItemError', [this, reason, loadedItem]); }.bind(this)); }, open: function(index) { this.adjustViewport(); this.addBodyPadding(); var container = this.getViewerContainer(); this.baseContainer.appendChild(container); BX.focus(container); this.showPanel(); var component = BX.ZIndexManager.getComponent(container); if (!component) { BX.ZIndexManager.register(container, { overlay: this.actionPanel.getPanelContainer(), overlayGap: 1 }); } BX.ZIndexManager.bringToFront(container); this.show(index, { asFirstToShow: true }); this.bindEvents(); this._isOpen = true; }, getPanelWrapper: function() { if (!this.layout.panel) { this.layout.panel = BX.create('div', { props: { className: 'ui-viewer-panel' } }); } return this.layout.panel; }, showPanel: function() { this.actionPanel.layout.container.style.background = 'none'; this.actionPanel.draw(); this.actionPanel.showPanel(); }, resetActionsInPanelByItem: function (item) { this.actionPanel.removeItems(); this.actionPanel.addItems( this.convertItemActionsToPanelItems(item) ); }, hideCurrentItem: function() { if (this.getCurrentItem()) { var classList = this.layout.container.classList; var containerModifiers = this.getCurrentItem().listContainerModifiers(); if (containerModifiers.length) { classList.remove.apply(classList, containerModifiers); } this.getCurrentItem().beforeHide(); } BX.cleanNode(this.layout.itemContainer); }, updateControls: function() { if (!this.allowToUseCycleMode() && this.currentIndex + 1 >= this.items.length) { BX.addClass(this.getNextButton(), 'ui-viewer-navigation-hide'); } else { BX.removeClass(this.getNextButton(), 'ui-viewer-navigation-hide'); } if (!this.allowToUseCycleMode() && this.currentIndex === 0) { BX.addClass(this.getPrevButton(), 'ui-viewer-navigation-hide'); } else { BX.removeClass(this.getPrevButton(), 'ui-viewer-navigation-hide'); } }, /** * @return {BX.UI.Viewer.Item} */ getCurrentItem: function () { return this.getItemByIndex(this.currentIndex); }, /** * @param {HTMLElement} node * @return {int} */ getIndexByNode: function (node) { var nodeIndex = null; this.items.forEach(function (item, index) { if (item.sourceNode === node) { nodeIndex = index; } }); return nodeIndex; }, /** * * @param index * @returns BX.UI.Viewer.Item */ getItemByIndex: function (index) { index = parseInt(index, 10); BX.onCustomEvent('BX.UI.Viewer.Controller:onGetItemByIndex', [this, index]); if (index < 0 || (index - 1) > this.items.length) { return null; } return this.items[index]; }, handleClickOnItemContainer: function (event) { if (this.getCurrentItem() instanceof BX.UI.Viewer.Image) { this.showNext(); } }, allowToUseCycleMode: function () { var cycleMode = this.cycleMode; var groupBy = this.getCurrentItem().getGroupBy(); if (this.optionsByGroup[groupBy] && this.optionsByGroup[groupBy].hasOwnProperty('cycleMode')) { cycleMode = this.optionsByGroup[groupBy].cycleMode; } return this.items.length > 1 && cycleMode; }, showNext: function () { var index = this.currentIndex + 1; if (this.allowToUseCycleMode() && index >= this.items.length) { index = 0; } this.show(index); }, showPrev: function () { var index = this.currentIndex - 1; if (this.allowToUseCycleMode() && index === -1) { index = this.items.length - 1; } this.show(index); }, close: function () { this._isOpen = false; BX.onCustomEvent('BX.UI.Viewer.Controller:onClose', [this]); BX.addClass(this.layout.container, 'ui-viewer-hide'); this.restoreViewport(); this.hideCurrentItem(); BX.bind(this.layout.container, 'transitionend', function() { BX.ZIndexManager.unregister(this.layout.container); BX.remove(this.layout.container); BX.removeClass(this.layout.container, 'ui-viewer-hide'); BX.unbindAll(this.layout.container); this.actionPanel.hidePanel(); this.unLockScroll(); this.unbindEvents(); this.disableReadingMode(); if(this.isBodyPaddingAdded) { this.removeBodyPadding(); } }.bind(this)); // this.items = null; // this.currentIndex = null; // this.layout.container = null; // this.layout.close = null; }, showLoading: function (options) { options = options || {}; options.zIndex = BX.type.isNumber(options.zIndex)? options.zIndex : -1; this.layout.inner.appendChild(this.getLoader()); this.setTextOnLoading(options.text || ''); this.layout.loader.style.zIndex = options.zIndex; }, setTextOnLoading: function (text) { this.layout.loaderText.textContent = text; }, hideLoading: function () { BX.remove(this.layout.loader); }, lockScroll: function() { BX.addClass(document.body, 'ui-viewer-lock-body'); }, unLockScroll: function() { BX.removeClass(document.body, 'ui-viewer-lock-body'); }, adjustViewerHeight: function() { if(!this.layout.container || BX.browser.IsMobile()) return; this.layout.container.style.height = document.documentElement.clientHeight + 'px'; }, getViewerContainer: function() { if (!this.layout.container) { this.layout.container = BX.create('div', { props: { className: 'ui-viewer', tabIndex: 22081990 }, style: { height: window.clientHeight + 'px' }, children: [ this.layout.inner = BX.create('div', { props: { className: 'ui-viewer-inner' }, children: [ this.getItemContainer() ] }), this.getCloseButton(), this.getPrevButton(), this.getNextButton(), this.getPanelWrapper() ] }); } return this.layout.container; }, getItemContainer: function() { if (!this.layout.itemContainer) { this.layout.itemContainer = BX.create('div', { props: { className: 'ui-viewer-inner-content' } }); } return this.layout.itemContainer }, handleTouchStart: function(event) { var touchObject = event.changedTouches[0]; this.swipeDirection = null; this.startX = touchObject.pageX; this.startY = touchObject.pageY; this.startTime = (new Date()).getTime(); // event.preventDefault(); }, handleTouchEnd: function(event) { var touchObject = event.changedTouches[0]; var allowedTime = 300; var threshold = 80; var restraint = 100; var distanceX = touchObject.pageX - this.startX; var distanceY = touchObject.pageY - this.startY; var elapsedTime = (new Date()).getTime() - this.startTime; if (elapsedTime <= allowedTime) { if (Math.abs(distanceX) >= threshold && Math.abs(distanceY) <= restraint) { this.swipeDirection = (distanceX < 0) ? 'left' : 'right'; } // else if (Math.abs(distanceY) >= threshold && Math.abs(distanceX) <= restraint) // { // this.swipeDirection = (distanceY < 0) ? 'up' : 'down'; // } } switch (this.swipeDirection) { case 'left': this.showPrev(); break; case 'right': this.showNext(); break; } // event.preventDefault(); }, isOnTop: function () { if (!this.isOpen()) { return false; } if (BX.getClass('BXIM.messenger') && BXIM.messenger.popupMessenger) { return true; } if (!BX.getClass('BX.SidePanel.Instance') || !BX.SidePanel.Instance.getTopSlider()) { return true; } return this.getZindex() > BX.SidePanel.Instance.getTopSlider().getZindex(); }, handleKeyPress: function (event) { if (!this.isOnTop()) { return; } if (event.metaKey) { return; } switch (event.code) { case 'Space': case 'ArrowRight': this.showNext(); event.preventDefault(); event.stopPropagation(); break; case 'ArrowLeft': this.showPrev(); event.preventDefault(); event.stopPropagation(); break; case 'Escape': this.close(); event.preventDefault(); event.stopPropagation(); break; } this.getCurrentItem().handleKeyPress(event); }, setOptionsByGroup: function (groupBy, options) { this.optionsByGroup[groupBy] = options; return this; }, getCachedData: function(id) { return this.cachedData[id]; }, setCachedData: function(id, data) { this.cachedData[id] = data; }, unsetCachedData: function(id) { this.cachedData[id] = null; }, /** * @param {String} type * @param {String} className */ addType: function (type, className) { return BX.UI.Viewer.addType(type, className); } }; /** * @extends {BX.UI.Viewer.Controller} * @param options * @constructor */ BX.UI.Viewer.InlineController = function (options) { options = options || {}; BX.UI.Viewer.Controller.apply(this, arguments); }; BX.UI.Viewer.InlineController.prototype = { __proto__: BX.UI.Viewer.Controller.prototype, constructor: BX.UI.Viewer.Controller, adjustViewport: function(){}, addBodyPadding: function(){}, adjustZindex: function(){}, showPanel: function(){}, adjustViewerHeight: function(){}, // showLoading: function(){}, /** * @param {HTMLElement} node */ renderItemByNode: function (node) { if (!node) { return; } this.buildItemListByNode(node).then(function(items){ if (items.length === 0) { return; } this.setItems(items).then(function(){ this.open(0); }.bind(this)); }.bind(this)); }, getViewerContainer: function() { if (!this.layout.container) { //this.layout.inner? for showLoading this.layout.container = this.layout.inner = BX.create('div', { props: { className: 'ui-viewer-inner' }, children: [ this.getItemContainer() ] }); } return this.layout.container; }, handleClickOnItemContainer: function(){}, handleKeyPress: function(){}, }; /** * @param type * @param {HTMLElement} node * @return {BX.UI.Viewer.Item} */ BX.UI.Viewer.buildItemByTypeAndNode = function (type, node) { var item = new type(); if (!(item instanceof BX.UI.Viewer.Item)) { throw new Error("BX.UI.Viewer.buildItemByTypeAndNode: 'item' has to be instance of BX.UI.Viewer.Item."); } item.bindSourceNode(node); item.setPropertiesByNode(node); item.setActions(BX.UI.Viewer.Instance.refineItemActions(item)); return item; }; /** * @param {HTMLElement} node * @returns {BX.UI.Viewer.Item} */ BX.UI.Viewer.buildItemByNode = function (node) { if (!BX.type.isDomNode(node)) { throw new Error("BX.UI.Viewer.buildItemByNode: 'node' has to be DomNode."); } var typeCode = node.dataset.viewerType; if (!typeCode && node.tagName.toLowerCase() === 'img') { typeCode = 'image'; } BX.UI.Viewer.triggerEventToFindTypeClass(typeCode); var className = types[typeCode]; if (className) { return BX.UI.Viewer.buildItemByTypeAndNode(BX.getClass(className), node); } if (node.dataset.viewerTypeClass) { if (!BX.getClass(node.dataset.viewerTypeClass)) { throw new Error("BX.UI.Viewer.buildItemByNode: could not find class " + node.dataset.viewerTypeClass); } return BX.UI.Viewer.buildItemByTypeAndNode(BX.getClass(node.dataset.viewerTypeClass), node); } console.warn("BX.UI.Viewer.buildItemByNode: could not find class to build type {" + typeCode + "}"); return BX.UI.Viewer.buildItemByTypeAndNode(BX.getClass(types.unknown), node); }; var types = { image: 'BX.UI.Viewer.Image', plainText: 'BX.UI.Viewer.PlainText', unknown: 'BX.UI.Viewer.Unknown', video: 'BX.UI.Viewer.Video', audio: 'BX.UI.Viewer.Audio', document: 'BX.UI.Viewer.Document', code: 'BX.UI.Viewer.HightlightCode' }; /** * @param {String} type * @param {String} className */ BX.UI.Viewer.addType = function (type, className) { types[type] = className; }; BX.UI.Viewer.triggerEventToFindTypeClass = function (type) { BX.onCustomEvent('BX.UI.Viewer.Controller:onFindType', [BX.UI.Viewer.Instance, type]); }; /** * @param {HTMLElement} container * @param filter * @returns {BX.Promise} */ BX.UI.Viewer.bind = function (container, filter) { if (!BX.type.isDomNode(container)) { throw new Error("BX.UI.Viewer.bind: 'container' has to be DomNode."); } if (!BX.type.isPlainObject(filter) && !BX.type.isFunction(filter)) { filter = function(node) { return BX.type.isElementNode(node) && node.dataset.hasOwnProperty('viewer'); }; } BX.bindDelegate(container, 'click', filter, function(event) { var nodes = BX.findChildren(container, filter, true); var indexToShow = 0; var targetNode = BX.getEventTarget(event); if (targetNode.tagName !== 'A' && targetNode.closest('a[target="_blank"]')) { return false; } var items = nodes.map(function(node, index) { if (node === targetNode) { indexToShow = index; } return BX.UI.Viewer.buildItemByNode(node); }); BX.UI.Viewer.Instance.setItems(items).then(function () { BX.UI.Viewer.Instance.open(indexToShow); }); event.preventDefault(); }); }; var shouldLoadExtensions = function(extension) { if (extension === 'disk.viewer.actions' && BX.getClass('BX.Disk.Viewer.Actions')) { return false; } if (extension === 'disk.viewer.document-item' && BX.getClass('BX.Disk.Viewer.DocumentItem')) { return false; } return true; } var instance = null; /** * @memberOf BX.UI.Viewer * @name BX.UI.Viewer#Instance * @type BX.UI.Viewer.Controller * @static * @readonly */ Object.defineProperty(BX.UI.Viewer, 'Instance', { enumerable: false, get: function() { if (window.top !== window && BX.getClass('window.top.BX.UI.Viewer.Instance')) { return window.top.BX.UI.Viewer.Instance; } if (instance === null) { instance = new BX.UI.Viewer.Controller({}); } return instance; } }); window.document.addEventListener('click', function(event) { if (event.button !== 0) { return; } if (window.top !== window && !BX.getClass('window.top.BX.UI.Viewer.Instance')) { top.BX.loadExt('ui.viewer').then(function () { top.BX.UI.Viewer.Instance.handleDocumentClick(event); }); } else { top.BX.UI.Viewer.Instance.handleDocumentClick(event); } }, true); //We have to show viewer only in top window not in iframe. //So we try to load ui.viewer to the top window if there is no viewer. if (window.top !== window && !BX.getClass('window.top.BX.UI.Viewer.Instance')) { top.BX.loadExt('ui.viewer'); } })();