`
xichao1929
  • 浏览: 40206 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

透明界面实现

阅读更多

想做一个透明的界面,无奈Java中的虽然有JFrame.setOpacity();但是这个东西,不好用。后来发现是通过抓取屏幕背景作为自己的背景图片。然后实现伪透明。桌面背景是很有可能更换的,因此要在适当的时机抓取新的屏幕背景最为自己的背景。这个似乎可以用线程来实现,但是感觉很消耗资源,毕竟桌面背景不是每隔一段时间就更新的。先解释一个屏幕绘制的这个过程。首先是抓取整个屏幕,然后我们选择合适的坐标,将我们的界面所覆盖的那一部分屏幕区域,绘制上去。这样当我们的界面移动时,只要修改适当的坐标,以取得当前覆盖的屏幕区域进行绘制。下面是这个代码片段。有一个地方在解释一下。

           

 

 

 

    @Override

     public void paintComponent(Graphics g) {

     super.paintComponent(g);

     //this.paint(g);

    

     Point pos = this.getLocationOnScreen();

     Point offset = new Point(-pos.x, -pos.y);

     // 有一个问题就是为什么这边要用复数

     g.drawImage(background, offset.x, offset.y, null);

     }

   我们看到这个 offset.xoffset.y是负数,也许我们感到困惑。这个只要搞张图片,用这个方法放在JFrame上,实现一下就明白了。

 

 

 

 

我们分析一下抓取屏幕背景的过程:当我们的界面抓取屏幕背景并作为自己的背景进行绘制后,如果屏幕的背景发生了变化,那么一定是我们的界面失去了焦点。这个时候操作的不在是我们的界面,因此可以将其隐藏起来。而当我们要操作我们的界面时,即获得焦点时,我们这时候再抓取屏幕背景就可以了。因此不需要用线程的,用监听器就可以。除了要用到焦点的监听器之外,我们还需要组件监听器,这个是监听界面的移动,大小的变化等,以及时重绘。这个思路是没有问题的。在实现的过程中可能会出现一些小问题。一些让我们头痛的BUG

第一个BUG:我们可能会把自己本身作为背景的一部分抓取过来,这就露馅了。于是当我们的界面失去焦点时,我们必须把这个界面隐藏起来,放在屏幕之外的位置。这样抓取的时候,就不会露馅了。

第二个BUG:我们思路是这样的,windowsGainedFocusWindowEvent e)即得到焦点的监听器,方法的内容是这样的,抓取屏幕背景,进行重绘。windowsLostFocusWindowEvent e)失去焦点监听器,方法的内容是,将界面隐藏起来。我们的步骤是这样的,我们抓取屏幕背景是在这样的情况发生的,我们失去了焦点,界面隐藏,然后得到焦点,抓取屏幕,绘制屏幕。但是这个步骤在程序运行刚开始时是不成立的,即程序一开始我们就直接得到了焦点,这个时候我们抓取屏幕,但是由于我们的界面还没有来的及隐藏,这就会出现第一个BUG的情况。解决方法是:设置一个标志fresh,是否抓取屏幕。

还有第三个BUG。这个我们先等一下,看一下解决了上述两个问题之后的效果。

下面是完整代码(两个类):

 

<!--EndFragment-->

package transparentUI2;

import java.awt.Point;
import java.awt.event.ComponentEvent;
import java.awt.event.ComponentListener;
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import java.awt.event.WindowEvent;
import java.awt.event.WindowFocusListener;

import javax.swing.JFrame;

public class TestUI extends JFrame implements WindowFocusListener,
		ComponentListener  {
	private JPaneUI jp;
	private boolean fresh = false;
	private Point tem_point;
	 

	public static void main(String args[]) {
		new TestUI();
	}

	public TestUI() {
		this.setSize(200, 300);
		this.setLocationRelativeTo(null);
		this.setDefaultCloseOperation(3);
		jp = new JPaneUI();
		this.add(jp);
		this.addComponentListener(this);
		this.addWindowFocusListener(this);
		 
		this.setVisible(true);
 
	}

	@Override
	public void windowGainedFocus(WindowEvent e) {
		// TODO Auto-generated method stub
		this.refresh();
		fresh = false;
	}
 
	@Override
	public void windowLostFocus(WindowEvent e) {
		// TODO Auto-generated method stub

		fresh = true;
		tem_point = this.getLocation();// 记录屏幕消失之前的坐标
		this.setLocation(-2000, -2000);
	}

	public void refresh() {
      
		if (fresh) {
		 
			jp.updateBackground();
			// 此时P应该重新被赋值了 p存放应该就是刷新之前的位置
			this.setLocation(tem_point);
			if (tem_point.x < 0 || tem_point.y < 0)
				this.setLocation(0, 0);
			jp.repaint();

		}
	}

	 
	@Override
	public void componentResized(ComponentEvent e) {
		// TODO Auto-generated method stub

		this.repaint();
	}

	@Override
	public void componentMoved(ComponentEvent e) {
		// TODO Auto-generated method stub
		// this.updateBackground();
		this.repaint();
	}

	@Override
	public void componentShown(ComponentEvent e) {
		// TODO Auto-generated method stub
		System.out.println("是不是在执行这个方法");
	// 	refresh();
		 this.repaint();
	}

	@Override
	public void componentHidden(ComponentEvent e) {
		// TODO Auto-generated method stub

	}
 

}

 

 

package transparentUI2;

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.JFrame;
import javax.swing.JPanel;

public class JPaneUI extends JPanel {
	private JFrame frame;
	private Image background;
	private Point final_point;

	public JPaneUI() {
		updateBackground();
	}

	// 获得当前屏幕快照
	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) {
		}

	}

	@Override
	 public void paintComponent(Graphics g) {
	 super.paintComponent(g);
	 //this.paint(g);
	
	 Point pos = this.getLocationOnScreen();
	 Point offset = new Point(-pos.x, -pos.y);
	 // 有一个问题就是为什么这边要用复数
	 g.drawImage(background, offset.x, offset.y, null);
	 }

 

现在将第三个BUG

本来是这样的我们操作另一个界面时,我们的界面会失去焦点,而等到我们操作我们的界面时,我们才能获得焦点。但是现在有一个例外的情况,就是我们操作的另一个界面的最小化或者是关闭的按钮,这样我们的界面在失去焦点之后,马上得到焦点。就是因为这个时间很短,所以我们抓取屏幕的动作,抓到不改抓到的东西就是其他界面消失过程中的图像。我们的界面消失有一个渐变的过程,这个过程被我们抓到了,然后的结果就是这个样子。如图:

<!--EndFragment-->

 

 

第一张是word的界面,当最小化时,我们的界面就会捕捉到他的一个渐变的过程。这个确实很烦人。解决办法就是,遇到这种情况时,系统多抓几次屏幕。但是会出另一个问题,就是屏幕会闪烁,于是双缓冲也得来。

下面是完整的代码:

package transparentUI;

import java.awt.Point;
import java.awt.event.ComponentEvent;
import java.awt.event.ComponentListener;
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import java.awt.event.WindowEvent;
import java.awt.event.WindowFocusListener;

import javax.swing.JFrame;

public class TestUI extends JFrame implements WindowFocusListener,
		ComponentListener, MouseListener {
	private JPaneUI jp;
	private boolean fresh = false;
	private Point tem_point;
	private int many_fresh;

	public static void main(String args[]) {
		new TestUI();
	}

	public TestUI() {
		this.setSize(200, 300);
		this.setLocationRelativeTo(null);
		this.setDefaultCloseOperation(3);
		jp = new JPaneUI();
		this.add(jp);
		this.addComponentListener(this);
		this.addWindowFocusListener(this);
		this.addMouseListener(this);
		this.setVisible(true);
 
	}

	@Override
	public void windowGainedFocus(WindowEvent e) {
		// TODO Auto-generated method stub
		this.refresh();
		fresh = false;
	}
 
	@Override
	public void windowLostFocus(WindowEvent e) {
		// TODO Auto-generated method stub

		fresh = true;
		tem_point = this.getLocation();// 记录屏幕消失之前的坐标
		this.setLocation(-2000, -2000);
	}

	public void refresh() {
      // int i=6;
		if (fresh) {
			if(many_fresh<=1){
			jp.updateBackground();
			// 此时P应该重新被赋值了 p存放应该就是刷新之前的位置
			this.setLocation(tem_point);
			if (tem_point.x < 0 || tem_point.y < 0)
				this.setLocation(0, 0);
			jp.repaint();
			}
			while (many_fresh > 1) {
				//p = this.getLocationOnScreen();
				this.setLocation(-2000,-2000);
				
				jp.updateBackground();
				// 此时P应该重新被赋值了 p存放应该就是刷新之前的位置
				this.setLocation(tem_point);
				if (tem_point.x < 0 || tem_point.y < 0)
					this.setLocation(0, 0);
				jp.repaint();
				try {
					Thread.sleep(5);
				} catch (InterruptedException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
				
				many_fresh--;
			}

		}
	}

	 
	@Override
	public void componentResized(ComponentEvent e) {
		// TODO Auto-generated method stub

		this.repaint();
	}

	@Override
	public void componentMoved(ComponentEvent e) {
		// TODO Auto-generated method stub
		// this.updateBackground();
		this.repaint();
	}

	@Override
	public void componentShown(ComponentEvent e) {
		// TODO Auto-generated method stub
		System.out.println("是不是在执行这个方法");
	// 	refresh();
		 this.repaint();
	}

	@Override
	public void componentHidden(ComponentEvent e) {
		// TODO Auto-generated method stub

	}

	@Override
	public void mouseClicked(MouseEvent e) {
		// TODO Auto-generated method stub

	}

	@Override
	public void mousePressed(MouseEvent e) {
		// TODO Auto-generated method stub

	}

	@Override
	public void mouseReleased(MouseEvent e) {
		// TODO Auto-generated method stub

	}

	@Override
	public void mouseEntered(MouseEvent e) {
		// TODO Auto-generated method stub
		many_fresh = 1;// 这个是不能改的
		System.out.println("mouseEntered这个应该执行了吧"+many_fresh);
	}

	@Override
	public void mouseExited(MouseEvent e) {
		// TODO Auto-generated method stub
		
		many_fresh = 6;// 这个时候获得焦点就要多次刷新,6次是试出来的比较好的刷新效果,当然5次或者其他的这个自己设置,
		// 不要纠结这个数字,可以更改
		System.out.println("这个应该执行了吧mouseExited"+many_fresh);
	}

}

 

 

 

package transparentUI;

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 javax.swing.JFrame;
import javax.swing.JPanel;

public class JPaneUI extends JPanel {
	 
	private Image background;
 

	public JPaneUI() {
		updateBackground();
	}

	// 获得当前屏幕快照
	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) {
		}

	}

	@Override
	 public void paintComponent(Graphics g) {
	 super.paintComponent(g);
	 //this.paint(g);
	
	 Point pos = this.getLocationOnScreen();
	 Point offset = new Point(-pos.x, -pos.y);
	 // 有一个问题就是为什么这边要用复数
	 g.drawImage(background, offset.x, offset.y, null);
	 }
 

	public void update(Graphics g) {
		Point pos = this.getLocationOnScreen();
		Point offset = new Point(-pos.x, -pos.y);
		Graphics gg = background.getGraphics();
		paint(gg);
		gg.dispose();
		g.drawImage(background, offset.x, offset.y, null);
		//this.repaint();
	}

}

 

 

<!--EndFragment-->

 

 

<!--EndFragment-->

<!--EndFragment-->

0
2
分享到:
评论
3 楼 yw57705916 2012-10-10  
这实现没人敢用
2 楼 xichao1929 2012-10-06  
freezingsky 写道
我依然觉得JAVA在背景透明这一块,实现起来压力不小,而且性能消耗实在不敢让人恭维。

   我总感觉所有的事情都要发生变化,总会有人让某些事情发生改变,30年之后,我变老了时候,依然会相信会有一种新的语言产生。 Java的世界不是专制,不需要恭维;应该是民主的,可以进谏,可以改变。
1 楼 freezingsky 2012-10-06  
我依然觉得JAVA在背景透明这一块,实现起来压力不小,而且性能消耗实在不敢让人恭维。

相关推荐

    C# WinForm 不规则透明界面美化 效果 DEMO

     功能:透明界面美化, 透明按钮切换,用户信息传递,最大化,最小化,关闭,移动窗口,定时器,弹窗,页面传值 3右下角浮动弹窗 功能:透明界面美化, 透明按钮切换,关闭,移动窗口,页面传值 详细 :...

    java 透明界面 java 透明界面

    ### Java 实现透明界面知识点详解 #### 一、概述 在Java中,通过使用`com.sun.awt.AWTUtilities`类中的方法,可以实现窗口透明效果。本文将详细讲解如何使用Java编写一个简单程序来实现透明界面,并允许用户通过...

    Labview自定义标题栏并透明化界面

    通过学习和分析这些文件,你可以更深入地理解如何在LabVIEW中实现自定义标题栏和透明化界面,同时也可以为自己的项目提供参考。记得在实验过程中保存备份,以防误操作导致数据丢失。 总之,自定义LabVIEW界面不仅...

    半透明界面DEMO

    标题中的“半透明界面DEMO”指的是一个演示项目,它展示了如何实现类似QQ窗口的具有半透明效果的用户界面。半透明界面设计在现代软件应用中非常流行,它能够为用户提供更加美观且交互性强的体验。半透明效果可以使...

    易语言玻璃透明界面

    在“易语言玻璃透明界面”这个主题中,我们主要探讨的是如何利用易语言来实现Windows操作系统中的玻璃透明效果,这种效果通常会给用户带来更加现代和高级的视觉体验。 首先,我们要理解“玻璃透明界面”的概念。在...

    易语言吉祥如意透明界面.7z

    在《易语言吉祥如意透明界面》这个压缩包中,可能包含了实现透明界面的相关代码示例、教程文档或者预设的透明界面模板。开发者可以通过学习这些资源,掌握如何在易语言中创建透明界面,提升自己在界面设计方面的能力...

    易语言吉祥如意透明界面

    在"易语言吉祥如意透明界面"这个主题中,我们主要探讨的是如何使用易语言来实现一个具有特定视觉效果的用户界面,即"吉祥如意透明界面"。 透明界面是现代应用程序中常见的一种设计风格,它能让用户界面更加美观,...

    VS界面美化全透明背景

    【VS界面美化全透明背景】是一个关于Visual Studio(VS)个性化设置的主题,主要涉及如何通过插件和定制来实现编辑器界面的美化,尤其是创建全透明背景的效果,从而提升编程时的工作体验。在Visual Studio这样的专业...

    Android侧滑实现主界面半透明

    本主题将深入探讨如何实现"Android侧滑实现主界面半透明"这一功能,包括侧滑机制、半透明效果的设置以及相关注意事项。 首先,侧滑功能通常通过滑动手势检测来实现。Android SDK提供了SlidingDrawer和...

    VC++实现界面半透明

    用VC++实现界面半透明效果,是不是觉得很炫啊!用VC++实现界面半透明效果,是不是觉得很炫啊!摘要必须大于50个字! 声明:请确保您上传的内容合法合规,涉及侵权内容将会被移除,详见《CSDN资源共享规则》

    通过透明的View展示透明的界面

    在Android开发中,创建透明界面是一种常见的需求,可以为用户提供独特的视觉体验。本文将深入探讨如何利用透明的View来实现这一效果,同时介绍相关的技术点和注意事项。 首先,要理解Android中的透明度概念。在...

    易语言源码易语言吉祥如意透明界面.rar

    "易语言吉祥如意透明界面"是一个使用易语言编写的源码项目,其主要目标是实现一个具有透明效果的用户界面,给用户带来更加美观和独特的交互体验。 在Windows操作系统中,通常的窗口界面是由标题栏、边框、菜单、...

    Direct界面透明模块.rar

    Direct界面透明模块是计算机编程领域中的一个特定主题,主要涉及Windows操作系统下的用户界面...开发者可以研究这些源码,了解如何使用易语言和DirectUI技术来实现透明界面,从而增强他们的编程技能和软件开发能力。

    QQ透明界面修改器

    因此,一些第三方软件工具应运而生,例如“QQ透明界面修改器”,它为用户提供了更广泛的个性化定制空间。 QQ透明界面修改器允许用户根据个人喜好调整QQ界面的多个方面,包括但不限于窗口透明度、图标样式等。用户...

    PyQt实现界面翻转切换效果

    PyQt实现界面翻转切换效果是用qt的场景功能来实现的,用到了QGraphicsView,QGraphicsLinearLayout,QGraphicsWidget等有关qt场景的库。算是对qt场景的一个小小的尝试,涉及内容不深,程序效果并是随心所欲,需要去...

    Qt多界面动画切换实现

    在本文中,我们将深入探讨如何在Qt环境中实现多界面间的动画切换,特别是在使用Qt5.9.1 LTS版本时。Qt Widgets库为我们提供了丰富的功能,用于构建多界面应用程序,并通过优雅的动画效果来提升用户体验。 首先,...

    VB 100%纯透明的窗体界面设计一例.rar

    此例中的"VB 100%纯透明的窗体界面设计"展示了如何实现一个几乎不可见的窗体,只有内容显示而没有传统的边框和背景。这种设计可以用于创建独特的用户界面,如悬浮窗口或高度定制的应用程序视图。 要实现这样的效果...

    QQ制作素材(聊天界面+透明主题)

    本资源包“QQ制作素材(聊天界面+透明主题)”提供了丰富的素材,旨在帮助开发者或设计师创建出美观且用户体验良好的QQ聊天界面,以及实现独特的透明主题效果。 首先,我们需要了解QQ聊天界面设计的基本要素。这通常...

    XP下依然能实现界面透明

    在Windows XP系统中,虽然微软默认并未开启桌面窗口的Alpha透明效果,但通过编程技术,尤其是使用C#这种强大的编程语言,我们完全可以在该系统上实现界面透明效果,即所谓的"Alhpa效果"。本篇文章将深入探讨如何在XP...

    visual c++ vc多窗口分体界面 仿千千静听的透明歌词.zip

    在本项目中,我们主要探讨如何使用Visual C++(VC++)来实现一个多窗口分体界面,同时融入了类似于千千静听的透明歌词效果。这个项目涉及到的主要知识点包括MFC(Microsoft Foundation Classes)、对话框类(Dialog ...

Global site tag (gtag.js) - Google Analytics