/* Copyright 2017 Mozilla Foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.PDFPresentationMode = undefined; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); var _ui_utils = require('./ui_utils'); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var DELAY_BEFORE_RESETTING_SWITCH_IN_PROGRESS = 1500; var DELAY_BEFORE_HIDING_CONTROLS = 3000; var ACTIVE_SELECTOR = 'pdfPresentationMode'; var CONTROLS_SELECTOR = 'pdfPresentationModeControls'; var MOUSE_SCROLL_COOLDOWN_TIME = 50; var PAGE_SWITCH_THRESHOLD = 0.1; var SWIPE_MIN_DISTANCE_THRESHOLD = 50; var SWIPE_ANGLE_THRESHOLD = Math.PI / 6; var PDFPresentationMode = function () { function PDFPresentationMode(options) { var _this = this; _classCallCheck(this, PDFPresentationMode); this.container = options.container; this.viewer = options.viewer || options.container.firstElementChild; this.pdfViewer = options.pdfViewer; this.eventBus = options.eventBus; var contextMenuItems = options.contextMenuItems || null; this.active = false; this.args = null; this.contextMenuOpen = false; this.mouseScrollTimeStamp = 0; this.mouseScrollDelta = 0; this.touchSwipeState = null; if (contextMenuItems) { contextMenuItems.contextFirstPage.addEventListener('click', function () { _this.contextMenuOpen = false; _this.eventBus.dispatch('firstpage'); }); contextMenuItems.contextLastPage.addEventListener('click', function () { _this.contextMenuOpen = false; _this.eventBus.dispatch('lastpage'); }); contextMenuItems.contextPageRotateCw.addEventListener('click', function () { _this.contextMenuOpen = false; _this.eventBus.dispatch('rotatecw'); }); contextMenuItems.contextPageRotateCcw.addEventListener('click', function () { _this.contextMenuOpen = false; _this.eventBus.dispatch('rotateccw'); }); } } _createClass(PDFPresentationMode, [{ key: 'request', value: function request() { if (this.switchInProgress || this.active || !this.viewer.hasChildNodes()) { return false; } this._addFullscreenChangeListeners(); this._setSwitchInProgress(); this._notifyStateChange(); if (this.container.requestFullscreen) { this.container.requestFullscreen(); } else if (this.container.mozRequestFullScreen) { this.container.mozRequestFullScreen(); } else if (this.container.webkitRequestFullscreen) { this.container.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT); } else if (this.container.msRequestFullscreen) { this.container.msRequestFullscreen(); } else { return false; } this.args = { page: this.pdfViewer.currentPageNumber, previousScale: this.pdfViewer.currentScaleValue }; return true; } }, { key: '_mouseWheel', value: function _mouseWheel(evt) { if (!this.active) { return; } evt.preventDefault(); var delta = (0, _ui_utils.normalizeWheelEventDelta)(evt); var currentTime = new Date().getTime(); var storedTime = this.mouseScrollTimeStamp; if (currentTime > storedTime && currentTime - storedTime < MOUSE_SCROLL_COOLDOWN_TIME) { return; } if (this.mouseScrollDelta > 0 && delta < 0 || this.mouseScrollDelta < 0 && delta > 0) { this._resetMouseScrollState(); } this.mouseScrollDelta += delta; if (Math.abs(this.mouseScrollDelta) >= PAGE_SWITCH_THRESHOLD) { var totalDelta = this.mouseScrollDelta; this._resetMouseScrollState(); var success = totalDelta > 0 ? this._goToPreviousPage() : this._goToNextPage(); if (success) { this.mouseScrollTimeStamp = currentTime; } } } }, { key: '_goToPreviousPage', value: function _goToPreviousPage() { var page = this.pdfViewer.currentPageNumber; if (page <= 1) { return false; } this.pdfViewer.currentPageNumber = page - 1; return true; } }, { key: '_goToNextPage', value: function _goToNextPage() { var page = this.pdfViewer.currentPageNumber; if (page >= this.pdfViewer.pagesCount) { return false; } this.pdfViewer.currentPageNumber = page + 1; return true; } }, { key: '_notifyStateChange', value: function _notifyStateChange() { this.eventBus.dispatch('presentationmodechanged', { source: this, active: this.active, switchInProgress: !!this.switchInProgress }); } }, { key: '_setSwitchInProgress', value: function _setSwitchInProgress() { var _this2 = this; if (this.switchInProgress) { clearTimeout(this.switchInProgress); } this.switchInProgress = setTimeout(function () { _this2._removeFullscreenChangeListeners(); delete _this2.switchInProgress; _this2._notifyStateChange(); }, DELAY_BEFORE_RESETTING_SWITCH_IN_PROGRESS); } }, { key: '_resetSwitchInProgress', value: function _resetSwitchInProgress() { if (this.switchInProgress) { clearTimeout(this.switchInProgress); delete this.switchInProgress; } } }, { key: '_enter', value: function _enter() { var _this3 = this; this.active = true; this._resetSwitchInProgress(); this._notifyStateChange(); this.container.classList.add(ACTIVE_SELECTOR); setTimeout(function () { _this3.pdfViewer.currentPageNumber = _this3.args.page; _this3.pdfViewer.currentScaleValue = 'page-fit'; }, 0); this._addWindowListeners(); this._showControls(); this.contextMenuOpen = false; this.container.setAttribute('contextmenu', 'viewerContextMenu'); window.getSelection().removeAllRanges(); } }, { key: '_exit', value: function _exit() { var _this4 = this; var page = this.pdfViewer.currentPageNumber; this.container.classList.remove(ACTIVE_SELECTOR); setTimeout(function () { _this4.active = false; _this4._removeFullscreenChangeListeners(); _this4._notifyStateChange(); _this4.pdfViewer.currentScaleValue = _this4.args.previousScale; _this4.pdfViewer.currentPageNumber = page; _this4.args = null; }, 0); this._removeWindowListeners(); this._hideControls(); this._resetMouseScrollState(); this.container.removeAttribute('contextmenu'); this.contextMenuOpen = false; } }, { key: '_mouseDown', value: function _mouseDown(evt) { if (this.contextMenuOpen) { this.contextMenuOpen = false; evt.preventDefault(); return; } if (evt.button === 0) { var isInternalLink = evt.target.href && evt.target.classList.contains('internalLink'); if (!isInternalLink) { evt.preventDefault(); this.pdfViewer.currentPageNumber += evt.shiftKey ? -1 : 1; } } } }, { key: '_contextMenu', value: function _contextMenu() { this.contextMenuOpen = true; } }, { key: '_showControls', value: function _showControls() { var _this5 = this; if (this.controlsTimeout) { clearTimeout(this.controlsTimeout); } else { this.container.classList.add(CONTROLS_SELECTOR); } this.controlsTimeout = setTimeout(function () { _this5.container.classList.remove(CONTROLS_SELECTOR); delete _this5.controlsTimeout; }, DELAY_BEFORE_HIDING_CONTROLS); } }, { key: '_hideControls', value: function _hideControls() { if (!this.controlsTimeout) { return; } clearTimeout(this.controlsTimeout); this.container.classList.remove(CONTROLS_SELECTOR); delete this.controlsTimeout; } }, { key: '_resetMouseScrollState', value: function _resetMouseScrollState() { this.mouseScrollTimeStamp = 0; this.mouseScrollDelta = 0; } }, { key: '_touchSwipe', value: function _touchSwipe(evt) { if (!this.active) { return; } if (evt.touches.length > 1) { this.touchSwipeState = null; return; } switch (evt.type) { case 'touchstart': this.touchSwipeState = { startX: evt.touches[0].pageX, startY: evt.touches[0].pageY, endX: evt.touches[0].pageX, endY: evt.touches[0].pageY }; break; case 'touchmove': if (this.touchSwipeState === null) { return; } this.touchSwipeState.endX = evt.touches[0].pageX; this.touchSwipeState.endY = evt.touches[0].pageY; evt.preventDefault(); break; case 'touchend': if (this.touchSwipeState === null) { return; } var delta = 0; var dx = this.touchSwipeState.endX - this.touchSwipeState.startX; var dy = this.touchSwipeState.endY - this.touchSwipeState.startY; var absAngle = Math.abs(Math.atan2(dy, dx)); if (Math.abs(dx) > SWIPE_MIN_DISTANCE_THRESHOLD && (absAngle <= SWIPE_ANGLE_THRESHOLD || absAngle >= Math.PI - SWIPE_ANGLE_THRESHOLD)) { delta = dx; } else if (Math.abs(dy) > SWIPE_MIN_DISTANCE_THRESHOLD && Math.abs(absAngle - Math.PI / 2) <= SWIPE_ANGLE_THRESHOLD) { delta = dy; } if (delta > 0) { this._goToPreviousPage(); } else if (delta < 0) { this._goToNextPage(); } break; } } }, { key: '_addWindowListeners', value: function _addWindowListeners() { this.showControlsBind = this._showControls.bind(this); this.mouseDownBind = this._mouseDown.bind(this); this.mouseWheelBind = this._mouseWheel.bind(this); this.resetMouseScrollStateBind = this._resetMouseScrollState.bind(this); this.contextMenuBind = this._contextMenu.bind(this); this.touchSwipeBind = this._touchSwipe.bind(this); window.addEventListener('mousemove', this.showControlsBind); window.addEventListener('mousedown', this.mouseDownBind); window.addEventListener('wheel', this.mouseWheelBind); window.addEventListener('keydown', this.resetMouseScrollStateBind); window.addEventListener('contextmenu', this.contextMenuBind); window.addEventListener('touchstart', this.touchSwipeBind); window.addEventListener('touchmove', this.touchSwipeBind); window.addEventListener('touchend', this.touchSwipeBind); } }, { key: '_removeWindowListeners', value: function _removeWindowListeners() { window.removeEventListener('mousemove', this.showControlsBind); window.removeEventListener('mousedown', this.mouseDownBind); window.removeEventListener('wheel', this.mouseWheelBind); window.removeEventListener('keydown', this.resetMouseScrollStateBind); window.removeEventListener('contextmenu', this.contextMenuBind); window.removeEventListener('touchstart', this.touchSwipeBind); window.removeEventListener('touchmove', this.touchSwipeBind); window.removeEventListener('touchend', this.touchSwipeBind); delete this.showControlsBind; delete this.mouseDownBind; delete this.mouseWheelBind; delete this.resetMouseScrollStateBind; delete this.contextMenuBind; delete this.touchSwipeBind; } }, { key: '_fullscreenChange', value: function _fullscreenChange() { if (this.isFullscreen) { this._enter(); } else { this._exit(); } } }, { key: '_addFullscreenChangeListeners', value: function _addFullscreenChangeListeners() { this.fullscreenChangeBind = this._fullscreenChange.bind(this); window.addEventListener('fullscreenchange', this.fullscreenChangeBind); window.addEventListener('mozfullscreenchange', this.fullscreenChangeBind); window.addEventListener('webkitfullscreenchange', this.fullscreenChangeBind); window.addEventListener('MSFullscreenChange', this.fullscreenChangeBind); } }, { key: '_removeFullscreenChangeListeners', value: function _removeFullscreenChangeListeners() { window.removeEventListener('fullscreenchange', this.fullscreenChangeBind); window.removeEventListener('mozfullscreenchange', this.fullscreenChangeBind); window.removeEventListener('webkitfullscreenchange', this.fullscreenChangeBind); window.removeEventListener('MSFullscreenChange', this.fullscreenChangeBind); delete this.fullscreenChangeBind; } }, { key: 'isFullscreen', get: function get() { return !!(document.fullscreenElement || document.mozFullScreen || document.webkitIsFullScreen || document.msFullscreenElement); } }]); return PDFPresentationMode; }(); exports.PDFPresentationMode = PDFPresentationMode;