/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ "use strict"; // This is loaded into all XUL windows. Wrap in a block to prevent // leaking to window scope. { const { AppConstants } = ChromeUtils.import( "resource://gre/modules/AppConstants.jsm" ); class MozMenuPopup extends MozElements.MozElementMixin(XULPopupElement) { constructor() { super(); this.AUTOSCROLL_INTERVAL = 25; this.NOT_DRAGGING = 0; this.DRAG_OVER_BUTTON = -1; this.DRAG_OVER_POPUP = 1; this._draggingState = this.NOT_DRAGGING; this._scrollTimer = 0; this.addEventListener("popupshowing", event => { if (event.target != this) { return; } // Make sure we generated shadow DOM to place menuitems into. this.shadowRoot; }); this.attachShadow({ mode: "open" }); } connectedCallback() { if (this.delayConnectedCallback() || this.hasConnected) { return; } this.hasConnected = true; if (this.parentNode && this.parentNode.localName == "menulist") { this._setUpMenulistPopup(); } } initShadowDOM() { // Retarget events from shadow DOM arrowscrollbox to the host. this.scrollBox.addEventListener("scroll", ev => this.dispatchEvent(new Event("scroll")) ); this.scrollBox.addEventListener("overflow", ev => this.dispatchEvent(new Event("overflow")) ); this.scrollBox.addEventListener("underflow", ev => this.dispatchEvent(new Event("underflow")) ); } get shadowRoot() { // We generate shadow DOM lazily on popupshowing event to avoid extra load // on the system during browser startup. if (!super.shadowRoot.firstElementChild) { super.shadowRoot.appendChild(this.fragment); this.initShadowDOM(); } return super.shadowRoot; } get fragment() { if (!this.constructor.hasOwnProperty("_fragment")) { this.constructor._fragment = MozXULElement.parseXULToFragment( this.markup ); } return document.importNode(this.constructor._fragment, true); } get markup() { return ` ${this.styles} `; } get styles() { let s = ` :host(.in-menulist) arrowscrollbox::part(scrollbutton-up), :host(.in-menulist) arrowscrollbox::part(scrollbutton-down) { display: none; } :host(.in-menulist) arrowscrollbox::part(scrollbox) { overflow: auto; margin: 0; } :host(.in-menulist) arrowscrollbox::part(scrollbox-clip) { overflow: visible; } `; switch (AppConstants.platform) { case "macosx": s += ` :host(.in-menulist) arrowscrollbox { padding: 0; } `; break; default: break; } return s; } get scrollBox() { if (!this._scrollBox) { this._scrollBox = this.shadowRoot.querySelector("arrowscrollbox"); } return this._scrollBox; } /** * Adds event listeners for a MozMenuPopup inside a menulist element. */ _setUpMenulistPopup() { // Access shadow root to generate menupoup shadow DOMs. We do generate // shadow DOM on popupshowing, but it doesn't work for HTML:selects, // which are implemented via menulist elements living in the main process. // So make them a special case then. this.shadowRoot; this.classList.add("in-menulist"); this.addEventListener("popupshown", () => { // Enable drag scrolling even when the mouse wasn't used. The // mousemove handler will remove it if the mouse isn't down. this._enableDragScrolling(false); }); this.addEventListener("popuphidden", () => { this._draggingState = this.NOT_DRAGGING; this._clearScrollTimer(); this.releaseCapture(); this.scrollBox.scrollbox.scrollTop = 0; }); this.addEventListener("mousedown", event => { if (event.button != 0) { return; } if ( this.state == "open" && (event.target.localName == "menuitem" || event.target.localName == "menu" || event.target.localName == "menucaption") ) { this._enableDragScrolling(true); } }); this.addEventListener("mouseup", event => { if (event.button != 0) { return; } this._draggingState = this.NOT_DRAGGING; this._clearScrollTimer(); }); this.addEventListener("mousemove", event => { if (!this._draggingState) { return; } this._clearScrollTimer(); // If the user released the mouse before the menupopup opens, we will // still be capturing, so check that the button is still pressed. If // not, release the capture and do nothing else. This also handles if // the dropdown was opened via the keyboard. if (!(event.buttons & 1)) { this._draggingState = this.NOT_DRAGGING; this.releaseCapture(); return; } // If dragging outside the top or bottom edge of the menupopup, but // within the menupopup area horizontally, scroll the list in that // direction. The _draggingState flag is used to ensure that scrolling // does not start until the mouse has moved over the menupopup first, // preventing scrolling while over the dropdown button. let popupRect = this.getOuterScreenRect(); if ( event.screenX >= popupRect.left && event.screenX <= popupRect.right ) { if (this._draggingState == this.DRAG_OVER_BUTTON) { if ( event.screenY > popupRect.top && event.screenY < popupRect.bottom ) { this._draggingState = this.DRAG_OVER_POPUP; } } if ( this._draggingState == this.DRAG_OVER_POPUP && (event.screenY <= popupRect.top || event.screenY >= popupRect.bottom) ) { let scrollAmount = event.screenY <= popupRect.top ? -1 : 1; this.scrollBox.scrollByIndex(scrollAmount, true); let win = this.ownerGlobal; this._scrollTimer = win.setInterval(() => { this.scrollBox.scrollByIndex(scrollAmount, true); }, this.AUTOSCROLL_INTERVAL); } } }); this._menulistPopupIsSetUp = true; } _enableDragScrolling(overItem) { if (!this._draggingState) { this.setCaptureAlways(); this._draggingState = overItem ? this.DRAG_OVER_POPUP : this.DRAG_OVER_BUTTON; } } _clearScrollTimer() { if (this._scrollTimer) { this.ownerGlobal.clearInterval(this._scrollTimer); this._scrollTimer = 0; } } } customElements.define("menupopup", MozMenuPopup); MozElements.MozMenuPopup = MozMenuPopup; }