Commit e877a03e authored by Joel Martin's avatar Joel Martin

Have RFB and Xpra co-exist in same tree.

vnc.html for RFB, xpra.html for loading Xpra. This could perhaps be
combined into one html file with a UI selector..
parent 07927521
...@@ -21,15 +21,16 @@ var that = {}, // Public API methods ...@@ -21,15 +21,16 @@ var that = {}, // Public API methods
// Pre-declare private functions used before definitions (jslint) // Pre-declare private functions used before definitions (jslint)
init_vars, updateState, fail, handle_message, init_vars, updateState, fail, handle_message,
framebufferUpdate, print_stats, init_msg, normal_msg, framebufferUpdate, print_stats,
pixelFormat, clientEncodings, fbUpdateRequest, fbUpdateRequests,
keyEvent, pointerEvent, clientCutText,
getTightCLength, extract_data_uri, getTightCLength, extract_data_uri,
keyPress, mouseButton, mouseMove, keyPress, mouseButton, mouseMove,
checkEvents, // Overridable for testing checkEvents, // Overridable for testing
bencode, bdecode, encode_packet, decode_packet,
// //
// Private RFB namespace variables // Private RFB namespace variables
...@@ -45,7 +46,25 @@ var that = {}, // Public API methods ...@@ -45,7 +46,25 @@ var that = {}, // Public API methods
rfb_auth_scheme= '', rfb_auth_scheme= '',
packetHandlers = {}, // In preference order
encodings = [
['COPYRECT', 0x01 ],
['TIGHT', 0x07 ],
['TIGHT_PNG', -260 ],
['HEXTILE', 0x05 ],
['RRE', 0x02 ],
['RAW', 0x00 ],
['DesktopSize', -223 ],
['Cursor', -239 ],
// Psuedo-encoding settings
//['JPEG_quality_lo', -32 ],
['JPEG_quality_med', -26 ],
//['JPEG_quality_hi', -23 ],
//['compress_lo', -255 ],
['compress_hi', -247 ],
['last_rect', -224 ]
],
encHandlers = {}, encHandlers = {},
encNames = {}, encNames = {},
...@@ -60,22 +79,21 @@ var that = {}, // Public API methods ...@@ -60,22 +79,21 @@ var that = {}, // Public API methods
disconnTimer = null, // disconnection timer disconnTimer = null, // disconnection timer
msgTimer = null, // queued handle_message timer msgTimer = null, // queued handle_message timer
zlib = null, // zlib encoder/decoder // Frame buffer update state
FBU = {
windows = {}, // managed windows rects : 0,
subrects : 0, // RRE
send_packets = [], // pending packets to send (mouse movements) lines : 0, // RAW
tiles : 0, // HEXTILE
raw_packets = {}, // partially received raw packets bytes : 0,
x : 0,
cur_packet_recv_time = 0, // record when we received the packet we are currently handling y : 0,
width : 0,
keycodes = {}, height : 0,
encoding : 0,
stats = { subencoding : -1,
last_ping_echoed_time: 0, background : null,
server_ping_latency: [], zlibs : [] // TIGHT zlib streams
client_ping_latency: []
}, },
fb_Bpp = 4, fb_Bpp = 4,
...@@ -130,7 +148,7 @@ Util.conf_defaults(conf, that, defaults, [ ...@@ -130,7 +148,7 @@ Util.conf_defaults(conf, that, defaults, [
['viewportDrag', 'rw', 'bool', false, 'Move the viewport on mouse drags'], ['viewportDrag', 'rw', 'bool', false, 'Move the viewport on mouse drags'],
['check_rate', 'rw', 'int', 217, 'Timing (ms) of send/receive check'], ['check_rate', 'rw', 'int', 217, 'Timing (ms) of send/receive check'],
['ping_rate', 'rw', 'int', 10000, 'Timing (ms) of ping sending'], ['fbu_req_rate', 'rw', 'int', 1413, 'Timing (ms) of frameBufferUpdate requests'],
// Callback functions // Callback functions
['onUpdateState', 'rw', 'func', function() { }, ['onUpdateState', 'rw', 'func', function() { },
...@@ -177,6 +195,7 @@ that.get_keyboard = function() { return keyboard; }; ...@@ -177,6 +195,7 @@ that.get_keyboard = function() { return keyboard; };
that.get_mouse = function() { return mouse; }; that.get_mouse = function() { return mouse; };
// //
// Setup routines // Setup routines
// //
...@@ -187,6 +206,12 @@ function constructor() { ...@@ -187,6 +206,12 @@ function constructor() {
var i, rmode; var i, rmode;
Util.Debug(">> RFB.constructor"); Util.Debug(">> RFB.constructor");
// Create lookup tables based encoding number
for (i=0; i < encodings.length; i+=1) {
encHandlers[encodings[i][1]] = encHandlers[encodings[i][0]];
encNames[encodings[i][1]] = encodings[i][0];
encStats[encodings[i][1]] = [0, 0];
}
// Initialize display, mouse, keyboard, and websock // Initialize display, mouse, keyboard, and websock
try { try {
display = new Display({'target': conf.target}); display = new Display({'target': conf.target});
...@@ -205,10 +230,8 @@ function constructor() { ...@@ -205,10 +230,8 @@ function constructor() {
ws = new Websock(); ws = new Websock();
ws.on('message', handle_message); ws.on('message', handle_message);
ws.on('open', function() { ws.on('open', function() {
Util.Info("connected")
if (rfb_state === "connect") { if (rfb_state === "connect") {
ws.send(helloPacket()); updateState('ProtocolVersion', "Starting VNC handshake");
updateState('hello', "Starting Xpra handshake");
} else { } else {
fail("Got unexpected WebSockets connection"); fail("Got unexpected WebSockets connection");
} }
...@@ -291,43 +314,45 @@ init_vars = function() { ...@@ -291,43 +314,45 @@ init_vars = function() {
/* Reset state */ /* Reset state */
ws.init(); ws.init();
FBU.rects = 0;
FBU.subrects = 0; // RRE and HEXTILE
FBU.lines = 0; // RAW
FBU.tiles = 0; // HEXTILE
FBU.zlibs = []; // TIGHT zlib encoders
mouse_buttonMask = 0; mouse_buttonMask = 0;
mouse_arr = []; mouse_arr = [];
zlib = new TINF(); // Clear the per connection encoding stats
zlib.init(); for (i=0; i < encodings.length; i+=1) {
encStats[encodings[i][1]][0] = 0;
var keynames = { }
32:'space', 33:'exclam', 35:'numbersign', 36:'dollar',
37:'percent', 38:'ampersand', 40:'parenleft', 41:'parenright', for (i=0; i < 4; i++) {
42:'asterisk', 43:'plus', 45:'minus', 61:'equal', //FBU.zlibs[i] = new InflateStream();
94:'asciicircum', 95:'underscore', FBU.zlibs[i] = new TINF();
96:'grave', 126:'asciitilde' }; FBU.zlibs[i].init();
// Symbols are wrong
for (i=32; i < 127; i++) {
var keyname = keynames[i] ? keynames[i] : String.fromCharCode(i);
keycodes[i] = [i, keyname, i, 0, 0];
} }
// Some whacky ones
keycodes[64] = [34, 'at', 34, 0, 0];
keycodes[126] = [126, 'asciitilde', 49, 0, 0];
// Special codes
keycodes[65505] = [65505, 'Shift_L', 65505, 0, 0];
keycodes[65506] = [65506, 'Shift_R', 65506, 0, 0];
keycodes[65507] = [65507, 'Control_L', 65507, 0, 0];
keycodes[65508] = [65508, 'Control_R', 65508, 0, 0];
keycodes[65513] = [65513, 'Alt_L', 65513, 0, 0];
keycodes[65514] = [65514, 'Alt_R', 65514, 0, 0];
keycodes[65288] = [65288, 'BackSpace', 65288, 0, 0];
keycodes[65289] = [65289, 'Tab', 65289, 0, 0];
keycodes[65293] = [65293, 'Return', 65293, 0, 0];
}; };
// Print statistics // Print statistics
print_stats = function() { print_stats = function() {
var i, s; var i, s;
Util.Info("Encoding stats for this connection:"); Util.Info("Encoding stats for this connection:");
for (i=0; i < encodings.length; i+=1) {
s = encStats[encodings[i][1]];
if ((s[0] + s[1]) > 0) {
Util.Info(" " + encodings[i][0] + ": " +
s[0] + " rects");
}
}
Util.Info("Encoding stats since page load:");
for (i=0; i < encodings.length; i+=1) {
s = encStats[encodings[i][1]];
if ((s[0] + s[1]) > 0) {
Util.Info(" " + encodings[i][0] + ": " +
s[1] + " rects");
}
}
}; };
// //
...@@ -345,8 +370,14 @@ print_stats = function() { ...@@ -345,8 +370,14 @@ print_stats = function() {
* failed - abnormal disconnect * failed - abnormal disconnect
* fatal - failed to load page, or fatal error * fatal - failed to load page, or fatal error
* *
* Xpra protocol initialization states: * RFB protocol initialization states:
* hello * ProtocolVersion
* Security
* Authentication
* password - waiting for password, not part of RFB
* SecurityResult
* ClientInitialization - not triggered by server message
* ServerInitialization (to normal)
*/ */
updateState = function(state, statusMsg) { updateState = function(state, statusMsg) {
var func, cmsg, oldstate = rfb_state; var func, cmsg, oldstate = rfb_state;
...@@ -496,50 +527,54 @@ fail = function(msg) { ...@@ -496,50 +527,54 @@ fail = function(msg) {
handle_message = function() { handle_message = function() {
//Util.Debug(">> handle_message ws.rQlen(): " + ws.rQlen()); //Util.Debug(">> handle_message ws.rQlen(): " + ws.rQlen());
//Util.Debug("ws.rQslice(0,20): " + ws.rQslice(0,20) + " (" + ws.rQlen() + ")"); //Util.Debug("ws.rQslice(0,20): " + ws.rQslice(0,20) + " (" + ws.rQlen() + ")");
if (cur_packet_recv_time === 0) { if (ws.rQlen() === 0) {
cur_packet_recv_time = (new Date()).getTime(); Util.Warn("handle_message called on empty receive queue");
} return;
var packet = decode_packet(ws.get_rQ(), ws.get_rQi());
if (packet.length > 0) {
// Remove the processed data from the queue
ws.set_rQi(ws.get_rQi() + packet.length);
}
if (packet.data) {
// full packet has been received, so process it
var ptype = packet.data[0].replace(/-/, "_"),
handler = packetHandlers[ptype];
if (handler) {
handler(packet.data);
} else {
Util.Warn("no handler defined for packet type '" + ptype + "'");
}
// we got a whole packet so reset the packet received timestamp
cur_packet_recv_time = 0;
} }
switch (rfb_state) {
if (ws.rQlen() >= 8) { case 'disconnected':
// if we have at least 8 bytes remaining, then requeue case 'failed':
// ourselves, but use setTimeout to give other events a chance Util.Error("Got data while disconnected");
// to run break;
if (msgTimer === null) { case 'normal':
Util.Debug("More data to process, creating timer"); if (normal_msg() && ws.rQlen() > 0) {
msgTimer = setTimeout(function () { // true means we can continue processing
msgTimer = null; // Give other events a chance to run
handle_message(); if (msgTimer === null) {
}, 1); Util.Debug("More data to process, creating timer");
} else { msgTimer = setTimeout(function () {
Util.Debug("More data to process, existing timer"); msgTimer = null;
handle_message();
}, 10);
} else {
Util.Debug("More data to process, existing timer");
}
} }
break;
default:
init_msg();
break;
} }
}; };
function genDES(password, challenge) {
var i, passwd = [];
for (i=0; i < password.length; i += 1) {
passwd.push(password.charCodeAt(i));
}
return (new DES(passwd)).encrypt(challenge);
}
function flushClient() { function flushClient() {
if (send_packets.length > 0) { if (mouse_arr.length > 0) {
for (var i=0; i < send_packets.length; i++) { //send(mouse_arr.concat(fbUpdateRequests()));
ws.send(send_packets[i]); ws.send(mouse_arr);
} setTimeout(function() {
send_packets = []; ws.send(fbUpdateRequests());
}, 50);
mouse_arr = [];
return true; return true;
} else { } else {
return false; return false;
...@@ -550,22 +585,25 @@ function flushClient() { ...@@ -550,22 +585,25 @@ function flushClient() {
checkEvents = function() { checkEvents = function() {
var now; var now;
if (rfb_state === 'normal' && !viewportDragging) { if (rfb_state === 'normal' && !viewportDragging) {
flushClient(); if (! flushClient()) {
now = new Date().getTime();
if (now > last_req_time + conf.fbu_req_rate) {
last_req_time = now;
ws.send(fbUpdateRequests());
}
}
} }
setTimeout(checkEvents, conf.check_rate); setTimeout(checkEvents, conf.check_rate);
}; };
keyPress = function(keysym, down, evt) { keyPress = function(keysym, down) {
var arr, packet; var arr;
if (conf.view_only) { return; } // View only, skip keyboard events if (conf.view_only) { return; } // View only, skip keyboard events
packet = keyActionPacket(1, keysym, down, evt) arr = keyEvent(keysym, down);
arr = arr.concat(fbUpdateRequests());
if (!packet) { return false; } ws.send(arr);
send_packets.push(packet);
flushClient();
}; };
mouseButton = function(x, y, down, bmask) { mouseButton = function(x, y, down, bmask) {
...@@ -590,14 +628,13 @@ mouseButton = function(x, y, down, bmask) { ...@@ -590,14 +628,13 @@ mouseButton = function(x, y, down, bmask) {
if (conf.view_only) { return; } // View only, skip mouse events if (conf.view_only) { return; } // View only, skip mouse events
var button = 1; // TODO: translate bmask to button # mouse_arr = mouse_arr.concat(
send_packets.push(pointerActionPacket( pointerEvent(display.absX(x), display.absY(y)) );
1, button, down, display.absX(x), display.absY(y)));
flushClient(); flushClient();
}; };
mouseMove = function(x, y) { mouseMove = function(x, y) {
Util.Debug('>> mouseMove ' + x + "," + y); //Util.Debug('>> mouseMove ' + x + "," + y);
var deltaX, deltaY; var deltaX, deltaY;
if (viewportDragging) { if (viewportDragging) {
...@@ -615,408 +652,1130 @@ mouseMove = function(x, y) { ...@@ -615,408 +652,1130 @@ mouseMove = function(x, y) {
if (conf.view_only) { return; } // View only, skip mouse events if (conf.view_only) { return; } // View only, skip mouse events
send_packets.push(pointerPositionPacket( mouse_arr = mouse_arr.concat(
1, display.absX(x), display.absY(y))); pointerEvent(display.absX(x), display.absY(y)) );
}; };
// //
// Xpra protocol routines // Server message handlers
// //
function bencode(data) { // RFB/VNC initialisation message handler
switch (typeof(data)) { init_msg = function() {
case "number": //Util.Debug(">> init_msg [rfb_state '" + rfb_state + "']");
return "i" + data + "e";
break; var strlen, reason, length, sversion, cversion, repeaterID,
case "string": i, types, num_types, challenge, response, bpp, depth,
return data.length + ":" + data; big_endian, red_max, green_max, blue_max, red_shift,
green_shift, blue_shift, true_color, name_length, is_repeater;
//Util.Debug("ws.rQ (" + ws.rQlen() + ") " + ws.rQslice(0));
switch (rfb_state) {
case 'ProtocolVersion' :
if (ws.rQlen() < 12) {
return fail("Incomplete protocol version");
}
sversion = ws.rQshiftStr(12).substr(4,7);
Util.Info("Server ProtocolVersion: " + sversion);
is_repeater = 0;
switch (sversion) {
case "000.000": is_repeater = 1; break; // UltraVNC repeater
case "003.003": rfb_version = 3.3; break;
case "003.006": rfb_version = 3.3; break; // UltraVNC
case "003.889": rfb_version = 3.3; break; // Apple Remote Desktop
case "003.007": rfb_version = 3.7; break;
case "003.008": rfb_version = 3.8; break;
case "004.000": rfb_version = 3.8; break; // Intel AMT KVM
case "004.001": rfb_version = 3.8; break; // RealVNC 4.6
default:
return fail("Invalid server version " + sversion);
}
if (is_repeater) {
repeaterID = conf.repeaterID;
while (repeaterID.length < 250) {
repeaterID += "\0";
}
ws.send_string(repeaterID);
break;
}
if (rfb_version > rfb_max_version) {
rfb_version = rfb_max_version;
}
if (! test_mode) {
sendTimer = setInterval(function() {
// Send updates either at a rate of one update
// every 50ms, or whatever slower rate the network
// can handle.
ws.flush();
}, 50);
}
cversion = "00" + parseInt(rfb_version,10) +
".00" + ((rfb_version * 10) % 10);
ws.send_string("RFB " + cversion + "\n");
updateState('Security', "Sent ProtocolVersion: " + cversion);
break; break;
case "object":
if (Array.isArray(data)) { case 'Security' :
var res = "l"; if (rfb_version >= 3.7) {
for (var i = 0; i < data.length; i++) { // Server sends supported list, client decides
res += bencode(data[i]); num_types = ws.rQshift8();
if (ws.rQwait("security type", num_types, 1)) { return false; }
if (num_types === 0) {
strlen = ws.rQshift32();
reason = ws.rQshiftStr(strlen);
return fail("Security failure: " + reason);
} }
return res + "e"; rfb_auth_scheme = 0;
} else { types = ws.rQshiftBytes(num_types);
var res = "d", Util.Debug("Server security types: " + types);
keys = Object.keys(data).sort(); for (i=0; i < types.length; i+=1) {
for (var i = 0; i < keys.length; i++) { if ((types[i] > rfb_auth_scheme) && (types[i] < 3)) {
var k = keys[i], rfb_auth_scheme = types[i];
v = data[k]; }
res += bencode(k); }
res += bencode(v); if (rfb_auth_scheme === 0) {
return fail("Unsupported security types: " + types);
} }
return res + "e";
ws.send([rfb_auth_scheme]);
} else {
// Server decides
if (ws.rQwait("security scheme", 4)) { return false; }
rfb_auth_scheme = ws.rQshift32();
} }
updateState('Authentication',
"Authenticating using scheme: " + rfb_auth_scheme);
init_msg(); // Recursive fallthrough (workaround JSLint complaint)
break; break;
default:
throw("Unknown encode_data type: " + typeof(data));
}
}
function bdecode(raw, f) { // Triggered by fallthough, not by server message
var ret, res; case 'Authentication' :
f = (typeof(f) === "undefined") ? 0 : f; //Util.Debug("Security auth scheme: " + rfb_auth_scheme);
switch (raw[f]) { switch (rfb_auth_scheme) {
case 'i': case 0: // connection failed
var end = raw.indexOf('e', f+1); if (ws.rQwait("auth reason", 4)) { return false; }
if (end < 0) { strlen = ws.rQshift32();
throw("Error decoding integer value"); reason = ws.rQshiftStr(strlen);
return fail("Auth failure: " + reason);
case 1: // no authentication
if (rfb_version >= 3.8) {
updateState('SecurityResult');
return;
}
// Fall through to ClientInitialisation
break;
case 2: // VNC authentication
if (rfb_password.length === 0) {
// Notify via both callbacks since it is kind of
// a RFB state change and a UI interface issue.
updateState('password', "Password Required");
conf.onPasswordRequired(that);
return;
}
if (ws.rQwait("auth challenge", 16)) { return false; }
challenge = ws.rQshiftBytes(16);
//Util.Debug("Password: " + rfb_password);
//Util.Debug("Challenge: " + challenge +
// " (" + challenge.length + ")");
response = genDES(rfb_password, challenge);
//Util.Debug("Response: " + response +
// " (" + response.length + ")");
//Util.Debug("Sending DES encrypted auth response");
ws.send(response);
updateState('SecurityResult');
return;
default:
fail("Unsupported auth scheme: " + rfb_auth_scheme);
return;
} }
var num = parseInt(raw.substring(f+1, end), 10); updateState('ClientInitialisation', "No auth required");
res = num; init_msg(); // Recursive fallthrough (workaround JSLint complaint)
f = end;
break; break;
case 'l':
res = []; case 'SecurityResult' :
f += 1; if (ws.rQwait("VNC auth response ", 4)) { return false; }
while (raw[f] !== 'e') { switch (ws.rQshift32()) {
ret = bdecode(raw, f); case 0: // OK
res.push(ret[0]); // Fall through to ClientInitialisation
f = ret[1]; break;
case 1: // failed
if (rfb_version >= 3.8) {
length = ws.rQshift32();
if (ws.rQwait("SecurityResult reason", length, 8)) {
return false;
}
reason = ws.rQshiftStr(length);
fail(reason);
} else {
fail("Authentication failed");
}
return;
case 2: // too-many
return fail("Too many auth attempts");
} }
updateState('ClientInitialisation', "Authentication OK");
init_msg(); // Recursive fallthrough (workaround JSLint complaint)
break; break;
case 'd':
var k, lastkey = null; // Triggered by fallthough, not by server message
res = {}; case 'ClientInitialisation' :
f += 1; ws.send([conf.shared ? 1 : 0]); // ClientInitialisation
while (raw[f] !== 'e') { updateState('ServerInitialisation', "Authentication OK");
ret = bdecode(raw, f); break;
k = ret[0];
f = ret[1]; case 'ServerInitialisation' :
if (lastkey !== null && lastkey >= k) { if (ws.rQwait("server initialization", 24)) { return false; }
throw("Unsorted keys found while decoding dict type");
} /* Screen size */
lastkey = k; fb_width = ws.rQshift16();
ret = bdecode(raw, f); fb_height = ws.rQshift16();
res[k] = ret[0];
f = ret[1]; /* PIXEL_FORMAT */
bpp = ws.rQshift8();
depth = ws.rQshift8();
big_endian = ws.rQshift8();
true_color = ws.rQshift8();
red_max = ws.rQshift16();
green_max = ws.rQshift16();
blue_max = ws.rQshift16();
red_shift = ws.rQshift8();
green_shift = ws.rQshift8();
blue_shift = ws.rQshift8();
ws.rQshiftStr(3); // padding
Util.Info("Screen: " + fb_width + "x" + fb_height +
", bpp: " + bpp + ", depth: " + depth +
", big_endian: " + big_endian +
", true_color: " + true_color +
", red_max: " + red_max +
", green_max: " + green_max +
", blue_max: " + blue_max +
", red_shift: " + red_shift +
", green_shift: " + green_shift +
", blue_shift: " + blue_shift);
if (big_endian !== 0) {
Util.Warn("Server native endian is not little endian");
}
if (red_shift !== 16) {
Util.Warn("Server native red-shift is not 16");
}
if (blue_shift !== 0) {
Util.Warn("Server native blue-shift is not 0");
} }
/* Connection name/title */
name_length = ws.rQshift32();
fb_name = ws.rQshiftStr(name_length);
if (conf.true_color && fb_name === "Intel(r) AMT KVM")
{
Util.Warn("Intel AMT KVM only support 8/16 bit depths. Disabling true color");
conf.true_color = false;
}
display.set_true_color(conf.true_color);
conf.onFBResize(that, fb_width, fb_height);
display.resize(fb_width, fb_height);
keyboard.grab();
mouse.grab();
if (conf.true_color) {
fb_Bpp = 4;
fb_depth = 3;
} else {
fb_Bpp = 1;
fb_depth = 1;
}
response = pixelFormat();
response = response.concat(clientEncodings());
response = response.concat(fbUpdateRequests());
timing.fbu_rt_start = (new Date()).getTime();
timing.pixels = 0;
ws.send(response);
/* Start pushing/polling */
setTimeout(checkEvents, conf.check_rate);
if (conf.encrypt) {
updateState('normal', "Connected (encrypted) to: " + fb_name);
} else {
updateState('normal', "Connected (unencrypted) to: " + fb_name);
}
break;
}
//Util.Debug("<< init_msg");
};
/* Normal RFB/VNC server message handler */
normal_msg = function() {
//Util.Debug(">> normal_msg");
var ret = true, msg_type, length, text,
c, first_colour, num_colours, red, green, blue;
if (FBU.rects > 0) {
msg_type = 0;
} else {
msg_type = ws.rQshift8();
}
switch (msg_type) {
case 0: // FramebufferUpdate
ret = framebufferUpdate(); // false means need more data
break;
case 1: // SetColourMapEntries
Util.Debug("SetColourMapEntries");
ws.rQshift8(); // Padding
first_colour = ws.rQshift16(); // First colour
num_colours = ws.rQshift16();
if (ws.rQwait("SetColourMapEntries", num_colours*6, 6)) { return false; }
for (c=0; c < num_colours; c+=1) {
red = ws.rQshift16();
//Util.Debug("red before: " + red);
red = parseInt(red / 256, 10);
//Util.Debug("red after: " + red);
green = parseInt(ws.rQshift16() / 256, 10);
blue = parseInt(ws.rQshift16() / 256, 10);
display.set_colourMap([blue, green, red], first_colour + c);
}
Util.Debug("colourMap: " + display.get_colourMap());
Util.Info("Registered " + num_colours + " colourMap entries");
//Util.Debug("colourMap: " + display.get_colourMap());
break;
case 2: // Bell
Util.Debug("Bell");
conf.onBell(that);
break;
case 3: // ServerCutText
Util.Debug("ServerCutText");
if (ws.rQwait("ServerCutText header", 7, 1)) { return false; }
ws.rQshiftBytes(3); // Padding
length = ws.rQshift32();
if (ws.rQwait("ServerCutText", length, 8)) { return false; }
text = ws.rQshiftStr(length);
conf.clipboardReceive(that, text); // Obsolete
conf.onClipboard(that, text);
break; break;
default: default:
if (!raw[f].match(/^[0-9]/)) { fail("Disconnected: illegal server message type " + msg_type);
throw("Unknown decoding type: " + raw[f]); Util.Debug("ws.rQslice(0,30):" + ws.rQslice(0,30));
break;
}
//Util.Debug("<< normal_msg");
return ret;
};
framebufferUpdate = function() {
var now, hdr, fbu_rt_diff, ret = true;
if (FBU.rects === 0) {
//Util.Debug("New FBU: ws.rQslice(0,20): " + ws.rQslice(0,20));
if (ws.rQwait("FBU header", 3)) {
ws.rQunshift8(0); // FBU msg_type
return false;
} }
var end = raw.indexOf(':', f+1); ws.rQshift8(); // padding
if (end < 0) { FBU.rects = ws.rQshift16();
throw("Error decoding string value"); //Util.Debug("FramebufferUpdate, rects:" + FBU.rects);
FBU.bytes = 0;
timing.cur_fbu = 0;
if (timing.fbu_rt_start > 0) {
now = (new Date()).getTime();
Util.Info("First FBU latency: " + (now - timing.fbu_rt_start));
} }
var len = parseInt(raw.substring(f, end), 10);
res = raw.substr(end+1, len);
f = end+len;
} }
return [res, f+1];
}
/* while (FBU.rects > 0) {
function encode_packet(data) { if (rfb_state !== "normal") {
var encoded_data = bencode(data), return false;
payload_size = encoded_data.length, }
packet_size = 8 + payload_size + (4-payload_size%4), if (ws.rQwait("FBU", FBU.bytes)) { return false; }
packet8 = new Uint8Array(packet_size), if (FBU.bytes === 0) {
payload8 = new Uint8Array(packet8.buffer, 8), if (ws.rQwait("rect header", 12)) { return false; }
packet32 = new Uint32Array(packet8.buffer, 0); /* New FramebufferUpdate */
packet8[0] = 80; // 'P'.charCodeAt(0) hdr = ws.rQshiftBytes(12);
packet8[1] = 0; // protocol flags FBU.x = (hdr[0] << 8) + hdr[1];
packet8[2] = 0; // compression level FBU.y = (hdr[2] << 8) + hdr[3];
packet8[3] = 0; // packet index FBU.width = (hdr[4] << 8) + hdr[5];
packet32[1] = payload_size; // packet payload size FBU.height = (hdr[6] << 8) + hdr[7];
FBU.encoding = parseInt((hdr[8] << 24) + (hdr[9] << 16) +
for (var i=0; i < payload_size; i++) { (hdr[10] << 8) + hdr[11], 10);
payload8[i] = encoded_data.charCodeAt(i);
} conf.onFBUReceive(that,
{'x': FBU.x, 'y': FBU.y,
'width': FBU.width, 'height': FBU.height,
'encoding': FBU.encoding,
'encodingName': encNames[FBU.encoding]});
if (encNames[FBU.encoding]) {
// Debug:
/*
var msg = "FramebufferUpdate rects:" + FBU.rects;
msg += " x: " + FBU.x + " y: " + FBU.y;
msg += " width: " + FBU.width + " height: " + FBU.height;
msg += " encoding:" + FBU.encoding;
msg += "(" + encNames[FBU.encoding] + ")";
msg += ", ws.rQlen(): " + ws.rQlen();
Util.Debug(msg);
*/
} else {
fail("Disconnected: unsupported encoding " +
FBU.encoding);
return false;
}
}
console.log("packet8: ", packet8, ", payload: ", encoded_data); timing.last_fbu = (new Date()).getTime();
return packet8;
} ret = encHandlers[FBU.encoding]();
*/
now = (new Date()).getTime();
function encode_packet(data) { timing.cur_fbu += (now - timing.last_fbu);
//console.log("send packet data:", data);
var encoded_data = bencode(data), if (ret) {
payload_size = encoded_data.length, encStats[FBU.encoding][0] += 1;
packet = []; encStats[FBU.encoding][1] += 1;
timing.pixels += FBU.width * FBU.height;
packet.push8(80); // 'P'.charCodeAt(0) }
packet.push8(0); // protocol flags
packet.push8(0); // compression level if (timing.pixels >= (fb_width * fb_height)) {
packet.push8(0); // packet index if (((FBU.width === fb_width) &&
packet.push32(payload_size); // packet payload size (FBU.height === fb_height)) ||
(timing.fbu_rt_start > 0)) {
// Convert to array timing.full_fbu_total += timing.cur_fbu;
// TODO: remmove this step timing.full_fbu_cnt += 1;
for (var i=0; i < payload_size; i++) { Util.Info("Timing of full FBU, cur: " +
packet.push8(encoded_data.charCodeAt(i)); timing.cur_fbu + ", total: " +
timing.full_fbu_total + ", cnt: " +
timing.full_fbu_cnt + ", avg: " +
(timing.full_fbu_total /
timing.full_fbu_cnt));
}
if (timing.fbu_rt_start > 0) {
fbu_rt_diff = now - timing.fbu_rt_start;
timing.fbu_rt_total += fbu_rt_diff;
timing.fbu_rt_cnt += 1;
Util.Info("full FBU round-trip, cur: " +
fbu_rt_diff + ", total: " +
timing.fbu_rt_total + ", cnt: " +
timing.fbu_rt_cnt + ", avg: " +
(timing.fbu_rt_total /
timing.fbu_rt_cnt));
timing.fbu_rt_start = 0;
}
}
if (! ret) {
return ret; // false ret means need more data
}
} }
return packet; conf.onFBUComplete(that,
} {'x': FBU.x, 'y': FBU.y,
'width': FBU.width, 'height': FBU.height,
'encoding': FBU.encoding,
'encodingName': encNames[FBU.encoding]});
return true; // We finished this FBU
};
//
// FramebufferUpdate encodings
//
encHandlers.RAW = function display_raw() {
//Util.Debug(">> display_raw (" + ws.rQlen() + " bytes)");
var cur_y, cur_height;
var inflate = function(data, offset) { if (FBU.lines === 0) {
zlib.reset(); FBU.lines = FBU.height;
var inflated = zlib.uncompress(data, offset);
if (inflated.status !== 0) {
throw("Invalid data in zlib stream");
} }
return inflated.data; FBU.bytes = FBU.width * fb_Bpp; // At least a line
if (ws.rQwait("RAW", FBU.bytes)) { return false; }
cur_y = FBU.y + (FBU.height - FBU.lines);
cur_height = Math.min(FBU.lines,
Math.floor(ws.rQlen()/(FBU.width * fb_Bpp)));
display.blitImage(FBU.x, cur_y, FBU.width, cur_height,
ws.get_rQ(), ws.get_rQi());
ws.rQshiftBytes(FBU.width * cur_height * fb_Bpp);
FBU.lines -= cur_height;
if (FBU.lines > 0) {
FBU.bytes = FBU.width * fb_Bpp; // At least another line
} else {
FBU.rects -= 1;
FBU.bytes = 0;
}
//Util.Debug("<< display_raw (" + ws.rQlen() + " bytes)");
return true;
};
encHandlers.COPYRECT = function display_copy_rect() {
//Util.Debug(">> display_copy_rect");
var old_x, old_y;
FBU.bytes = 4;
if (ws.rQwait("COPYRECT", 4)) { return false; }
display.renderQ_push({
'type': 'copy',
'old_x': ws.rQshift16(),
'old_y': ws.rQshift16(),
'x': FBU.x,
'y': FBU.y,
'width': FBU.width,
'height': FBU.height});
FBU.rects -= 1;
FBU.bytes = 0;
return true;
}; };
function decode_packet(arr, idx) { encHandlers.RRE = function display_rre() {
var packet = {}; //Util.Debug(">> display_rre (" + ws.rQlen() + " bytes)");
idx = (typeof(idx) === "undefined") ? 0 : idx; var color, x, y, width, height, chunk;
if (arr.length - idx < 8) {
return {'length': 0}; if (FBU.subrects === 0) {
FBU.bytes = 4+fb_Bpp;
if (ws.rQwait("RRE", 4+fb_Bpp)) { return false; }
FBU.subrects = ws.rQshift32();
color = ws.rQshiftBytes(fb_Bpp); // Background
display.fillRect(FBU.x, FBU.y, FBU.width, FBU.height, color);
} }
packet.magic = arr[idx]; while ((FBU.subrects > 0) && (ws.rQlen() >= (fb_Bpp + 8))) {
packet.flags = arr[idx+1]; color = ws.rQshiftBytes(fb_Bpp);
packet.level = arr[idx+2]; x = ws.rQshift16();
packet.index = arr[idx+3]; y = ws.rQshift16();
packet.size = (arr[idx+4] << 24) + width = ws.rQshift16();
(arr[idx+5] << 16) + height = ws.rQshift16();
(arr[idx+6] << 8) + display.fillRect(FBU.x + x, FBU.y + y, width, height, color);
(arr[idx+7]); FBU.subrects -= 1;
packet.length = 8 + packet.size;
packet.data = null;
if (arr.length - idx < packet.length) {
return {'length': 0};
} }
//Util.Debug(" display_rre: rects: " + FBU.rects +
// ", FBU.subrects: " + FBU.subrects);
var new_arr, offset = idx + 8; if (FBU.subrects > 0) {
if (packet.level > 0) { chunk = Math.min(rre_chunk_sz, FBU.subrects);
new_arr = inflate(arr, offset); FBU.bytes = (fb_Bpp + 8) * chunk;
} else { } else {
// Convert to string FBU.rects -= 1;
// TODO: remove this step FBU.bytes = 0;
new_arr = arr.slice(offset, offset + packet.size);
} }
var str = String.fromCharCode.apply(null, new_arr); //Util.Debug("<< display_rre, FBU.bytes: " + FBU.bytes);
return true;
};
if (packet.index > 0) { encHandlers.HEXTILE = function display_hextile() {
raw_packets[packet.index] = str; //Util.Debug(">> display_hextile");
// packet.data stays null indicating incomplete packet var subencoding, subrects, color, cur_tile,
} else { tile_x, x, w, tile_y, y, h, xy, s, sx, sy, wh, sw, sh,
packet.data = bdecode(str, 0)[0]; rQ = ws.get_rQ(), rQi = ws.get_rQi();
// Insert any raw packets into place
for (var idx in raw_packets) { if (FBU.tiles === 0) {
packet.data[idx] = raw_packets[idx]; FBU.tiles_x = Math.ceil(FBU.width/16);
} FBU.tiles_y = Math.ceil(FBU.height/16);
raw_packets = {}; FBU.total_tiles = FBU.tiles_x * FBU.tiles_y;
FBU.tiles = FBU.total_tiles;
} }
return packet;
} /* FBU.bytes comes in as 1, ws.rQlen() at least 1 */
while (FBU.tiles > 0) {
FBU.bytes = 1;
if (ws.rQwait("HEXTILE subencoding", FBU.bytes)) { return false; }
subencoding = rQ[rQi]; // Peek
if (subencoding > 30) { // Raw
fail("Disconnected: illegal hextile subencoding " + subencoding);
//Util.Debug("ws.rQslice(0,30):" + ws.rQslice(0,30));
return false;
}
subrects = 0;
cur_tile = FBU.total_tiles - FBU.tiles;
tile_x = cur_tile % FBU.tiles_x;
tile_y = Math.floor(cur_tile / FBU.tiles_x);
x = FBU.x + tile_x * 16;
y = FBU.y + tile_y * 16;
w = Math.min(16, (FBU.x + FBU.width) - x);
h = Math.min(16, (FBU.y + FBU.height) - y);
/* Figure out how much we are expecting */
if (subencoding & 0x01) { // Raw
//Util.Debug(" Raw subencoding");
FBU.bytes += w * h * fb_Bpp;
} else {
if (subencoding & 0x02) { // Background
FBU.bytes += fb_Bpp;
}
if (subencoding & 0x04) { // Foreground
FBU.bytes += fb_Bpp;
}
if (subencoding & 0x08) { // AnySubrects
FBU.bytes += 1; // Since we aren't shifting it off
if (ws.rQwait("hextile subrects header", FBU.bytes)) { return false; }
subrects = rQ[rQi + FBU.bytes-1]; // Peek
if (subencoding & 0x10) { // SubrectsColoured
FBU.bytes += subrects * (fb_Bpp + 2);
} else {
FBU.bytes += subrects * 2;
}
}
}
// /*
// Client packet generation routines Util.Debug(" tile:" + cur_tile + "/" + (FBU.total_tiles - 1) +
// " (" + tile_x + "," + tile_y + ")" +
" [" + x + "," + y + "]@" + w + "x" + h +
", subenc:" + subencoding +
"(last: " + FBU.lastsubencoding + "), subrects:" +
subrects +
", ws.rQlen():" + ws.rQlen() + ", FBU.bytes:" + FBU.bytes +
" last:" + ws.rQslice(FBU.bytes-10, FBU.bytes) +
" next:" + ws.rQslice(FBU.bytes-1, FBU.bytes+10));
*/
if (ws.rQwait("hextile", FBU.bytes)) { return false; }
/* We know the encoding and have a whole tile */
FBU.subencoding = rQ[rQi];
rQi += 1;
if (FBU.subencoding === 0) {
if (FBU.lastsubencoding & 0x01) {
/* Weird: ignore blanks after RAW */
Util.Debug(" Ignoring blank after RAW");
} else {
display.fillRect(x, y, w, h, FBU.background);
}
} else if (FBU.subencoding & 0x01) { // Raw
display.blitImage(x, y, w, h, rQ, rQi);
rQi += FBU.bytes - 1;
} else {
if (FBU.subencoding & 0x02) { // Background
FBU.background = rQ.slice(rQi, rQi + fb_Bpp);
rQi += fb_Bpp;
}
if (FBU.subencoding & 0x04) { // Foreground
FBU.foreground = rQ.slice(rQi, rQi + fb_Bpp);
rQi += fb_Bpp;
}
function helloPacket () { display.startTile(x, y, w, h, FBU.background);
var flat_keycodes = [], if (FBU.subencoding & 0x08) { // AnySubrects
capabilities; subrects = rQ[rQi];
for(var code in keycodes) { rQi += 1;
flat_keycodes.push(keycodes[code]); for (s = 0; s < subrects; s += 1) {
if (FBU.subencoding & 0x10) { // SubrectsColoured
color = rQ.slice(rQi, rQi + fb_Bpp);
rQi += fb_Bpp;
} else {
color = FBU.foreground;
}
xy = rQ[rQi];
rQi += 1;
sx = (xy >> 4);
sy = (xy & 0x0f);
wh = rQ[rQi];
rQi += 1;
sw = (wh >> 4) + 1;
sh = (wh & 0x0f) + 1;
display.subTile(sx, sy, sw, sh, color);
}
}
display.finishTile();
}
ws.set_rQi(rQi);
FBU.lastsubencoding = FBU.subencoding;
FBU.bytes = 0;
FBU.tiles -= 1;
} }
capabilities = {'encoding': 'png',
'encodings': ['png', 'jpeg'],
'version': '0.9.0',
'xkbmap_keycodes': flat_keycodes};
return encode_packet(['hello', capabilities]);
}
function keyActionPacket(wid, keysym, down, evt) { if (FBU.tiles === 0) {
var keydata = keycodes[keysym], FBU.rects -= 1;
keyname = "",
modifiers = [],
keyval = 0,
str = "", // not used
keycode = 0,
group = 0, // not used
is_modifier = 0; // not used
if (!keydata) {
Util.Warn("ignoring undefined keysym " + keysym + " full event: " + evt);
return false;
} }
keyname = keydata[1];
keyval = keydata[0]; //Util.Debug("<< display_hextile");
keycode = keydata[2]; return true;
if (evt.shiftKey) { };
modifiers.push('shift');
// Get 'compact length' header and data size
getTightCLength = function (arr) {
var header = 1, data = 0;
data += arr[0] & 0x7f;
if (arr[0] & 0x80) {
header += 1;
data += (arr[1] & 0x7f) << 7;
if (arr[1] & 0x80) {
header += 1;
data += arr[2] << 14;
}
} }
if (evt.ctrlKey) { return [header, data];
modifiers.push('control'); };
function display_tight(isTightPNG) {
//Util.Debug(">> display_tight");
if (fb_depth === 1) {
fail("Tight protocol handler only implements true color mode");
} }
if (evt.altKey) {
modifiers.push('alt'); var ctl, cmode, clength, color, img, data;
var filterId = -1, resetStreams = 0, streamId = -1;
var rQ = ws.get_rQ(), rQi = ws.get_rQi();
FBU.bytes = 1; // compression-control byte
if (ws.rQwait("TIGHT compression-control", FBU.bytes)) { return false; }
var checksum = function(data) {
var sum=0, i;
for (i=0; i<data.length;i++) {
sum += data[i];
if (sum > 65536) sum -= 65536;
}
return sum;
} }
Util.Info("send key-action for window " + wid + ", keyname: " + keyname + ", keyval: " + keyval + ", modifiers: " + modifiers);
return encode_packet(['key-action', wid, keyname, down,
modifiers, keyval, str, keycode, group, is_modifier]);
}
function pointerPositionPacket(wid, x, y) { var decompress = function(data) {
var modifiers = [], for (var i=0; i<4; i++) {
buttons = [], if ((resetStreams >> i) & 1) {
rx = windows[wid].x + x, FBU.zlibs[i].reset();
ry = windows[wid].y + y; Util.Info("Reset zlib stream " + i);
return encode_packet(['pointer-position', wid, }
[rx, ry], modifiers, buttons]); }
} var uncompressed = FBU.zlibs[streamId].uncompress(data, 0);
if (uncompressed.status !== 0) {
Util.Error("Invalid data in zlib stream");
}
//Util.Warn("Decompressed " + data.length + " to " +
// uncompressed.data.length + " checksums " +
// checksum(data) + ":" + checksum(uncompressed.data));
function pointerActionPacket(wid, button, down, x, y) { return uncompressed.data;
var modifiers = [], }
buttons = [],
rx = windows[wid].x + x,
ry = windows[wid].y + y;
return encode_packet(['button-action', wid, button, down,
[rx, ry], modifiers, buttons]);
}
function pingPacket() { var indexedToRGB = function (data, numColors, palette, width, height) {
var now_ms = (new Date()).getTime(); // Convert indexed (palette based) image data to RGB
Util.Info("send ping now_ms: " + now_ms); // TODO: reduce number of calculations inside loop
return encode_packet(['ping', now_ms]); var dest = [];
} var x, y, b, w, w1, dp, sp;
if (numColors === 2) {
w = Math.floor((width + 7) / 8);
w1 = Math.floor(width / 8);
for (y = 0; y < height; y++) {
for (x = 0; x < w1; x++) {
for (b = 7; b >= 0; b--) {
dp = (y*width + x*8 + 7-b) * 3;
sp = (data[y*w + x] >> b & 1) * 3;
dest[dp ] = palette[sp ];
dest[dp+1] = palette[sp+1];
dest[dp+2] = palette[sp+2];
}
}
for (b = 7; b >= 8 - width % 8; b--) {
dp = (y*width + x*8 + 7-b) * 3;
sp = (data[y*w + x] >> b & 1) * 3;
dest[dp ] = palette[sp ];
dest[dp+1] = palette[sp+1];
dest[dp+2] = palette[sp+2];
}
}
} else {
for (y = 0; y < height; y++) {
for (x = 0; x < width; x++) {
dp = (y*width + x) * 3;
sp = data[y*width + x] * 3;
dest[dp ] = palette[sp ];
dest[dp+1] = palette[sp+1];
dest[dp+2] = palette[sp+2];
}
}
}
return dest;
};
var handlePalette = function() {
var numColors = rQ[rQi + 2] + 1;
var paletteSize = numColors * fb_depth;
FBU.bytes += paletteSize;
if (ws.rQwait("TIGHT palette " + cmode, FBU.bytes)) { return false; }
var bpp = (numColors <= 2) ? 1 : 8;
var rowSize = Math.floor((FBU.width * bpp + 7) / 8);
var raw = false;
if (rowSize * FBU.height < 12) {
raw = true;
clength = [0, rowSize * FBU.height];
} else {
clength = getTightCLength(ws.rQslice(3 + paletteSize,
3 + paletteSize + 3));
}
FBU.bytes += clength[0] + clength[1];
if (ws.rQwait("TIGHT " + cmode, FBU.bytes)) { return false; }
// // Shift ctl, filter id, num colors, palette entries, and clength off
// ws.rQshiftBytes(3);
// var palette = ws.rQshiftBytes(paletteSize);
ws.rQshiftBytes(clength[0]);
function send_ping () { if (raw) {
ws.send(pingPacket()); data = ws.rQshiftBytes(clength[1]);
if (rfb_state === 'normal') { } else {
setTimeout(send_ping, conf.ping_rate); data = decompress(ws.rQshiftBytes(clength[1]));
}
// Convert indexed (palette based) image data to RGB
var rgb = indexedToRGB(data, numColors, palette, FBU.width, FBU.height);
// Add it to the render queue
display.renderQ_push({
'type': 'blitRgb',
'data': rgb,
'x': FBU.x,
'y': FBU.y,
'width': FBU.width,
'height': FBU.height});
return true;
} }
}
// var handleCopy = function() {
// Server packet receive handlers var raw = false;
// var uncompressedSize = FBU.width * FBU.height * fb_depth;
if (uncompressedSize < 12) {
raw = true;
clength = [0, uncompressedSize];
} else {
clength = getTightCLength(ws.rQslice(1, 4));
}
FBU.bytes = 1 + clength[0] + clength[1];
if (ws.rQwait("TIGHT " + cmode, FBU.bytes)) { return false; }
packetHandlers.hello = function process_hello (data) { // Shift ctl, clength off
Util.Info("got hello: " + data); ws.rQshiftBytes(1 + clength[0]);
ws.send(encode_packet(["set_deflate", 0]));
updateState('normal', "Connected");
/* Start pushing/polling */ if (raw) {
setTimeout(checkEvents, conf.check_rate); data = ws.rQshiftBytes(clength[1]);
} else {
data = decompress(ws.rQshiftBytes(clength[1]));
}
/* Start sending pings to the server */ display.renderQ_push({
send_ping(); 'type': 'blitRgb',
}; 'data': data,
'x': FBU.x,
'y': FBU.y,
'width': FBU.width,
'height': FBU.height});
return true;
}
packetHandlers.new_window = function process_new_window (data) { ctl = ws.rQpeek8();
Util.Info("got new-window: " + data);
var wid = data[1], // Keep tight reset bits
x = data[2], resetStreams = ctl & 0xF;
y = data[3],
w = data[4], // Figure out filter
h = data[5], ctl = ctl >> 4;
props = data[6], streamId = ctl & 0x3;
loc = data[7];
if (wid !== 1) { if (ctl === 0x08) cmode = "fill";
// TODO: don't ignore other windows else if (ctl === 0x09) cmode = "jpeg";
Util.Warn("ignoring new-window for window ID " + wid); else if (ctl === 0x0A) cmode = "png";
return; else if (ctl & 0x04) cmode = "filter";
else if (ctl < 0x04) cmode = "copy";
else return fail("Illegal tight compression received, ctl: " + ctl);
if (isTightPNG && (cmode === "filter" || cmode === "copy")) {
return fail("filter/copy received in tightPNG mode");
} }
if (w !== fb_width || h !== fb_height) { switch (cmode) {
fb_width = w; // fill uses fb_depth because TPIXELs drop the padding byte
fb_height = h; case "fill": FBU.bytes += fb_depth; break; // TPIXEL
conf.onFBResize(that, fb_width, fb_height); case "jpeg": FBU.bytes += 3; break; // max clength
display.resize(fb_width, fb_height); case "png": FBU.bytes += 3; break; // max clength
timing.fbu_rt_start = (new Date()).getTime(); case "filter": FBU.bytes += 2; break; // filter id + num colors if palette
case "copy": break;
} }
windows[wid] = {x: x, y: y, w: w, h: h, props: props, loc: loc}; if (ws.rQwait("TIGHT " + cmode, FBU.bytes)) { return false; }
//ws.send(encode_packet(["configure-window", wid, x, y, w, h, loc]));
ws.send(encode_packet(["map-window", wid, x, y, w, h, loc])); //Util.Debug(" ws.rQslice(0,20): " + ws.rQslice(0,20) + " (" + ws.rQlen() + ")");
ws.send(encode_packet(["focus", wid])); //Util.Debug(" cmode: " + cmode);
// Determine FBU.bytes
switch (cmode) {
case "fill":
ws.rQshift8(); // shift off ctl
color = ws.rQshiftBytes(fb_depth);
display.renderQ_push({
'type': 'fill',
'x': FBU.x,
'y': FBU.y,
'width': FBU.width,
'height': FBU.height,
'color': [color[2], color[1], color[0]] });
break;
case "png":
case "jpeg":
clength = getTightCLength(ws.rQslice(1, 4));
FBU.bytes = 1 + clength[0] + clength[1]; // ctl + clength size + jpeg-data
if (ws.rQwait("TIGHT " + cmode, FBU.bytes)) { return false; }
// We have everything, render it
//Util.Debug(" jpeg, ws.rQlen(): " + ws.rQlen() + ", clength[0]: " +
// clength[0] + ", clength[1]: " + clength[1]);
ws.rQshiftBytes(1 + clength[0]); // shift off ctl + compact length
img = new Image();
img.src = "data:image/" + cmode +
extract_data_uri(ws.rQshiftBytes(clength[1]));
display.renderQ_push({
'type': 'img',
'img': img,
'x': FBU.x,
'y': FBU.y});
img = null;
break;
case "filter":
filterId = rQ[rQi + 1];
if (filterId === 1) {
if (!handlePalette()) { return false; }
} else {
// Filter 0, Copy could be valid here, but servers don't send it as an explicit filter
// Filter 2, Gradient is valid but not used if jpeg is enabled
throw("Unsupported tight subencoding received, filter: " + filterId);
}
break;
case "copy":
if (!handleCopy()) { return false; }
break;
}
FBU.bytes = 0;
FBU.rects -= 1;
//Util.Debug(" ending ws.rQslice(0,20): " + ws.rQslice(0,20) + " (" + ws.rQlen() + ")");
//Util.Debug("<< display_tight_png");
return true;
}
extract_data_uri = function(arr) {
//var i, stra = [];
//for (i=0; i< arr.length; i += 1) {
// stra.push(String.fromCharCode(arr[i]));
//}
//return "," + escape(stra.join(''));
return ";base64," + Base64.encode(arr);
};
encHandlers.TIGHT = function () { return display_tight(false); };
encHandlers.TIGHT_PNG = function () { return display_tight(true); };
encHandlers.last_rect = function last_rect() {
//Util.Debug(">> last_rect");
FBU.rects = 0;
//Util.Debug("<< last_rect");
return true;
};
encHandlers.DesktopSize = function set_desktopsize() {
Util.Debug(">> set_desktopsize");
fb_width = FBU.width;
fb_height = FBU.height;
conf.onFBResize(that, fb_width, fb_height);
display.resize(fb_width, fb_height); display.resize(fb_width, fb_height);
keyboard.grab(); timing.fbu_rt_start = (new Date()).getTime();
mouse.grab(); // Send a new non-incremental request
ws.send(fbUpdateRequests());
FBU.bytes = 0;
FBU.rects -= 1;
Util.Debug("<< set_desktopsize");
return true;
}; };
packetHandlers.ping_echo = function process_ping_echo (data) { encHandlers.Cursor = function set_cursor() {
Util.Info("got ping_echo: " + data); var x, y, w, h, pixelslength, masklength;
var echoedtime = data[1], Util.Debug(">> set_cursor");
l1 = data[2], x = FBU.x; // hotspot-x
l2 = data[3], y = FBU.y; // hotspot-y
l3 = data[4], w = FBU.width;
cl = data[5], h = FBU.height;
now_ms = (new Date()).getTime(),
sl = -1; pixelslength = w * h * fb_Bpp;
stats.last_ping_echoed_time = echoedtime; masklength = Math.floor((w + 7) / 8) * h;
sl = now_ms - echoedtime;
stats.server_ping_latency.push([now_ms, sl]); FBU.bytes = pixelslength + masklength;
// Keep 100 entries if (ws.rQwait("cursor encoding", FBU.bytes)) { return false; }
if (stats.server_ping_latency.length > 100) {
stats.server_ping_latency.splice(0, stats.server_ping_latency.length-100); //Util.Debug(" set_cursor, x: " + x + ", y: " + y + ", w: " + w + ", h: " + h);
}
display.changeCursor(ws.rQshiftBytes(pixelslength),
ws.rQshiftBytes(masklength),
x, y, w, h);
FBU.bytes = 0;
FBU.rects -= 1;
if (cl >= 0) { Util.Debug("<< set_cursor");
stats.client_ping_latency.push([now_ms, cl]); return true;
// Keep 100 entries };
if (stats.client_ping_latency.length > 100) {
stats.client_ping_latency.splice(0, stats.client_ping_latency.length-100); encHandlers.JPEG_quality_lo = function set_jpeg_quality() {
Util.Error("Server sent jpeg_quality pseudo-encoding");
};
encHandlers.compress_lo = function set_compress_level() {
Util.Error("Server sent compress level pseudo-encoding");
};
/*
* Client message routines
*/
pixelFormat = function() {
//Util.Debug(">> pixelFormat");
var arr;
arr = [0]; // msg-type
arr.push8(0); // padding
arr.push8(0); // padding
arr.push8(0); // padding
arr.push8(fb_Bpp * 8); // bits-per-pixel
arr.push8(fb_depth * 8); // depth
arr.push8(0); // little-endian
arr.push8(conf.true_color ? 1 : 0); // true-color
arr.push16(255); // red-max
arr.push16(255); // green-max
arr.push16(255); // blue-max
arr.push8(16); // red-shift
arr.push8(8); // green-shift
arr.push8(0); // blue-shift
arr.push8(0); // padding
arr.push8(0); // padding
arr.push8(0); // padding
//Util.Debug("<< pixelFormat");
return arr;
};
clientEncodings = function() {
//Util.Debug(">> clientEncodings");
var arr, i, encList = [];
for (i=0; i<encodings.length; i += 1) {
if ((encodings[i][0] === "Cursor") &&
(! conf.local_cursor)) {
Util.Debug("Skipping Cursor pseudo-encoding");
// TODO: remove this when we have tight+non-true-color
} else if ((encodings[i][0] === "TIGHT") &&
(! conf.true_color)) {
Util.Warn("Skipping tight, only support with true color");
} else {
//Util.Debug("Adding encoding: " + encodings[i][0]);
encList.push(encodings[i][1]);
} }
} }
arr = [2]; // msg-type
arr.push8(0); // padding
arr.push16(encList.length); // encoding count
for (i=0; i < encList.length; i += 1) {
arr.push32(encList[i]);
}
//Util.Debug("<< clientEncodings: " + arr);
return arr;
}; };
packetHandlers.ping = function process_ping (data) { fbUpdateRequest = function(incremental, x, y, xw, yw) {
Util.Info("got ping: " + data); //Util.Debug(">> fbUpdateRequest");
var echotime = data[1], if (typeof(x) === "undefined") { x = 0; }
l1 = 500, l2 = 500, l3 = 500, // fake load-averages if (typeof(y) === "undefined") { y = 0; }
sl = -1, if (typeof(xw) === "undefined") { xw = fb_width; }
sl_len = stats.server_ping_latency.length; if (typeof(yw) === "undefined") { yw = fb_height; }
if (sl_len > 0) { var arr;
sl = stats.server_ping_latency[sl_len-1][1]; arr = [3]; // msg-type
arr.push8(incremental);
arr.push16(x);
arr.push16(y);
arr.push16(xw);
arr.push16(yw);
//Util.Debug("<< fbUpdateRequest");
return arr;
};
// Based on clean/dirty areas, generate requests to send
fbUpdateRequests = function() {
var cleanDirty = display.getCleanDirtyReset(),
arr = [], i, cb, db;
cb = cleanDirty.cleanBox;
if (cb.w > 0 && cb.h > 0) {
// Request incremental for clean box
arr = arr.concat(fbUpdateRequest(1, cb.x, cb.y, cb.w, cb.h));
}
for (i = 0; i < cleanDirty.dirtyBoxes.length; i++) {
db = cleanDirty.dirtyBoxes[i];
// Force all (non-incremental for dirty box
arr = arr.concat(fbUpdateRequest(0, db.x, db.y, db.w, db.h));
} }
Util.Info("send ping_echo sl: " + sl); return arr;
ws.send(encode_packet(["ping_echo", echotime,
l1, l2, l3, sl]));
}; };
packetHandlers.draw = function process_draw (data) {
Util.Info("got draw #" + data[8] + " for window " + data[1] + ": " + data.slice(2,7));
var wid = data[1], keyEvent = function(keysym, down) {
x = data[2], //Util.Debug(">> keyEvent, keysym: " + keysym + ", down: " + down);
y = data[3], var arr;
w = data[4], arr = [4]; // msg-type
h = data[5], arr.push8(down);
coding = data[6], arr.push16(0);
raw = data[7], arr.push32(keysym);
damage_seq = data[8], //Util.Debug("<< keyEvent");
rowstride = data[9], return arr;
client_opts = data[10], };
decode_time,
img; pointerEvent = function(x, y) {
if (wid !== 1) { //Util.Debug(">> pointerEvent, x,y: " + x + "," + y +
// TODO: handle other windows // " , mask: " + mouse_buttonMask);
Util.Warn("ignoring draw for window ID " + wid); var arr;
return; arr = [5]; // msg-type
arr.push8(mouse_buttonMask);
arr.push16(x);
arr.push16(y);
//Util.Debug("<< pointerEvent");
return arr;
};
clientCutText = function(text) {
//Util.Debug(">> clientCutText");
var arr, i, n;
arr = [6]; // msg-type
arr.push8(0); // padding
arr.push8(0); // padding
arr.push8(0); // padding
arr.push32(text.length);
n = text.length;
for (i=0; i < n; i+=1) {
arr.push(text.charCodeAt(i));
} }
img = new Image(); //Util.Debug("<< clientCutText:" + arr);
img.src = "data:image/" + coding + ";base64," + window.btoa(raw); return arr;
display.renderQ_push({
'type': 'img',
'img': img,
'x': x,
'y': y});
img = null;
// based on _do_draw, draw_region, do_draw_region, paint_png, etc
decode_time = ((new Date()).getTime() - cur_packet_recv_time)*1000;
Util.Info("send damage-sequence #" + damage_seq + " for window " + wid + ", w: " + w + ", h: " + h + ", decode_time: " + decode_time);
ws.send(encode_packet(['damage-sequence', damage_seq,
wid, w, h, decode_time]));
}; };
// //
// Public API interface functions // Public API interface functions
// //
...@@ -1044,17 +1803,55 @@ that.disconnect = function() { ...@@ -1044,17 +1803,55 @@ that.disconnect = function() {
//Util.Debug("<< disconnect"); //Util.Debug("<< disconnect");
}; };
that.sendPassword = function(passwd) {
rfb_password = passwd;
rfb_state = "Authentication";
setTimeout(init_msg, 1);
};
that.sendCtrlAltDel = function() {
if (rfb_state !== "normal" || conf.view_only) { return false; }
Util.Info("Sending Ctrl-Alt-Del");
var arr = [];
arr = arr.concat(keyEvent(0xFFE3, 1)); // Control
arr = arr.concat(keyEvent(0xFFE9, 1)); // Alt
arr = arr.concat(keyEvent(0xFFFF, 1)); // Delete
arr = arr.concat(keyEvent(0xFFFF, 0)); // Delete
arr = arr.concat(keyEvent(0xFFE9, 0)); // Alt
arr = arr.concat(keyEvent(0xFFE3, 0)); // Control
arr = arr.concat(fbUpdateRequests());
ws.send(arr);
};
// Send a key press. If 'down' is not specified then send a down key
// followed by an up key.
that.sendKey = function(code, down) {
if (rfb_state !== "normal" || conf.view_only) { return false; }
var arr = [];
if (typeof down !== 'undefined') {
Util.Info("Sending key code (" + (down ? "down" : "up") + "): " + code);
arr = arr.concat(keyEvent(code, down ? 1 : 0));
} else {
Util.Info("Sending key code (down + up): " + code);
arr = arr.concat(keyEvent(code, 1));
arr = arr.concat(keyEvent(code, 0));
}
arr = arr.concat(fbUpdateRequests());
ws.send(arr);
};
that.clipboardPasteFrom = function(text) {
if (rfb_state !== "normal") { return; }
//Util.Debug(">> clipboardPasteFrom: " + text.substr(0,40) + "...");
ws.send(clientCutText(text));
//Util.Debug("<< clipboardPasteFrom");
};
// Override internal functions for testing // Override internal functions for testing
that.testMode = function(override_send, data_mode) { that.testMode = function(override_send, data_mode) {
test_mode = true; test_mode = true;
that.recv_message = ws.testMode(override_send, data_mode); that.recv_message = ws.testMode(override_send, data_mode);
// Allow debug calls to this
that.encode_packet = encode_packet;
that.decode_packet = decode_packet;
that.bdecode = bdecode;
that.bencode = bencode;
checkEvents = function () { /* Stub Out */ }; checkEvents = function () { /* Stub Out */ };
that.connect = function(host, port, password) { that.connect = function(host, port, password) {
rfb_host = host; rfb_host = host;
......
...@@ -10,9 +10,6 @@ ...@@ -10,9 +10,6 @@
/*jslint white: false, browser: true */ /*jslint white: false, browser: true */
/*global window, $D, Util, WebUtil, RFB, Display */ /*global window, $D, Util, WebUtil, RFB, Display */
// Load supporting scripts
window.onscriptsload = function () { UI.load(); };
var UI = { var UI = {
rfb_state : 'loaded', rfb_state : 'loaded',
...@@ -66,9 +63,9 @@ start: function(callback) { ...@@ -66,9 +63,9 @@ start: function(callback) {
UI.initSetting('path', 'websockify'); UI.initSetting('path', 'websockify');
UI.initSetting('repeaterID', ''); UI.initSetting('repeaterID', '');
UI.rfb = RFB({'target': $D('noVNC_canvas'), UI.rfb = UI.Client({'target': $D('noVNC_canvas'),
'onUpdateState': UI.updateState, 'onUpdateState': UI.updateState,
'onClipboard': UI.clipReceive}); 'onClipboard': UI.clipReceive});
UI.updateVisualState(); UI.updateVisualState();
// Unfocus clipboard when over the VNC area // Unfocus clipboard when over the VNC area
......
/*
* noVNC: HTML5 VNC client
* Copyright (C) 2012 Joel Martin
* Licensed under MPL 2.0 (see LICENSE.txt)
*
* See README.md for usage and integration instructions.
*
* TIGHT decoder portion:
* (c) 2012 Michael Tinglof, Joe Balaz, Les Piech (Mercuri.ca)
*/
/*jslint white: false, browser: true, bitwise: false, plusplus: false */
/*global window, Util, Display, Keyboard, Mouse, Websock, Websock_native, Base64, DES */
function Xpra(defaults) {
"use strict";
var that = {}, // Public API methods
conf = {}, // Configuration attributes
// Pre-declare private functions used before definitions (jslint)
init_vars, updateState, fail, handle_message, print_stats,
checkEvents, // Overridable for testing
keyPress, mouseButton, mouseMove,
bencode, bdecode, encode_packet, decode_packet,
//
// Private namespace variables
//
// connection related
connHost = '',
connPort = 5900,
connPassword = '',
connPath = '',
connState = 'disconnected',
rfb_max_version= 3.8,
rfb_auth_scheme= '',
packetHandlers = {},
encHandlers = {},
encNames = {},
encStats = {}, // [rectCnt, rectCntTot]
ws = null, // Websock object
display = null, // Display object
keyboard = null, // Keyboard input handler object
mouse = null, // Mouse input handler object
sendTimer = null, // Send Queue check timer
connTimer = null, // connection timer
disconnTimer = null, // disconnection timer
msgTimer = null, // queued handle_message timer
zlib = null, // zlib encoder/decoder
windows = {}, // managed windows
send_packets = [], // pending packets to send (mouse movements)
raw_packets = {}, // partially received raw packets
cur_packet_recv_time = 0, // record when we received the packet we are currently handling
keycodes = {},
stats = {
last_ping_echoed_time: 0,
server_ping_latency: [],
client_ping_latency: []
},
fb_Bpp = 4,
fb_depth = 3,
fb_width = 0,
fb_height = 0,
fb_name = "",
last_req_time = 0,
rre_chunk_sz = 100,
timing = {
last_fbu : 0,
fbu_total : 0,
fbu_total_cnt : 0,
full_fbu_total : 0,
full_fbu_cnt : 0,
fbu_rt_start : 0,
fbu_rt_total : 0,
fbu_rt_cnt : 0,
pixels : 0
},
test_mode = false,
def_con_timeout = Websock_native ? 2 : 5,
/* Mouse state */
mouse_buttonMask = 0,
mouse_arr = [],
viewportDragging = false,
viewportDragPos = {};
// Configuration attributes
Util.conf_defaults(conf, that, defaults, [
['target', 'wo', 'dom', null, 'VNC display rendering Canvas object'],
['focusContainer', 'wo', 'dom', document, 'DOM element that captures keyboard input'],
['encrypt', 'rw', 'bool', false, 'Use TLS/SSL/wss encryption'],
['true_color', 'rw', 'bool', true, 'Request true color pixel data'],
['local_cursor', 'rw', 'bool', false, 'Request locally rendered cursor'],
['shared', 'rw', 'bool', true, 'Request shared mode'],
['view_only', 'rw', 'bool', false, 'Disable client mouse/keyboard'],
['connectTimeout', 'rw', 'int', def_con_timeout, 'Time (s) to wait for connection'],
['disconnectTimeout', 'rw', 'int', 3, 'Time (s) to wait for disconnection'],
// UltraVNC repeater ID to connect to
['repeaterID', 'rw', 'str', '', 'RepeaterID to connect to'],
['viewportDrag', 'rw', 'bool', false, 'Move the viewport on mouse drags'],
['check_rate', 'rw', 'int', 217, 'Timing (ms) of send/receive check'],
['ping_rate', 'rw', 'int', 10000, 'Timing (ms) of ping sending'],
// Callback functions
['onUpdateState', 'rw', 'func', function() { },
'onUpdateState(rfb, state, oldstate, statusMsg): RFB state update/change '],
['onPasswordRequired', 'rw', 'func', function() { },
'onPasswordRequired(rfb): VNC password is required '],
['onClipboard', 'rw', 'func', function() { },
'onClipboard(rfb, text): RFB clipboard contents received'],
['onBell', 'rw', 'func', function() { },
'onBell(rfb): RFB Bell message received '],
['onFBUReceive', 'rw', 'func', function() { },
'onFBUReceive(rfb, fbu): RFB FBU received but not yet processed '],
['onFBUComplete', 'rw', 'func', function() { },
'onFBUComplete(rfb, fbu): RFB FBU received and processed '],
['onFBResize', 'rw', 'func', function() { },
'onFBResize(rfb, width, height): frame buffer resized'],
// These callback names are deprecated
['updateState', 'rw', 'func', function() { },
'obsolete, use onUpdateState'],
['clipboardReceive', 'rw', 'func', function() { },
'obsolete, use onClipboard']
]);
// Override/add some specific configuration getters/setters
that.set_local_cursor = function(cursor) {
if ((!cursor) || (cursor in {'0':1, 'no':1, 'false':1})) {
conf.local_cursor = false;
} else {
if (display.get_cursor_uri()) {
conf.local_cursor = true;
} else {
Util.Warn("Browser does not support local cursor");
}
}
};
// These are fake configuration getters
that.get_display = function() { return display; };
that.get_keyboard = function() { return keyboard; };
that.get_mouse = function() { return mouse; };
//
// Setup routines
//
// Create the public API interface and initialize values that stay
// constant across connect/disconnect
function constructor() {
var i, rmode;
Util.Debug(">> RFB.constructor");
// Initialize display, mouse, keyboard, and websock
try {
display = new Display({'target': conf.target});
} catch (exc) {
Util.Error("Display exception: " + exc);
updateState('fatal', "No working Display");
}
keyboard = new Keyboard({'target': conf.focusContainer,
'onKeyPress': keyPress});
mouse = new Mouse({'target': conf.target,
'onMouseButton': mouseButton,
'onMouseMove': mouseMove});
rmode = display.get_render_mode();
ws = new Websock();
ws.on('message', handle_message);
ws.on('open', function() {
Util.Info("connected")
if (connState === "connect") {
ws.send(helloPacket());
updateState('hello', "Starting Xpra handshake");
} else {
fail("Got unexpected WebSockets connection");
}
});
ws.on('close', function(e) {
Util.Warn("WebSocket on-close event");
var msg = "";
if (e.code) {
msg = " (code: " + e.code;
if (e.reason) {
msg += ", reason: " + e.reason;
}
msg += ")";
}
if (connState === 'disconnect') {
updateState('disconnected', 'VNC disconnected' + msg);
} else if (connState === 'ProtocolVersion') {
fail('Failed to connect to server' + msg);
} else if (connState in {'failed':1, 'disconnected':1}) {
Util.Error("Received onclose while disconnected" + msg);
} else {
fail('Server disconnected' + msg);
}
});
ws.on('error', function(e) {
Util.Warn("WebSocket on-error event");
//fail("WebSock reported an error");
});
init_vars();
/* Check web-socket-js if no builtin WebSocket support */
if (Websock_native) {
Util.Info("Using native WebSockets");
updateState('loaded', 'noVNC ready: native WebSockets, ' + rmode);
} else {
Util.Warn("Using web-socket-js bridge. Flash version: " +
Util.Flash.version);
if ((! Util.Flash) ||
(Util.Flash.version < 9)) {
updateState('fatal', "WebSockets or <a href='http://get.adobe.com/flashplayer'>Adobe Flash<\/a> is required");
} else if (document.location.href.substr(0, 7) === "file://") {
updateState('fatal',
"'file://' URL is incompatible with Adobe Flash");
} else {
updateState('loaded', 'noVNC ready: WebSockets emulation, ' + rmode);
}
}
Util.Debug("<< RFB.constructor");
return that; // Return the public API interface
}
function connect() {
Util.Debug(">> RFB.connect");
var uri;
if (typeof UsingSocketIO !== "undefined") {
uri = "http://" + connHost + ":" + connPort + "/" + connPath;
} else {
if (conf.encrypt) {
uri = "wss://";
} else {
uri = "ws://";
}
uri += connHost + ":" + connPort + "/" + connPath;
}
Util.Info("connecting to " + uri);
// TODO: make protocols a configurable
ws.open(uri, ['binary', 'base64']);
Util.Debug("<< RFB.connect");
}
// Initialize variables that are reset before each connection
init_vars = function() {
var i;
/* Reset state */
ws.init();
mouse_buttonMask = 0;
mouse_arr = [];
zlib = new TINF();
zlib.init();
var keynames = {
32:'space', 33:'exclam', 35:'numbersign', 36:'dollar',
37:'percent', 38:'ampersand', 40:'parenleft', 41:'parenright',
42:'asterisk', 43:'plus', 45:'minus', 61:'equal',
94:'asciicircum', 95:'underscore',
96:'grave', 126:'asciitilde' };
// Symbols are wrong
for (i=32; i < 127; i++) {
var keyname = keynames[i] ? keynames[i] : String.fromCharCode(i);
keycodes[i] = [i, keyname, i, 0, 0];
}
// Some whacky ones
keycodes[64] = [34, 'at', 34, 0, 0];
keycodes[126] = [126, 'asciitilde', 49, 0, 0];
// Special codes
keycodes[65505] = [65505, 'Shift_L', 65505, 0, 0];
keycodes[65506] = [65506, 'Shift_R', 65506, 0, 0];
keycodes[65507] = [65507, 'Control_L', 65507, 0, 0];
keycodes[65508] = [65508, 'Control_R', 65508, 0, 0];
keycodes[65513] = [65513, 'Alt_L', 65513, 0, 0];
keycodes[65514] = [65514, 'Alt_R', 65514, 0, 0];
keycodes[65288] = [65288, 'BackSpace', 65288, 0, 0];
keycodes[65289] = [65289, 'Tab', 65289, 0, 0];
keycodes[65293] = [65293, 'Return', 65293, 0, 0];
};
// Print statistics
print_stats = function() {
var i, s;
Util.Info("Encoding stats for this connection:");
};
//
// Utility routines
//
/*
* Page states:
* loaded - page load, equivalent to disconnected
* disconnected - idle state
* connect - starting to connect (to ProtocolVersion)
* normal - connected
* disconnect - starting to disconnect
* failed - abnormal disconnect
* fatal - failed to load page, or fatal error
*
* Xpra protocol initialization states:
* hello
*/
updateState = function(state, statusMsg) {
var func, cmsg, oldstate = connState;
if (state === oldstate) {
/* Already here, ignore */
Util.Debug("Already in state '" + state + "', ignoring.");
return;
}
/*
* These are disconnected states. A previous connect may
* asynchronously cause a connection so make sure we are closed.
*/
if (state in {'disconnected':1, 'loaded':1, 'connect':1,
'disconnect':1, 'failed':1, 'fatal':1}) {
if (sendTimer) {
clearInterval(sendTimer);
sendTimer = null;
}
if (msgTimer) {
clearInterval(msgTimer);
msgTimer = null;
}
if (display && display.get_context()) {
keyboard.ungrab();
mouse.ungrab();
display.defaultCursor();
if ((Util.get_logging() !== 'debug') ||
(state === 'loaded')) {
// Show noVNC logo on load and when disconnected if
// debug is off
display.clear();
}
}
ws.close();
}
if (oldstate === 'fatal') {
Util.Error("Fatal error, cannot continue");
}
if ((state === 'failed') || (state === 'fatal')) {
func = Util.Error;
} else {
func = Util.Warn;
}
cmsg = typeof(statusMsg) !== 'undefined' ? (" Msg: " + statusMsg) : "";
func("New state '" + state + "', was '" + oldstate + "'." + cmsg);
if ((oldstate === 'failed') && (state === 'disconnected')) {
// Do disconnect action, but stay in failed state
connState = 'failed';
} else {
connState = state;
}
if (connTimer && (connState !== 'connect')) {
Util.Debug("Clearing connect timer");
clearInterval(connTimer);
connTimer = null;
}
if (disconnTimer && (connState !== 'disconnect')) {
Util.Debug("Clearing disconnect timer");
clearInterval(disconnTimer);
disconnTimer = null;
}
switch (state) {
case 'normal':
if ((oldstate === 'disconnected') || (oldstate === 'failed')) {
Util.Error("Invalid transition from 'disconnected' or 'failed' to 'normal'");
}
break;
case 'connect':
connTimer = setTimeout(function () {
fail("Connect timeout");
}, conf.connectTimeout * 1000);
init_vars();
connect();
// WebSocket.onopen transitions to 'ProtocolVersion'
break;
case 'disconnect':
if (! test_mode) {
disconnTimer = setTimeout(function () {
fail("Disconnect timeout");
}, conf.disconnectTimeout * 1000);
}
print_stats();
// WebSocket.onclose transitions to 'disconnected'
break;
case 'failed':
if (oldstate === 'disconnected') {
Util.Error("Invalid transition from 'disconnected' to 'failed'");
}
if (oldstate === 'normal') {
Util.Error("Error while connected.");
}
if (oldstate === 'init') {
Util.Error("Error while initializing.");
}
// Make sure we transition to disconnected
setTimeout(function() { updateState('disconnected'); }, 50);
break;
default:
// No state change action to take
}
if ((oldstate === 'failed') && (state === 'disconnected')) {
// Leave the failed message
conf.updateState(that, state, oldstate); // Obsolete
conf.onUpdateState(that, state, oldstate);
} else {
conf.updateState(that, state, oldstate, statusMsg); // Obsolete
conf.onUpdateState(that, state, oldstate, statusMsg);
}
};
fail = function(msg) {
updateState('failed', msg);
return false;
};
handle_message = function() {
//Util.Debug(">> handle_message ws.rQlen(): " + ws.rQlen());
//Util.Debug("ws.rQslice(0,20): " + ws.rQslice(0,20) + " (" + ws.rQlen() + ")");
if (cur_packet_recv_time === 0) {
cur_packet_recv_time = (new Date()).getTime();
}
var packet = decode_packet(ws.get_rQ(), ws.get_rQi());
if (packet.length > 0) {
// Remove the processed data from the queue
ws.set_rQi(ws.get_rQi() + packet.length);
}
if (packet.data) {
// full packet has been received, so process it
var ptype = packet.data[0].replace(/-/, "_"),
handler = packetHandlers[ptype];
if (handler) {
handler(packet.data);
} else {
Util.Warn("no handler defined for packet type '" + ptype + "'");
}
// we got a whole packet so reset the packet received timestamp
cur_packet_recv_time = 0;
}
if (ws.rQlen() >= 8) {
// if we have at least 8 bytes remaining, then requeue
// ourselves, but use setTimeout to give other events a chance
// to run
if (msgTimer === null) {
Util.Debug("More data to process, creating timer");
msgTimer = setTimeout(function () {
msgTimer = null;
handle_message();
}, 1);
} else {
Util.Debug("More data to process, existing timer");
}
}
};
function flushClient() {
if (send_packets.length > 0) {
for (var i=0; i < send_packets.length; i++) {
ws.send(send_packets[i]);
}
send_packets = [];
return true;
} else {
return false;
}
}
// overridable for testing
checkEvents = function() {
var now;
if (connState === 'normal' && !viewportDragging) {
flushClient();
}
setTimeout(checkEvents, conf.check_rate);
};
keyPress = function(keysym, down, evt) {
var arr, packet;
if (conf.view_only) { return; } // View only, skip keyboard events
packet = keyActionPacket(1, keysym, down, evt)
if (!packet) { return false; }
send_packets.push(packet);
flushClient();
};
mouseButton = function(x, y, down, bmask) {
if (down) {
mouse_buttonMask |= bmask;
} else {
mouse_buttonMask ^= bmask;
}
if (conf.viewportDrag) {
if (down && !viewportDragging) {
viewportDragging = true;
viewportDragPos = {'x': x, 'y': y};
// Skip sending mouse events
return;
} else {
viewportDragging = false;
ws.send(fbUpdateRequests()); // Force immediate redraw
}
}
if (conf.view_only) { return; } // View only, skip mouse events
var button = 1; // TODO: translate bmask to button #
send_packets.push(pointerActionPacket(
1, button, down, display.absX(x), display.absY(y)));
flushClient();
};
mouseMove = function(x, y) {
Util.Debug('>> mouseMove ' + x + "," + y);
var deltaX, deltaY;
if (viewportDragging) {
//deltaX = x - viewportDragPos.x; // drag viewport
deltaX = viewportDragPos.x - x; // drag frame buffer
//deltaY = y - viewportDragPos.y; // drag viewport
deltaY = viewportDragPos.y - y; // drag frame buffer
viewportDragPos = {'x': x, 'y': y};
display.viewportChange(deltaX, deltaY);
// Skip sending mouse events
return;
}
if (conf.view_only) { return; } // View only, skip mouse events
send_packets.push(pointerPositionPacket(
1, display.absX(x), display.absY(y)));
};
//
// Xpra protocol routines
//
function bencode(data) {
switch (typeof(data)) {
case "number":
return "i" + data + "e";
break;
case "string":
return data.length + ":" + data;
break;
case "object":
if (Array.isArray(data)) {
var res = "l";
for (var i = 0; i < data.length; i++) {
res += bencode(data[i]);
}
return res + "e";
} else {
var res = "d",
keys = Object.keys(data).sort();
for (var i = 0; i < keys.length; i++) {
var k = keys[i],
v = data[k];
res += bencode(k);
res += bencode(v);
}
return res + "e";
}
break;
default:
throw("Unknown encode_data type: " + typeof(data));
}
}
function bdecode(raw, f) {
var ret, res;
f = (typeof(f) === "undefined") ? 0 : f;
switch (raw[f]) {
case 'i':
var end = raw.indexOf('e', f+1);
if (end < 0) {
throw("Error decoding integer value");
}
var num = parseInt(raw.substring(f+1, end), 10);
res = num;
f = end;
break;
case 'l':
res = [];
f += 1;
while (raw[f] !== 'e') {
ret = bdecode(raw, f);
res.push(ret[0]);
f = ret[1];
}
break;
case 'd':
var k, lastkey = null;
res = {};
f += 1;
while (raw[f] !== 'e') {
ret = bdecode(raw, f);
k = ret[0];
f = ret[1];
if (lastkey !== null && lastkey >= k) {
throw("Unsorted keys found while decoding dict type");
}
lastkey = k;
ret = bdecode(raw, f);
res[k] = ret[0];
f = ret[1];
}
break;
default:
if (!raw[f].match(/^[0-9]/)) {
throw("Unknown decoding type: " + raw[f]);
}
var end = raw.indexOf(':', f+1);
if (end < 0) {
throw("Error decoding string value");
}
var len = parseInt(raw.substring(f, end), 10);
res = raw.substr(end+1, len);
f = end+len;
}
return [res, f+1];
}
/*
function encode_packet(data) {
var encoded_data = bencode(data),
payload_size = encoded_data.length,
packet_size = 8 + payload_size + (4-payload_size%4),
packet8 = new Uint8Array(packet_size),
payload8 = new Uint8Array(packet8.buffer, 8),
packet32 = new Uint32Array(packet8.buffer, 0);
packet8[0] = 80; // 'P'.charCodeAt(0)
packet8[1] = 0; // protocol flags
packet8[2] = 0; // compression level
packet8[3] = 0; // packet index
packet32[1] = payload_size; // packet payload size
for (var i=0; i < payload_size; i++) {
payload8[i] = encoded_data.charCodeAt(i);
}
console.log("packet8: ", packet8, ", payload: ", encoded_data);
return packet8;
}
*/
function encode_packet(data) {
//console.log("send packet data:", data);
var encoded_data = bencode(data),
payload_size = encoded_data.length,
packet = [];
packet.push8(80); // 'P'.charCodeAt(0)
packet.push8(0); // protocol flags
packet.push8(0); // compression level
packet.push8(0); // packet index
packet.push32(payload_size); // packet payload size
// Convert to array
// TODO: remmove this step
for (var i=0; i < payload_size; i++) {
packet.push8(encoded_data.charCodeAt(i));
}
return packet;
}
var inflate = function(data, offset) {
zlib.reset();
var inflated = zlib.uncompress(data, offset);
if (inflated.status !== 0) {
throw("Invalid data in zlib stream");
}
return inflated.data;
};
function decode_packet(arr, idx) {
var packet = {};
idx = (typeof(idx) === "undefined") ? 0 : idx;
if (arr.length - idx < 8) {
return {'length': 0};
}
packet.magic = arr[idx];
packet.flags = arr[idx+1];
packet.level = arr[idx+2];
packet.index = arr[idx+3];
packet.size = (arr[idx+4] << 24) +
(arr[idx+5] << 16) +
(arr[idx+6] << 8) +
(arr[idx+7]);
packet.length = 8 + packet.size;
packet.data = null;
if (arr.length - idx < packet.length) {
return {'length': 0};
}
var new_arr, offset = idx + 8;
if (packet.level > 0) {
new_arr = inflate(arr, offset);
} else {
// Convert to string
// TODO: remove this step
new_arr = arr.slice(offset, offset + packet.size);
}
var str = String.fromCharCode.apply(null, new_arr);
if (packet.index > 0) {
raw_packets[packet.index] = str;
// packet.data stays null indicating incomplete packet
} else {
packet.data = bdecode(str, 0)[0];
// Insert any raw packets into place
for (var idx in raw_packets) {
packet.data[idx] = raw_packets[idx];
}
raw_packets = {};
}
return packet;
}
//
// Client packet generation routines
//
function helloPacket () {
var flat_keycodes = [],
capabilities;
for(var code in keycodes) {
flat_keycodes.push(keycodes[code]);
}
capabilities = {'encoding': 'png',
'encodings': ['png', 'jpeg'],
'version': '0.9.0',
'xkbmap_keycodes': flat_keycodes};
return encode_packet(['hello', capabilities]);
}
function keyActionPacket(wid, keysym, down, evt) {
var keydata = keycodes[keysym],
keyname = "",
modifiers = [],
keyval = 0,
str = "", // not used
keycode = 0,
group = 0, // not used
is_modifier = 0; // not used
if (!keydata) {
Util.Warn("ignoring undefined keysym " + keysym + " full event: " + evt);
return false;
}
keyname = keydata[1];
keyval = keydata[0];
keycode = keydata[2];
if (evt.shiftKey) {
modifiers.push('shift');
}
if (evt.ctrlKey) {
modifiers.push('control');
}
if (evt.altKey) {
modifiers.push('alt');
}
Util.Info("send key-action for window " + wid + ", keyname: " + keyname + ", keyval: " + keyval + ", modifiers: " + modifiers);
return encode_packet(['key-action', wid, keyname, down,
modifiers, keyval, str, keycode, group, is_modifier]);
}
function pointerPositionPacket(wid, x, y) {
var modifiers = [],
buttons = [],
rx = windows[wid].x + x,
ry = windows[wid].y + y;
return encode_packet(['pointer-position', wid,
[rx, ry], modifiers, buttons]);
}
function pointerActionPacket(wid, button, down, x, y) {
var modifiers = [],
buttons = [],
rx = windows[wid].x + x,
ry = windows[wid].y + y;
return encode_packet(['button-action', wid, button, down,
[rx, ry], modifiers, buttons]);
}
function pingPacket() {
var now_ms = (new Date()).getTime();
Util.Info("send ping now_ms: " + now_ms);
return encode_packet(['ping', now_ms]);
}
//
//
//
function send_ping () {
ws.send(pingPacket());
if (connState === 'normal') {
setTimeout(send_ping, conf.ping_rate);
}
}
//
// Server packet receive handlers
//
packetHandlers.hello = function process_hello (data) {
Util.Info("got hello: " + data);
ws.send(encode_packet(["set_deflate", 0]));
updateState('normal', "Connected");
/* Start pushing/polling */
setTimeout(checkEvents, conf.check_rate);
/* Start sending pings to the server */
send_ping();
};
packetHandlers.new_window = function process_new_window (data) {
Util.Info("got new-window: " + data);
var wid = data[1],
x = data[2],
y = data[3],
w = data[4],
h = data[5],
props = data[6],
loc = data[7];
if (wid !== 1) {
// TODO: don't ignore other windows
Util.Warn("ignoring new-window for window ID " + wid);
return;
}
if (w !== fb_width || h !== fb_height) {
fb_width = w;
fb_height = h;
conf.onFBResize(that, fb_width, fb_height);
display.resize(fb_width, fb_height);
timing.fbu_rt_start = (new Date()).getTime();
}
windows[wid] = {x: x, y: y, w: w, h: h, props: props, loc: loc};
//ws.send(encode_packet(["configure-window", wid, x, y, w, h, loc]));
ws.send(encode_packet(["map-window", wid, x, y, w, h, loc]));
ws.send(encode_packet(["focus", wid]));
display.resize(fb_width, fb_height);
keyboard.grab();
mouse.grab();
};
packetHandlers.ping_echo = function process_ping_echo (data) {
Util.Info("got ping_echo: " + data);
var echoedtime = data[1],
l1 = data[2],
l2 = data[3],
l3 = data[4],
cl = data[5],
now_ms = (new Date()).getTime(),
sl = -1;
stats.last_ping_echoed_time = echoedtime;
sl = now_ms - echoedtime;
stats.server_ping_latency.push([now_ms, sl]);
// Keep 100 entries
if (stats.server_ping_latency.length > 100) {
stats.server_ping_latency.splice(0, stats.server_ping_latency.length-100);
}
if (cl >= 0) {
stats.client_ping_latency.push([now_ms, cl]);
// Keep 100 entries
if (stats.client_ping_latency.length > 100) {
stats.client_ping_latency.splice(0, stats.client_ping_latency.length-100);
}
}
};
packetHandlers.ping = function process_ping (data) {
Util.Info("got ping: " + data);
var echotime = data[1],
l1 = 500, l2 = 500, l3 = 500, // fake load-averages
sl = -1,
sl_len = stats.server_ping_latency.length;
if (sl_len > 0) {
sl = stats.server_ping_latency[sl_len-1][1];
}
Util.Info("send ping_echo sl: " + sl);
ws.send(encode_packet(["ping_echo", echotime,
l1, l2, l3, sl]));
};
packetHandlers.draw = function process_draw (data) {
Util.Info("got draw #" + data[8] + " for window " + data[1] + ": " + data.slice(2,7));
var wid = data[1],
x = data[2],
y = data[3],
w = data[4],
h = data[5],
coding = data[6],
raw = data[7],
damage_seq = data[8],
rowstride = data[9],
client_opts = data[10],
decode_time,
img;
if (wid !== 1) {
// TODO: handle other windows
Util.Warn("ignoring draw for window ID " + wid);
return;
}
img = new Image();
img.src = "data:image/" + coding + ";base64," + window.btoa(raw);
display.renderQ_push({
'type': 'img',
'img': img,
'x': x,
'y': y});
img = null;
// based on _do_draw, draw_region, do_draw_region, paint_png, etc
decode_time = ((new Date()).getTime() - cur_packet_recv_time)*1000;
Util.Info("send damage-sequence #" + damage_seq + " for window " + wid + ", w: " + w + ", h: " + h + ", decode_time: " + decode_time);
ws.send(encode_packet(['damage-sequence', damage_seq,
wid, w, h, decode_time]));
};
//
// Public API interface functions
//
that.connect = function(host, port, password, path) {
//Util.Debug(">> connect");
connHost = host;
connPort = port;
connPassword = (password !== undefined) ? password : "";
connPath = (path !== undefined) ? path : "";
if ((!connHost) || (!connPort)) {
return fail("Must set host and port");
}
updateState('connect');
//Util.Debug("<< connect");
};
that.disconnect = function() {
//Util.Debug(">> disconnect");
updateState('disconnect', 'Disconnecting');
//Util.Debug("<< disconnect");
};
// Override internal functions for testing
that.testMode = function(override_send, data_mode) {
test_mode = true;
that.recv_message = ws.testMode(override_send, data_mode);
// Allow debug calls to this
that.encode_packet = encode_packet;
that.decode_packet = decode_packet;
that.bdecode = bdecode;
that.bencode = bencode;
checkEvents = function () { /* Stub Out */ };
that.connect = function(host, port, password) {
connHost = host;
connPort = port;
connPassword = password;
init_vars();
updateState('ProtocolVersion', "Starting VNC handshake");
};
};
return constructor(); // Return the public API interface
} // End of Xpra()
...@@ -181,6 +181,10 @@ ...@@ -181,6 +181,10 @@
<script> <script>
Util.load_scripts(["webutil.js", "base64.js", "websock.js", "des.js", Util.load_scripts(["webutil.js", "base64.js", "websock.js", "des.js",
"input.js", "display.js", "jsunzip.js", "rfb.js", "ui.js"]); "input.js", "display.js", "jsunzip.js", "rfb.js", "ui.js"]);
window.onscriptsload = function () {
UI.Client = RFB;
UI.load();
};
</script> </script>
</body> </body>
</html> </html>
<!DOCTYPE html>
<html>
<head>
<!--
noVNC example: simple example using default UI
Copyright (C) 2012 Joel Martin
noVNC is licensed under the MPL 2.0 (see LICENSE.txt)
This file is licensed under the 2-Clause BSD license (see LICENSE.txt).
Connect parameters are provided in query string:
http://example.com/?host=HOST&port=PORT&encrypt=1&true_color=1
-->
<title>noVNC</title>
<meta charset="utf-8">
<!-- Always force latest IE rendering engine (even in intranet) & Chrome Frame
Remove this if you use the .htaccess -->
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
<!-- Apple iOS Safari settings -->
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no">
<meta name="apple-mobile-web-app-capable" content="yes" />
<meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" />
<!-- App Start Icon -->
<link rel="apple-touch-startup-image" href="images/screen_320x460.png" />
<!-- For iOS devices set the icon to use if user bookmarks app on their homescreen -->
<link rel="apple-touch-icon" href="images/screen_57x57.png">
<!--
<link rel="apple-touch-icon-precomposed" href="images/screen_57x57.png" />
-->
<!-- Stylesheets -->
<link rel="stylesheet" href="include/base.css" />
<link rel="alternate stylesheet" href="include/black.css" TITLE="Black" />
<link rel="alternate stylesheet" href="include/blue.css" TITLE="Blue" />
<!--
<script type='text/javascript'
src='http://getfirebug.com/releases/lite/1.2/firebug-lite-compressed.js'></script>
-->
</head>
<body>
<div id="noVNC-control-bar">
<div id="noVNC-menu-bar" style="display:none;">
</div>
<!--noVNC Mobile Device only Buttons-->
<div class="noVNC-buttons-left">
<input type="image" src="images/drag.png"
id="noVNC_view_drag_button" class="noVNC_status_button"
title="Move/Drag Viewport">
<div id="noVNC_mobile_buttons">
<input type="image" src="images/mouse_none.png"
id="noVNC_mouse_button0" class="noVNC_status_button">
<input type="image" src="images/mouse_left.png"
id="noVNC_mouse_button1" class="noVNC_status_button">
<input type="image" src="images/mouse_middle.png"
id="noVNC_mouse_button2" class="noVNC_status_button">
<input type="image" src="images/mouse_right.png"
id="noVNC_mouse_button4" class="noVNC_status_button">
<input type="image" src="images/keyboard.png"
id="showKeyboard" class="noVNC_status_button"
value="Keyboard" title="Show Keyboard"/>
<input type="email"
autocapitalize="off" autocorrect="off"
id="keyboardinput" class="noVNC_status_button"/>
</div>
</div>
<!--noVNC Buttons-->
<div class="noVNC-buttons-right">
<input type="image" src="images/ctrlaltdel.png"
id="sendCtrlAltDelButton" class="noVNC_status_button"
title="Send Ctrl-Alt-Del" />
<input type="image" src="images/clipboard.png"
id="clipboardButton" class="noVNC_status_button"
title="Clipboard" />
<input type="image" src="images/settings.png"
id="settingsButton" class="noVNC_status_button"
title="Settings" />
<input type="image" src="images/connect.png"
id="connectButton" class="noVNC_status_button"
title="Connect" />
<input type="image" src="images/disconnect.png"
id="disconnectButton" class="noVNC_status_button"
title="Disconnect" />
</div>
<!-- Description Panel -->
<!-- Shown by default when hosted at for kanaka.github.com -->
<div id="noVNC_description" style="display:none;" class="">
noVNC is a browser based VNC client implemented using HTML5 Canvas
and WebSockets. You will either need a VNC server with WebSockets
support (such as <a href="http://libvncserver.sourceforge.net/">libvncserver</a>)
or you will need to use
<a href="https://github.com/kanaka/websockify">websockify</a>
to bridge between your browser and VNC server. See the noVNC
<a href="https://github.com/kanaka/noVNC">README</a>
and <a href="http://kanaka.github.com/noVNC">website</a>
for more information.
<br />
<input id="descriptionButton" type="button" value="Close">
</div>
<!-- Clipboard Panel -->
<div id="noVNC_clipboard" class="triangle-right top">
<textarea id="noVNC_clipboard_text" rows=5>
</textarea>
<br />
<input id="noVNC_clipboard_clear_button" type="button"
value="Clear">
</div>
<!-- Settings Panel -->
<div id="noVNC_settings" class="triangle-right top">
<span id="noVNC_settings_menu">
<ul>
<li><input id="noVNC_encrypt" type="checkbox"> Encrypt</li>
<li><input id="noVNC_true_color" type="checkbox" checked> True Color</li>
<li><input id="noVNC_cursor" type="checkbox"> Local Cursor</li>
<li><input id="noVNC_clip" type="checkbox"> Clip to Window</li>
<li><input id="noVNC_shared" type="checkbox"> Shared Mode</li>
<li><input id="noVNC_view_only" type="checkbox"> View Only</li>
<li><input id="noVNC_connectTimeout" type="input"> Connect Timeout (s)</li>
<li><input id="noVNC_path" type="input" value="websockify"> Path</li>
<li><input id="noVNC_repeaterID" type="input" value=""> Repeater ID</li>
<hr>
<!-- Stylesheet selection dropdown -->
<li><label><strong>Style: </strong>
<select id="noVNC_stylesheet" name="vncStyle">
<option value="default">default</option>
</select></label>
</li>
<!-- Logging selection dropdown -->
<li><label><strong>Logging: </strong>
<select id="noVNC_logging" name="vncLogging">
</select></label>
</li>
<hr>
<li><input type="button" id="noVNC_apply" value="Apply"></li>
</ul>
</span>
</div>
<!-- Connection Panel -->
<div id="noVNC_controls" class="triangle-right top">
<ul>
<li><label><strong>Host: </strong><input id="noVNC_host" /></label></li>
<li><label><strong>Port: </strong><input id="noVNC_port" /></label></li>
<li><label><strong>Password: </strong><input id="noVNC_password" type="password" /></label></li>
<li><input id="noVNC_connect_button" type="button" value="Connect"></li>
</ul>
</div>
</div> <!-- End of noVNC-control-bar -->
<div id="noVNC_screen">
<div id="noVNC_screen_pad"></div>
<div id="noVNC_status_bar" class="noVNC_status_bar">
<div id="noVNC_status">Loading</div>
</div>
<h1 id="noVNC_logo"><span>no</span><br />VNC</h1>
<!-- HTML5 Canvas -->
<div id="noVNC_container">
<canvas id="noVNC_canvas" width="640px" height="20px">
Canvas not supported.
</canvas>
</div>
</div>
<script src="include/util.js"></script>
<script>
Util.load_scripts(["webutil.js", "base64.js", "websock.js", "des.js",
"input.js", "display.js", "jsunzip.js", "xpra.js", "ui.js"]);
window.onscriptsload = function () {
UI.Client = Xpra;
UI.load();
};
</script>
</body>
</html>
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment