/** * @provides javelin-scrollbar * @requires javelin-install * javelin-dom * javelin-stratcom * javelin-vector * @javelin */ /** * Provides an aesthetic scrollbar. * * This shoves an element's scrollbar under a hidden overflow and draws a * pretty looking fake one in its place. This makes complex UIs with multiple * independently scrollable panels less hideous by (a) making the scrollbar * itself prettier and (b) reclaiming the space occupied by the scrollbar. * * Note that on OSX the heavy scrollbars are normally drawn only if you have * a mouse connected. OSX uses more aesthetic touchpad scrollbars normally, * which these scrollbars emulate. * * This class was initially adapted from "Trackpad Scroll Emulator", by * Jonathan Nicol. See <https://github.com/jnicol/trackpad-scroll-emulator>. */ JX.install('Scrollbar', { construct: function(frame) { this._frame = frame; JX.DOM.listen(frame, 'load', null, JX.bind(this, this._onload)); this._onload(); // Before doing anything, check if the scrollbar control has a measurable // width. If it doesn't, we're already in an environment with an aesthetic // scrollbar (like Safari on OSX with no mouse connected, or an iPhone) // and we don't need to do anything. if (JX.Scrollbar._getScrollbarControlWidth() === 0) { return; } // Wrap the frame content in a bunch of nodes. The frame itself stays on // the outside so that any positioning information the node had isn't // disrupted. // We put a "viewport" node inside of it, which is what actually scrolls. // This is the node that gets a scrollbar, but we make the viewport very // slightly too wide for the frame. That hides the scrollbar underneath // the edge of the frame. // We put a "content" node inside of the viewport. This allows us to // measure the content height so we can resize and offset the scrollbar // handle properly. // We move all the actual frame content into the "content" node. So it // ends up wrapped by the "content" node, then by the "viewport" node, // and finally by the original "frame" node. JX.DOM.alterClass(frame, 'jx-scrollbar-frame', true); var content = JX.$N('div', {className: 'jx-scrollbar-content'}); while (frame.firstChild) { JX.DOM.appendContent(content, frame.firstChild); } var viewport = JX.$N('div', {className: 'jx-scrollbar-viewport'}, content); JX.DOM.appendContent(frame, viewport); this._viewport = viewport; this._content = content; // The handle is the visible node which you can click and drag. this._handle = JX.$N('div', {className: 'jx-scrollbar-handle'}); // The bar is the area the handle slides up and down in. this._bar = JX.$N('div', {className: 'jx-scrollbar-bar'}, this._handle); JX.DOM.prependContent(frame, this._bar); JX.DOM.listen(this._handle, 'mousedown', null, JX.bind(this, this._ondrag)); JX.DOM.listen(this._bar, 'mousedown', null, JX.bind(this, this._onjump)); JX.enableDispatch(document.body, 'mouseenter'); JX.DOM.listen(viewport, 'mouseenter', null, JX.bind(this, this._onenter)); JX.DOM.listen(frame, 'scroll', null, JX.bind(this, this._onscroll)); // Enabling dispatch for this event on `window` allows us to scroll even // if the mouse cursor is dragged outside the window in at least some // browsers (for example, Safari on OSX). JX.enableDispatch(window, 'mousemove'); JX.Stratcom.listen('mousemove', null, JX.bind(this, this._onmove)); JX.Stratcom.listen('mouseup', null, JX.bind(this, this._ondrop)); JX.Stratcom.listen('resize', null, JX.bind(this, this._onresize)); this._resizeViewport(); this._resizeBar(); }, statics: { _controlWidth: null, /** * Compute the width of the browser's scrollbar control, in pixels. */ _getScrollbarControlWidth: function() { var self = JX.Scrollbar; if (self._controlWidth === null) { var tmp = JX.$N('div', {className: 'jx-scrollbar-test'}, '-'); document.body.appendChild(tmp); var d1 = JX.Vector.getDim(tmp); tmp.style.overflowY = 'scroll'; var d2 = JX.Vector.getDim(tmp); JX.DOM.remove(tmp); self._controlWidth = (d2.x - d1.x); } return self._controlWidth; }, /** * Get the margin width required to avoid double scrollbars. * * For most browsers which render a real scrollbar control, this is 0. * Adjacent elements may touch the edge of the content directly without * overlapping. * * On OSX with a trackpad, scrollbars are only drawn when content is * scrolled. Content panes with internal scrollbars may overlap adjacent * scrollbars if they are not laid out with a margin. * * @return int Control margin width in pixels. */ getScrollbarControlMargin: function() { var self = JX.Scrollbar; // If this browser and OS don't render a real scrollbar control, we // need to leave a margin. Generally, this is OSX with no mouse attached. if (self._getScrollbarControlWidth() === 0) { return 12; } return 0; } }, members: { _frame: null, _viewport: null, _content: null, _bar: null, _handle: null, _timeout: null, _dragOrigin: null, _scrollOrigin: null, _lastHeight: null, /** * Mark this content as the scroll frame. * * This changes the behavior of the @{class:JX.DOM} scroll functions so the * continue to work properly if the main page content is reframed to scroll * independently. */ setAsScrollFrame: function() { if (this._viewport) { // If we activated the scrollbar, the viewport and content nodes become // the new scroll and content frames. JX.DOM.setContentFrame(this._viewport, this._content); // If nothing is focused, or the document body is focused, change focus // to the viewport. This makes the arrow keys, spacebar, and page // up/page down keys work immediately after the page loads, without // requiring a click. // Focusing the <div /> itself doesn't work on any browser, so we // add a fake, focusable element and focus that instead. var focus = document.activeElement; if (!focus || focus == window.document.body) { var link = JX.$N('a', {href: '#', className: 'jx-scrollbar-link'}); JX.DOM.listen(link, 'blur', null, function() { // When the user clicks anything else, remove this. try { JX.DOM.remove(link); } catch (ignored) { // We can get a second blur event, likey related to T447. // Fix doesn't seem trivial so just ignore it. } }); JX.DOM.listen(link, 'click', null, function(e) { // Don't respond to clicks. Since the link isn't visible, this // most likely means the user hit enter or something like that. e.kill(); }); JX.DOM.prependContent(this._viewport, link); JX.DOM.focus(link); } } else { // Otherwise, the unaltered content frame is both the scroll frame and // content frame. JX.DOM.setContentFrame(this._frame, this._frame); } }, /** * After the user scrolls the page, show the scrollbar to give them * feedback about their position. */ _onscroll: function() { this._showBar(); }, /** * When the user mouses over the viewport, show the scrollbar. */ _onenter: function() { this._showBar(); }, /** * When the user resizes the window, recalculate everything. */ _onresize: function() { this._resizeViewport(); this._resizeBar(); }, /** * When the user clicks the bar area (but not the handle), jump up or * down a page. */ _onjump: function(e) { if (e.getTarget() === this._handle) { return; } var distance = JX.Vector.getDim(this._viewport).y * (7/8); var epos = JX.$V(e); var hpos = JX.$V(this._handle); if (epos.y > hpos.y) { this._viewport.scrollTop += distance; } else { this._viewport.scrollTop -= distance; } }, /** * When the user clicks the scroll handle, begin dragging it. */ _ondrag: function(e) { e.kill(); // Store the position where the drag started. this._dragOrigin = JX.$V(e); // Store the original position of the handle. this._scrollOrigin = this._viewport.scrollTop; }, /** * As the user drags the scroll handle up or down, scroll the viewport. */ _onmove: function(e) { if (this._dragOrigin === null) { return; } var p = JX.$V(e); var offset = (p.y - this._dragOrigin.y); var ratio = offset / JX.Vector.getDim(this._bar).y; var adjust = ratio * JX.Vector.getDim(this._content).y; if (this._shouldSnapback()) { if (Math.abs(p.x - this._dragOrigin.x) > 140) { adjust = 0; } } this._viewport.scrollTop = this._scrollOrigin + adjust; }, /** * Should the scrollbar snap back to the original position if the user * drags the mouse away to the left or right, perpendicular to the * scrollbar? * * Scrollbars have this behavior on Windows, but not on OSX or Linux. */ _shouldSnapback: function() { // Since this is an OS-specific behavior, detect the OS. We can't // reasonably use feature detection here. return (navigator.platform.indexOf('Win') > -1); }, /** * When the user releases the mouse after a drag, stop moving the * viewport. */ _ondrop: function() { this._dragOrigin = null; // Reset the timer to hide the bar. this._showBar(); }, /** * Something inside the frame fired a load event. * * The typical case is that an image loaded. This may have changed the * height of the scroll area, and we may want to make adjustments. */ _onload: function() { var viewport = this.getViewportNode(); var height = viewport.scrollHeight; var visible = JX.Vector.getDim(viewport).y; if (this._lastHeight !== null && this._lastHeight != height) { // If the viewport was scrollable and was scrolled down to near the // bottom, scroll it down to account for the new height. The effect // of this rule is to keep panels like the chat column scrolled to // the bottom as images load into the thread. if (viewport.scrollTop > 0) { if ((viewport.scrollTop + visible + 64) >= this._lastHeight) { viewport.scrollTop += (height - this._lastHeight); } } } this._lastHeight = height; }, /** * Shove the scrollbar on the viewport under the edge of the frame so the * user can't see it. */ _resizeViewport: function() { var fdim = JX.Vector.getDim(this._frame); fdim.x += JX.Scrollbar._getScrollbarControlWidth(); fdim.setDim(this._viewport); }, /** * Figure out the correct size and offset of the scrollbar handle. */ _resizeBar: function() { // We're hiding and showing the bar itself, not just the handle, because // pages that contain other panels may have scrollbars underneath the // bar. If we don't hide the bar, it ends up eating clicks targeting // these panels. // Because the bar may be hidden, we can't measure it. Measure the // viewport instead. var cdim = JX.Vector.getDim(this._content); var spos = JX.Vector.getAggregateScrollForNode(this._viewport); var vdim = JX.Vector.getDim(this._viewport); var ratio = (vdim.y / cdim.y); // We're scaling things down very slightly to leave a 2px margin at // either end of the scroll gutter, so the bar doesn't quite bump up // against the chrome. ratio = ratio * (vdim.y / (vdim.y + 4)); var offset = Math.round(ratio * spos.y) + 2; var size = Math.floor(ratio * vdim.y); if (size < cdim.y) { this._handle.style.top = offset + 'px'; this._handle.style.height = size + 'px'; JX.DOM.show(this._bar); } else { JX.DOM.hide(this._bar); } }, /** * Show the scrollbar for the next second. */ _showBar: function() { this._resizeBar(); JX.DOM.alterClass(this._handle, 'jx-scrollbar-visible', true); this._clearTimeout(); this._timeout = setTimeout(JX.bind(this, this._hideBar), 1000); }, /** * Hide the scrollbar. */ _hideBar: function() { if (this._dragOrigin !== null) { // If we're currently dragging the handle, we never want to hide // it. return; } JX.DOM.alterClass(this._handle, 'jx-scrollbar-visible', false); this._clearTimeout(); }, /** * Clear the scrollbar hide timeout, if one is set. */ _clearTimeout: function() { if (this._timeout) { clearTimeout(this._timeout); this._timeout = null; } }, getContentNode: function() { return this._content || this._frame; }, getViewportNode: function() { return this._viewport || this._frame; }, scrollTo: function(scroll) { if (this._viewport !== null) { this._viewport.scrollTop = scroll; } else { this._frame.scrollTop = scroll; } return this; } } });