1 # this file is used by the client and server.
3 # work around lack of module system in the browser:
8 my_exports = window.terminal
12 constructor: (width, height) ->
19 @a = 0x000007 # cursor attributes
21 @saved_normal_screen = null
24 resize: (width, height) ->
25 # FIXME: write a version that retains some of the data
37 # pass data from stdout
39 return unless data?.length > 0
40 if @partial.length > 0
41 data = @partial + data
43 parts = data.split(/\x1b\[/)
45 if -1 is @escape_sequence_length parts[parts.length - 1]
46 @partial = parts.pop()
48 for i in [0...parts.length]
52 @update_sequence_then_text parts[i]
60 # move (newly cleared) top line to the bottom
63 tmp = @attributes.shift()
65 # slide cursor up with rest of text
74 # str has no escape sequences
76 return unless str.length > 0
80 @update_text " ".substr(@x % 8)
85 when '\x08' # backspace
92 when '\x0a', '\x0b' # lf, vertical tab (same thing)
98 @attributes[@y][@x] = @a
102 set_attribute_bits: (mask, value) ->
103 @a = ((@a & ~mask) | value)
105 # we're supposed to ignore leeding zeros, and while we're at it, lets swap
106 # in the default for blank or missing values
107 fix_esc_arg: (value, deef_alt) ->
108 if value? and value != ''
109 while value[0] is '0' and value.length > 1
110 value = value.substr 1
115 # set cursor position (one based)
116 csi_H: (row, column) ->
117 # handle blank/missing args and convert to 0 base
118 row = @fix_esc_arg(row, 1) - 1
119 column = @fix_esc_arg(column, 1) - 1
124 else if column >= @width
135 # clear to screen edge(es)
136 csi_J: (direction) ->
137 switch @fix_esc_arg direction, '0'
138 when '0' # erase down
139 # rest of current line
142 for row in [@y...@height]
143 for i in [0...@width]
145 @attributes[row][i] = @a
147 # beginning of current line
151 for i in [0...@width]
153 @attributes[row][i] = @a
154 when '2' # erase everything
155 for row in [0...@height]
156 for i in [0...@width]
158 @attributes[row][i] = @a
160 console.log "confusing arg for csi_J: #{direction}"
163 # clear (some or all of) current line
164 csi_K: (direction) ->
165 switch @fix_esc_arg direction, '0'
166 when '0' # erase to right
167 for i in [@x...@width]
169 @attributes[@y][i] = @a
170 when '1' # erase to left
171 # @x can equal @width (after printing to right-most column)
178 @attributes[@y][i] = @a
179 when '2' # erase whole line
180 for i in [0...@width]
182 @attributes[@y][i] = @a
184 console.log "confusing arg for csi_K: #{direction}"
191 switch @fix_esc_arg i, ''
193 if @saved_normal_screen?
194 console.log "ignoring request to switch to the alt screen because we're already on the alt screen"
196 @saved_normal_screen = [@x, @y, @text, @attributes]
199 for y in [0...@height]
202 for x in [0...@width]
204 @attributes[y].push 0
209 switch @fix_esc_arg i, ''
211 if not @saved_normal_screen?
212 console.log "ignoring request to switch to the normal screen because we're already on the normal screen"
214 @x = @saved_normal_screen[0]
215 @y = @saved_normal_screen[1]
216 @text = @saved_normal_screen[2]
217 @attributes = @saved_normal_screen[3]
218 @saved_normal_screen = null
220 # set color, bold, underline, etc
224 args.push @fix_esc_arg i, '0'
226 while args.length > 0
229 # remove all style/color
235 @set_attribute_bits 0x010000, 0x010000
236 when '3' # italic (rare)
237 @set_attribute_bits 0x200000, 0x200000
239 @set_attribute_bits 0x020000, 0x020000
241 @set_attribute_bits 0x040000, 0x040000
243 @set_attribute_bits 0x080000, 0x080000
245 @set_attribute_bits 0x100000, 0x100000
247 # disable style attributes
248 when '21' # not bold (rare)
249 @set_attribute_bits 0x010000, 0
251 @set_attribute_bits 0x010000, 0
252 when '23' # not italic (rare)
253 @set_attribute_bits 0x200000, 0
254 when '24' # not underline
255 @set_attribute_bits 0x020000, 0
256 when '25' # not blink
257 @set_attribute_bits 0x040000, 0
258 when '27' # not inverse
259 @set_attribute_bits 0x080000, 0
260 when '28' # not invisible
261 @set_attribute_bits 0x100000, 0
263 when '100' # reset colors but not other attributes
264 @set_attribute_bits 0xffff, 0x0007
268 @set_attribute_bits 0xff, 0x00
270 @set_attribute_bits 0xff, 0x01
272 @set_attribute_bits 0xff, 0x02
273 when '33' # fg yellow
274 @set_attribute_bits 0xff, 0x03
276 @set_attribute_bits 0xff, 0x04
277 when '35' # fg magenta
278 @set_attribute_bits 0xff, 0x05
280 @set_attribute_bits 0xff, 0x06
281 when '37', '39' # fg white (39 is default)
282 @set_attribute_bits 0xff, 0x07
285 if args.length >= 2 and args[0] is '5'
287 @set_attribute_bits 0xff, (0xff & args.shift())
289 @set_attribute_bits 0x20000, 0x20000
293 @set_attribute_bits 0xff00, 0x0000
295 @set_attribute_bits 0xff00, 0x0100
297 @set_attribute_bits 0xff00, 0x0200
298 when '43' # bg yellow
299 @set_attribute_bits 0xff00, 0x0300
301 @set_attribute_bits 0xff00, 0x0400
302 when '45' # bg magenta
303 @set_attribute_bits 0xff00, 0x0500
305 @set_attribute_bits 0xff00, 0x0600
307 @set_attribute_bits 0xff00, 0x0700
308 when '49' # bg default
309 @set_attribute_bits 0xff00, 0x0000
312 if args.length >= 2 and args[0] is '5'
314 @set_attribute_bits 0xff00, ((0xff & args.shift()) << 8)
316 @set_attribute_bits 0x20000, 0x20000
319 when '90' # fg bright black
320 @set_attribute_bits 0xff, 0x08
321 when '91' # fg bright red
322 @set_attribute_bits 0xff, 0x09
323 when '92' # fg bright green
324 @set_attribute_bits 0xff, 0x0a
325 when '93' # fg bright yellow
326 @set_attribute_bits 0xff, 0x0b
327 when '94' # fg bright blue
328 @set_attribute_bits 0xff, 0x0c
329 when '95' # fg bright magenta
330 @set_attribute_bits 0xff, 0x0d
331 when '96' # fg bright cyan
332 @set_attribute_bits 0xff, 0x0e
333 when '97' # fg bright white
334 @set_attribute_bits 0xff, 0x0f
337 when '100' # bg bright black
338 @set_attribute_bits 0xff, 0x08
339 when '101' # bg bright red
340 @set_attribute_bits 0xff, 0x09
341 when '102' # bg bright green
342 @set_attribute_bits 0xff, 0x0a
343 when '103' # bg bright yellow
344 @set_attribute_bits 0xff, 0x0b
345 when '104' # bg bright blue
346 @set_attribute_bits 0xff, 0x0c
347 when '105' # bg bright magenta
348 @set_attribute_bits 0xff, 0x0d
349 when '106' # bg bright cyan
350 @set_attribute_bits 0xff, 0x0e
351 when '107' # bg bright white
352 @set_attribute_bits 0xff, 0x0f
355 # if we don't recognize the style, go back to default
356 console.log "unrecognized csi_m arg: \"#{arg}\""
360 # str is the whole escape sequence (minus the esc[ prefix)
361 update_sequence: (str) ->
366 command = @[prefix + str.substr(str.length - 1)]
368 console.log "Unrecognized sequence: ESC[#{str}"
370 args = str.substr(0, str.length - 1).split ';'
371 command.call this, args...
373 update_sequence_then_text: (str) ->
374 len = @escape_sequence_length str
376 console.log "couldn't find escape sequence here: #{str.substr 0, 25}"
377 @update_text "ESC[" + str
379 @update_sequence str.substr 0, len
380 @update_text str.substr len
382 escape_sequence_length: (str) ->
383 parts = str.match(/^[0-9;?]{0,25}./)
384 return -1 unless parts?
385 return parts[0].length
387 my_exports.new = (width, height) ->
388 return new Terminal width, height