JasonWoof Got questions, comments, patches, etc.? Contact Jason Woofenden
init: encode terminal state as terminal output
[watch-my-terminal.git] / server.coffee
1 handler = (req, res) ->
2         reply_err = (req, res, msg) ->
3                 res.writeHead(200, 'Content-Type': 'text/plain')
4                 return res.end("Error loading #{req.url} \"#{msg}\"")
5         switch req.url
6                 when '/', '/index.html'
7                         filename = __dirname + '/index.html'
8                         type = 'text/html'
9                 when '/jquery.js'
10                         filename = '/usr/share/javascript/jquery/jquery.min.js'
11                         type = 'text/javascript'
12                 when '/terminal.js'
13                         filename = __dirname + '/terminal.coffee'
14                         type = 'text/javascript'
15                 when '/client.js'
16                         filename = __dirname + '/client.coffee'
17                         type = 'text/javascript'
18                 else
19                         return reply_err req, res
20         fs.readFile filename, 'utf8', (err, data) ->
21                 return reply_err req, res, err if err
22
23                 if filename.substr(filename.length - 7) is '.coffee'
24                         try
25                                 data = coffee.compile data
26                         catch e
27                                 return reply_err req, res, "server faild to compile #{filename}: #{JSON.stringify(e)}"
28
29                 res.writeHead(200, 'Content-Type': type)
30                 res.end(data)
31
32 coffee = require 'coffee-script'
33 app = require('http').createServer(handler)
34 io = require('socket.io').listen(app)
35 fs = require('fs')
36 terminal = require('./terminal.coffee')
37
38 # SETTINGS
39 app.listen(2218)
40 term = terminal.new(104, 66)
41
42 sockets = []
43
44 io.sockets.on 'connection', (socket) ->
45         sockets.push socket
46         socket.on 'disconnect', ->
47                 for i in [i...sockets.length]
48                         if sockets[i] is socket
49                                 sockets.splice i, 1
50                                 return
51
52
53         enc_color = (prefix, c) ->
54                 if c < 8
55                         return "#{prefix}#{c}"
56                 if c < 16
57                         return "#{prefix + 6}#{c - 8}"
58                 return "#{prefix}8;5;#{c}"
59
60         attr_diff = (a, b) ->
61                 xo = a ^ b
62                 parts = []
63                 if (xo & 0xff)
64                         parts.push enc_color 3, (b & 0xff)
65                 if (xo & 0xff00)
66                         parts.push enc_color 4, ((b & 0xff00) >> 8)
67                 for [bit, code] in [[0x010000, '1'], [0x200000, '3'], [0x020000, '4'], [0x040000, '5'], [0x080000, '7'], [0x100000, '8']]
68                         if (xo & bit)
69                                 if (b & bit)
70                                         parts.push code
71                                 else
72                                         parts.push '2' + code
73                 if parts.length
74                         return "\x1b[#{parts.join ';'}m"
75                 else
76                         return ''
77
78         a = 0x07
79         state = ''
80         # FIXME handle alt screen
81         for y in [0...term.height]
82                 for x in [0...term.width]
83                         if term.attributes[y][x] isnt a
84                                 state += attr_diff a, term.attributes[y][x]
85                                 a = term.attributes[y][x]
86                         state += term.text[y][x]
87                 if y < term.height - 1
88                         state += '\n'
89         state += attr_diff a, term.a
90         state += "\x1b[#{term.y + 1};#{term.x + 1}H"
91         unless term.cursor_visible
92                 state += "\x1b[?25l"
93         unless term.scroll_top is 0 and term.scroll_bottom is term.height - 1
94                 state += "\x1b[#{term.scroll_top};#{term.scroll_bottom}r"
95         socket.emit 'init', width: term.width, height: term.height, text: state
96
97 process.stdin.resume()
98 process.stdin.setEncoding 'utf8'
99
100 process.stdin.on 'data', (data) ->
101         term.update data
102         for s in sockets
103                 s.emit 'data', data
104
105 process.stdin.on 'end', -> process.exit()