mirror of
https://github.com/gyurix1968/guacamole-client.git
synced 2025-09-07 13:41:21 +00:00
Extracted tunnel, mouse, and keyboard handling from GuacamoleClient
This commit is contained in:
@@ -14,14 +14,9 @@
|
|||||||
* GNU Affero General Public License for more details.
|
* GNU Affero General Public License for more details.
|
||||||
*
|
*
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
function GuacamoleClient(display, tunnelURL) {
|
function GuacamoleClient(display, tunnel) {
|
||||||
|
|
||||||
var TUNNEL_CONNECT = tunnelURL + "?connect";
|
|
||||||
var TUNNEL_READ = tunnelURL + "?read";
|
|
||||||
var TUNNEL_WRITE = tunnelURL + "?write";
|
|
||||||
|
|
||||||
var STATE_IDLE = 0;
|
var STATE_IDLE = 0;
|
||||||
var STATE_CONNECTING = 1;
|
var STATE_CONNECTING = 1;
|
||||||
@@ -32,7 +27,8 @@ function GuacamoleClient(display, tunnelURL) {
|
|||||||
|
|
||||||
var currentState = STATE_IDLE;
|
var currentState = STATE_IDLE;
|
||||||
var stateChangeHandler = null;
|
var stateChangeHandler = null;
|
||||||
var pollResponse = 1; // Default to polling - will be turned off automatically if not needed
|
|
||||||
|
tunnel.setInstructionHandler(doInstruction);
|
||||||
|
|
||||||
// Display must be relatively positioned for mouse to be handled properly
|
// Display must be relatively positioned for mouse to be handled properly
|
||||||
display.style.position = "relative";
|
display.style.position = "relative";
|
||||||
@@ -66,7 +62,7 @@ function GuacamoleClient(display, tunnelURL) {
|
|||||||
|
|
||||||
var cursorHidden = 0;
|
var cursorHidden = 0;
|
||||||
|
|
||||||
function redrawCursor() {
|
function redrawCursor(x, y) {
|
||||||
|
|
||||||
// Hide hardware cursor
|
// Hide hardware cursor
|
||||||
if (cursorHidden == 0) {
|
if (cursorHidden == 0) {
|
||||||
@@ -78,8 +74,8 @@ function GuacamoleClient(display, tunnelURL) {
|
|||||||
cursor.clearRect(cursorRectX, cursorRectY, cursorRectW, cursorRectH);
|
cursor.clearRect(cursorRectX, cursorRectY, cursorRectW, cursorRectH);
|
||||||
|
|
||||||
// Update rect
|
// Update rect
|
||||||
cursorRectX = mouse.getX() - cursorHotspotX;
|
cursorRectX = x - cursorHotspotX;
|
||||||
cursorRectY = mouse.getY() - cursorHotspotY;
|
cursorRectY = y - cursorHotspotY;
|
||||||
cursorRectW = cursorImage.width;
|
cursorRectW = cursorImage.width;
|
||||||
cursorRectH = cursorImage.height;
|
cursorRectH = cursorImage.height;
|
||||||
|
|
||||||
@@ -87,90 +83,28 @@ function GuacamoleClient(display, tunnelURL) {
|
|||||||
cursor.drawImage(cursorRectX, cursorRectY, cursorImage);
|
cursor.drawImage(cursorRectX, cursorRectY, cursorImage);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.sendKeyEvent = function(pressed, keysym) {
|
||||||
|
|
||||||
|
|
||||||
/*****************************************/
|
|
||||||
/*** Keyboard ***/
|
|
||||||
/*****************************************/
|
|
||||||
|
|
||||||
var keyboard = new GuacamoleKeyboard(document);
|
|
||||||
|
|
||||||
this.disableKeyboard = function() {
|
|
||||||
keyboard.setKeyPressedHandler(null);
|
|
||||||
keyboard.setKeyReleasedHandler(null);
|
|
||||||
};
|
|
||||||
|
|
||||||
this.enableKeyboard = function() {
|
|
||||||
keyboard.setKeyPressedHandler(
|
|
||||||
function (keysym) {
|
|
||||||
sendKeyEvent(1, keysym);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
keyboard.setKeyReleasedHandler(
|
|
||||||
function (keysym) {
|
|
||||||
sendKeyEvent(0, keysym);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
// Enable keyboard by default
|
|
||||||
this.enableKeyboard();
|
|
||||||
|
|
||||||
function sendKeyEvent(pressed, keysym) {
|
|
||||||
// Do not send requests if not connected
|
// Do not send requests if not connected
|
||||||
if (!isConnected())
|
if (!isConnected())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
sendMessage("key:" + keysym + "," + pressed + ";");
|
tunnel.sendMessage("key:" + keysym + "," + pressed + ";");
|
||||||
}
|
}
|
||||||
|
|
||||||
this.pressKey = function(keysym) {
|
this.sendMouseState = function(mouseState) {
|
||||||
sendKeyEvent(1, keysym);
|
|
||||||
};
|
|
||||||
|
|
||||||
this.releaseKey = function(keysym) {
|
|
||||||
sendKeyEvent(0, keysym);
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
/*****************************************/
|
|
||||||
/*** Mouse ***/
|
|
||||||
/*****************************************/
|
|
||||||
|
|
||||||
var mouse = new GuacamoleMouse(display);
|
|
||||||
mouse.setButtonPressedHandler(
|
|
||||||
function(mouseState) {
|
|
||||||
sendMouseState(mouseState);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
mouse.setButtonReleasedHandler(
|
|
||||||
function(mouseState) {
|
|
||||||
sendMouseState(mouseState);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
mouse.setMovementHandler(
|
|
||||||
function(mouseState) {
|
|
||||||
|
|
||||||
// Draw client-side cursor
|
|
||||||
if (cursorImage != null) {
|
|
||||||
redrawCursor();
|
|
||||||
}
|
|
||||||
|
|
||||||
sendMouseState(mouseState);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
function sendMouseState(mouseState) {
|
|
||||||
|
|
||||||
// Do not send requests if not connected
|
// Do not send requests if not connected
|
||||||
if (!isConnected())
|
if (!isConnected())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
// Draw client-side cursor
|
||||||
|
if (cursorImage != null) {
|
||||||
|
redrawCursor(
|
||||||
|
mouseState.getX(),
|
||||||
|
mouseState.getY()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
// Build mask
|
// Build mask
|
||||||
var buttonMask = 0;
|
var buttonMask = 0;
|
||||||
if (mouseState.getLeft()) buttonMask |= 1;
|
if (mouseState.getLeft()) buttonMask |= 1;
|
||||||
@@ -180,80 +114,19 @@ function GuacamoleClient(display, tunnelURL) {
|
|||||||
if (mouseState.getDown()) buttonMask |= 16;
|
if (mouseState.getDown()) buttonMask |= 16;
|
||||||
|
|
||||||
// Send message
|
// Send message
|
||||||
sendMessage("mouse:" + mouseState.getX() + "," + mouseState.getY() + "," + buttonMask + ";");
|
tunnel.sendMessage("mouse:" + mouseState.getX() + "," + mouseState.getY() + "," + buttonMask + ";");
|
||||||
}
|
}
|
||||||
|
|
||||||
var sendingMessages = 0;
|
|
||||||
var outputMessageBuffer = "";
|
|
||||||
|
|
||||||
function sendMessage(message) {
|
|
||||||
|
|
||||||
// Add event to queue, restart send loop if finished.
|
|
||||||
outputMessageBuffer += message;
|
|
||||||
if (sendingMessages == 0)
|
|
||||||
sendPendingMessages();
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
function sendPendingMessages() {
|
|
||||||
|
|
||||||
if (outputMessageBuffer.length > 0) {
|
|
||||||
|
|
||||||
sendingMessages = 1;
|
|
||||||
|
|
||||||
var message_xmlhttprequest = new XMLHttpRequest();
|
|
||||||
message_xmlhttprequest.open("POST", TUNNEL_WRITE);
|
|
||||||
message_xmlhttprequest.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
|
|
||||||
message_xmlhttprequest.setRequestHeader("Content-length", outputMessageBuffer.length);
|
|
||||||
|
|
||||||
// Once response received, send next queued event.
|
|
||||||
message_xmlhttprequest.onreadystatechange = function() {
|
|
||||||
if (message_xmlhttprequest.readyState == 4)
|
|
||||||
sendPendingMessages();
|
|
||||||
}
|
|
||||||
|
|
||||||
message_xmlhttprequest.send(outputMessageBuffer);
|
|
||||||
outputMessageBuffer = ""; // Clear buffer
|
|
||||||
|
|
||||||
}
|
|
||||||
else
|
|
||||||
sendingMessages = 0;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*****************************************/
|
|
||||||
/*** Clipboard ***/
|
|
||||||
/*****************************************/
|
|
||||||
|
|
||||||
this.setClipboard = function(data) {
|
this.setClipboard = function(data) {
|
||||||
|
|
||||||
// Do not send requests if not connected
|
// Do not send requests if not connected
|
||||||
if (!isConnected())
|
if (!isConnected())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
sendMessage("clipboard:" + escapeGuacamoleString(data) + ";");
|
tunnel.sendMessage("clipboard:" + tunnel.escapeGuacamoleString(data) + ";");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Handlers
|
||||||
function desaturateFilter(data, width, height) {
|
|
||||||
|
|
||||||
for (var i=0; i<data.length; i+=4) {
|
|
||||||
|
|
||||||
// Get RGB values
|
|
||||||
var r = data[i];
|
|
||||||
var g = data[i+1];
|
|
||||||
var b = data[i+2];
|
|
||||||
|
|
||||||
// Desaturate
|
|
||||||
var v = Math.max(r, g, b) / 2;
|
|
||||||
data[i] = v;
|
|
||||||
data[i+1] = v;
|
|
||||||
data[i+2] = v;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
var nameHandler = null;
|
var nameHandler = null;
|
||||||
this.setNameHandler = function(handler) {
|
this.setNameHandler = function(handler) {
|
||||||
@@ -265,217 +138,11 @@ function GuacamoleClient(display, tunnelURL) {
|
|||||||
errorHandler = handler;
|
errorHandler = handler;
|
||||||
};
|
};
|
||||||
|
|
||||||
var errorEncountered = 0;
|
|
||||||
function showError(error) {
|
|
||||||
// Only display first error (avoid infinite error loops)
|
|
||||||
if (errorEncountered == 0) {
|
|
||||||
errorEncountered = 1;
|
|
||||||
|
|
||||||
disconnect();
|
|
||||||
|
|
||||||
// Show error by desaturating display
|
|
||||||
for (var i=0; i<layers.length; i++) {
|
|
||||||
layers[i].filter(desaturateFilter);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (errorHandler)
|
|
||||||
errorHandler(error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function handleErrors(message) {
|
|
||||||
var errors = message.getErrors();
|
|
||||||
for (var errorIndex=0; errorIndex<errors.length; errorIndex++)
|
|
||||||
showError(errors[errorIndex].getMessage());
|
|
||||||
}
|
|
||||||
|
|
||||||
var clipboardHandler = null;
|
var clipboardHandler = null;
|
||||||
var requests = 0;
|
|
||||||
|
|
||||||
this.setClipboardHandler = function(handler) {
|
this.setClipboardHandler = function(handler) {
|
||||||
clipboardHandler = handler;
|
clipboardHandler = handler;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
function handleResponse(xmlhttprequest) {
|
|
||||||
|
|
||||||
var interval = null;
|
|
||||||
var nextRequest = null;
|
|
||||||
|
|
||||||
var dataUpdateEvents = 0;
|
|
||||||
var instructionStart = 0;
|
|
||||||
var startIndex = 0;
|
|
||||||
|
|
||||||
function parseResponse() {
|
|
||||||
|
|
||||||
// Start next request as soon as possible
|
|
||||||
if (xmlhttprequest.readyState >= 2 && nextRequest == null)
|
|
||||||
nextRequest = makeRequest();
|
|
||||||
|
|
||||||
// Parse stream when data is received and when complete.
|
|
||||||
if (xmlhttprequest.readyState == 3 ||
|
|
||||||
xmlhttprequest.readyState == 4) {
|
|
||||||
|
|
||||||
// Also poll every 30ms (some browsers don't repeatedly call onreadystatechange for new data)
|
|
||||||
if (pollResponse == 1) {
|
|
||||||
if (xmlhttprequest.readyState == 3 && interval == null)
|
|
||||||
interval = setInterval(parseResponse, 30);
|
|
||||||
else if (xmlhttprequest.readyState == 4 && interval != null)
|
|
||||||
clearInterval(interval);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Halt on error during request
|
|
||||||
if (xmlhttprequest.status == 0) {
|
|
||||||
showError("Request canceled by browser.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
else if (xmlhttprequest.status != 200) {
|
|
||||||
showError("Error during request (HTTP " + xmlhttprequest.status + "): " + xmlhttprequest.statusText);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var current = xmlhttprequest.responseText;
|
|
||||||
var instructionEnd;
|
|
||||||
|
|
||||||
while ((instructionEnd = current.indexOf(";", startIndex)) != -1) {
|
|
||||||
|
|
||||||
// Start next search at next instruction
|
|
||||||
startIndex = instructionEnd+1;
|
|
||||||
|
|
||||||
var instruction = current.substr(instructionStart,
|
|
||||||
instructionEnd - instructionStart);
|
|
||||||
|
|
||||||
instructionStart = startIndex;
|
|
||||||
|
|
||||||
var opcodeEnd = instruction.indexOf(":");
|
|
||||||
|
|
||||||
var opcode;
|
|
||||||
var parameters;
|
|
||||||
if (opcodeEnd == -1) {
|
|
||||||
opcode = instruction;
|
|
||||||
parameters = new Array();
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
opcode = instruction.substr(0, opcodeEnd);
|
|
||||||
parameters = instruction.substr(opcodeEnd+1).split(",");
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we're done parsing, handle the next response.
|
|
||||||
if (opcode.length == 0) {
|
|
||||||
|
|
||||||
if (isConnected()) {
|
|
||||||
delete xmlhttprequest;
|
|
||||||
if (nextRequest)
|
|
||||||
handleResponse(nextRequest);
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Call instruction handler.
|
|
||||||
doInstruction(opcode, parameters);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Start search at end of string.
|
|
||||||
startIndex = current.length;
|
|
||||||
|
|
||||||
delete instruction;
|
|
||||||
delete parameters;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// If response polling enabled, attempt to detect if still
|
|
||||||
// necessary (via wrapping parseResponse())
|
|
||||||
if (pollResponse == 1) {
|
|
||||||
xmlhttprequest.onreadystatechange = function() {
|
|
||||||
|
|
||||||
// If we receive two or more readyState==3 events,
|
|
||||||
// there is no need to poll.
|
|
||||||
if (xmlhttprequest.readyState == 3) {
|
|
||||||
dataUpdateEvents++;
|
|
||||||
if (dataUpdateEvents >= 2) {
|
|
||||||
pollResponse = 0;
|
|
||||||
xmlhttprequest.onreadystatechange = parseResponse;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
parseResponse();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Otherwise, just parse
|
|
||||||
else
|
|
||||||
xmlhttprequest.onreadystatechange = parseResponse;
|
|
||||||
|
|
||||||
parseResponse();
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
function makeRequest() {
|
|
||||||
|
|
||||||
// Download self
|
|
||||||
var xmlhttprequest = new XMLHttpRequest();
|
|
||||||
xmlhttprequest.open("POST", TUNNEL_READ);
|
|
||||||
xmlhttprequest.send(null);
|
|
||||||
|
|
||||||
return xmlhttprequest;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
function escapeGuacamoleString(str) {
|
|
||||||
|
|
||||||
var escapedString = "";
|
|
||||||
|
|
||||||
for (var i=0; i<str.length; i++) {
|
|
||||||
|
|
||||||
var c = str.charAt(i);
|
|
||||||
if (c == ",")
|
|
||||||
escapedString += "\\c";
|
|
||||||
else if (c == ";")
|
|
||||||
escapedString += "\\s";
|
|
||||||
else if (c == "\\")
|
|
||||||
escapedString += "\\\\";
|
|
||||||
else
|
|
||||||
escapedString += c;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
return escapedString;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
function unescapeGuacamoleString(str) {
|
|
||||||
|
|
||||||
var unescapedString = "";
|
|
||||||
|
|
||||||
for (var i=0; i<str.length; i++) {
|
|
||||||
|
|
||||||
var c = str.charAt(i);
|
|
||||||
if (c == "\\" && i<str.length-1) {
|
|
||||||
|
|
||||||
var escapeChar = str.charAt(++i);
|
|
||||||
if (escapeChar == "c")
|
|
||||||
unescapedString += ",";
|
|
||||||
else if (escapeChar == "s")
|
|
||||||
unescapedString += ";";
|
|
||||||
else if (escapeChar == "\\")
|
|
||||||
unescapedString += "\\";
|
|
||||||
else
|
|
||||||
unescapedString += "\\" + escapeChar;
|
|
||||||
|
|
||||||
}
|
|
||||||
else
|
|
||||||
unescapedString += c;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
return unescapedString;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// Layers
|
// Layers
|
||||||
var displayWidth = 0;
|
var displayWidth = 0;
|
||||||
var displayHeight = 0;
|
var displayHeight = 0;
|
||||||
@@ -484,6 +151,10 @@ function GuacamoleClient(display, tunnelURL) {
|
|||||||
var buffers = new Array();
|
var buffers = new Array();
|
||||||
var cursor = null;
|
var cursor = null;
|
||||||
|
|
||||||
|
this.getLayers = function() {
|
||||||
|
return layers;
|
||||||
|
}
|
||||||
|
|
||||||
function getLayer(index) {
|
function getLayer(index) {
|
||||||
|
|
||||||
// If negative index, use buffer
|
// If negative index, use buffer
|
||||||
@@ -546,15 +217,15 @@ function GuacamoleClient(display, tunnelURL) {
|
|||||||
var instructionHandlers = {
|
var instructionHandlers = {
|
||||||
|
|
||||||
"error": function(parameters) {
|
"error": function(parameters) {
|
||||||
showError(unescapeGuacamoleString(parameters[0]));
|
if (errorHandler) errorHandler(tunnel.unescapeGuacamoleString(parameters[0]));
|
||||||
},
|
},
|
||||||
|
|
||||||
"name": function(parameters) {
|
"name": function(parameters) {
|
||||||
nameHandler(unescapeGuacamoleString(parameters[0]));
|
if (nameHandler) nameHandler(tunnel.unescapeGuacamoleString(parameters[0]));
|
||||||
},
|
},
|
||||||
|
|
||||||
"clipboard": function(parameters) {
|
"clipboard": function(parameters) {
|
||||||
clipboardHandler(unescapeGuacamoleString(parameters[0]));
|
if (clipboardHandler) clipboardHandler(tunnel.unescapeGuacamoleString(parameters[0]));
|
||||||
},
|
},
|
||||||
|
|
||||||
"size": function(parameters) {
|
"size": function(parameters) {
|
||||||
@@ -652,17 +323,8 @@ function GuacamoleClient(display, tunnelURL) {
|
|||||||
this.connect = function() {
|
this.connect = function() {
|
||||||
|
|
||||||
setState(STATE_CONNECTING);
|
setState(STATE_CONNECTING);
|
||||||
|
tunnel.connect();
|
||||||
// Start tunnel and connect synchronously
|
|
||||||
var connect_xmlhttprequest = new XMLHttpRequest();
|
|
||||||
connect_xmlhttprequest.open("POST", TUNNEL_CONNECT, false);
|
|
||||||
connect_xmlhttprequest.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
|
|
||||||
connect_xmlhttprequest.setRequestHeader("Content-length", 0);
|
|
||||||
connect_xmlhttprequest.send(null);
|
|
||||||
|
|
||||||
// Start reading data
|
|
||||||
setState(STATE_WAITING);
|
setState(STATE_WAITING);
|
||||||
handleResponse(makeRequest());
|
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -673,16 +335,8 @@ function GuacamoleClient(display, tunnelURL) {
|
|||||||
if (currentState != STATE_DISCONNECTED
|
if (currentState != STATE_DISCONNECTED
|
||||||
&& currentState != STATE_DISCONNECTING) {
|
&& currentState != STATE_DISCONNECTING) {
|
||||||
|
|
||||||
var message = "disconnect;";
|
|
||||||
setState(STATE_DISCONNECTING);
|
setState(STATE_DISCONNECTING);
|
||||||
|
tunnel.sendMessage("disconnect;");
|
||||||
// Send disconnect message (synchronously... as necessary until handoff is implemented)
|
|
||||||
var disconnect_xmlhttprequest = new XMLHttpRequest();
|
|
||||||
disconnect_xmlhttprequest.open("POST", TUNNEL_WRITE, false);
|
|
||||||
disconnect_xmlhttprequest.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
|
|
||||||
disconnect_xmlhttprequest.setRequestHeader("Content-length", message.length);
|
|
||||||
disconnect_xmlhttprequest.send(message);
|
|
||||||
|
|
||||||
setState(STATE_DISCONNECTED);
|
setState(STATE_DISCONNECTED);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -150,43 +150,6 @@ function GuacamoleMouse(element) {
|
|||||||
handleScroll(e);
|
handleScroll(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
function MouseEvent(x, y, left, middle, right, up, down) {
|
|
||||||
|
|
||||||
this.getX = function() {
|
|
||||||
return x;
|
|
||||||
};
|
|
||||||
|
|
||||||
this.getY = function() {
|
|
||||||
return y;
|
|
||||||
};
|
|
||||||
|
|
||||||
this.getLeft = function() {
|
|
||||||
return left;
|
|
||||||
};
|
|
||||||
|
|
||||||
this.getMiddle = function() {
|
|
||||||
return middle;
|
|
||||||
};
|
|
||||||
|
|
||||||
this.getRight = function() {
|
|
||||||
return right;
|
|
||||||
};
|
|
||||||
|
|
||||||
this.getUp = function() {
|
|
||||||
return up;
|
|
||||||
};
|
|
||||||
|
|
||||||
this.getDown = function() {
|
|
||||||
return down;
|
|
||||||
};
|
|
||||||
|
|
||||||
this.toString = function() {
|
|
||||||
return (mouseIndex++) + "," + x + "," + y + "," + left + "," + middle + "," + right + "," + up + "," + down;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
var buttonPressedHandler = null;
|
var buttonPressedHandler = null;
|
||||||
var buttonReleasedHandler = null;
|
var buttonReleasedHandler = null;
|
||||||
var movementHandler = null;
|
var movementHandler = null;
|
||||||
@@ -203,3 +166,35 @@ function GuacamoleMouse(element) {
|
|||||||
this.getRightButton = function() {return mouseRightButton;};
|
this.getRightButton = function() {return mouseRightButton;};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function MouseEvent(x, y, left, middle, right, up, down) {
|
||||||
|
|
||||||
|
this.getX = function() {
|
||||||
|
return x;
|
||||||
|
};
|
||||||
|
|
||||||
|
this.getY = function() {
|
||||||
|
return y;
|
||||||
|
};
|
||||||
|
|
||||||
|
this.getLeft = function() {
|
||||||
|
return left;
|
||||||
|
};
|
||||||
|
|
||||||
|
this.getMiddle = function() {
|
||||||
|
return middle;
|
||||||
|
};
|
||||||
|
|
||||||
|
this.getRight = function() {
|
||||||
|
return right;
|
||||||
|
};
|
||||||
|
|
||||||
|
this.getUp = function() {
|
||||||
|
return up;
|
||||||
|
};
|
||||||
|
|
||||||
|
this.getDown = function() {
|
||||||
|
return down;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
||||||
|
266
guacamole-common-js/src/main/resources/tunnel.js
Normal file
266
guacamole-common-js/src/main/resources/tunnel.js
Normal file
@@ -0,0 +1,266 @@
|
|||||||
|
|
||||||
|
/*
|
||||||
|
* Guacamole - Clientless Remote Desktop
|
||||||
|
* Copyright (C) 2010 Michael Jumper
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
*/
|
||||||
|
|
||||||
|
function GuacamoleHTTPTunnel(tunnelURL) {
|
||||||
|
|
||||||
|
var TUNNEL_CONNECT = tunnelURL + "?connect";
|
||||||
|
var TUNNEL_READ = tunnelURL + "?read";
|
||||||
|
var TUNNEL_WRITE = tunnelURL + "?write";
|
||||||
|
|
||||||
|
var pollResponse = 1; // Default to polling - will be turned off automatically if not needed
|
||||||
|
var instructionHandler = null;
|
||||||
|
|
||||||
|
var sendingMessages = 0;
|
||||||
|
var outputMessageBuffer = "";
|
||||||
|
|
||||||
|
function sendMessage(message) {
|
||||||
|
|
||||||
|
// Add event to queue, restart send loop if finished.
|
||||||
|
outputMessageBuffer += message;
|
||||||
|
if (sendingMessages == 0)
|
||||||
|
sendPendingMessages();
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
function sendPendingMessages() {
|
||||||
|
|
||||||
|
if (outputMessageBuffer.length > 0) {
|
||||||
|
|
||||||
|
sendingMessages = 1;
|
||||||
|
|
||||||
|
var message_xmlhttprequest = new XMLHttpRequest();
|
||||||
|
message_xmlhttprequest.open("POST", TUNNEL_WRITE);
|
||||||
|
message_xmlhttprequest.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
|
||||||
|
message_xmlhttprequest.setRequestHeader("Content-length", outputMessageBuffer.length);
|
||||||
|
|
||||||
|
// Once response received, send next queued event.
|
||||||
|
message_xmlhttprequest.onreadystatechange = function() {
|
||||||
|
if (message_xmlhttprequest.readyState == 4)
|
||||||
|
sendPendingMessages();
|
||||||
|
}
|
||||||
|
|
||||||
|
message_xmlhttprequest.send(outputMessageBuffer);
|
||||||
|
outputMessageBuffer = ""; // Clear buffer
|
||||||
|
|
||||||
|
}
|
||||||
|
else
|
||||||
|
sendingMessages = 0;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function handleResponse(xmlhttprequest) {
|
||||||
|
|
||||||
|
var interval = null;
|
||||||
|
var nextRequest = null;
|
||||||
|
|
||||||
|
var dataUpdateEvents = 0;
|
||||||
|
var instructionStart = 0;
|
||||||
|
var startIndex = 0;
|
||||||
|
|
||||||
|
function parseResponse() {
|
||||||
|
|
||||||
|
// Start next request as soon as possible
|
||||||
|
if (xmlhttprequest.readyState >= 2 && nextRequest == null)
|
||||||
|
nextRequest = makeRequest();
|
||||||
|
|
||||||
|
// Parse stream when data is received and when complete.
|
||||||
|
if (xmlhttprequest.readyState == 3 ||
|
||||||
|
xmlhttprequest.readyState == 4) {
|
||||||
|
|
||||||
|
// Also poll every 30ms (some browsers don't repeatedly call onreadystatechange for new data)
|
||||||
|
if (pollResponse == 1) {
|
||||||
|
if (xmlhttprequest.readyState == 3 && interval == null)
|
||||||
|
interval = setInterval(parseResponse, 30);
|
||||||
|
else if (xmlhttprequest.readyState == 4 && interval != null)
|
||||||
|
clearInterval(interval);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Halt on error during request
|
||||||
|
if (xmlhttprequest.status == 0) {
|
||||||
|
showError("Request canceled by browser.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else if (xmlhttprequest.status != 200) {
|
||||||
|
showError("Error during request (HTTP " + xmlhttprequest.status + "): " + xmlhttprequest.statusText);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var current = xmlhttprequest.responseText;
|
||||||
|
var instructionEnd;
|
||||||
|
|
||||||
|
while ((instructionEnd = current.indexOf(";", startIndex)) != -1) {
|
||||||
|
|
||||||
|
// Start next search at next instruction
|
||||||
|
startIndex = instructionEnd+1;
|
||||||
|
|
||||||
|
var instruction = current.substr(instructionStart,
|
||||||
|
instructionEnd - instructionStart);
|
||||||
|
|
||||||
|
instructionStart = startIndex;
|
||||||
|
|
||||||
|
var opcodeEnd = instruction.indexOf(":");
|
||||||
|
|
||||||
|
var opcode;
|
||||||
|
var parameters;
|
||||||
|
if (opcodeEnd == -1) {
|
||||||
|
opcode = instruction;
|
||||||
|
parameters = new Array();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
opcode = instruction.substr(0, opcodeEnd);
|
||||||
|
parameters = instruction.substr(opcodeEnd+1).split(",");
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we're done parsing, handle the next response.
|
||||||
|
if (opcode.length == 0) {
|
||||||
|
|
||||||
|
delete xmlhttprequest;
|
||||||
|
if (nextRequest)
|
||||||
|
handleResponse(nextRequest);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call instruction handler.
|
||||||
|
if (instructionHandler != null)
|
||||||
|
instructionHandler(opcode, parameters);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start search at end of string.
|
||||||
|
startIndex = current.length;
|
||||||
|
|
||||||
|
delete instruction;
|
||||||
|
delete parameters;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// If response polling enabled, attempt to detect if still
|
||||||
|
// necessary (via wrapping parseResponse())
|
||||||
|
if (pollResponse == 1) {
|
||||||
|
xmlhttprequest.onreadystatechange = function() {
|
||||||
|
|
||||||
|
// If we receive two or more readyState==3 events,
|
||||||
|
// there is no need to poll.
|
||||||
|
if (xmlhttprequest.readyState == 3) {
|
||||||
|
dataUpdateEvents++;
|
||||||
|
if (dataUpdateEvents >= 2) {
|
||||||
|
pollResponse = 0;
|
||||||
|
xmlhttprequest.onreadystatechange = parseResponse;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
parseResponse();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Otherwise, just parse
|
||||||
|
else
|
||||||
|
xmlhttprequest.onreadystatechange = parseResponse;
|
||||||
|
|
||||||
|
parseResponse();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function makeRequest() {
|
||||||
|
|
||||||
|
// Download self
|
||||||
|
var xmlhttprequest = new XMLHttpRequest();
|
||||||
|
xmlhttprequest.open("POST", TUNNEL_READ);
|
||||||
|
xmlhttprequest.send(null);
|
||||||
|
|
||||||
|
return xmlhttprequest;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
function connect() {
|
||||||
|
|
||||||
|
// Start tunnel and connect synchronously
|
||||||
|
var connect_xmlhttprequest = new XMLHttpRequest();
|
||||||
|
connect_xmlhttprequest.open("POST", TUNNEL_CONNECT, false);
|
||||||
|
connect_xmlhttprequest.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
|
||||||
|
connect_xmlhttprequest.setRequestHeader("Content-length", 0);
|
||||||
|
connect_xmlhttprequest.send(null);
|
||||||
|
|
||||||
|
// Start reading data
|
||||||
|
handleResponse(makeRequest());
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
// External API
|
||||||
|
this.connect = connect;
|
||||||
|
this.sendMessage = sendMessage;
|
||||||
|
this.setInstructionHandler = function(handler) {
|
||||||
|
instructionHandler = handler;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.escapeGuacamoleString = function(str) {
|
||||||
|
|
||||||
|
var escapedString = "";
|
||||||
|
|
||||||
|
for (var i=0; i<str.length; i++) {
|
||||||
|
|
||||||
|
var c = str.charAt(i);
|
||||||
|
if (c == ",")
|
||||||
|
escapedString += "\\c";
|
||||||
|
else if (c == ";")
|
||||||
|
escapedString += "\\s";
|
||||||
|
else if (c == "\\")
|
||||||
|
escapedString += "\\\\";
|
||||||
|
else
|
||||||
|
escapedString += c;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return escapedString;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
this.unescapeGuacamoleString = function(str) {
|
||||||
|
|
||||||
|
var unescapedString = "";
|
||||||
|
|
||||||
|
for (var i=0; i<str.length; i++) {
|
||||||
|
|
||||||
|
var c = str.charAt(i);
|
||||||
|
if (c == "\\" && i<str.length-1) {
|
||||||
|
|
||||||
|
var escapeChar = str.charAt(++i);
|
||||||
|
if (escapeChar == "c")
|
||||||
|
unescapedString += ",";
|
||||||
|
else if (escapeChar == "s")
|
||||||
|
unescapedString += ";";
|
||||||
|
else if (escapeChar == "\\")
|
||||||
|
unescapedString += "\\";
|
||||||
|
else
|
||||||
|
unescapedString += "\\" + escapeChar;
|
||||||
|
|
||||||
|
}
|
||||||
|
else
|
||||||
|
unescapedString += c;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return unescapedString;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Reference in New Issue
Block a user