123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389 |
- /**
- * scrollablebox.js - scrollable box element for blessed
- * Copyright (c) 2013-2015, Christopher Jeffrey and contributors (MIT License).
- * https://github.com/chjj/blessed
- */
- /**
- * Modules
- */
- var Node = require('./node');
- var Box = require('./box');
- /**
- * ScrollableBox
- */
- function ScrollableBox(options) {
- var self = this;
- if (!(this instanceof Node)) {
- return new ScrollableBox(options);
- }
- options = options || {};
- Box.call(this, options);
- if (options.scrollable === false) {
- return this;
- }
- this.scrollable = true;
- this.childOffset = 0;
- this.childBase = 0;
- this.baseLimit = options.baseLimit || Infinity;
- this.alwaysScroll = options.alwaysScroll;
- this.scrollbar = options.scrollbar;
- if (this.scrollbar) {
- this.scrollbar.ch = this.scrollbar.ch || ' ';
- this.style.scrollbar = this.style.scrollbar || this.scrollbar.style;
- if (!this.style.scrollbar) {
- this.style.scrollbar = {};
- this.style.scrollbar.fg = this.scrollbar.fg;
- this.style.scrollbar.bg = this.scrollbar.bg;
- this.style.scrollbar.bold = this.scrollbar.bold;
- this.style.scrollbar.underline = this.scrollbar.underline;
- this.style.scrollbar.inverse = this.scrollbar.inverse;
- this.style.scrollbar.invisible = this.scrollbar.invisible;
- }
- //this.scrollbar.style = this.style.scrollbar;
- if (this.track || this.scrollbar.track) {
- this.track = this.scrollbar.track || this.track;
- this.style.track = this.style.scrollbar.track || this.style.track;
- this.track.ch = this.track.ch || ' ';
- this.style.track = this.style.track || this.track.style;
- if (!this.style.track) {
- this.style.track = {};
- this.style.track.fg = this.track.fg;
- this.style.track.bg = this.track.bg;
- this.style.track.bold = this.track.bold;
- this.style.track.underline = this.track.underline;
- this.style.track.inverse = this.track.inverse;
- this.style.track.invisible = this.track.invisible;
- }
- this.track.style = this.style.track;
- }
- // Allow controlling of the scrollbar via the mouse:
- if (options.mouse) {
- this.on('mousedown', function(data) {
- if (self._scrollingBar) {
- // Do not allow dragging on the scrollbar:
- delete self.screen._dragging;
- delete self._drag;
- return;
- }
- var x = data.x - self.aleft;
- var y = data.y - self.atop;
- if (x === self.width - self.iright - 1) {
- // Do not allow dragging on the scrollbar:
- delete self.screen._dragging;
- delete self._drag;
- var perc = (y - self.itop) / (self.height - self.iheight);
- self.setScrollPerc(perc * 100 | 0);
- self.screen.render();
- var smd, smu;
- self._scrollingBar = true;
- self.onScreenEvent('mousedown', smd = function(data) {
- var y = data.y - self.atop;
- var perc = y / self.height;
- self.setScrollPerc(perc * 100 | 0);
- self.screen.render();
- });
- // If mouseup occurs out of the window, no mouseup event fires, and
- // scrollbar will drag again on mousedown until another mouseup
- // occurs.
- self.onScreenEvent('mouseup', smu = function() {
- self._scrollingBar = false;
- self.removeScreenEvent('mousedown', smd);
- self.removeScreenEvent('mouseup', smu);
- });
- }
- });
- }
- }
- if (options.mouse) {
- this.on('wheeldown', function() {
- self.scroll(self.height / 2 | 0 || 1);
- self.screen.render();
- });
- this.on('wheelup', function() {
- self.scroll(-(self.height / 2 | 0) || -1);
- self.screen.render();
- });
- }
- if (options.keys && !options.ignoreKeys) {
- this.on('keypress', function(ch, key) {
- if (key.name === 'up' || (options.vi && key.name === 'k')) {
- self.scroll(-1);
- self.screen.render();
- return;
- }
- if (key.name === 'down' || (options.vi && key.name === 'j')) {
- self.scroll(1);
- self.screen.render();
- return;
- }
- if (options.vi && key.name === 'u' && key.ctrl) {
- self.scroll(-(self.height / 2 | 0) || -1);
- self.screen.render();
- return;
- }
- if (options.vi && key.name === 'd' && key.ctrl) {
- self.scroll(self.height / 2 | 0 || 1);
- self.screen.render();
- return;
- }
- if (options.vi && key.name === 'b' && key.ctrl) {
- self.scroll(-self.height || -1);
- self.screen.render();
- return;
- }
- if (options.vi && key.name === 'f' && key.ctrl) {
- self.scroll(self.height || 1);
- self.screen.render();
- return;
- }
- if (options.vi && key.name === 'g' && !key.shift) {
- self.scrollTo(0);
- self.screen.render();
- return;
- }
- if (options.vi && key.name === 'g' && key.shift) {
- self.scrollTo(self.getScrollHeight());
- self.screen.render();
- return;
- }
- });
- }
- this.on('parsed content', function() {
- self._recalculateIndex();
- });
- self._recalculateIndex();
- }
- ScrollableBox.prototype.__proto__ = Box.prototype;
- ScrollableBox.prototype.type = 'scrollable-box';
- // XXX Potentially use this in place of scrollable checks elsewhere.
- ScrollableBox.prototype.__defineGetter__('reallyScrollable', function() {
- if (this.shrink) return this.scrollable;
- return this.getScrollHeight() > this.height;
- });
- ScrollableBox.prototype._scrollBottom = function() {
- if (!this.scrollable) return 0;
- // We could just calculate the children, but we can
- // optimize for lists by just returning the items.length.
- if (this._isList) {
- return this.items ? this.items.length : 0;
- }
- if (this.lpos && this.lpos._scrollBottom) {
- return this.lpos._scrollBottom;
- }
- var bottom = this.children.reduce(function(current, el) {
- // el.height alone does not calculate the shrunken height, we need to use
- // getCoords. A shrunken box inside a scrollable element will not grow any
- // larger than the scrollable element's context regardless of how much
- // content is in the shrunken box, unless we do this (call getCoords
- // without the scrollable calculation):
- // See: $ node test/widget-shrink-fail-2.js
- if (!el.detached) {
- var lpos = el._getCoords(false, true);
- if (lpos) {
- return Math.max(current, el.rtop + (lpos.yl - lpos.yi));
- }
- }
- return Math.max(current, el.rtop + el.height);
- }, 0);
- // XXX Use this? Makes .getScrollHeight() useless!
- // if (bottom < this._clines.length) bottom = this._clines.length;
- if (this.lpos) this.lpos._scrollBottom = bottom;
- return bottom;
- };
- ScrollableBox.prototype.setScroll =
- ScrollableBox.prototype.scrollTo = function(offset, always) {
- // XXX
- // At first, this appeared to account for the first new calculation of childBase:
- this.scroll(0);
- return this.scroll(offset - (this.childBase + this.childOffset), always);
- };
- ScrollableBox.prototype.getScroll = function() {
- return this.childBase + this.childOffset;
- };
- ScrollableBox.prototype.scroll = function(offset, always) {
- if (!this.scrollable) return;
- if (this.detached) return;
- // Handle scrolling.
- var visible = this.height - this.iheight
- , base = this.childBase
- , d
- , p
- , t
- , b
- , max
- , emax;
- if (this.alwaysScroll || always) {
- // Semi-workaround
- this.childOffset = offset > 0
- ? visible - 1 + offset
- : offset;
- } else {
- this.childOffset += offset;
- }
- if (this.childOffset > visible - 1) {
- d = this.childOffset - (visible - 1);
- this.childOffset -= d;
- this.childBase += d;
- } else if (this.childOffset < 0) {
- d = this.childOffset;
- this.childOffset += -d;
- this.childBase += d;
- }
- if (this.childBase < 0) {
- this.childBase = 0;
- } else if (this.childBase > this.baseLimit) {
- this.childBase = this.baseLimit;
- }
- // Find max "bottom" value for
- // content and descendant elements.
- // Scroll the content if necessary.
- if (this.childBase === base) {
- return this.emit('scroll');
- }
- // When scrolling text, we want to be able to handle SGR codes as well as line
- // feeds. This allows us to take preformatted text output from other programs
- // and put it in a scrollable text box.
- this.parseContent();
- // XXX
- // max = this.getScrollHeight() - (this.height - this.iheight);
- max = this._clines.length - (this.height - this.iheight);
- if (max < 0) max = 0;
- emax = this._scrollBottom() - (this.height - this.iheight);
- if (emax < 0) emax = 0;
- this.childBase = Math.min(this.childBase, Math.max(emax, max));
- if (this.childBase < 0) {
- this.childBase = 0;
- } else if (this.childBase > this.baseLimit) {
- this.childBase = this.baseLimit;
- }
- // Optimize scrolling with CSR + IL/DL.
- p = this.lpos;
- // Only really need _getCoords() if we want
- // to allow nestable scrolling elements...
- // or if we **really** want shrinkable
- // scrolling elements.
- // p = this._getCoords();
- if (p && this.childBase !== base && this.screen.cleanSides(this)) {
- t = p.yi + this.itop;
- b = p.yl - this.ibottom - 1;
- d = this.childBase - base;
- if (d > 0 && d < visible) {
- // scrolled down
- this.screen.deleteLine(d, t, t, b);
- } else if (d < 0 && -d < visible) {
- // scrolled up
- d = -d;
- this.screen.insertLine(d, t, t, b);
- }
- }
- return this.emit('scroll');
- };
- ScrollableBox.prototype._recalculateIndex = function() {
- var max, emax;
- if (this.detached || !this.scrollable) {
- return 0;
- }
- // XXX
- // max = this.getScrollHeight() - (this.height - this.iheight);
- max = this._clines.length - (this.height - this.iheight);
- if (max < 0) max = 0;
- emax = this._scrollBottom() - (this.height - this.iheight);
- if (emax < 0) emax = 0;
- this.childBase = Math.min(this.childBase, Math.max(emax, max));
- if (this.childBase < 0) {
- this.childBase = 0;
- } else if (this.childBase > this.baseLimit) {
- this.childBase = this.baseLimit;
- }
- };
- ScrollableBox.prototype.resetScroll = function() {
- if (!this.scrollable) return;
- this.childOffset = 0;
- this.childBase = 0;
- return this.emit('scroll');
- };
- ScrollableBox.prototype.getScrollHeight = function() {
- return Math.max(this._clines.length, this._scrollBottom());
- };
- ScrollableBox.prototype.getScrollPerc = function(s) {
- var pos = this.lpos || this._getCoords();
- if (!pos) return s ? -1 : 0;
- var height = (pos.yl - pos.yi) - this.iheight
- , i = this.getScrollHeight()
- , p;
- if (height < i) {
- if (this.alwaysScroll) {
- p = this.childBase / (i - height);
- } else {
- p = (this.childBase + this.childOffset) / (i - 1);
- }
- return p * 100;
- }
- return s ? -1 : 0;
- };
- ScrollableBox.prototype.setScrollPerc = function(i) {
- // XXX
- // var m = this.getScrollHeight();
- var m = Math.max(this._clines.length, this._scrollBottom());
- return this.scrollTo((i / 100) * m | 0);
- };
- /**
- * Expose
- */
- module.exports = ScrollableBox;
|