Signed-off-by: Jeremy White <jwhite@xxxxxxxxxxxxxxx> --- cursor.js | 14 ++++++++ display.js | 5 +-- enums.js | 24 ++++++++++++- main.js | 112 ++++++++++++++++++++++++++++++++++++++++++++++++++++------- spice.css | 28 ++++++++------- spicemsg.js | 77 ++++++++++++++++++++++++++++++++++++++++ 6 files changed, 232 insertions(+), 28 deletions(-) diff --git a/cursor.js b/cursor.js index cafdf65..71e941d 100644 --- a/cursor.js +++ b/cursor.js @@ -80,6 +80,20 @@ SpiceCursorConn.prototype.process_channel_message = function(msg) return true; } + if (msg.type == SPICE_MSG_CURSOR_RESET) + { + DEBUG > 1 && console.log("SpiceMsgCursorReset"); + document.getElementById(this.parent.screen_id).style.cursor = "auto"; + return true; + } + + if (msg.type == SPICE_MSG_CURSOR_INVAL_ALL) + { + DEBUG > 1 && console.log("SpiceMsgCursorInvalAll"); + // FIXME - There may be something useful to do here... + return true; + } + return false; } diff --git a/display.js b/display.js index 6d7dafc..2aa5985 100644 --- a/display.js +++ b/display.js @@ -446,8 +446,9 @@ SpiceDisplayConn.prototype.process_channel_message = function(msg) /* This .save() is done entirely to enable SPICE_MSG_DISPLAY_RESET */ canvas.context.save(); document.getElementById(this.parent.screen_id).appendChild(canvas); - document.getElementById(this.parent.screen_id).setAttribute('width', m.surface.width); - document.getElementById(this.parent.screen_id).setAttribute('height', m.surface.height); + + /* We're going to leave width dynamic, but correctly set the height */ + document.getElementById(this.parent.screen_id).style.height = m.surface.height + "px"; this.hook_events(); } return true; diff --git a/enums.js b/enums.js index 7d41226..d99b38e 100644 --- a/enums.js +++ b/enums.js @@ -71,7 +71,12 @@ var SPICE_MSG_MAIN_MIGRATE_SWITCH_HOST = 111; var SPICE_MSG_MAIN_MIGRATE_END = 112; var SPICE_MSG_MAIN_NAME = 113; var SPICE_MSG_MAIN_UUID = 114; -var SPICE_MSG_END_MAIN = 115; +var SPICE_MSG_MAIN_AGENT_CONNECTED_TOKENS = 115; +var SPICE_MSG_MAIN_MIGRATE_BEGIN_SEAMLESS = 116; +var SPICE_MSG_MAIN_MIGRATE_DST_SEAMLESS_ACK = 117; +var SPICE_MSG_MAIN_MIGRATE_DST_SEAMLESS_NACK = 118; +var SPICE_MSG_END_MAIN = 119; + var SPICE_MSGC_ACK_SYNC = 1; @@ -300,3 +305,20 @@ var SPICE_CURSOR_TYPE_ALPHA = 0, SPICE_CURSOR_TYPE_COLOR32 = 6; var SPICE_VIDEO_CODEC_TYPE_MJPEG = 1; + +var VD_AGENT_PROTOCOL = 1; + +var VD_AGENT_MOUSE_STATE = 1, + VD_AGENT_MONITORS_CONFIG = 2, + VD_AGENT_REPLY = 3, + VD_AGENT_CLIPBOARD = 4, + VD_AGENT_DISPLAY_CONFIG = 5, + VD_AGENT_ANNOUNCE_CAPABILITIES = 6, + VD_AGENT_CLIPBOARD_GRAB = 7, + VD_AGENT_CLIPBOARD_REQUEST = 8, + VD_AGENT_CLIPBOARD_RELEASE = 9, + VD_AGENT_FILE_XFER_START =10, + VD_AGENT_FILE_XFER_STATUS =11, + VD_AGENT_FILE_XFER_DATA =12, + VD_AGENT_CLIENT_DISCONNECTED =13, + VD_AGENT_MAX_CLIPBOARD =14; diff --git a/main.js b/main.js index 3656a8d..7928685 100644 --- a/main.js +++ b/main.js @@ -78,15 +78,11 @@ SpiceMainConn.prototype.process_channel_message = function(msg) " ; ram_hint " + this.main_init.ram_hint); } - this.mouse_mode = this.main_init.current_mouse_mode; - if (this.main_init.current_mouse_mode != SPICE_MOUSE_MODE_CLIENT && - (this.main_init.supported_mouse_modes & SPICE_MOUSE_MODE_CLIENT)) - { - var mode_request = new SpiceMsgcMainMouseModeRequest(SPICE_MOUSE_MODE_CLIENT); - var mr = new SpiceMiniData(); - mr.build_msg(SPICE_MSGC_MAIN_MOUSE_MODE_REQUEST, mode_request); - this.send_msg(mr); - } + this.handle_mouse_mode(this.main_init.current_mouse_mode, + this.main_init.supported_mouse_modes); + + if (this.main_init.agent_connected) + this.connect_agent(); var attach = new SpiceMiniData; attach.type = SPICE_MSGC_MAIN_ATTACH_CHANNELS; @@ -99,9 +95,7 @@ SpiceMainConn.prototype.process_channel_message = function(msg) { var mode = new SpiceMsgMainMouseMode(msg.data); DEBUG > 0 && this.log_info("Mouse supported modes " + mode.supported_modes + "; current " + mode.current_mode); - this.mouse_mode = mode.current_mode; - if (this.inputs) - this.inputs.mouse_mode = mode.current_mode; + this.handle_mouse_mode(mode.current_mode, mode.supported_modes); return true; } @@ -144,6 +138,19 @@ SpiceMainConn.prototype.process_channel_message = function(msg) return true; } + if (msg.type == SPICE_MSG_MAIN_AGENT_CONNECTED || + msg.type == SPICE_MSG_MAIN_AGENT_CONNECTED_TOKENS) + { + this.connect_agent(); + return true; + } + + if (msg.type == SPICE_MSG_MAIN_AGENT_DISCONNECTED) + { + this.agent_connected = false; + return true; + } + return false; } @@ -177,3 +184,84 @@ SpiceMainConn.prototype.stop = function(msg) this.extra_channels[e].cleanup(); this.extra_channels = undefined; } + +SpiceMainConn.prototype.resize_window = function(flags, width, height, depth, x, y) +{ + if (this.agent_connected > 0) + { + var monitors_config = new VDAgentMonitorsConfig(flags, width, height, depth, x, y); + var agent_data = new SpiceMsgcMainAgentData(VD_AGENT_MONITORS_CONFIG, monitors_config); + var mr = new SpiceMiniData(); + mr.build_msg(SPICE_MSGC_MAIN_AGENT_DATA, agent_data); + this.send_msg(mr); + } +} + +SpiceMainConn.prototype.connect_agent = function() +{ + this.agent_connected = true; + + var agent_start = new SpiceMsgcMainAgentStart(0); + var mr = new SpiceMiniData(); + mr.build_msg(SPICE_MSGC_MAIN_AGENT_START, agent_start); + this.send_msg(mr); + + window.addEventListener('resize', handle_resize); + window.spice_connection = this; + + resize_helper(this); +} + +SpiceMainConn.prototype.handle_mouse_mode = function(current, supported) +{ + this.mouse_mode = current; + if (current != SPICE_MOUSE_MODE_CLIENT && (supported & SPICE_MOUSE_MODE_CLIENT)) + { + var mode_request = new SpiceMsgcMainMouseModeRequest(SPICE_MOUSE_MODE_CLIENT); + var mr = new SpiceMiniData(); + mr.build_msg(SPICE_MSGC_MAIN_MOUSE_MODE_REQUEST, mode_request); + this.send_msg(mr); + } + + if (this.inputs) + this.inputs.mouse_mode = current; +} + +function resize_helper(sc) +{ + var w = document.getElementById(sc.screen_id).clientWidth; + var h = document.getElementById(sc.screen_id).clientHeight; + + var m = document.getElementById(sc.message_id); + + /* Resize vertically; basically we leave a 20 pixel margin + at the bottom, and use the position of the message window + to figure out how to resize */ + var hd = window.innerHeight - m.offsetHeight - m.offsetTop - 20; + + /* Xorg requires height be a multiple of 8; round up */ + h = h + hd; + if (h % 8 > 0) + h += (8 - (h % 8)); + + /* Xorg requires width be a multiple of 8; round up */ + if (w % 8 > 0) + w += (8 - (w % 8)); + + + sc.resize_window(0, w, h, 32, 0, 0); + sc.spice_resize_timer = undefined; +} + +function handle_resize(e) +{ + var sc = window.spice_connection; + + if (sc && sc.spice_resize_timer) + { + window.clearTimeout(sc.spice_resize_timer); + sc.spice_resize_timer = undefined; + } + + sc.spice_resize_timer = window.setTimeout(resize_helper, 200, sc); +} diff --git a/spice.css b/spice.css index 450cf05..5d092ba 100644 --- a/spice.css +++ b/spice.css @@ -11,11 +11,9 @@ body #login { - width: 790px; - margin-top: 20px; + width: 95%; margin-left: auto; margin-right: auto; - padding: 2px 10px 2px 20px; border: 1px solid #999999; background: -webkit-gradient(linear, left top, left bottom, from(#fff), to(#24414e)); background: -moz-linear-gradient(top, #fff, #24414e); @@ -28,7 +26,7 @@ body { display: inline-block; margin-right: 5px; - padding-right: 10px; + padding: 2px 10px 2px 20px; border-right: 1px solid #999999; font-size: 20px; font-weight: bolder; @@ -66,16 +64,12 @@ body #spice-area { - text-align: center; -} -.spice-screen -{ - display: inline-block; - padding: 10px; - min-width: 800px; - min-height: 600px; + height: 100%; + width: 95%; + padding: 0; + margin-left: auto; + margin-right: auto; border: solid #222222 1px; - background-color: #333333; -webkit-box-shadow: 5px 5px 5px rgba(0, 0, 0, 0.2); -moz-box-shadow: 5px 5px 5px rgba(0, 0, 0, 0.2); box-shadow: 5px 5px 5px rgba(0, 0, 0, 0.2); @@ -83,6 +77,14 @@ body -webkit-border-radius: 10px; border-radius: 10px; } +.spice-screen +{ + min-height: 600px; + height: 100%; + margin: 10px; + padding: 0; + background-color: #333333; +} .spice-message { width: 700px; height: 50px; diff --git a/spicemsg.js b/spicemsg.js index 78371bc..c64f5a3 100644 --- a/spicemsg.js +++ b/spicemsg.js @@ -402,6 +402,83 @@ SpiceMsgcMainMouseModeRequest.prototype = } } +function SpiceMsgcMainAgentStart(num_tokens) +{ + this.num_tokens = num_tokens; +} + +SpiceMsgcMainAgentStart.prototype = +{ + to_buffer: function(a, at) + { + at = at || 0; + var dv = new SpiceDataView(a); + dv.setUint32(at, this.num_tokens, true); at += 4; + }, + buffer_size: function() + { + return 4; + } +} + +function SpiceMsgcMainAgentData(type, data) +{ + this.protocol = VD_AGENT_PROTOCOL; + this.type = type; + this.opaque = 0; + this.size = data.buffer_size(); + this.data = data; +} + +SpiceMsgcMainAgentData.prototype = +{ + to_buffer: function(a, at) + { + at = at || 0; + var dv = new SpiceDataView(a); + dv.setUint32(at, this.protocol, true); at += 4; + dv.setUint32(at, this.type, true); at += 4; + dv.setUint64(at, this.opaque, true); at += 8; + dv.setUint32(at, this.size, true); at += 4; + this.data.to_buffer(a, at); + }, + buffer_size: function() + { + return 4 + 4 + 8 + 4 + this.data.buffer_size(); + } +} + +function VDAgentMonitorsConfig(flags, width, height, depth, x, y) +{ + this.num_mon = 1; + this.flags = flags; + this.width = width; + this.height = height; + this.depth = depth; + this.x = x; + this.y = y; +} + +VDAgentMonitorsConfig.prototype = +{ + to_buffer: function(a, at) + { + at = at || 0; + var dv = new SpiceDataView(a); + dv.setUint32(at, this.num_mon, true); at += 4; + dv.setUint32(at, this.flags, true); at += 4; + dv.setUint32(at, this.height, true); at += 4; + dv.setUint32(at, this.width, true); at += 4; + dv.setUint32(at, this.depth, true); at += 4; + dv.setUint32(at, this.x, true); at += 4; + dv.setUint32(at, this.y, true); at += 4; + }, + buffer_size: function() + { + return 28; + } +} + function SpiceMsgNotify(a, at) { this.from_buffer(a, at); -- 1.7.10.4 _______________________________________________ Spice-devel mailing list Spice-devel@xxxxxxxxxxxxxxxxxxxxx http://lists.freedesktop.org/mailman/listinfo/spice-devel