]> git.proxmox.com Git - mirror_xterm.js.git/blame - src/Viewport.ts
Add comments explaining actions preventing x from wrapping
[mirror_xterm.js.git] / src / Viewport.ts
CommitLineData
7ff03bb4 1/**
1d300911 2 * @license MIT
7ff03bb4
DI
3 */
4
e0d98711 5import { ITerminal } from './Interfaces';
74483fb2 6import { CharMeasure } from './utils/CharMeasure';
e0d98711 7
7ff03bb4
DI
8/**
9 * Represents the viewport of a terminal, the visible area within the larger buffer of output.
10 * Logic for the virtual scroll bar is included in this object.
7ff03bb4 11 */
e0d98711 12export class Viewport {
e0d98711
DI
13 private currentRowHeight: number;
14 private lastRecordedBufferLength: number;
15 private lastRecordedViewportHeight: number;
7ff03bb4 16
81dc48ef
DI
17 /**
18 * Creates a new Viewport.
19 * @param terminal The terminal this viewport belongs to.
20 * @param viewportElement The DOM element acting as the viewport.
21 * @param scrollArea The DOM element acting as the scroll area.
22 * @param charMeasureElement A DOM element used to measure the character size of. the terminal.
23 */
e0d98711
DI
24 constructor(
25 private terminal: ITerminal,
26 private viewportElement: HTMLElement,
27 private scrollArea: HTMLElement,
74483fb2 28 private charMeasure: CharMeasure
e0d98711
DI
29 ) {
30 this.currentRowHeight = 0;
31 this.lastRecordedBufferLength = 0;
32 this.lastRecordedViewportHeight = 0;
7ff03bb4 33
e0d98711
DI
34 this.terminal.on('scroll', this.syncScrollArea.bind(this));
35 this.terminal.on('resize', this.syncScrollArea.bind(this));
36 this.viewportElement.addEventListener('scroll', this.onScroll.bind(this));
7ff03bb4 37
e0d98711 38 this.syncScrollArea();
7ff03bb4
DI
39 }
40
e0d98711
DI
41 /**
42 * Refreshes row height, setting line-height, viewport height and scroll area height if
43 * necessary.
44 * @param charSize A character size measurement bounding rect object, if it doesn't exist it will
45 * be created.
46 */
74483fb2
DI
47 private refresh(): void {
48 if (this.charMeasure.height > 0) {
ada96a83 49 const rowHeightChanged = this.charMeasure.height !== this.currentRowHeight;
e0d98711 50 if (rowHeightChanged) {
74483fb2
DI
51 this.currentRowHeight = this.charMeasure.height;
52 this.viewportElement.style.lineHeight = this.charMeasure.height + 'px';
53 this.terminal.rowContainer.style.lineHeight = this.charMeasure.height + 'px';
e0d98711 54 }
d2ef19df 55 const viewportHeightChanged = this.lastRecordedViewportHeight !== this.terminal.rows;
e0d98711
DI
56 if (rowHeightChanged || viewportHeightChanged) {
57 this.lastRecordedViewportHeight = this.terminal.rows;
74483fb2 58 this.viewportElement.style.height = this.charMeasure.height * this.terminal.rows + 'px';
e0d98711 59 }
74483fb2 60 this.scrollArea.style.height = (this.charMeasure.height * this.lastRecordedBufferLength) + 'px';
e0d98711 61 }
7ff03bb4 62 }
7ff03bb4 63
e0d98711
DI
64 /**
65 * Updates dimensions and synchronizes the scroll area if necessary.
66 */
67 public syncScrollArea(): void {
68 if (this.lastRecordedBufferLength !== this.terminal.lines.length) {
69 // If buffer height changed
70 this.lastRecordedBufferLength = this.terminal.lines.length;
71 this.refresh();
72 } else if (this.lastRecordedViewportHeight !== this.terminal.rows) {
73 // If viewport height changed
74 this.refresh();
75 } else {
76 // If size has changed, refresh viewport
74483fb2 77 if (this.charMeasure.height !== this.currentRowHeight) {
9385b930 78 this.refresh();
e0d98711
DI
79 }
80 }
7ff03bb4 81
e0d98711 82 // Sync scrollTop
d2ef19df 83 const scrollTop = this.terminal.ydisp * this.currentRowHeight;
e0d98711
DI
84 if (this.viewportElement.scrollTop !== scrollTop) {
85 this.viewportElement.scrollTop = scrollTop;
86 }
7ff03bb4 87 }
e0d98711
DI
88
89 /**
90 * Handles scroll events on the viewport, calculating the new viewport and requesting the
91 * terminal to scroll to it.
92 * @param ev The scroll event.
93 */
94 private onScroll(ev: Event) {
d2ef19df
DI
95 const newRow = Math.round(this.viewportElement.scrollTop / this.currentRowHeight);
96 const diff = newRow - this.terminal.ydisp;
e0d98711 97 this.terminal.scrollDisp(diff, true);
7ff03bb4 98 }
7ff03bb4 99
e0d98711
DI
100 /**
101 * Handles mouse wheel events by adjusting the viewport's scrollTop and delegating the actual
102 * scrolling to `onScroll`, this event needs to be attached manually by the consumer of
103 * `Viewport`.
104 * @param ev The mouse wheel event.
105 */
106 public onWheel(ev: WheelEvent) {
107 if (ev.deltaY === 0) {
108 // Do nothing if it's not a vertical scroll event
109 return;
110 }
111 // Fallback to WheelEvent.DOM_DELTA_PIXEL
d2ef19df 112 let multiplier = 1;
e0d98711
DI
113 if (ev.deltaMode === WheelEvent.DOM_DELTA_LINE) {
114 multiplier = this.currentRowHeight;
115 } else if (ev.deltaMode === WheelEvent.DOM_DELTA_PAGE) {
116 multiplier = this.currentRowHeight * this.terminal.rows;
117 }
118 this.viewportElement.scrollTop += ev.deltaY * multiplier;
119 // Prevent the page from scrolling when the terminal scrolls
120 ev.preventDefault();
121 };
122}