网络编程 | 站长之家 | 网页制作 | 图形图象 | 操作系统 | 冲浪宝典 | 软件教学 | 网络办公 | 邮件系统 | 网络安全 | 认证考试 | 系统进程
Firefox | IE | Maxthon | 迅雷 | 电驴 | BitComet | FlashGet | QQ | QQ空间 | Vista | 输入法 | Ghost | Word | Excel | wps | Powerpoint
asp | .net | php | jsp | Sql | c# | Ajax | xml | Dreamweaver | FrontPages | Javascript | css | photoshop | fireworks | Flash | Cad | Discuz!
当前位置 > 网站建设学院 > 网络编程 > Java
Tag:注入,存储过程,分页,安全,优化,xmlhttp,fso,jmail,application,session,防盗链,stream,无组件,组件,md5,乱码,缓存,加密,验证码,算法,cookies,ubb,正则表达式,水印,索引,日志,压缩,base64,url重写,上传,控件,Web.config,JDBC,函数,内存,PDF,迁移,结构,破解,编译,配置,进程,分词,IIS,Apache,Tomcat,phpmyadmin,Gzip,触发器,socket
网络编程:ASP教程,ASP.NET教程,PHP教程,JSP教程,C#教程,数据库,XML教程,Ajax,Java,Perl,Shell,VB教程,Delphi,C/C++教程,软件工程,J2EE/J2ME,移动开发
本月文章推荐
.当主线程崩溃而其它线程继续运行.
.无状态Enterprise JavaBeans的观.
.JavaI/OAPI之性能分析(上).
.Java 学习之道.
.违例的限制.
.JBuilderclass文件的Module引用.
.使用XML小服务程序配置文件配置小.
.Java 实践: 用 XQuery 进行屏幕搜.
.Java中鲜为人知的缺点(上).
.漫谈数据库的中庸思想.
.Java小技巧:关于Cookie的操作.
.JAVA入门教程:第一章.
.J2ME概述.
.关于J2EE中软件基础结构的瓶颈.
.Huihoo.org发布了JFoxMX的最新版.
.Java核心代码例程之:DateFormat.
..NET应用程序的授权模型使系统更.
.感受JDO 2.0查询语言的特点.
.实战 J2EE 开发购物网站 创建数据.
.IOPCBrowseServerAddressSpace的.

Java 2D开发技巧之“灯光与阴影”

发表日期:2008-1-5


一、 引言

  在本文中,我们将向你展示如何为扁平外形添加一种灯光效果以实现一种类3D外观。

  也许你比较满足于自己的文字表达能力,但一幅图片往往能够产生更好的效果。对于图形处理来说,也是如此;不妨请参考一下图1中的两种图形。在本文中,我将向你展示如何克服左边扁平外形所带来的烦恼而以一种更为光滑的更具舒适感的外形代替。

Java 2D开发技巧之“灯光与阴影”(图一)
图1.普通扁平外形与施加Java 2D效果后的外形

  二、 实现技术分析

  借助于适当的颜色,你可以使用本文中介绍的技术来模拟一种彩色光闪耀"越过"你的外形,从而生成一种微妙的发光效果。我们是如何实现这一效果的呢?请分析下面的代码;在方法drawBorderGlow()上面的注释更为细致地介绍了要害实现方法:

import java.awt.geom.*;
import java.awt.image.*;
private static final Color clrHi = new Color(255, 229, 63);
private static final Color clrLo = new Color(255, 105, 0);
private static final Color clrGlowInnerHi = new Color(253, 239, 175, 148);
private static final Color clrGlowInnerLo = new Color(255, 209, 0);
private static final Color clrGlowOuterHi = new Color(253, 239, 175, 124);
private static final Color clrGlowOuterLo = new Color(255, 179, 0);
private Shape createClipShape() {
 float border = 20.0f;
 float x1 = border;
 float y1 = border;
 float x2 = width - border;
 float y2 = height - border;
 float adj = 3.0f; //帮助圆化类锐的拐角
 float arc = 8.0f;
 float dcx = 0.18f * width;
 float cx1 = x1-dcx;
 float cy1 = 0.40f * height;
 float cx2 = x1+dcx;
 float cy2 = 0.50f * height;
 GeneralPath gp = new GeneralPath();
 gp.moveTo(x1-adj, y1+adj);
 gp.quadTo(x1, y1, x1+adj, y1);
 gp.lineTo(x2-arc, y1);
 gp.quadTo(x2, y1, x2, y1+arc);
 gp.lineTo(x2, y2-arc);
 gp.quadTo(x2, y2, x2-arc, y2);
 gp.lineTo(x1+adj, y2);
 gp.quadTo(x1, y2, x1, y2-adj);
 gp.curveTo(cx2, cy2, cx1, cy1, x1-adj, y1+adj);
 gp.closePath();
 return gp;
}
private BufferedImage createClipImage(Shape s) {
 // 创建一半透明的中间图像,我们可以使用它来实现软修剪效果
 GraphicsConfiguration gc = g.getDeviceConfiguration();
 BufferedImage img = gc.createCompatibleImage(width, height, Transparency.TRANSLUCENT);
 Graphics2D g2 = img.createGraphics();
 //清除图像,这样所有的像素都具有零alpha
 g2.setComposite(AlphaComposite.Clear);
 g2.fillRect(0, 0, width, height);
 // 把我们的修剪外形生成到图像上。注重,我们启动了
 // 反走样功能以实现软修剪效果。你可以
 //尝试注释掉启动反走样的这一行,那么
 //你会看到通常的生硬的修剪效果.
 g2.setComposite(AlphaComposite.Src);
 g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
 g2.setColor(Color.WHITE);
 g2.fill(s);
 g2.dispose();
 return img;
}
private static Color getMixedColor(Color c1, float pct1, Color c2, float pct2) {
 float[] clr1 = c1.getComponents(null);
 float[] clr2 = c2.getComponents(null);
 for (int i = 0; i < clr1.length; i++) {
  clr1[i] = (clr1[i] * pct1) + (clr2[i] * pct2);
 }
 return new Color(clr1[0], clr1[1], clr1[2], clr1[3]);
}
//下面是实现技巧:为了实现发光效果,我们开始使用一种"内部"颜色粗笔
//和笔划需要的外形。然后,我们不断地把笔变细,
//并且不断地移向"外部"颜色,
//并且不断地提高颜色的不透明度以便使其朝向外形的内部看上去暗淡。
//我们使用已经生成到我们的目的图像上的"修剪外形",这样以来,
//SRC_ATOP规则就会修剪在我们的外形外部的笔划部分。
private void paintBorderGlow(Graphics2D g2, int glowWidth) {
 int gw = glowWidth*2;
 for (int i=gw; i >= 2; i-=2) {
  float pct = (float)(gw - i) / (gw - 1);
  Color mixHi = getMixedColor(clrGlowInnerHi, pct,clrGlowOuterHi, 1.0f - pct);
  Color mixLo = getMixedColor(clrGlowInnerLo, pct,clrGlowOuterLo, 1.0f - pct);
  g2.setPaint(new GradientPaint(0.0f, height*0.25f, mixHi,0.0f, height, mixLo));
  //g2.setColor(Color.WHITE);
  g2.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_ATOP, pct));
  g2.setStroke(new BasicStroke(i));
  g2.draw(clipShape);
 }
}
Shape clipShape = createClipShape();
//Shape clipShape = new Ellipse2D.Float(width/4, height/4, width/2, height/2);
//把背景清除为白色
g.setColor(Color.WHITE);
g.fillRect(0, 0, width, height);
//设置修剪外形
BufferedImage clipImage = createClipImage(clipShape);
Graphics2D g2 = clipImage.createGraphics();
//使用渐变填充外形
g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
g2.setComposite(AlphaComposite.SrcAtop);
g2.setPaint(new GradientPaint(0, 0, clrHi, 0, height, clrLo));
g2.fill(clipShape);
//应用边界发光效果
paintBorderGlow(g2, 8);
g2.dispose();
g.drawImage(clipImage, 0, 0, null);
  注重,在上面的例子中,我把一些可选的代码行加上了注释。你可以去掉这些注释并观察它们对生成效果的影响。

  注重:聪明的读者可以已经注重到,上面应用于paintBorderGlow()方法中的技术也可以用于沿外形添加一种投影效果。你不妨先猜测一下如何实现这一点……好,时间到!不是在外形的顶部生成边缘(记住,修剪能够确保笔划仅影响外形的内部),我们可以预先绕着我们的外形生成一种可变的灰色边界。这意味着,阴影笔划将出现在我们的外形的外边;阴影笔划的内部将会通过我们的外形而有效地生成。

  你可以把下面的一些代码插入到上面的例子中以便在相应的同一个外形上添加一种阴影边界效果:


上一篇:用Spring更好地处理Struts动作三种整合 人气:620
下一篇:将DBMS存储过程封装为会话EJB组件中的方法(2) 人气:929
浏览全部Java的内容 Dreamweaver插件下载 网页广告代码 祝你圣诞节快乐 2009年新年快乐