程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> J2ME >> 在MIDP2.0中操作圖片像素

在MIDP2.0中操作圖片像素

編輯:J2ME

正文

   我們知道,在MIDP1.0中,除非我們利用特定廠商的API(比如Nokia),我們是沒法對圖片的像素進行操作的,但是在MIDP2.0中,Image和Graphics的功能都大大增強了。比如,我們可以獲取Image的所有像素值,然後利用程序來修改這些像素(比如說ARGB各自的值),最後再把修改後的像素圖繪制出來。通過直接操作圖片像素,我們就獲得了一種很強大的能力,用編程的方式實現出很多有趣的效果來,而不用額外制作新圖片。比如說透明度漸變,顏色反轉等。下面就是2個例子,分別實現透明度漸變和顏色反轉的功能。

例題一: 透明度漸變效果的實現

   給定一張圖片,假如我們想實現這麼一種效果:圖片由全透明狀態逐漸清晰,最後達到正常狀態。要實現這一個過程,我們首先要獲取該圖片的所有像素值,逐步讓這些像素的alpha值從0轉變到正常,每改變圖片的所有像素值一次,我們就請求刷屏一次,把最新的像素圖畫出來,這樣我們就能實現透明度漸變的效果了。代碼實現如下:
  1. import Java.io.IOException;
  2. import Javax.microedition.lcdui.Canvas;
  3. import Javax.microedition.lcdui.Display;
  4. import Javax.microedition.lcdui.Graphics;
  5. import Javax.microedition.lcdui.Image;
  6. import Javax.microedition.midlet.MIDlet;
  7. import Javax.microedition.midlet.MIDletStateChangeException;
  8. /**
  9.  * 
  10.  * @author JagIE
  11.  * 
  12.  */
  13. public class ShadowMIDlet extends MIDlet {
  14.     Canvas c = new ShadowCanvas();
  15.     public ShadowMIDlet() {
  16.     }
  17.     protected void startApp() throws MIDletStateChangeException {
  18.         Display.getDisplay(this).setCurrent(c);
  19.     }
  20.     protected void pauseApp() {
  21.         // TODO Auto-generated method stub
  22.     }
  23.     protected void destroyApp(boolean arg0) throws MIDletStateChangeException {
  24.         // TODO Auto-generated method stub
  25.     }
  26. }
  27. /**
  28.  * 
  29.  * @author JagIE
  30.  * 
  31.  */
  32. class ShadowCanvas extends Canvas implements Runnable {
  33.     int w, h;
  34.     // 原始圖片
  35.     Image srcImage;
  36.     // 原始圖片的像素數組
  37.     int[] srcRgbImage;
  38.     // 漸變圖片的像素數組
  39.     int[] shadowRgbImage;
  40.     int imgWidth, imgHeight;
  41.     int count;
  42.     public ShadowCanvas() {
  43.         w = this.getWidth();
  44.         h = this.getHeight();
  45.         try {
  46.             srcImage = Image.createImage("/av.png");
  47.         } catch (IOException e) {
  48.             // TODO Auto-generated catch block
  49.             e.printStackTrace();
  50.         }
  51.         imgWidth = srcImage.getWidth();
  52.         imgHeight = srcImage.getHeight();
  53.         // 制造原始圖片的像素數組,用一個int來代表每一個像素,按位表示方式是:0xAARRGGBB
  54.         srcRgbImage = new int[imgWidth * imgHeight];
  55.         // 獲取原始圖片的所有像素,參見MIDP APPI文檔
  56.         srcImage.getRGB(srcRgbImage, 0, imgWidth, 0, 0, imgWidth, imgHeight);
  57.         shadowRgbImage = new int[srcRgbImage.length];
  58.         System.arraycopy(srcRgbImage, 0, shadowRgbImage, 0,
  59.                 shadowRgbImage.length);
  60.         // 漸變圖片的所有像素已開始都是全透明的
  61.         for (int i = 0; i < shadowRgbImage.length; i++) {
  62.             shadowRgbImage[i] &= 0x00ffffff;
  63.         }
  64.         new Thread(this).start();
  65.     }
  66.     public void paint(Graphics g) {
  67.         g.setColor(0, 0, 0);
  68.         g.fillRect(0, 0, w, h);
  69.         // 繪制漸變圖片
  70.         g.drawRGB(shadowRgbImage, 0, imgWidth, (w - imgWidth) / 2,
  71.                 (h - imgHeight) / 2, imgWidth, imgHeight, true);
  72.         g.setColor(0, 255, 0);
  73.         g.drawString("count=" + count, w / 2, 30, Graphics.HCENTER
  74.                 | Graphics.TOP);
  75.     }
  76.     public void run() {
  77.         while (true) {
  78.             boolean changed = false;
  79.             // 改變漸變圖片的每一個像素
  80.             for (int i = 0; i < shadowRgbImage.length; i++) {
  81.                 // 獲取漸變圖片的某一像素的alpha值
  82.                 int alpha = (shadowRgbImage[i] & 0xff000000) >>> 24;
  83.                 // 原始圖片的對應像素的alpha值
  84.                 int oldAlpha = (srcRgbImage[i] & 0xff000000) >>> 24;
  85.                 if (alpha < oldAlpha) {
  86.                     // alpha值++
  87.                     shadowRgbImage[i] = ((alpha + 1) << 24)
  88.                             | (shadowRgbImage[i] & 0x00ffffff);
  89.                     changed = true;
  90.                 }
  91.             }
  92.             try {
  93.                 Thread.sleep(10);
  94.             } catch (InterruptedException e) {
  95.                 // TODO Auto-generated catch block
  96.                 e.printStackTrace();
  97.             }
  98.             count++;
  99.             repaint();
  100.             // 當所有像素的alpha值都達到原始值後,線程運行結束
  101.             if (!changed) {
  102.                 System.out.println("over");
  103.                 break;
  104.             }
  105.         }
  106.     }
  107. }

透明度漸變效果如下:
  
  

例題二:顏色反轉


   在手機游戲中,我們經常會碰上這樣一種情況,比如我方飛機和敵方飛機外觀是完全一樣的,唯一的區別就是顏色不同,比如說敵方飛機是紅色的,而我方飛機是綠色的。在MIDP1.0中,我們就只好制作2張圖片來表示2種飛機,自然,這樣會造成jar空間的極大浪費。但是在MIDP2.0中,通過對圖片直接進行像素操作,反轉RGB中的一個值,我們只需要一張圖片就夠了,樣例代碼如下:
ColorMIDlet.Java

  1. import Java.io.IOException;
  2. import Javax.microedition.lcdui.Canvas;
  3. import Javax.microedition.lcdui.Display;
  4. import Javax.microedition.lcdui.Graphics;
  5. import Javax.microedition.lcdui.Image;
  6. import Javax.microedition.midlet.MIDlet;
  7. import Javax.microedition.midlet.MIDletStateChangeException;
  8. /**
  9.  * 
  10.  * @author JagIE
  11.  * 
  12.  */
  13. public class ColorMIDlet extends MIDlet {
  14.     Canvas c = new ColorCanvas();
  15.     public ColorMIDlet() {
  16.         super();
  17.         // TODO Auto-generated constructor stub
  18.     }
  19.     protected void startApp() throws MIDletStateChangeException {
  20.         Display.getDisplay(this).setCurrent(c);
  21.     }
  22.     protected void pauseApp() {
  23.         // TODO Auto-generated method stub
  24.     }
  25.     protected void destroyApp(boolean arg0) throws MIDletStateChangeException {
  26.         // TODO Auto-generated method stub
  27.     }
  28. }
  29. /**
  30.  * 
  31.  * @author JagIE
  32.  * 
  33.  */
  34. class ColorCanvas extends Canvas {
  35.     Image srcImage;
  36.     int[] targetImage1;
  37.     int[] targetImage2;
  38.     public ColorCanvas() {
  39.         try {
  40.             srcImage = Image.createImage("/av.png");
  41.         } catch (IOException e) {
  42.             // TODO Auto-generated catch block
  43.             e.printStackTrace();
  44.         }
  45.         targetImage1 = GraphicsUtil.flipImageColor(srcImage,
  46.                 GraphicsUtil.SHIFT_RED_TO_BLUE);
  47.         targetImage2 = GraphicsUtil.flipImageColor(srcImage,
  48.                 GraphicsUtil.SHIFT_RED_TO_GREEN);
  49.     }
  50.     public void paint(Graphics g) {
  51.         g.setColor(0, 0, 0);
  52.         g.fillRect(0, 0, this.getWidth(), this.getHeight());
  53.         g.setColor(0x00ff00);
  54.         g.drawString("origin:", getWidth() / 2, 0, Graphics.TOP
  55.                 | Graphics.HCENTER);
  56.         g.drawImage(srcImage, 30, 20, Graphics.LEFT | Graphics.TOP);
  57.         g.drawString("SHIFT_RED_TO_BLUE:", getWidth() / 2,
  58.                 srcImage.getHeight() + 20, Graphics.TOP | Graphics.HCENTER);
  59.         g.drawRGB(targetImage1, 0, srcImage.getWidth(), 30, srcImage
  60.                 .getHeight() + 40, srcImage.getWidth(), srcImage.getHeight(),
  61.                 true);
  62.         g.drawString("SHIFT_RED_TO_GREEN:", getWidth() / 2, srcImage
  63.                 .getHeight() * 2 + 40, Graphics.TOP | Graphics.HCENTER);
  64.         g.drawRGB(targetImage2, 0, srcImage.getWidth(), 30, srcImage
  65.                 .getHeight() * 2 + 60, srcImage.getWidth(), srcImage
  66.                 .getHeight(), true);
  67.     }
  68. }

GraphicsUtil.Java

  1. import Javax.microedition.lcdui.Image;
  2. /**
  3.  * 
  4.  * @author JagIE
  5.  * 
  6.  */
  7. public class GraphicsUtil {
  8.     public static final int SHIFT_RED_TO_GREEN = 0;
  9.     public static final int SHIFT_RED_TO_BLUE = 1;
  10.     public static final int SHIFT_GREEN_TO_BLUE = 2;
  11.     public static final int SHIFT_GREEN_TO_RED = 3;
  12.     public static final int SHIFT_BLUE_TO_RED = 4;
  13.     public static final int SHIFT_BLUE_TO_GREEN = 5;
  14.     public static int[] flipImageColor(Image source, int shiftType) {
  15.         // we start by getting the image data into an int array - the number
  16.         // of 32-bit ints is equal to the width multiplIEd by the height
  17.         int[] rgbData = new int[(source.getWidth() * source.getHeight())];
  18.         source.getRGB(rgbData, 0, source.getWidth(), 0, 0, source.getWidth(),
  19.                 source.getHeight());
  20.         // now go through every pixel and adjust it's color
  21.         for (int i = 0; i < rgbData.length; i++) {
  22.             int p = rgbData[i];
  23.             // split out the different byte components of the pixel by
  24.             // applying
  25.             // a mask so we only get what we need, then shift it to make it
  26.             // a normal number we can play around with
  27.             int a = ((p & 0xff000000) >> 24);
  28.             int r = ((p & 0x00ff0000) >> 16);
  29.             int g = ((p & 0x0000ff00) >> 8);
  30.             int b = ((p & 0x000000ff) >> 0);
  31.             int ba = a, br = r, bb = b, bg = g; // backup copIEs
  32.             // flip the colors around according to the Operation required
  33.             switch (shiftType) {
  34.             case SHIFT_RED_TO_GREEN:
  35.                 g = r;
  36.                 r = bg;
  37.                 break;
  38.             case SHIFT_RED_TO_BLUE:
  39.                 b = r;
  40.                 r = bb;
  41.                 break;
  42.             case SHIFT_GREEN_TO_BLUE:
  43.                 g = b;
  44.                 b = bg;
  45.                 break;
  46.             case SHIFT_GREEN_TO_RED:
  47.                 g = r;
  48.                 r = bg;
  49.                 break;
  50.             case SHIFT_BLUE_TO_RED:
  51.                 b = r;
  52.                 r = bb;
  53.                 break;
  54.             case SHIFT_BLUE_TO_GREEN:
  55.                 b = g;
  56.                 g = bb;
  57.                 break;
  58.             }
  59.             // shift all our values back in
  60.             rgbData[i] = (a << 24) | (r << 16) | (g << 8) | b;
  61.         }
  62.         return rgbData;
  63.     }
  64. }
  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved