<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    小魚(yú)的空氣

    記錄我所思

    用java打造任意形狀窗口和透明窗口



    圖形界面開(kāi)發(fā)對(duì)于Java來(lái)說(shuō)并非它的長(zhǎng)項(xiàng),開(kāi)發(fā)者經(jīng)常會(huì)碰到各種各樣的限制,比如,如何打造一款任意形狀的窗口如何可以透過(guò)窗口顯示它覆蓋下的內(nèi)容

    考慮到Java并沒(méi)有被設(shè)計(jì)成支持以上的功能,所以,你能得到的永遠(yuǎn)是方方正正的窗口,毫無(wú)新意,當(dāng)然,我們可以通過(guò)JNI調(diào)用本地代碼來(lái)完成,但是這就失去了java可移植性的意義,那么,用純粹的java代碼如何實(shí)現(xiàn)以上兩種功能呢?

    下文提供了一個(gè)實(shí)現(xiàn)的參考

    預(yù)備知識(shí)
    1.java.awt.Robot,這個(gè)類(lèi)是一個(gè)功能非常強(qiáng)大的類(lèi),通過(guò)它我們可以控制鼠標(biāo)和鍵盤(pán)的響應(yīng),不過(guò)這里,我們只需要用到它的一個(gè)功能--截屏,也就是得到當(dāng)前屏幕的快照(screenshot)
    2.我們可以通過(guò)調(diào)用一個(gè)swing組件的paintComponent(Graphics g)方法用特定的Graphics為其定制特殊的外觀

    首先聲明的一點(diǎn)是,本文中的實(shí)現(xiàn)方法只是一個(gè)欺騙的手法,因?yàn)橐雽?shí)現(xiàn)前文中的功能,我們幾乎的重寫(xiě)一套Swing出來(lái),這里,簡(jiǎn)便的做法是,我們通過(guò)robot類(lèi)來(lái)獲得當(dāng)前屏幕的快照,然后貼在我們需要的窗口上,這樣,不就實(shí)現(xiàn)了透明的功能了嗎?順便提一下,幾年前日本發(fā)明的隱形衣也是依靠這種機(jī)制,這種衣服在身后附帶一套攝像機(jī),然后即時(shí)的將拍下的內(nèi)容顯示在衣服的正面,因此,當(dāng)別人看過(guò)來(lái)時(shí),仿佛就通過(guò)人和衣服看到了身后的場(chǎng)景^_^

    另外,還要感謝Joshua Chris leniz的Swing Hack一書(shū),以上充滿(mǎn)創(chuàng)新的方法正來(lái)自他的書(shū)中

    好咯,讓我們具體看一下細(xì)節(jié)的處理:
    第一,我們要得到當(dāng)前屏幕的快照,保存到一個(gè)Image對(duì)象[color=Red]background[/color]中:
    ? public void updateBackground() {
    ??try {
    ???Robot rbt = new Robot();
    ???Toolkit tk = Toolkit.getDefaultToolkit();
    ???Dimension dim = tk.getScreenSize();
    ???[color=Red]background [/color]= rbt.createScreenCapture(new Rectangle(0, 0, (int) dim
    ?????.getWidth(), (int) dim.getHeight()));
    ??} catch (Exception ex) {
    ?? }
    ?}

    第二,我們需要讓窗口顯示這個(gè)圖像,也就是說(shuō),讓窗口的背景圖像是這副圖像,以達(dá)到透明的欺騙效果:

    public void paintComponent(Graphics g) {
    ??Point pos = this.getLocationOnScreen();
    ??Point offset = new Point(-pos.x, -pos.y);
    ??g.drawImage([color=Red]background[/color], offset.x, offset.y, null);
    ?}

    在swing hack一書(shū)中,作者也給出了他的實(shí)現(xiàn),然而,運(yùn)行結(jié)果表明該實(shí)現(xiàn)存在很大的問(wèn)題:窗口經(jīng)常無(wú)法即時(shí)更新,往往背景變了,窗口里顯示的卻還是以前的背景。仔細(xì)研究了他的代碼,我發(fā)現(xiàn)了問(wèn)題的根結(jié),同時(shí)也是這種實(shí)現(xiàn)技術(shù)的關(guān)鍵要素--如何讓窗口在正確的時(shí)機(jī)里更新顯示,下面,我們會(huì)討論這一點(diǎn)

    第三,更新窗口外觀
    前兩步的操作,只能得到一副當(dāng)前屏幕的快照,一旦背景變化了,或者窗口移動(dòng)了,變化大小了,那么我們制作的窗口將永遠(yuǎn)無(wú)法和和屏幕背景聯(lián)合成整體,也就失去了透明的效果;同時(shí),我們也不可能每時(shí)每刻都調(diào)用
    updateBackground() 方法獲得最新的背景,可行的方法是,通過(guò)對(duì)事件的監(jiān)聽(tīng)來(lái)選擇適當(dāng)?shù)臅r(shí)間來(lái)更新外觀

    我們應(yīng)該可以達(dá)到這三點(diǎn)共識(shí)
    1。窗口移動(dòng)或改變大小時(shí),背景圖像一般是不會(huì)發(fā)生變化的,我們不需要得到新的屏幕快照,只用將以前得到的背景中適當(dāng)?shù)牟糠仲N到窗口上,調(diào)用repaint()方法就足已
    2。要獲得最新的屏幕快照,必須先將窗口隱藏起來(lái),調(diào)用updateBackground() 得到圖像后再把窗口顯示,我們可以用refresh方法來(lái)表示
    ?refresh(){
    ??? frame.hide();
    ??? updateBackground() ;
    ?? frame.show();
    }
    3。如果背景改變了,那么一定是別的windows程序獲得了或失去了事件焦點(diǎn),也就是說(shuō)我們關(guān)注的窗口將觸發(fā)焦點(diǎn)得失事件,這個(gè)時(shí)候需要調(diào)用refresh() 得到最新的屏幕快照


    看到這里,你或許認(rèn)為已經(jīng)沒(méi)有技術(shù)難點(diǎn)了,然而,此時(shí)才是我們[color=Red]最需要關(guān)注的地方[/color]:
    參看第三點(diǎn),我們需要在窗口得失焦點(diǎn)時(shí)調(diào)用refresh() 方法;參看第一點(diǎn),我們調(diào)用refresh() 方法時(shí)需要先將窗口隱藏,然后再顯示。于是問(wèn)題來(lái)了,在隱藏和顯示窗口時(shí),同樣會(huì)觸發(fā)得失焦點(diǎn)事件,得失焦點(diǎn)事件又將觸發(fā)新的隱藏和顯示窗口事件(為了得到新的屏幕快照),這就使程序陷入了死循環(huán)中,我們必須加以控制,使得第二次觸發(fā)焦點(diǎn)得失事件時(shí)不調(diào)用refresh()方法

    作者的辦法是加一個(gè)線程來(lái)控制,通過(guò)判斷時(shí)間間隔長(zhǎng)短來(lái)決定是否調(diào)用refresh()方法,可是,這個(gè)條件是程序非常的不穩(wěn)定,因?yàn)橥{(diào)用時(shí)間會(huì)根據(jù)系統(tǒng)繁忙度而改變,使得需要更新時(shí)不能更新,不需要更新的時(shí)候反而更新了
    因此,我決定采取新的解決方案,能不能隱藏/顯示窗口時(shí)不觸發(fā)得失焦點(diǎn)事件呢?
    解決方法很簡(jiǎn)單,我拋開(kāi)了傳統(tǒng)的setVisible()或者show(),hide()方法,而是使用setLocation()方法,因?yàn)檎{(diào)用setLocation()方法時(shí)窗口不會(huì)失去焦點(diǎn),同時(shí),只要用類(lèi)似setLocation(-2000,-2000)方法也同樣可以輕松的讓窗口在屏幕中消失
    下面是我的全部代碼:


    import java.awt.BorderLayout;
    import java.awt.Dimension;
    import java.awt.Graphics;
    import java.awt.Image;
    import java.awt.Point;
    import java.awt.Rectangle;
    import java.awt.Robot;
    import java.awt.Toolkit;
    import java.awt.event.ComponentEvent;
    import java.awt.event.ComponentListener;
    import java.awt.event.WindowEvent;
    import java.awt.event.WindowFocusListener;

    import javax.swing.JButton;
    import javax.swing.JComponent;
    import javax.swing.JFrame;
    import javax.swing.JLabel;
    import javax.swing.UIManager;

    import com.birosoft.liquid.LiquidLookAndFeel;

    public class TestEvent extends JComponent
    ??implements?ComponentListener,WindowFocusListener {
    ?
    ?private JFrame frame;

    ?private boolean start = false;

    ?private Image background;

    ?private Point p;

    ?// 獲得當(dāng)前屏幕快照
    ?public void updateBackground() {
    ??try {
    ???Robot rbt = new Robot();
    ???Toolkit tk = Toolkit.getDefaultToolkit();
    ???Dimension dim = tk.getScreenSize();
    ???background = rbt.createScreenCapture(new Rectangle(0, 0, (int) dim
    ?????.getWidth(), (int) dim.getHeight()));
    ??} catch (Exception ex) {
    ???// p(ex.toString());
    ???// 此方法沒(méi)有申明過(guò) ,因?yàn)闊o(wú)法得知上下文 。因?yàn)椴挥绊憟?zhí)行效果 ,先注釋掉它 ex.printStackTrace();
    ??}

    ?}

    ?// 將窗口掉離出屏幕以獲得純粹的背景圖象
    ?public void refresh() {
    ??if (start == true) {
    ???this.updateBackground();
    ???frame.setLocation(p);
    ???if (p.x < 0 || p.y < 0)
    ????frame.setLocation(0, 0);
    ???this.repaint();
    ??}
    ?}

    ?public void componentHidden(ComponentEvent e) {
    ??System.out.println("Hidden");
    ?}

    ?// 窗口移動(dòng)時(shí)
    ?public void componentMoved(ComponentEvent e) {
    ??System.out.println("moved");
    ??this.repaint();
    ?}

    ?// 窗口改變大小時(shí)
    ?public void componentResized(ComponentEvent e) {
    ??System.out.println("resized");
    ??this.repaint();
    ?}

    ?public void componentShown(ComponentEvent e) {
    ??System.out.println("shown");
    ?}

    ?// 窗口得到焦點(diǎn)后,用refresh()方法更新界面
    ?public void windowGainedFocus(WindowEvent e) {
    ??System.out.println("gainedFocus");
    ??refresh();
    ??start = false;
    ?}

    ?// 窗口失去焦點(diǎn)后,將其移出屏幕
    ?public void windowLostFocus(WindowEvent e) {
    ??System.out.println("lostFocus");
    ??if (frame.isShowing() == true) {
    ???System.out.println("visible");
    ??} else {
    ???System.out.println("invisible");
    ??}
    ??start = true;
    ??p = frame.getLocation();
    ??frame.setLocation(-2000, -2000);
    ?}

    ?public TestEvent(JFrame frame) {
    ??super();
    ??this.frame = frame;
    ??updateBackground();
    ??this.setSize(200, 120);
    ??this.setVisible(true);
    ??frame.addComponentListener(this);
    ??frame.addWindowFocusListener(this);

    ?}

    ?// 繪制外觀,注意,其中 pos,offset 是為了將特定部分的圖象貼到窗口上
    ?public void paintComponent(Graphics g) {
    ??Point pos = this.getLocationOnScreen();
    ??Point offset = new Point(-pos.x, -pos.y);
    ??g.drawImage(background, offset.x, offset.y, null);
    ?}

    ?/**
    ? * @param args
    ? */
    ?public static void main(String[] args) {
    ??try {
    ???// UIManager.setLookAndFeel("org.fife.plaf.Office2003.Office2003LookAndFeel");
    ???// UIManager.setLookAndFeel("org.fife.plaf.OfficeXP.OfficeXPLookAndFeel");
    ???// UIManager.setLookAndFeel("org.fife.plaf.OfficeXP.OfficeXPLookAndFeel");
    ???UIManager.setLookAndFeel("com.birosoft.liquid.LiquidLookAndFeel");
    ???LiquidLookAndFeel.setLiquidDecorations(true);
    ???// LiquidLookAndFeel.setLiquidDecorations(true, "mac");
    ???// UIManager.setLookAndFeel(new SubstanceLookAndFeel());
    ???// UIManager.setLookAndFeel(new SmoothLookAndFeel());
    ???// UIManager.setLookAndFeel(new QuaquaLookAndFeel());
    ???// UIManager.put("swing.boldMetal", false);
    ???if (System.getProperty("substancelaf.useDecorations") == null) {
    ????JFrame.setDefaultLookAndFeelDecorated(true);
    ????// JDialog.setDefaultLookAndFeelDecorated(true);
    ???}
    ???System.setProperty("sun.awt.noerasebackground", "true");
    ???// SubstanceLookAndFeel.setCurrentTheme(new
    ???// SubstanceLightAquaTheme());

    ???// UIManager.setLookAndFeel("org.fife.plaf.VisualStudio2005.VisualStudio2005LookAndFeel");
    ??} catch (Exception e) {
    ???System.err.println("Oops!? Something went wrong!");
    ??}

    ??JFrame frame = new JFrame("Transparent Window");
    ??TestEvent t = new TestEvent(frame);
    ??t.setLayout(new BorderLayout());
    ??JButton button = new JButton("This is a button");
    ??t.add("North", button);
    ??JLabel label = new JLabel("This is a label");
    ??t.add("South", label);
    ??frame.getContentPane().add("Center", t);
    ??frame.pack();
    ??frame.setSize(150, 100);
    ??frame.show();
    ??frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    ??// t.start=true;
    ?}

    }


    當(dāng)然,以上代碼還存在許多不足,比如在移動(dòng)窗口時(shí)性能會(huì)有影響,應(yīng)該可以通過(guò)線程來(lái)控制其更新的頻率來(lái)提升效率,希望大家能一起研究一下,改好了記得通知我哦

    ps:你也許還會(huì)說(shuō),e?怎么沒(méi)講任意形狀窗口如何打造?
    ?????? 呵呵,其實(shí)只要把窗口按上述方法設(shè)置成透明,再去掉邊框,再換上自己的邊框,不就完成了馬?
    ???? 相信聰明的各位一定很容易就辦到咯

    另外,我還在下面附帶了作者的大論,想了解更多的同學(xué)也可以去看看

    http://www.matrix.org.cn/resource/article/44/44186_Swing.html

    posted on 2006-10-23 17:03 小魚(yú) 閱讀(991) 評(píng)論(0)  編輯  收藏


    只有注冊(cè)用戶(hù)登錄后才能發(fā)表評(píng)論。


    網(wǎng)站導(dǎo)航:
     
    <2025年5月>
    27282930123
    45678910
    11121314151617
    18192021222324
    25262728293031
    1234567

    導(dǎo)航

    統(tǒng)計(jì)

    常用鏈接

    留言簿(3)

    我參與的團(tuán)隊(duì)

    隨筆檔案

    文章檔案

    搜索

    最新評(píng)論

    主站蜘蛛池模板: 国产亚洲美女精品久久| 国产性生交xxxxx免费| 一级做α爱过程免费视频 | 久久成人永久免费播放| 亚洲日韩AV无码一区二区三区人| 久久久久亚洲精品美女| 精品国产香蕉伊思人在线在线亚洲一区二区| 丁香花免费完整高清观看| 久章草在线精品视频免费观看| 一区免费在线观看| 在线观看亚洲免费| 亚洲精品国产第一综合99久久| 亚洲成av人片在线看片| 亚洲AV日韩AV永久无码久久| 亚洲综合伊人久久大杳蕉| 亚洲精品国产日韩无码AV永久免费网| 好吊妞998视频免费观看在线| 69成人免费视频| 91九色视频无限观看免费| 久久精品毛片免费观看| 日本免费一区二区三区| 久久国产精品成人免费| 日韩a级无码免费视频| 一个人看的www免费视频在线观看 一个人免费视频观看在线www | 亚洲AV无码精品蜜桃| 亚洲成a人片在线观看播放| 久久精品国产亚洲AV无码偷窥| 亚洲va国产va天堂va久久| 久久亚洲国产午夜精品理论片| 亚洲国产精品高清久久久| 亚洲无人区一区二区三区| 在线亚洲人成电影网站色www| 精品国产亚洲男女在线线电影 | 亚洲欧洲日韩不卡| 内射干少妇亚洲69XXX| 亚洲自偷自偷精品| 亚洲一卡2卡4卡5卡6卡在线99| 亚洲AV无码精品蜜桃| 亚洲欧美日韩中文字幕一区二区三区| 天堂亚洲国产中文在线| 亚洲av成人无码网站…|