Initial commit
[yaffs-website] / node_modules / gauge / index.js
1 'use strict'
2 var Plumbing = require('./plumbing.js')
3 var hasUnicode = require('has-unicode')
4 var hasColor = require('./has-color.js')
5 var onExit = require('signal-exit')
6 var defaultThemes = require('./themes')
7 var setInterval = require('./set-interval.js')
8 var process = require('./process.js')
9 var setImmediate = require('./set-immediate')
10
11 module.exports = Gauge
12
13 function callWith (obj, method) {
14   return function () {
15     return method.call(obj)
16   }
17 }
18
19 function Gauge (arg1, arg2) {
20   var options, writeTo
21   if (arg1 && arg1.write) {
22     writeTo = arg1
23     options = arg2 || {}
24   } else if (arg2 && arg2.write) {
25     writeTo = arg2
26     options = arg1 || {}
27   } else {
28     writeTo = process.stderr
29     options = arg1 || arg2 || {}
30   }
31
32   this._status = {
33     spun: 0,
34     section: '',
35     subsection: ''
36   }
37   this._paused = false // are we paused for back pressure?
38   this._disabled = true // are all progress bar updates disabled?
39   this._showing = false // do we WANT the progress bar on screen
40   this._onScreen = false // IS the progress bar on screen
41   this._needsRedraw = false // should we print something at next tick?
42   this._hideCursor = options.hideCursor == null ? true : options.hideCursor
43   this._fixedFramerate = options.fixedFramerate == null
44     ? !(/^v0\.8\./.test(process.version))
45     : options.fixedFramerate
46   this._lastUpdateAt = null
47   this._updateInterval = options.updateInterval == null ? 50 : options.updateInterval
48
49   this._themes = options.themes || defaultThemes
50   this._theme = options.theme
51   var theme = this._computeTheme(options.theme)
52   var template = options.template || [
53     {type: 'progressbar', length: 20},
54     {type: 'activityIndicator', kerning: 1, length: 1},
55     {type: 'section', kerning: 1, default: ''},
56     {type: 'subsection', kerning: 1, default: ''}
57   ]
58   this.setWriteTo(writeTo, options.tty)
59   var PlumbingClass = options.Plumbing || Plumbing
60   this._gauge = new PlumbingClass(theme, template, this.getWidth())
61
62   this._$$doRedraw = callWith(this, this._doRedraw)
63   this._$$handleSizeChange = callWith(this, this._handleSizeChange)
64
65   this._cleanupOnExit = options.cleanupOnExit == null || options.cleanupOnExit
66   this._removeOnExit = null
67
68   if (options.enabled || (options.enabled == null && this._tty && this._tty.isTTY)) {
69     this.enable()
70   } else {
71     this.disable()
72   }
73 }
74 Gauge.prototype = {}
75
76 Gauge.prototype.isEnabled = function () {
77   return !this._disabled
78 }
79
80 Gauge.prototype.setTemplate = function (template) {
81   this._gauge.setTemplate(template)
82   if (this._showing) this._requestRedraw()
83 }
84
85 Gauge.prototype._computeTheme = function (theme) {
86   if (!theme) theme = {}
87   if (typeof theme === 'string') {
88     theme = this._themes.getTheme(theme)
89   } else if (theme && (Object.keys(theme).length === 0 || theme.hasUnicode != null || theme.hasColor != null)) {
90     var useUnicode = theme.hasUnicode == null ? hasUnicode() : theme.hasUnicode
91     var useColor = theme.hasColor == null ? hasColor : theme.hasColor
92     theme = this._themes.getDefault({hasUnicode: useUnicode, hasColor: useColor, platform: theme.platform})
93   }
94   return theme
95 }
96
97 Gauge.prototype.setThemeset = function (themes) {
98   this._themes = themes
99   this.setTheme(this._theme)
100 }
101
102 Gauge.prototype.setTheme = function (theme) {
103   this._gauge.setTheme(this._computeTheme(theme))
104   if (this._showing) this._requestRedraw()
105   this._theme = theme
106 }
107
108 Gauge.prototype._requestRedraw = function () {
109   this._needsRedraw = true
110   if (!this._fixedFramerate) this._doRedraw()
111 }
112
113 Gauge.prototype.getWidth = function () {
114   return ((this._tty && this._tty.columns) || 80) - 1
115 }
116
117 Gauge.prototype.setWriteTo = function (writeTo, tty) {
118   var enabled = !this._disabled
119   if (enabled) this.disable()
120   this._writeTo = writeTo
121   this._tty = tty ||
122     (writeTo === process.stderr && process.stdout.isTTY && process.stdout) ||
123     (writeTo.isTTY && writeTo) ||
124     this._tty
125   if (this._gauge) this._gauge.setWidth(this.getWidth())
126   if (enabled) this.enable()
127 }
128
129 Gauge.prototype.enable = function () {
130   if (!this._disabled) return
131   this._disabled = false
132   if (this._tty) this._enableEvents()
133   if (this._showing) this.show()
134 }
135
136 Gauge.prototype.disable = function () {
137   if (this._disabled) return
138   if (this._showing) {
139     this._lastUpdateAt = null
140     this._showing = false
141     this._doRedraw()
142     this._showing = true
143   }
144   this._disabled = true
145   if (this._tty) this._disableEvents()
146 }
147
148 Gauge.prototype._enableEvents = function () {
149   if (this._cleanupOnExit) {
150     this._removeOnExit = onExit(callWith(this, this.disable))
151   }
152   this._tty.on('resize', this._$$handleSizeChange)
153   if (this._fixedFramerate) {
154     this.redrawTracker = setInterval(this._$$doRedraw, this._updateInterval)
155     if (this.redrawTracker.unref) this.redrawTracker.unref()
156   }
157 }
158
159 Gauge.prototype._disableEvents = function () {
160   this._tty.removeListener('resize', this._$$handleSizeChange)
161   if (this._fixedFramerate) clearInterval(this.redrawTracker)
162   if (this._removeOnExit) this._removeOnExit()
163 }
164
165 Gauge.prototype.hide = function (cb) {
166   if (this._disabled) return cb && process.nextTick(cb)
167   if (!this._showing) return cb && process.nextTick(cb)
168   this._showing = false
169   this._doRedraw()
170   cb && setImmediate(cb)
171 }
172
173 Gauge.prototype.show = function (section, completed) {
174   this._showing = true
175   if (typeof section === 'string') {
176     this._status.section = section
177   } else if (typeof section === 'object') {
178     var sectionKeys = Object.keys(section)
179     for (var ii = 0; ii < sectionKeys.length; ++ii) {
180       var key = sectionKeys[ii]
181       this._status[key] = section[key]
182     }
183   }
184   if (completed != null) this._status.completed = completed
185   if (this._disabled) return
186   this._requestRedraw()
187 }
188
189 Gauge.prototype.pulse = function (subsection) {
190   this._status.subsection = subsection || ''
191   this._status.spun ++
192   if (this._disabled) return
193   if (!this._showing) return
194   this._requestRedraw()
195 }
196
197 Gauge.prototype._handleSizeChange = function () {
198   this._gauge.setWidth(this._tty.columns - 1)
199   this._requestRedraw()
200 }
201
202 Gauge.prototype._doRedraw = function () {
203   if (this._disabled || this._paused) return
204   if (!this._fixedFramerate) {
205     var now = Date.now()
206     if (this._lastUpdateAt && now - this._lastUpdateAt < this._updateInterval) return
207     this._lastUpdateAt = now
208   }
209   if (!this._showing && this._onScreen) {
210     this._onScreen = false
211     var result = this._gauge.hide()
212     if (this._hideCursor) {
213       result += this._gauge.showCursor()
214     }
215     return this._writeTo.write(result)
216   }
217   if (!this._showing && !this._onScreen) return
218   if (this._showing && !this._onScreen) {
219     this._onScreen = true
220     this._needsRedraw = true
221     if (this._hideCursor) {
222       this._writeTo.write(this._gauge.hideCursor())
223     }
224   }
225   if (!this._needsRedraw) return
226   if (!this._writeTo.write(this._gauge.show(this._status))) {
227     this._paused = true
228     this._writeTo.on('drain', callWith(this, function () {
229       this._paused = false
230       this._doRedraw()
231     }))
232   }
233 }