Java / Swing:系统剪贴板的所有权

时间:2011-08-27 17:12:56

标签: java image swing awt clipboard

我正在编写一个小型Java程序,它应该运行一个外部程序,将图像复制到系统剪贴板(即Windows 7“剪切工具”),等待它完成,将图像从剪贴板保存到磁盘并将URL(可从中访问图像)复制到剪贴板。简而言之,它应该是:

  1. 运行外部工具并等待它
  2. 从剪贴板复制图像
  3. 将字符串复制到剪贴板
  4. 这个,我的程序完全可以做到。但是,我想使用Swing / AWT来呈现用户界面。我正在使用系统托盘图标,但为了简单起见,它也可以是框架中的JButton。单击该按钮时,应执行上述过程。第一次这样做,它应该工作。将图像复制,粘贴到磁盘,并将字符串复制到剪贴板。然后,第二次单击按钮,就好像我的程序没有意识到剪贴板已经更新,因为它仍然从第一次看到自己的字符串。之后,我的剪贴板处理类失去了所有权,实际上,每次尝试该过程都会失败。

    import java.awt.Toolkit;
    import java.awt.datatransfer.Clipboard;
    import java.awt.datatransfer.ClipboardOwner;
    import java.awt.datatransfer.DataFlavor;
    import java.awt.datatransfer.StringSelection;
    import java.awt.datatransfer.Transferable;
    import java.awt.event.ActionEvent;
    import java.awt.event.ActionListener;
    import java.awt.image.BufferedImage;
    import java.io.IOException;
    
    import javax.swing.JButton;
    import javax.swing.JFrame;
    
    public class Main {
        private static BufferedImage image; //the image from clipboard to be saved
    
        public static void main(String[] args) throws InterruptedException, IOException {
            new GUI();
        }
    
        public static void run(String filename) throws IOException, InterruptedException {
            CBHandler cbh = new CBHandler();
    
            //run tool, tool will copy an image to system clipboard
            Process p = Runtime.getRuntime().exec("C:\\Windows\\system32\\SnippingTool.exe");
            p.waitFor();
    
            //copy image from clipboard
            image = cbh.getClipboard();
            if(image == null) {
                System.out.println("No image found in clipboard.");
                return;
            }
    
            //save image to disk...
    
            //copy file link to clipboard
            String link = "http://somedomain.com/" + filename;
            cbh.setClipboard(link);
        }
    }
    
    class CBHandler implements ClipboardOwner {
        public BufferedImage getClipboard() {
            Transferable t = Toolkit.getDefaultToolkit().getSystemClipboard().getContents(null);
    
            try {
                if(t.isDataFlavorSupported(DataFlavor.imageFlavor))
                    return (BufferedImage) t.getTransferData(DataFlavor.imageFlavor);
            }
            catch(Exception e) {
                e.printStackTrace();
            }
            return null;
        }
    
        public void setClipboard(String str) {
            StringSelection strsel = new StringSelection(str);
            Toolkit.getDefaultToolkit().getSystemClipboard().setContents(strsel, this);
        }
    
        @Override
        public void lostOwnership(Clipboard arg0, Transferable arg1) {
            System.out.println("Lost ownership!");
        }
    }
    
    class GUI extends JFrame {
        public GUI() {
            JButton button = new JButton("Run");
            button.addActionListener(new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent arg0) {
                    try {
                        Main.run("saveFile.png");
                    } catch (IOException e) {
                        e.printStackTrace();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            });
    
            add(button);
            pack();
            setVisible(true);
        }
    }
    

    如果您尝试运行它,请注意,在第二次运行时,仅在尝试复制图像后调用lostOwnership方法。我猜这是我的问题的根源,我不知道它为什么会发生,除了它只发生在由Swing事件触发时。任何解决这个问题的帮助都表示赞赏。

2 个答案:

答案 0 :(得分:0)

一个猜测:您正在AWT事件派发线程上进行整个处理(调用另一个进程)(例如直接从ActionListener或类似程序)。

剪贴板更改消息也将由EDT上的VM处理...但仅在完成按钮点击后才会处理。

道德:不要在EDT上做长期运行的东西(以及应该在事件队列中排队的东西) - 而是为此开始一个新的线程。

答案 1 :(得分:0)

理解失去的所有权问题的关键在于此行

Toolkit.getDefaultToolkit().getSystemClipboard().setContents(strsel, this);

您传递的第二个参数是ClipboardOwner。 clipboard.setContents的JavaDocs说

  

如果现有所有者与参数所有者不同,则通过在该所有者上调用ClipboardOwner.lostOwnership()通知该所有者它不再拥有剪贴板内容的所有权。 setContents()的实现是免费的,不直接从此方法调用lostOwnership()。例如,稍后可以在不同的线程上调用lostOwnership()。这同样适用于在此剪贴板上注册的FlavorListeners。

好的,那发生了什么?传入所有者时,剪贴板现在具有对该对象的引用。在这种情况下,它是CBHandler。然后创建一个新的并尝试再次设置内容。剪贴板然后回到旧的所有者(您的原始实例)并告诉它“嘿,您不再是所有者了”。

public synchronized void setContents(Transferable contents, ClipboardOwner owner) {
    final ClipboardOwner oldOwner = this.owner;
    final Transferable oldContents = this.contents;

    this.owner = owner;
    this.contents = contents;

    if (oldOwner != null && oldOwner != owner) {
        EventQueue.invokeLater(new Runnable() {
            public void run() {
                oldOwner.lostOwnership(Clipboard.this, oldContents);
            }
        });
    }
    fireFlavorsChanged();
}

你必须提供关于另一个问题的更多细节“就好像我的程序没有意识到剪贴板已经更新,因为它仍然从第一次看到它自己的字符串。”