/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved. * Copyright 2009-2013 Pierre Ossman for Cendio AB * Copyright (C) 2011-2013 D. R. Commander. All Rights Reserved. * Copyright (C) 2011-2015 Brian P. Hinz * * This is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This software 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this software; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, * USA. */ // // CConn // // Methods on CConn are called from both the GUI thread and the thread which // processes incoming RFB messages ("the RFB thread"). This means we need to // be careful with synchronization here. // // Any access to writer() must not only be synchronized, but we must also make // sure that the connection is in RFBSTATE_NORMAL. We are guaranteed this for // any code called after serverInit() has been called. Since the DesktopWindow // isn't created until then, any methods called only from DesktopWindow can // assume that we are in RFBSTATE_NORMAL. package com.tigervnc.vncviewer; import java.awt.*; import java.awt.event.*; import java.io.IOException; import java.io.InputStream; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.util.jar.Attributes; import java.util.jar.Manifest; import javax.swing.*; import javax.swing.ImageIcon; import java.net.InetSocketAddress; import java.net.SocketException; import java.util.*; import com.tigervnc.rdr.*; import com.tigervnc.rfb.*; import com.tigervnc.rfb.Point; import com.tigervnc.rfb.Exception; import com.tigervnc.network.Socket; import com.tigervnc.network.TcpSocket; public class CConn extends CConnection implements UserPasswdGetter, UserMsgBox, OptionsDialogCallback, FdInStreamBlockCallback, ActionListener { public final PixelFormat getPreferredPF() { return fullColourPF; } static final PixelFormat verylowColourPF = new PixelFormat(8, 3, false, true, 1, 1, 1, 2, 1, 0); static final PixelFormat lowColourPF = new PixelFormat(8, 6, false, true, 3, 3, 3, 4, 2, 0); static final PixelFormat mediumColourPF = new PixelFormat(8, 8, false, false, 7, 7, 3, 0, 3, 6); static final int KEY_LOC_SHIFT_R = 0; static final int KEY_LOC_SHIFT_L = 16; static final int SUPER_MASK = 1<<15; //////////////////////////////////////////////////////////////////// // The following methods are all called from the RFB thread public CConn(VncViewer viewer_, Socket sock_, String vncServerName) { sock = sock_; viewer = viewer_; pendingPFChange = false; currentEncoding = Encodings.encodingTight; lastServerEncoding = -1; fullColour = viewer.fullColour.getValue(); lowColourLevel = viewer.lowColourLevel.getValue(); autoSelect = viewer.autoSelect.getValue(); formatChange = false; encodingChange = false; fullScreen = viewer.fullScreen.getValue(); menuKeyCode = MenuKey.getMenuKeyCode(); options = new OptionsDialog(this); options.initDialog(); clipboardDialog = new ClipboardDialog(this); firstUpdate = true; pendingUpdate = false; continuousUpdates = false; forceNonincremental = true; supportsSyncFence = false; downKeySym = new HashMap(); setShared(viewer.shared.getValue()); upg = this; msg = this; String encStr = viewer.preferredEncoding.getValue(); int encNum = Encodings.encodingNum(encStr); if (encNum != -1) { currentEncoding = encNum; } cp.supportsDesktopResize = true; cp.supportsExtendedDesktopSize = true; cp.supportsSetDesktopSize = false; cp.supportsClientRedirect = true; cp.supportsDesktopRename = true; cp.supportsLocalCursor = viewer.useLocalCursor.getValue(); cp.customCompressLevel = viewer.customCompressLevel.getValue(); cp.compressLevel = viewer.compressLevel.getValue(); cp.noJpeg = viewer.noJpeg.getValue(); cp.qualityLevel = viewer.qualityLevel.getValue(); initMenu(); if (sock != null) { String name = sock.getPeerEndpoint(); vlog.info("Accepted connection from " + name); } else { if (vncServerName != null && !viewer.alwaysShowServerDialog.getValue()) { setServerName(Hostname.getHost(vncServerName)); setServerPort(Hostname.getPort(vncServerName)); } else { ServerDialog dlg = new ServerDialog(options, vncServerName, this); boolean ret = dlg.showDialog(); if (!ret) { close(); return; } setServerName(viewer.vncServerName.getValueStr()); setServerPort(viewer.vncServerPort.getValue()); } try { if (viewer.tunnel.getValue() || (viewer.via.getValue() != null)) { int localPort = TcpSocket.findFreeTcpPort(); if (localPort == 0) throw new Exception("Could not obtain free TCP port"); Tunnel.createTunnel(this, localPort); sock = new TcpSocket("localhost", localPort); } else { sock = new TcpSocket(getServerName(), getServerPort()); } } catch (java.lang.Exception e) { throw new Exception(e.getMessage()); } vlog.info("connected to host "+getServerName()+" port "+getServerPort()); } sock.inStream().setBlockCallback(this); setStreams(sock.inStream(), sock.outStream()); initialiseProtocol(); } public void refreshFramebuffer() { forceNonincremental = true; // Without fences, we cannot safely trigger an update request directly // but must wait for the next update to arrive. if (supportsSyncFence) requestNewUpdate(); } public boolean showMsgBox(int flags, String title, String text) { //StringBuffer titleText = new StringBuffer("VNC Viewer: "+title); return true; } // deleteWindow() is called when the user closes the desktop or menu windows. void deleteWindow() { if (viewport != null) viewport.dispose(); viewport = null; } // blockCallback() is called when reading from the socket would block. public void blockCallback() { try { synchronized(this) { wait(1); } } catch (java.lang.InterruptedException e) { throw new Exception(e.getMessage()); } } // getUserPasswd() is called by the CSecurity object when it needs us to read // a password from the user. public final boolean getUserPasswd(StringBuffer user, StringBuffer passwd) { String title = ("VNC Authentication [" +csecurity.description() + "]"); String passwordFileStr = viewer.passwordFile.getValue(); PasswdDialog dlg; if (user == null && !passwordFileStr.equals("")) { InputStream fp = null; try { fp = new FileInputStream(passwordFileStr); } catch(FileNotFoundException e) { throw new Exception("Opening password file failed"); } byte[] obfPwd = new byte[256]; try { fp.read(obfPwd); fp.close(); } catch(IOException e) { throw new Exception("Failed to read VncPasswd file"); } String PlainPasswd = VncAuth.unobfuscatePasswd(obfPwd); passwd.append(PlainPasswd); passwd.setLength(PlainPasswd.length()); return true; } if (user == null) { dlg = new PasswdDialog(title, (user == null), (passwd == null)); } else { if ((passwd == null) && viewer.sendLocalUsername.getValue()) { user.append((String)System.getProperties().get("user.name")); return true; } dlg = new PasswdDialog(title, viewer.sendLocalUsername.getValue(), (passwd == null)); } if (!dlg.showDialog()) return false; if (user != null) { if (viewer.sendLocalUsername.getValue()) { user.append((String)System.getProperties().get("user.name")); } else { user.append(dlg.userEntry.getText()); } } if (passwd != null) passwd.append(new String(dlg.passwdEntry.getPassword())); return true; } // CConnection callback methods // serverInit() is called when the serverInit message has been received. At // this point we create the desktop window and display it. We also tell the // server the pixel format and encodings to use and request the first update. public void serverInit() { super.serverInit(); // If using AutoSelect with old servers, start in FullColor // mode. See comment in autoSelectFormatAndEncoding. if (cp.beforeVersion(3, 8) && autoSelect) fullColour = true; serverPF = cp.pf(); desktop = new DesktopWindow(cp.width, cp.height, serverPF, this); fullColourPF = desktop.getPreferredPF(); // Force a switch to the format and encoding we'd like formatChange = true; encodingChange = true; // And kick off the update cycle requestNewUpdate(); // This initial update request is a bit of a corner case, so we need // to help out setting the correct format here. assert(pendingPFChange); desktop.setServerPF(pendingPF); cp.setPF(pendingPF); pendingPFChange = false; if (viewer.embed.getValue()) { setupEmbeddedFrame(); } else { recreateViewport(); } } void setupEmbeddedFrame() { UIManager.getDefaults().put("ScrollPane.ancestorInputMap", new UIDefaults.LazyInputMap(new Object[]{})); JScrollPane sp = new JScrollPane(); sp.setBorder(BorderFactory.createEmptyBorder(0, 0, 0, 0)); sp.getViewport().setBackground(Color.BLACK); InputMap im = sp.getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW); int ctrlAltShiftMask = Event.SHIFT_MASK | Event.CTRL_MASK | Event.ALT_MASK; if (im != null) { im.put(KeyStroke.getKeyStroke(KeyEvent.VK_UP, ctrlAltShiftMask), "unitScrollUp"); im.put(KeyStroke.getKeyStroke(KeyEvent.VK_DOWN, ctrlAltShiftMask), "unitScrollDown"); im.put(KeyStroke.getKeyStroke(KeyEvent.VK_LEFT, ctrlAltShiftMask), "unitScrollLeft"); im.put(KeyStroke.getKeyStroke(KeyEvent.VK_RIGHT, ctrlAltShiftMask), "unitScrollRight"); im.put(KeyStroke.getKeyStroke(KeyEvent.VK_PAGE_UP, ctrlAltShiftMask), "scrollUp"); im.put(KeyStroke.getKeyStroke(KeyEvent.VK_PAGE_DOWN, ctrlAltShiftMask), "scrollDown"); im.put(KeyStroke.getKeyStroke(KeyEvent.VK_HOME, ctrlAltShiftMask), "scrollLeft"); im.put(KeyStroke.getKeyStroke(KeyEvent.VK_END, ctrlAltShiftMask), "scrollRight"); } sp.setHorizontalScrollBarPolicy(ScrollPaneConstants.HORIZONTAL_SCROLLBAR_AS_NEEDED); sp.setVerticalScrollBarPolicy(ScrollPaneConstants.VERTICAL_SCROLLBAR_AS_NEEDED); desktop.setViewport(sp.getViewport()); viewer.getContentPane().removeAll(); viewer.add(sp); viewer.addFocusListener(new FocusAdapter() { public void focusGained(FocusEvent e) { if (desktop.isAncestorOf(viewer)) desktop.requestFocus(); } public void focusLost(FocusEvent e) { releaseDownKeys(); } }); viewer.validate(); desktop.requestFocus(); } // setDesktopSize() is called when the desktop size changes (including when // it is set initially). public void setDesktopSize(int w, int h) { super.setDesktopSize(w, h); resizeFramebuffer(); } // setExtendedDesktopSize() is a more advanced version of setDesktopSize() public void setExtendedDesktopSize(int reason, int result, int w, int h, ScreenSet layout) { super.setExtendedDesktopSize(reason, result, w, h, layout); if ((reason == screenTypes.reasonClient) && (result != screenTypes.resultSuccess)) { vlog.error("SetDesktopSize failed: " + result); return; } resizeFramebuffer(); } // clientRedirect() migrates the client to another host/port public void clientRedirect(int port, String host, String x509subject) { try { sock.close(); setServerPort(port); sock = new TcpSocket(host, port); vlog.info("Redirected to "+host+":"+port); VncViewer.newViewer(viewer, sock, true); } catch (java.lang.Exception e) { throw new Exception(e.getMessage()); } } // setName() is called when the desktop name changes public void setName(String name) { super.setName(name); if (viewport != null) { viewport.setTitle(name+" - TigerVNC"); } } // framebufferUpdateStart() is called at the beginning of an update. // Here we try to send out a new framebuffer update request so that the // next update can be sent out in parallel with us decoding the current // one. public void framebufferUpdateStart() { // Note: This might not be true if sync fences are supported pendingUpdate = false; requestNewUpdate(); } // framebufferUpdateEnd() is called at the end of an update. // For each rectangle, the FdInStream will have timed the speed // of the connection, allowing us to select format and encoding // appropriately, and then request another incremental update. public void framebufferUpdateEnd() { desktop.updateWindow(); if (firstUpdate) { int width, height; // We need fences to make extra update requests and continuous // updates "safe". See fence() for the next step. if (cp.supportsFence) writer().writeFence(fenceTypes.fenceFlagRequest | fenceTypes.fenceFlagSyncNext, 0, null); if (cp.supportsSetDesktopSize && viewer.desktopSize.getValue() != null && viewer.desktopSize.getValue().split("x").length == 2) { width = Integer.parseInt(viewer.desktopSize.getValue().split("x")[0]); height = Integer.parseInt(viewer.desktopSize.getValue().split("x")[1]); ScreenSet layout; layout = cp.screenLayout; if (layout.num_screens() == 0) layout.add_screen(new Screen()); else if (layout.num_screens() != 1) { while (true) { Iterator iter = layout.screens.iterator(); Screen screen = (Screen)iter.next(); if (!iter.hasNext()) break; layout.remove_screen(screen.id); } } Screen screen0 = (Screen)layout.screens.iterator().next(); screen0.dimensions.tl.x = 0; screen0.dimensions.tl.y = 0; screen0.dimensions.br.x = width; screen0.dimensions.br.y = height; writer().writeSetDesktopSize(width, height, layout); } firstUpdate = false; } // A format change has been scheduled and we are now past the update // with the old format. Time to active the new one. if (pendingPFChange) { desktop.setServerPF(pendingPF); cp.setPF(pendingPF); pendingPFChange = false; } // Compute new settings based on updated bandwidth values if (autoSelect) autoSelectFormatAndEncoding(); } // The rest of the callbacks are fairly self-explanatory... public void setColourMapEntries(int firstColour, int nColours, int[] rgbs) { desktop.setColourMapEntries(firstColour, nColours, rgbs); } public void bell() { if (viewer.acceptBell.getValue()) desktop.getToolkit().beep(); } public void serverCutText(String str, int len) { if (viewer.acceptClipboard.getValue()) clipboardDialog.serverCutText(str, len); } // We start timing on beginRect and stop timing on endRect, to // avoid skewing the bandwidth estimation as a result of the server // being slow or the network having high latency public void beginRect(Rect r, int encoding) { sock.inStream().startTiming(); if (encoding != Encodings.encodingCopyRect) { lastServerEncoding = encoding; } } public void endRect(Rect r, int encoding) { sock.inStream().stopTiming(); } public void fillRect(Rect r, int p) { desktop.fillRect(r.tl.x, r.tl.y, r.width(), r.height(), p); } public void imageRect(Rect r, Object p) { desktop.imageRect(r.tl.x, r.tl.y, r.width(), r.height(), p); } public void copyRect(Rect r, int sx, int sy) { desktop.copyRect(r.tl.x, r.tl.y, r.width(), r.height(), sx, sy); } public void setCursor(int width, int height, Point hotspot, int[] data, byte[] mask) { desktop.setCursor(width, height, hotspot, data, mask); } public void fence(int flags, int len, byte[] data) { // can't call super.super.fence(flags, len, data); cp.supportsFence = true; if ((flags & fenceTypes.fenceFlagRequest) != 0) { // We handle everything synchronously so we trivially honor these modes flags = flags & (fenceTypes.fenceFlagBlockBefore | fenceTypes.fenceFlagBlockAfter); writer().writeFence(flags, len, data); return; } if (len == 0) { // Initial probe if ((flags & fenceTypes.fenceFlagSyncNext) != 0) { supportsSyncFence = true; if (cp.supportsContinuousUpdates) { vlog.info("Enabling continuous updates"); continuousUpdates = true; writer().writeEnableContinuousUpdates(true, 0, 0, cp.width, cp.height); } } } else { // Pixel format change MemInStream memStream = new MemInStream(data, 0, len); PixelFormat pf = new PixelFormat(); pf.read(memStream); desktop.setServerPF(pf); cp.setPF(pf); } } private void resizeFramebuffer() { if (desktop == null) return; if (continuousUpdates) writer().writeEnableContinuousUpdates(true, 0, 0, cp.width, cp.height); if ((cp.width == 0) && (cp.height == 0)) return; if ((desktop.width() == cp.width) && (desktop.height() == cp.height)) return; desktop.resize(); if (viewer.embed.getValue()) { setupEmbeddedFrame(); } else { recreateViewport(); } } public void setEmbeddedFeatures(boolean s) { menu.restore.setEnabled(s); menu.minimize.setEnabled(s); menu.maximize.setEnabled(s); menu.fullScreen.setEnabled(s); menu.newConn.setEnabled(s); options.fullScreen.setEnabled(s); options.fullScreenAllMonitors.setEnabled(s); options.scalingFactor.setEnabled(s); } // recreateViewport() recreates our top-level window. This seems to be // better than attempting to resize the existing window, at least with // various X window managers. public void recreateViewport() { if (viewer.embed.getValue()) return; if (viewport != null) viewport.dispose(); viewport = new Viewport(cp.name(), this); viewport.setUndecorated(fullScreen); desktop.setViewport(viewport.getViewport()); reconfigureViewport(); if ((cp.width > 0) && (cp.height > 0)) viewport.setVisible(true); desktop.requestFocusInWindow(); } private void reconfigureViewport() { Dimension dpySize = viewport.getScreenSize(); int w = desktop.scaledWidth; int h = desktop.scaledHeight; if (fullScreen) { if (!viewer.fullScreenAllMonitors.getValue()) viewport.setExtendedState(JFrame.MAXIMIZED_BOTH); viewport.setBounds(viewport.getScreenBounds()); if (!viewer.fullScreenAllMonitors.getValue()) Viewport.setFullScreenWindow(viewport); } else { int wmDecorationWidth = viewport.getInsets().left + viewport.getInsets().right; int wmDecorationHeight = viewport.getInsets().top + viewport.getInsets().bottom; if (w + wmDecorationWidth >= dpySize.width) w = dpySize.width - wmDecorationWidth; if (h + wmDecorationHeight >= dpySize.height) h = dpySize.height - wmDecorationHeight; if (viewport.getExtendedState() == JFrame.MAXIMIZED_BOTH) { w = viewport.getSize().width; h = viewport.getSize().height; int x = viewport.getLocation().x; int y = viewport.getLocation().y; viewport.setGeometry(x, y, w, h); } else { int x = (dpySize.width - w - wmDecorationWidth) / 2; int y = (dpySize.height - h - wmDecorationHeight)/2; viewport.setExtendedState(JFrame.NORMAL); viewport.setGeometry(x, y, w, h); } Viewport.setFullScreenWindow(null); } } // autoSelectFormatAndEncoding() chooses the format and encoding appropriate // to the connection speed: // // First we wait for at least one second of bandwidth measurement. // // Above 16Mbps (i.e. LAN), we choose the second highest JPEG quality, // which should be perceptually lossless. // // If the bandwidth is below that, we choose a more lossy JPEG quality. // // If the bandwidth drops below 256 Kbps, we switch to palette mode. // // Note: The system here is fairly arbitrary and should be replaced // with something more intelligent at the server end. // private void autoSelectFormatAndEncoding() { long kbitsPerSecond = sock.inStream().kbitsPerSecond(); long timeWaited = sock.inStream().timeWaited(); boolean newFullColour = fullColour; int newQualityLevel = cp.qualityLevel; // Always use Tight if (currentEncoding != Encodings.encodingTight) { currentEncoding = Encodings.encodingTight; encodingChange = true; } // Check that we have a decent bandwidth measurement if ((kbitsPerSecond == 0) || (timeWaited < 100)) return; // Select appropriate quality level if (!cp.noJpeg) { if (kbitsPerSecond > 16000) newQualityLevel = 8; else newQualityLevel = 6; if (newQualityLevel != cp.qualityLevel) { vlog.info("Throughput "+kbitsPerSecond+ " kbit/s - changing to quality "+newQualityLevel); cp.qualityLevel = newQualityLevel; viewer.qualityLevel.setParam(Integer.toString(newQualityLevel)); encodingChange = true; } } if (cp.beforeVersion(3, 8)) { // Xvnc from TightVNC 1.2.9 sends out FramebufferUpdates with // cursors "asynchronously". If this happens in the middle of a // pixel format change, the server will encode the cursor with // the old format, but the client will try to decode it // according to the new format. This will lead to a // crash. Therefore, we do not allow automatic format change for // old servers. return; } // Select best color level newFullColour = (kbitsPerSecond > 256); if (newFullColour != fullColour) { vlog.info("Throughput "+kbitsPerSecond+ " kbit/s - full color is now "+ (newFullColour ? "enabled" : "disabled")); fullColour = newFullColour; formatChange = true; forceNonincremental = true; } } // requestNewUpdate() requests an update from the server, having set the // format and encoding appropriately. private void requestNewUpdate() { if (formatChange) { PixelFormat pf; /* Catch incorrect requestNewUpdate calls */ assert(!pendingUpdate || supportsSyncFence); if (fullColour) { pf = fullColourPF; } else { if (lowColourLevel == 0) { pf = verylowColourPF; } else if (lowColourLevel == 1) { pf = lowColourPF; } else { pf = mediumColourPF; } } if (supportsSyncFence) { // We let the fence carry the pixel format and switch once we // get the response back. That way we will be synchronised with // when the server switches. MemOutStream memStream = new MemOutStream(); pf.write(memStream); writer().writeFence(fenceTypes.fenceFlagRequest | fenceTypes.fenceFlagSyncNext, memStream.length(), (byte[])memStream.data()); } else { // New requests are sent out at the start of processing the last // one, so we cannot switch our internal format right now (doing so // would mean misdecoding the current update). pendingPFChange = true; pendingPF = pf; } String str = pf.print(); vlog.info("Using pixel format " + str); writer().writeSetPixelFormat(pf); formatChange = false; } checkEncodings(); if (forceNonincremental || !continuousUpdates) { pendingUpdate = true; writer().writeFramebufferUpdateRequest(new Rect(0, 0, cp.width, cp.height), !forceNonincremental); } forceNonincremental = false; } //////////////////////////////////////////////////////////////////// // The following methods are all called from the GUI thread // close() shuts down the socket, thus waking up the RFB thread. public void close() { if (closeListener != null) { viewer.embed.setParam(true); if (VncViewer.nViewers == 1) { JFrame f = (JFrame)JOptionPane.getFrameForComponent(viewer); if (f != null) f.dispatchEvent(new WindowEvent(f, WindowEvent.WINDOW_CLOSING)); } } deleteWindow(); shuttingDown = true; try { if (sock != null) sock.shutdown(); } catch (java.lang.Exception e) { throw new Exception(e.getMessage()); } } // Menu callbacks. These are guaranteed only to be called after serverInit() // has been called, since the menu is only accessible from the DesktopWindow private void initMenu() { menu = new F8Menu(this); } void showMenu(int x, int y) { String os = System.getProperty("os.name"); if (os.startsWith("Windows")) com.sun.java.swing.plaf.windows.WindowsLookAndFeel.setMnemonicHidden(false); menu.show(desktop, x, y); } void showAbout() { String pkgDate = ""; String pkgTime = ""; try { Manifest manifest = new Manifest(VncViewer.timestamp); Attributes attributes = manifest.getMainAttributes(); pkgDate = attributes.getValue("Package-Date"); pkgTime = attributes.getValue("Package-Time"); } catch (java.lang.Exception e) { } Window fullScreenWindow = Viewport.getFullScreenWindow(); if (fullScreenWindow != null) Viewport.setFullScreenWindow(null); String msg = String.format(VncViewer.aboutText, VncViewer.version, VncViewer.build, VncViewer.buildDate, VncViewer.buildTime); JOptionPane op = new JOptionPane(msg, JOptionPane.INFORMATION_MESSAGE, JOptionPane.DEFAULT_OPTION, VncViewer.logoIcon); JDialog dlg = op.createDialog(desktop, "About TigerVNC Viewer for Java"); dlg.setIconImage(VncViewer.frameIcon); dlg.setAlwaysOnTop(true); dlg.setVisible(true); if (fullScreenWindow != null) Viewport.setFullScreenWindow(fullScreenWindow); } void showInfo() { Window fullScreenWindow = Viewport.getFullScreenWindow(); if (fullScreenWindow != null) Viewport.setFullScreenWindow(null); String info = new String("Desktop name: %s%n"+ "Host: %s:%d%n"+ "Size: %dx%d%n"+ "Pixel format: %s%n"+ " (server default: %s)%n"+ "Requested encoding: %s%n"+ "Last used encoding: %s%n"+ "Line speed estimate: %d kbit/s%n"+ "Protocol version: %d.%d%n"+ "Security method: %s [%s]%n"); String msg = String.format(info, cp.name(), sock.getPeerName(), sock.getPeerPort(), cp.width, cp.height, desktop.getPF().print(), serverPF.print(), Encodings.encodingName(currentEncoding), Encodings.encodingName(lastServerEncoding), sock.inStream().kbitsPerSecond(), cp.majorVersion, cp.minorVersion, Security.secTypeName(csecurity.getType()), csecurity.description()); JOptionPane op = new JOptionPane(msg, JOptionPane.PLAIN_MESSAGE, JOptionPane.DEFAULT_OPTION); JDialog dlg = op.createDialog(desktop, "VNC connection info"); dlg.setIconImage(VncViewer.frameIcon); dlg.setAlwaysOnTop(true); dlg.setVisible(true); if (fullScreenWindow != null) Viewport.setFullScreenWindow(fullScreenWindow); } public void refresh() { writer().writeFramebufferUpdateRequest(new Rect(0,0,cp.width,cp.height), false); pendingUpdate = true; } // OptionsDialogCallback. setOptions() sets the options dialog's checkboxes // etc to reflect our flags. getOptions() sets our flags according to the // options dialog's checkboxes. They are both called from the GUI thread. // Some of the flags are also accessed by the RFB thread. I believe that // reading and writing boolean and int values in java is atomic, so there is // no need for synchronization. public void setOptions() { int digit; options.autoSelect.setSelected(autoSelect); options.fullColour.setSelected(fullColour); options.veryLowColour.setSelected(!fullColour && lowColourLevel == 0); options.lowColour.setSelected(!fullColour && lowColourLevel == 1); options.mediumColour.setSelected(!fullColour && lowColourLevel == 2); options.tight.setSelected(currentEncoding == Encodings.encodingTight); options.zrle.setSelected(currentEncoding == Encodings.encodingZRLE); options.hextile.setSelected(currentEncoding == Encodings.encodingHextile); options.raw.setSelected(currentEncoding == Encodings.encodingRaw); options.customCompressLevel.setSelected(viewer.customCompressLevel.getValue()); digit = 0 + viewer.compressLevel.getValue(); if (digit >= 0 && digit <= 9) { options.compressLevel.setSelectedItem(digit); } else { options.compressLevel.setSelectedItem(Integer.parseInt(viewer.compressLevel.getDefaultStr())); } options.noJpeg.setSelected(!viewer.noJpeg.getValue()); digit = 0 + viewer.qualityLevel.getValue(); if (digit >= 0 && digit <= 9) { options.qualityLevel.setSelectedItem(digit); } else { options.qualityLevel.setSelectedItem(Integer.parseInt(viewer.qualityLevel.getDefaultStr())); } options.viewOnly.setSelected(viewer.viewOnly.getValue()); options.acceptClipboard.setSelected(viewer.acceptClipboard.getValue()); options.sendClipboard.setSelected(viewer.sendClipboard.getValue()); options.menuKey.setSelectedItem(KeyEvent.getKeyText(MenuKey.getMenuKeyCode())); options.sendLocalUsername.setSelected(viewer.sendLocalUsername.getValue()); if (state() == RFBSTATE_NORMAL) { options.shared.setEnabled(false); options.secVeNCrypt.setEnabled(false); options.encNone.setEnabled(false); options.encTLS.setEnabled(false); options.encX509.setEnabled(false); options.x509ca.setEnabled(false); options.caButton.setEnabled(false); options.x509crl.setEnabled(false); options.crlButton.setEnabled(false); options.secIdent.setEnabled(false); options.secNone.setEnabled(false); options.secVnc.setEnabled(false); options.secPlain.setEnabled(false); options.sendLocalUsername.setEnabled(false); options.cfLoadButton.setEnabled(false); options.cfSaveAsButton.setEnabled(true); options.sshTunnel.setEnabled(false); options.sshUseGateway.setEnabled(false); options.sshUser.setEnabled(false); options.sshHost.setEnabled(false); options.sshPort.setEnabled(false); options.sshUseExt.setEnabled(false); options.sshClient.setEnabled(false); options.sshClientBrowser.setEnabled(false); options.sshArgsDefault.setEnabled(false); options.sshArgsCustom.setEnabled(false); options.sshArguments.setEnabled(false); options.sshConfig.setEnabled(false); options.sshConfigBrowser.setEnabled(false); options.sshKeyFile.setEnabled(false); options.sshKeyFileBrowser.setEnabled(false); } else { options.shared.setSelected(viewer.shared.getValue()); options.sendLocalUsername.setSelected(viewer.sendLocalUsername.getValue()); options.cfSaveAsButton.setEnabled(false); if (viewer.tunnel.getValue() || viewer.via.getValue() != null) options.sshTunnel.setSelected(true); if (viewer.via.getValue() != null) options.sshUseGateway.setSelected(true); options.sshUser.setText(Tunnel.getSshUser(this)); options.sshHost.setText(Tunnel.getSshHost(this)); options.sshPort.setText(Integer.toString(Tunnel.getSshPort(this))); options.sshUseExt.setSelected(viewer.extSSH.getValue()); File client = new File(viewer.extSSHClient.getValue()); if (client.exists() && client.canRead()) options.sshClient.setText(client.getAbsolutePath()); if (viewer.extSSHArgs.getValue() == null) { options.sshArgsDefault.setSelected(true); options.sshArguments.setText(""); } else { options.sshArgsCustom.setSelected(true); options.sshArguments.setText(viewer.extSSHArgs.getValue()); } File config = new File(viewer.sshConfig.getValue()); if (config.exists() && config.canRead()) options.sshConfig.setText(config.getAbsolutePath()); options.sshKeyFile.setText(Tunnel.getSshKeyFile(this)); /* Process non-VeNCrypt sectypes */ java.util.List secTypes = new ArrayList(); secTypes = Security.GetEnabledSecTypes(); for (Iterator i = secTypes.iterator(); i.hasNext();) { switch ((Integer)i.next()) { case Security.secTypeVeNCrypt: options.secVeNCrypt.setSelected(UserPreferences.getBool("viewer", "secVeNCrypt", true)); break; case Security.secTypeNone: options.encNone.setSelected(true); options.secNone.setSelected(UserPreferences.getBool("viewer", "secTypeNone", true)); break; case Security.secTypeVncAuth: options.encNone.setSelected(true); options.secVnc.setSelected(UserPreferences.getBool("viewer", "secTypeVncAuth", true)); break; } } /* Process VeNCrypt subtypes */ if (options.secVeNCrypt.isSelected()) { java.util.List secTypesExt = new ArrayList(); secTypesExt = Security.GetEnabledExtSecTypes(); for (Iterator iext = secTypesExt.iterator(); iext.hasNext();) { switch ((Integer)iext.next()) { case Security.secTypePlain: options.encNone.setSelected(UserPreferences.getBool("viewer", "encNone", true)); options.secPlain.setSelected(UserPreferences.getBool("viewer", "secPlain", true)); break; case Security.secTypeIdent: options.encNone.setSelected(UserPreferences.getBool("viewer", "encNone", true)); options.secIdent.setSelected(UserPreferences.getBool("viewer", "secIdent", true)); break; case Security.secTypeTLSNone: options.encTLS.setSelected(UserPreferences.getBool("viewer", "encTLS", true)); options.secNone.setSelected(UserPreferences.getBool("viewer", "secNone", true)); break; case Security.secTypeTLSVnc: options.encTLS.setSelected(UserPreferences.getBool("viewer", "encTLS", true)); options.secVnc.setSelected(UserPreferences.getBool("viewer", "secVnc", true)); break; case Security.secTypeTLSPlain: options.encTLS.setSelected(UserPreferences.getBool("viewer", "encTLS", true)); options.secPlain.setSelected(UserPreferences.getBool("viewer", "secPlain", true)); break; case Security.secTypeTLSIdent: options.encTLS.setSelected(UserPreferences.getBool("viewer", "encTLS", true)); options.secIdent.setSelected(UserPreferences.getBool("viewer", "secIdent", true)); break; case Security.secTypeX509None: options.encX509.setSelected(UserPreferences.getBool("viewer", "encX509", true)); options.secNone.setSelected(UserPreferences.getBool("viewer", "secNone", true)); break; case Security.secTypeX509Vnc: options.encX509.setSelected(UserPreferences.getBool("viewer", "encX509", true)); options.secVnc.setSelected(UserPreferences.getBool("viewer", "secVnc", true)); break; case Security.secTypeX509Plain: options.encX509.setSelected(UserPreferences.getBool("viewer", "encX509", true)); options.secPlain.setSelected(UserPreferences.getBool("viewer", "secPlain", true)); break; case Security.secTypeX509Ident: options.encX509.setSelected(UserPreferences.getBool("viewer", "encX509", true)); options.secIdent.setSelected(UserPreferences.getBool("viewer", "secIdent", true)); break; } } } File caFile = new File(viewer.x509ca.getValue()); if (caFile.exists() && caFile.canRead()) options.x509ca.setText(caFile.getAbsolutePath()); File crlFile = new File(viewer.x509crl.getValue()); if (crlFile.exists() && crlFile.canRead()) options.x509crl.setText(crlFile.getAbsolutePath()); options.encNone.setEnabled(options.secVeNCrypt.isSelected()); options.encTLS.setEnabled(options.secVeNCrypt.isSelected()); options.encX509.setEnabled(options.secVeNCrypt.isSelected()); options.x509ca.setEnabled(options.secVeNCrypt.isSelected() && options.encX509.isSelected()); options.caButton.setEnabled(options.secVeNCrypt.isSelected() && options.encX509.isSelected()); options.x509crl.setEnabled(options.secVeNCrypt.isSelected() && options.encX509.isSelected()); options.crlButton.setEnabled(options.secVeNCrypt.isSelected() && options.encX509.isSelected()); options.secIdent.setEnabled(options.secVeNCrypt.isSelected()); options.secPlain.setEnabled(options.secVeNCrypt.isSelected()); options.sendLocalUsername.setEnabled(options.secPlain.isSelected()|| options.secIdent.isSelected()); options.sshTunnel.setEnabled(true); options.sshUseGateway.setEnabled(options.sshTunnel.isSelected()); options.sshUser.setEnabled(options.sshTunnel.isSelected() && options.sshUseGateway.isEnabled() && options.sshUseGateway.isSelected()); options.sshHost.setEnabled(options.sshTunnel.isSelected() && options.sshUseGateway.isEnabled() && options.sshUseGateway.isSelected()); options.sshPort.setEnabled(options.sshTunnel.isSelected() && options.sshUseGateway.isEnabled() && options.sshUseGateway.isSelected()); options.sshUseExt.setEnabled(options.sshTunnel.isSelected()); options.sshClient.setEnabled(options.sshTunnel.isSelected() && options.sshUseExt.isEnabled() && options.sshUseExt.isSelected()); options.sshClientBrowser.setEnabled(options.sshTunnel.isSelected() && options.sshUseExt.isEnabled() && options.sshUseExt.isSelected()); options.sshArgsDefault.setEnabled(options.sshTunnel.isSelected() && options.sshUseExt.isEnabled() && options.sshUseExt.isSelected()); options.sshArgsCustom.setEnabled(options.sshTunnel.isSelected() && options.sshUseExt.isEnabled() && options.sshUseExt.isSelected()); options.sshArguments.setEnabled(options.sshTunnel.isSelected() && options.sshUseExt.isEnabled() && options.sshUseExt.isSelected() && options.sshArgsCustom.isSelected()); options.sshConfig.setEnabled(options.sshTunnel.isSelected() && options.sshUseExt.isEnabled() && !options.sshUseExt.isSelected()); options.sshConfigBrowser.setEnabled(options.sshTunnel.isSelected() && options.sshUseExt.isEnabled() && !options.sshUseExt.isSelected()); options.sshKeyFile.setEnabled(options.sshTunnel.isSelected() && options.sshUseExt.isEnabled() && !options.sshUseExt.isSelected()); options.sshKeyFileBrowser.setEnabled(options.sshTunnel.isSelected() && options.sshUseExt.isEnabled() && !options.sshUseExt.isSelected()); } options.fullScreen.setSelected(fullScreen); options.fullScreenAllMonitors.setSelected(viewer.fullScreenAllMonitors.getValue()); options.useLocalCursor.setSelected(viewer.useLocalCursor.getValue()); options.acceptBell.setSelected(viewer.acceptBell.getValue()); String scaleString = viewer.scalingFactor.getValue(); if (scaleString.equalsIgnoreCase("Auto")) { options.scalingFactor.setSelectedItem("Auto"); } else if(scaleString.equalsIgnoreCase("FixedRatio")) { options.scalingFactor.setSelectedItem("Fixed Aspect Ratio"); } else { digit = Integer.parseInt(scaleString); if (digit >= 1 && digit <= 1000) { options.scalingFactor.setSelectedItem(digit+"%"); } else { digit = Integer.parseInt(viewer.scalingFactor.getDefaultStr()); options.scalingFactor.setSelectedItem(digit+"%"); } int scaleFactor = Integer.parseInt(scaleString.substring(0, scaleString.length())); } if (viewer.desktopSize.getValue() != null && viewer.desktopSize.getValue().split("x").length == 2) { options.desktopSize.setSelected(true); String desktopWidth = viewer.desktopSize.getValue().split("x")[0]; options.desktopWidth.setText(desktopWidth); String desktopHeight = viewer.desktopSize.getValue().split("x")[1]; options.desktopHeight.setText(desktopHeight); } } public void getOptions() { autoSelect = options.autoSelect.isSelected(); if (fullColour != options.fullColour.isSelected()) { formatChange = true; forceNonincremental = true; } fullColour = options.fullColour.isSelected(); if (!fullColour) { int newLowColourLevel = (options.veryLowColour.isSelected() ? 0 : options.lowColour.isSelected() ? 1 : 2); if (newLowColourLevel != lowColourLevel) { lowColourLevel = newLowColourLevel; formatChange = true; forceNonincremental = true; } } int newEncoding = (options.zrle.isSelected() ? Encodings.encodingZRLE : options.hextile.isSelected() ? Encodings.encodingHextile : options.tight.isSelected() ? Encodings.encodingTight : Encodings.encodingRaw); if (newEncoding != currentEncoding) { currentEncoding = newEncoding; encodingChange = true; } viewer.customCompressLevel.setParam(options.customCompressLevel.isSelected()); if (cp.customCompressLevel != viewer.customCompressLevel.getValue()) { cp.customCompressLevel = viewer.customCompressLevel.getValue(); encodingChange = true; } if (Integer.parseInt(options.compressLevel.getSelectedItem().toString()) >= 0 && Integer.parseInt(options.compressLevel.getSelectedItem().toString()) <= 9) { viewer.compressLevel.setParam(options.compressLevel.getSelectedItem().toString()); } else { viewer.compressLevel.setParam(viewer.compressLevel.getDefaultStr()); } if (cp.compressLevel != viewer.compressLevel.getValue()) { cp.compressLevel = viewer.compressLevel.getValue(); encodingChange = true; } viewer.noJpeg.setParam(!options.noJpeg.isSelected()); if (cp.noJpeg != viewer.noJpeg.getValue()) { cp.noJpeg = viewer.noJpeg.getValue(); encodingChange = true; } viewer.qualityLevel.setParam(options.qualityLevel.getSelectedItem().toString()); if (cp.qualityLevel != viewer.qualityLevel.getValue()) { cp.qualityLevel = viewer.qualityLevel.getValue(); encodingChange = true; } if (!options.x509ca.getText().equals("")) CSecurityTLS.x509ca.setParam(options.x509ca.getText()); if (!options.x509crl.getText().equals("")) CSecurityTLS.x509crl.setParam(options.x509crl.getText()); viewer.sendLocalUsername.setParam(options.sendLocalUsername.isSelected()); viewer.viewOnly.setParam(options.viewOnly.isSelected()); viewer.acceptClipboard.setParam(options.acceptClipboard.isSelected()); viewer.sendClipboard.setParam(options.sendClipboard.isSelected()); viewer.acceptBell.setParam(options.acceptBell.isSelected()); String scaleString = options.scalingFactor.getSelectedItem().toString(); String oldScaleFactor = viewer.scalingFactor.getValue(); if (scaleString.equalsIgnoreCase("Fixed Aspect Ratio")) { scaleString = new String("FixedRatio"); } else if (scaleString.equalsIgnoreCase("Auto")) { scaleString = new String("Auto"); } else { scaleString=scaleString.substring(0, scaleString.length()-1); } if (!oldScaleFactor.equals(scaleString)) { viewer.scalingFactor.setParam(scaleString); if ((options.fullScreen.isSelected() == fullScreen) && (desktop != null)) recreateViewport(); } clipboardDialog.setSendingEnabled(viewer.sendClipboard.getValue()); viewer.menuKey.setParam(MenuKey.getMenuKeySymbols()[options.menuKey.getSelectedIndex()].name); F8Menu.f8.setText("Send "+KeyEvent.getKeyText(MenuKey.getMenuKeyCode())); setShared(options.shared.isSelected()); viewer.useLocalCursor.setParam(options.useLocalCursor.isSelected()); if (cp.supportsLocalCursor != viewer.useLocalCursor.getValue()) { cp.supportsLocalCursor = viewer.useLocalCursor.getValue(); encodingChange = true; if (desktop != null) desktop.resetLocalCursor(); } viewer.extSSH.setParam(options.sshUseExt.isSelected()); checkEncodings(); if (state() != RFBSTATE_NORMAL) { /* Process security types which don't use encryption */ if (options.encNone.isSelected()) { if (options.secNone.isSelected()) Security.EnableSecType(Security.secTypeNone); if (options.secVnc.isSelected()) Security.EnableSecType(Security.secTypeVncAuth); if (options.secPlain.isSelected()) Security.EnableSecType(Security.secTypePlain); if (options.secIdent.isSelected()) Security.EnableSecType(Security.secTypeIdent); } else { Security.DisableSecType(Security.secTypeNone); Security.DisableSecType(Security.secTypeVncAuth); Security.DisableSecType(Security.secTypePlain); Security.DisableSecType(Security.secTypeIdent); } /* Process security types which use TLS encryption */ if (options.encTLS.isSelected()) { if (options.secNone.isSelected()) Security.EnableSecType(Security.secTypeTLSNone); if (options.secVnc.isSelected()) Security.EnableSecType(Security.secTypeTLSVnc); if (options.secPlain.isSelected()) Security.EnableSecType(Security.secTypeTLSPlain); if (options.secIdent.isSelected()) Security.EnableSecType(Security.secTypeTLSIdent); } else { Security.DisableSecType(Security.secTypeTLSNone); Security.DisableSecType(Security.secTypeTLSVnc); Security.DisableSecType(Security.secTypeTLSPlain); Security.DisableSecType(Security.secTypeTLSIdent); } /* Process security types which use X509 encryption */ if (options.encX509.isSelected()) { if (options.secNone.isSelected()) Security.EnableSecType(Security.secTypeX509None); if (options.secVnc.isSelected()) Security.EnableSecType(Security.secTypeX509Vnc); if (options.secPlain.isSelected()) Security.EnableSecType(Security.secTypeX509Plain); if (options.secIdent.isSelected()) Security.EnableSecType(Security.secTypeX509Ident); } else { Security.DisableSecType(Security.secTypeX509None); Security.DisableSecType(Security.secTypeX509Vnc); Security.DisableSecType(Security.secTypeX509Plain); Security.DisableSecType(Security.secTypeX509Ident); } /* Process *None security types */ if (options.secNone.isSelected()) { if (options.encNone.isSelected()) Security.EnableSecType(Security.secTypeNone); if (options.encTLS.isSelected()) Security.EnableSecType(Security.secTypeTLSNone); if (options.encX509.isSelected()) Security.EnableSecType(Security.secTypeX509None); } else { Security.DisableSecType(Security.secTypeNone); Security.DisableSecType(Security.secTypeTLSNone); Security.DisableSecType(Security.secTypeX509None); } /* Process *Vnc security types */ if (options.secVnc.isSelected()) { if (options.encNone.isSelected()) Security.EnableSecType(Security.secTypeVncAuth); if (options.encTLS.isSelected()) Security.EnableSecType(Security.secTypeTLSVnc); if (options.encX509.isSelected()) Security.EnableSecType(Security.secTypeX509Vnc); } else { Security.DisableSecType(Security.secTypeVncAuth); Security.DisableSecType(Security.secTypeTLSVnc); Security.DisableSecType(Security.secTypeX509Vnc); } /* Process *Plain security types */ if (options.secPlain.isSelected()) { if (options.encNone.isSelected()) Security.EnableSecType(Security.secTypePlain); if (options.encTLS.isSelected()) Security.EnableSecType(Security.secTypeTLSPlain); if (options.encX509.isSelected()) Security.EnableSecType(Security.secTypeX509Plain); } else { Security.DisableSecType(Security.secTypePlain); Security.DisableSecType(Security.secTypeTLSPlain); Security.DisableSecType(Security.secTypeX509Plain); } /* Process *Ident security types */ if (options.secIdent.isSelected()) { if (options.encNone.isSelected()) Security.EnableSecType(Security.secTypeIdent); if (options.encTLS.isSelected()) Security.EnableSecType(Security.secTypeTLSIdent); if (options.encX509.isSelected()) Security.EnableSecType(Security.secTypeX509Ident); } else { Security.DisableSecType(Security.secTypeIdent); Security.DisableSecType(Security.secTypeTLSIdent); Security.DisableSecType(Security.secTypeX509Ident); } if (options.sshTunnel.isSelected()) { if (options.sshUseGateway.isSelected()) { String user = options.sshUser.getText(); String host = options.sshHost.getText(); String port = options.sshPort.getText(); viewer.via.setParam(user+"@"+host+":"+port); } else { viewer.tunnel.setParam(true); } } viewer.extSSH.setParam(options.sshUseExt.isSelected()); viewer.extSSHClient.setParam(options.sshClient.getText()); if (options.sshArgsCustom.isSelected()) viewer.extSSHArgs.setParam(options.sshArguments.getText()); viewer.sshConfig.setParam(options.sshConfig.getText()); viewer.sshKeyFile.setParam(options.sshKeyFile.getText()); } String desktopSize = (options.desktopSize.isSelected()) ? options.desktopWidth.getText() + "x" + options.desktopHeight.getText() : ""; viewer.desktopSize.setParam(desktopSize); if (options.fullScreen.isSelected() ^ fullScreen) { viewer.fullScreenAllMonitors.setParam(options.fullScreenAllMonitors.isSelected()); toggleFullScreen(); } else { if (viewer.fullScreenAllMonitors.getValue() != options.fullScreenAllMonitors.isSelected()) { viewer.fullScreenAllMonitors.setParam(options.fullScreenAllMonitors.isSelected()); if (desktop != null) recreateViewport(); } else { viewer.fullScreenAllMonitors.setParam(options.fullScreenAllMonitors.isSelected()); } } } public void toggleFullScreen() { if (viewer.embed.getValue()) return; fullScreen = !fullScreen; menu.fullScreen.setSelected(fullScreen); if (viewport != null) { if (!viewport.lionFSSupported()) { recreateViewport(); } else { viewport.toggleLionFS(); } } } // writeClientCutText() is called from the clipboard dialog public void writeClientCutText(String str, int len) { if (state() != RFBSTATE_NORMAL || shuttingDown) return; writer().writeClientCutText(str, len); } public void writeKeyEvent(int keysym, boolean down) { if (state() != RFBSTATE_NORMAL || shuttingDown) return; writer().writeKeyEvent(keysym, down); } public void writeKeyEvent(KeyEvent ev) { if (viewer.viewOnly.getValue() || shuttingDown) return; boolean down = (ev.getID() == KeyEvent.KEY_PRESSED); int keySym, keyCode = ev.getKeyCode(); // If neither the keyCode or keyChar are defined, then there's // really nothing that we can do with this. The fn key on OS-X // fires events like this when pressed but does not fire a // corresponding release event. if (keyCode == 0 && ev.getKeyChar() == KeyEvent.CHAR_UNDEFINED) return; if (!down) { Integer iter = downKeySym.get(keyCode); if (iter == null) { // Note that dead keys will raise this sort of error falsely // See https://bugs.openjdk.java.net/browse/JDK-6534883 vlog.error("Unexpected key release of keyCode "+keyCode); String fmt = ev.paramString().replaceAll("%","%%"); vlog.error(String.format(fmt.replaceAll(",","%n "))); return; } vlog.debug(String.format("Key released: 0x%04x => 0x%04x", keyCode, iter)); writeKeyEvent(iter, false); downKeySym.remove(keyCode); return; } keySym = Keysyms.translateKeyEvent(ev); if (keySym == Keysyms.VoidSymbol) return; boolean need_cheat = true; if (VncViewer.os.startsWith("windows")) { // Windows doesn't have a proper AltGr, but handles it using fake // Ctrl+Alt. Unfortunately X11 doesn't generally like the combination // Ctrl+Alt+AltGr, which we usually end up with when Xvnc tries to // get everything in the correct state. Cheat and temporarily release // Ctrl and Alt whenever we get a key with a symbol. if (KeyEvent.getKeyText(keyCode).isEmpty()) need_cheat = false; else if (!downKeySym.containsValue(Keysyms.Control_L) && !downKeySym.containsValue(Keysyms.Control_R)) need_cheat = false; else if (!downKeySym.containsValue(Keysyms.Alt_L) && !downKeySym.containsValue(Keysyms.Alt_R)) need_cheat = false; if (need_cheat) { vlog.info("Faking release of AltGr (Ctrl+Alt)"); if (downKeySym.containsValue(Keysyms.Control_L)) writeKeyEvent(Keysyms.Control_L, false); if (downKeySym.containsValue(Keysyms.Control_R)) writeKeyEvent(Keysyms.Control_R, false); if (downKeySym.containsValue(Keysyms.Alt_L)) writeKeyEvent(Keysyms.Alt_L, false); if (downKeySym.containsValue(Keysyms.Alt_R)) writeKeyEvent(Keysyms.Alt_R, false); } } vlog.debug(String.format("Key pressed: 0x%04x '%s' => 0x%04x", keyCode, Character.toString(ev.getKeyChar()), keySym)); downKeySym.put(keyCode, keySym); writeKeyEvent(keySym, down); if (VncViewer.os.startsWith("windows")) { if (need_cheat) { vlog.debug("Restoring AltGr state"); if (downKeySym.containsValue(Keysyms.Control_L)) writeKeyEvent(Keysyms.Control_L, true); if (downKeySym.containsValue(Keysyms.Control_R)) writeKeyEvent(Keysyms.Control_R, true); if (downKeySym.containsValue(Keysyms.Alt_L)) writeKeyEvent(Keysyms.Alt_L, true); if (downKeySym.containsValue(Keysyms.Alt_R)) writeKeyEvent(Keysyms.Alt_R, true); } } } public void writePointerEvent(MouseEvent ev) { if (state() != RFBSTATE_NORMAL || shuttingDown) return; switch (ev.getID()) { case MouseEvent.MOUSE_PRESSED: buttonMask = 1; if ((ev.getModifiers() & KeyEvent.ALT_MASK) != 0) buttonMask = 2; if ((ev.getModifiers() & KeyEvent.META_MASK) != 0) buttonMask = 4; break; case MouseEvent.MOUSE_RELEASED: buttonMask = 0; break; } if (cp.width != desktop.scaledWidth || cp.height != desktop.scaledHeight) { int sx = (desktop.scaleWidthRatio == 1.00) ? ev.getX() : (int)Math.floor(ev.getX() / desktop.scaleWidthRatio); int sy = (desktop.scaleHeightRatio == 1.00) ? ev.getY() : (int)Math.floor(ev.getY() / desktop.scaleHeightRatio); ev.translatePoint(sx - ev.getX(), sy - ev.getY()); } writer().writePointerEvent(new Point(ev.getX(), ev.getY()), buttonMask); } public void writeWheelEvent(MouseWheelEvent ev) { if (state() != RFBSTATE_NORMAL || shuttingDown) return; int x, y; int clicks = ev.getWheelRotation(); if (clicks < 0) { buttonMask = 8; } else { buttonMask = 16; } for (int i = 0; i < Math.abs(clicks); i++) { x = ev.getX(); y = ev.getY(); writer().writePointerEvent(new Point(x, y), buttonMask); buttonMask = 0; writer().writePointerEvent(new Point(x, y), buttonMask); } } synchronized void releaseDownKeys() { for (Map.Entry entry : downKeySym.entrySet()) writeKeyEvent(entry.getValue(), false); downKeySym.clear(); } // this is a special ActionListener passed in by the // Java Plug-in software to control applet's close behavior public void setCloseListener(ActionListener cl) { closeListener = cl; } public void actionPerformed(ActionEvent e) {} public Socket getSocket() { return sock; } //////////////////////////////////////////////////////////////////// // The following methods are called from both RFB and GUI threads // checkEncodings() sends a setEncodings message if one is needed. private void checkEncodings() { if (encodingChange && (writer() != null)) { vlog.info("Requesting " + Encodings.encodingName(currentEncoding) + " encoding"); writer().writeSetEncodings(currentEncoding, true); encodingChange = false; } } // the following never change so need no synchronization: // viewer object is only ever accessed by the GUI thread so needs no // synchronization (except for one test in DesktopWindow - see comment // there). VncViewer viewer; // access to desktop by different threads is specified in DesktopWindow // the following need no synchronization: public static UserPasswdGetter upg; public UserMsgBox msg; // shuttingDown is set by the GUI thread and only ever tested by the RFB // thread after the window has been destroyed. boolean shuttingDown = false; // reading and writing int and boolean is atomic in java, so no // synchronization of the following flags is needed: int lowColourLevel; // All menu, options, about and info stuff is done in the GUI thread (apart // from when constructed). F8Menu menu; OptionsDialog options; // clipboard sync issues? ClipboardDialog clipboardDialog; // the following are only ever accessed by the GUI thread: int buttonMask; private Socket sock; protected DesktopWindow desktop; // FIXME: should be private public PixelFormat serverPF; private PixelFormat fullColourPF; private boolean pendingPFChange; private PixelFormat pendingPF; private int currentEncoding, lastServerEncoding; private boolean formatChange; private boolean encodingChange; private boolean firstUpdate; private boolean pendingUpdate; private boolean continuousUpdates; private boolean forceNonincremental; private boolean supportsSyncFence; public int menuKeyCode; Viewport viewport; private boolean fullColour; private boolean autoSelect; boolean fullScreen; private HashMap downKeySym; public ActionListener closeListener = null; static LogWriter vlog = new LogWriter("CConn"); }