283 lines
9.9 KiB
TypeScript
283 lines
9.9 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the MIT License. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import * as dom from 'vs/base/browser/dom';
|
|
import { FastDomNode, createFastDomNode } from 'vs/base/browser/fastDomNode';
|
|
import { GlobalMouseMoveMonitor, IStandardMouseMoveEventData, standardMouseMoveMerger } from 'vs/base/browser/globalMouseMoveMonitor';
|
|
import { IMouseEvent, StandardWheelEvent } from 'vs/base/browser/mouseEvent';
|
|
import { ScrollbarArrow, ScrollbarArrowOptions } from 'vs/base/browser/ui/scrollbar/scrollbarArrow';
|
|
import { ScrollbarState } from 'vs/base/browser/ui/scrollbar/scrollbarState';
|
|
import { ScrollbarVisibilityController } from 'vs/base/browser/ui/scrollbar/scrollbarVisibilityController';
|
|
import { Widget } from 'vs/base/browser/ui/widget';
|
|
import * as platform from 'vs/base/common/platform';
|
|
import { INewScrollPosition, Scrollable, ScrollbarVisibility } from 'vs/base/common/scrollable';
|
|
|
|
/**
|
|
* The orthogonal distance to the slider at which dragging "resets". This implements "snapping"
|
|
*/
|
|
const MOUSE_DRAG_RESET_DISTANCE = 140;
|
|
|
|
export interface ISimplifiedMouseEvent {
|
|
buttons: number;
|
|
posx: number;
|
|
posy: number;
|
|
}
|
|
|
|
export interface ScrollbarHost {
|
|
onMouseWheel(mouseWheelEvent: StandardWheelEvent): void;
|
|
onDragStart(): void;
|
|
onDragEnd(): void;
|
|
}
|
|
|
|
export interface AbstractScrollbarOptions {
|
|
lazyRender: boolean;
|
|
host: ScrollbarHost;
|
|
scrollbarState: ScrollbarState;
|
|
visibility: ScrollbarVisibility;
|
|
extraScrollbarClassName: string;
|
|
scrollable: Scrollable;
|
|
}
|
|
|
|
export abstract class AbstractScrollbar extends Widget {
|
|
|
|
protected _host: ScrollbarHost;
|
|
protected _scrollable: Scrollable;
|
|
private _lazyRender: boolean;
|
|
protected _scrollbarState: ScrollbarState;
|
|
private _visibilityController: ScrollbarVisibilityController;
|
|
private _mouseMoveMonitor: GlobalMouseMoveMonitor<IStandardMouseMoveEventData>;
|
|
|
|
public domNode: FastDomNode<HTMLElement>;
|
|
public slider!: FastDomNode<HTMLElement>;
|
|
|
|
protected _shouldRender: boolean;
|
|
|
|
constructor(opts: AbstractScrollbarOptions) {
|
|
super();
|
|
this._lazyRender = opts.lazyRender;
|
|
this._host = opts.host;
|
|
this._scrollable = opts.scrollable;
|
|
this._scrollbarState = opts.scrollbarState;
|
|
this._visibilityController = this._register(new ScrollbarVisibilityController(opts.visibility, 'visible scrollbar ' + opts.extraScrollbarClassName, 'invisible scrollbar ' + opts.extraScrollbarClassName));
|
|
this._visibilityController.setIsNeeded(this._scrollbarState.isNeeded());
|
|
this._mouseMoveMonitor = this._register(new GlobalMouseMoveMonitor<IStandardMouseMoveEventData>());
|
|
this._shouldRender = true;
|
|
this.domNode = createFastDomNode(document.createElement('div'));
|
|
this.domNode.setAttribute('role', 'presentation');
|
|
this.domNode.setAttribute('aria-hidden', 'true');
|
|
|
|
this._visibilityController.setDomNode(this.domNode);
|
|
this.domNode.setPosition('absolute');
|
|
|
|
this.onmousedown(this.domNode.domNode, (e) => this._domNodeMouseDown(e));
|
|
}
|
|
|
|
// ----------------- creation
|
|
|
|
/**
|
|
* Creates the dom node for an arrow & adds it to the container
|
|
*/
|
|
protected _createArrow(opts: ScrollbarArrowOptions): void {
|
|
let arrow = this._register(new ScrollbarArrow(opts));
|
|
this.domNode.domNode.appendChild(arrow.bgDomNode);
|
|
this.domNode.domNode.appendChild(arrow.domNode);
|
|
}
|
|
|
|
/**
|
|
* Creates the slider dom node, adds it to the container & hooks up the events
|
|
*/
|
|
protected _createSlider(top: number, left: number, width: number | undefined, height: number | undefined): void {
|
|
this.slider = createFastDomNode(document.createElement('div'));
|
|
this.slider.setClassName('slider');
|
|
this.slider.setPosition('absolute');
|
|
this.slider.setTop(top);
|
|
this.slider.setLeft(left);
|
|
if (typeof width === 'number') {
|
|
this.slider.setWidth(width);
|
|
}
|
|
if (typeof height === 'number') {
|
|
this.slider.setHeight(height);
|
|
}
|
|
this.slider.setLayerHinting(true);
|
|
this.slider.setContain('strict');
|
|
|
|
this.domNode.domNode.appendChild(this.slider.domNode);
|
|
|
|
this.onmousedown(this.slider.domNode, (e) => {
|
|
if (e.leftButton) {
|
|
e.preventDefault();
|
|
this._sliderMouseDown(e, () => { /*nothing to do*/ });
|
|
}
|
|
});
|
|
|
|
this.onclick(this.slider.domNode, e => {
|
|
if (e.leftButton) {
|
|
e.stopPropagation();
|
|
}
|
|
});
|
|
}
|
|
|
|
// ----------------- Update state
|
|
|
|
protected _onElementSize(visibleSize: number): boolean {
|
|
if (this._scrollbarState.setVisibleSize(visibleSize)) {
|
|
this._visibilityController.setIsNeeded(this._scrollbarState.isNeeded());
|
|
this._shouldRender = true;
|
|
if (!this._lazyRender) {
|
|
this.render();
|
|
}
|
|
}
|
|
return this._shouldRender;
|
|
}
|
|
|
|
protected _onElementScrollSize(elementScrollSize: number): boolean {
|
|
if (this._scrollbarState.setScrollSize(elementScrollSize)) {
|
|
this._visibilityController.setIsNeeded(this._scrollbarState.isNeeded());
|
|
this._shouldRender = true;
|
|
if (!this._lazyRender) {
|
|
this.render();
|
|
}
|
|
}
|
|
return this._shouldRender;
|
|
}
|
|
|
|
protected _onElementScrollPosition(elementScrollPosition: number): boolean {
|
|
if (this._scrollbarState.setScrollPosition(elementScrollPosition)) {
|
|
this._visibilityController.setIsNeeded(this._scrollbarState.isNeeded());
|
|
this._shouldRender = true;
|
|
if (!this._lazyRender) {
|
|
this.render();
|
|
}
|
|
}
|
|
return this._shouldRender;
|
|
}
|
|
|
|
// ----------------- rendering
|
|
|
|
public beginReveal(): void {
|
|
this._visibilityController.setShouldBeVisible(true);
|
|
}
|
|
|
|
public beginHide(): void {
|
|
this._visibilityController.setShouldBeVisible(false);
|
|
}
|
|
|
|
public render(): void {
|
|
if (!this._shouldRender) {
|
|
return;
|
|
}
|
|
this._shouldRender = false;
|
|
|
|
this._renderDomNode(this._scrollbarState.getRectangleLargeSize(), this._scrollbarState.getRectangleSmallSize());
|
|
this._updateSlider(this._scrollbarState.getSliderSize(), this._scrollbarState.getArrowSize() + this._scrollbarState.getSliderPosition());
|
|
}
|
|
// ----------------- DOM events
|
|
|
|
private _domNodeMouseDown(e: IMouseEvent): void {
|
|
if (e.target !== this.domNode.domNode) {
|
|
return;
|
|
}
|
|
this._onMouseDown(e);
|
|
}
|
|
|
|
public delegateMouseDown(e: IMouseEvent): void {
|
|
let domTop = this.domNode.domNode.getClientRects()[0].top;
|
|
let sliderStart = domTop + this._scrollbarState.getSliderPosition();
|
|
let sliderStop = domTop + this._scrollbarState.getSliderPosition() + this._scrollbarState.getSliderSize();
|
|
let mousePos = this._sliderMousePosition(e);
|
|
if (sliderStart <= mousePos && mousePos <= sliderStop) {
|
|
// Act as if it was a mouse down on the slider
|
|
if (e.leftButton) {
|
|
e.preventDefault();
|
|
this._sliderMouseDown(e, () => { /*nothing to do*/ });
|
|
}
|
|
} else {
|
|
// Act as if it was a mouse down on the scrollbar
|
|
this._onMouseDown(e);
|
|
}
|
|
}
|
|
|
|
private _onMouseDown(e: IMouseEvent): void {
|
|
let offsetX: number;
|
|
let offsetY: number;
|
|
if (e.target === this.domNode.domNode && typeof e.browserEvent.offsetX === 'number' && typeof e.browserEvent.offsetY === 'number') {
|
|
offsetX = e.browserEvent.offsetX;
|
|
offsetY = e.browserEvent.offsetY;
|
|
} else {
|
|
const domNodePosition = dom.getDomNodePagePosition(this.domNode.domNode);
|
|
offsetX = e.posx - domNodePosition.left;
|
|
offsetY = e.posy - domNodePosition.top;
|
|
}
|
|
this._setDesiredScrollPositionNow(this._scrollbarState.getDesiredScrollPositionFromOffset(this._mouseDownRelativePosition(offsetX, offsetY)));
|
|
if (e.leftButton) {
|
|
e.preventDefault();
|
|
this._sliderMouseDown(e, () => { /*nothing to do*/ });
|
|
}
|
|
}
|
|
|
|
private _sliderMouseDown(e: IMouseEvent, onDragFinished: () => void): void {
|
|
const initialMousePosition = this._sliderMousePosition(e);
|
|
const initialMouseOrthogonalPosition = this._sliderOrthogonalMousePosition(e);
|
|
const initialScrollbarState = this._scrollbarState.clone();
|
|
this.slider.toggleClassName('active', true);
|
|
|
|
this._mouseMoveMonitor.startMonitoring(
|
|
e.target,
|
|
e.buttons,
|
|
standardMouseMoveMerger,
|
|
(mouseMoveData: IStandardMouseMoveEventData) => {
|
|
const mouseOrthogonalPosition = this._sliderOrthogonalMousePosition(mouseMoveData);
|
|
const mouseOrthogonalDelta = Math.abs(mouseOrthogonalPosition - initialMouseOrthogonalPosition);
|
|
|
|
if (platform.isWindows && mouseOrthogonalDelta > MOUSE_DRAG_RESET_DISTANCE) {
|
|
// The mouse has wondered away from the scrollbar => reset dragging
|
|
this._setDesiredScrollPositionNow(initialScrollbarState.getScrollPosition());
|
|
return;
|
|
}
|
|
|
|
const mousePosition = this._sliderMousePosition(mouseMoveData);
|
|
const mouseDelta = mousePosition - initialMousePosition;
|
|
this._setDesiredScrollPositionNow(initialScrollbarState.getDesiredScrollPositionFromDelta(mouseDelta));
|
|
},
|
|
() => {
|
|
this.slider.toggleClassName('active', false);
|
|
this._host.onDragEnd();
|
|
onDragFinished();
|
|
}
|
|
);
|
|
|
|
this._host.onDragStart();
|
|
}
|
|
|
|
private _setDesiredScrollPositionNow(_desiredScrollPosition: number): void {
|
|
|
|
let desiredScrollPosition: INewScrollPosition = {};
|
|
this.writeScrollPosition(desiredScrollPosition, _desiredScrollPosition);
|
|
|
|
this._scrollable.setScrollPositionNow(desiredScrollPosition);
|
|
}
|
|
|
|
public updateScrollbarSize(scrollbarSize: number): void {
|
|
this._updateScrollbarSize(scrollbarSize);
|
|
this._scrollbarState.setScrollbarSize(scrollbarSize);
|
|
this._shouldRender = true;
|
|
if (!this._lazyRender) {
|
|
this.render();
|
|
}
|
|
}
|
|
|
|
// ----------------- Overwrite these
|
|
|
|
protected abstract _renderDomNode(largeSize: number, smallSize: number): void;
|
|
protected abstract _updateSlider(sliderSize: number, sliderPosition: number): void;
|
|
|
|
protected abstract _mouseDownRelativePosition(offsetX: number, offsetY: number): number;
|
|
protected abstract _sliderMousePosition(e: ISimplifiedMouseEvent): number;
|
|
protected abstract _sliderOrthogonalMousePosition(e: ISimplifiedMouseEvent): number;
|
|
protected abstract _updateScrollbarSize(size: number): void;
|
|
|
|
public abstract writeScrollPosition(target: INewScrollPosition, scrollPosition: number): void;
|
|
}
|