有过UI设计经验的一定对2D图形渲染中的Color Gradient 或多或少有些接触,很多编程

语言也提供了Gradient的接口,但是想知道它是怎么实现的嘛?

本文介绍三种简单的颜色梯度变化算法,就可以很容易实现常见的梯度变化算法

三种都要求提供两个参数即起始颜色RGB值, 最终颜色RGB的值。

垂直梯度颜色变化,效果如下:

水平梯度颜色变化,效果如下:

水平与垂直两个方向叠加梯度变化效果如下:

算法代码及其解释

 

计算起始颜色和终点颜色RGB之间差值代码如下:

float rr = startColor[0] - endColor[0];

float gg = startColor[1] - endColor[1];

float bb = startColor[2] - endColor[2];

实现垂直梯度变化的代码如下:

r = endColor[0] + (int)(rr * ((float)row/255.0f) +0.5f);

g = endColor[1] + (int)(gg * ((float)row/255.0f) +0.5f);

b = endColor[2] + (int)(bb * ((float)row/255.0f) +0.5f);

实现水平梯度变化代码如下:

// set gradient color valuefor each pixel

r = endColor[0] + (int)(rr * ((float)col/255.0f) +0.5f);

g = endColor[1] + (int)(gg * ((float)col/255.0f) + 0.5f);

b = endColor[2] + (int)(bb * ((float)col/255.0f) +0.5f);

实现水平和垂直两个方向上Gradient叠加代码如下:

r = endColor[0] + (int)(rr * (((float)col * (float)row)/size) +0.5f);

g = endColor[1] + (int)(gg * (((float)col * (float)row)/size) +0.5f);

b = endColor[2] + (int)(bb * (((float)col * (float)row)/size) +0.5f);

程序对上面三种Gradient方法分别放在三个不同的方法中,根据参数调用。

程序的完全Java源代码如下:

  1. import java.awt.BorderLayout;
  2. import java.awt.Dimension;
  3. import java.awt.Graphics;
  4. import java.awt.Graphics2D;
  5. import java.awt.RenderingHints;
  6. import java.awt.image.BufferedImage;
  7. import javax.swing.JComponent;
  8. import javax.swing.JFrame;
  9. public class ColorGradientDemo extends JComponent {
  10. /**
  11. *
  12. */
  13. private static final long serialVersionUID = -4134440495899912869L;
  14. private BufferedImage image = null;
  15. protected void paintComponent(Graphics g) {
  16. Graphics2D g2 = (Graphics2D)g;
  17. g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
  18. g2.drawImage(getImage(4), 5, 5, image.getWidth(), image.getHeight(), null);
  19. }
  20. public BufferedImage getImage(int type) {
  21. if(image == null) {
  22. image = new BufferedImage(256, 256, BufferedImage.TYPE_INT_ARGB);
  23. int[] rgbData = new int[256*256];
  24. if(type == 1) {
  25. generateVGradientImage(rgbData);
  26. } else if(type == 2) {
  27. generateHGradientImage(rgbData);
  28. } else {
  29. generateHVGradientImage(rgbData);
  30. }
  31. setRGB(image, 0, 0, 256, 256, rgbData);
  32. }
  33. return image;
  34. }
  35. private void generateVGradientImage(int[] rgbData) {
  36. int[] startColor = getStartColor();
  37. int[] endColor = getEndColor();
  38. float rr = startColor[0] - endColor[0];
  39. float gg = startColor[1] - endColor[1];
  40. float bb = startColor[2] - endColor[2];
  41. int a=255;
  42. int r=0, g=0, b=0;
  43. int index = 0;
  44. for(int row=0; row<256; row++) {
  45. for(int col=0; col<256; col++) {
  46. // set random color value for each pixel
  47. // set gradient color value for each pixel
  48. r = endColor[0] + (int)(rr * ((float)row/255.0f) + 0.5f);
  49. g = endColor[1] + (int)(gg * ((float)row/255.0f) + 0.5f);
  50. b = endColor[2] + (int)(bb * ((float)row/255.0f) + 0.5f);
  51. rgbData[index] = ((a & 0xff) << 24) |
  52. ((r & 0xff) << 16)  |
  53. ((g & 0xff) << 8)   |
  54. ((b & 0xff));
  55. index++;
  56. }
  57. }
  58. }
  59. private void generateHGradientImage(int[] rgbData) {
  60. int[] startColor = getStartColor();
  61. int[] endColor = getEndColor();
  62. float rr = startColor[0] - endColor[0];
  63. float gg = startColor[1] - endColor[1];
  64. float bb = startColor[2] - endColor[2];
  65. int a=255;
  66. int r=0, g=0, b=0;
  67. int index = 0;
  68. for(int row=0; row<256; row++) {
  69. for(int col=0; col<256; col++) {
  70. // set gradient color value for each pixel
  71. r = endColor[0] + (int)(rr * ((float)col/255.0f) + 0.5f);
  72. g = endColor[1] + (int)(gg * ((float)col/255.0f) + 0.5f);
  73. b = endColor[2] + (int)(bb * ((float)col/255.0f) + 0.5f);
  74. rgbData[index] = ((a & 0xff) << 24) |
  75. ((r & 0xff) << 16)  |
  76. ((g & 0xff) << 8)   |
  77. ((b & 0xff));
  78. index++;
  79. }
  80. }
  81. }
  82. private void generateHVGradientImage(int[] rgbData) {
  83. int[] startColor = getStartColor();
  84. int[] endColor = getEndColor();
  85. float rr = startColor[0] - endColor[0];
  86. float gg = startColor[1] - endColor[1];
  87. float bb = startColor[2] - endColor[2];
  88. int a=255;
  89. int r=0, g=0, b=0;
  90. int index = 0;
  91. float size = (float)Math.pow(255.0d, 2.0);
  92. for(int row=0; row<256; row++) {
  93. for(int col=0; col<256; col++) {
  94. // set random color value for each pixel
  95. r = endColor[0] + (int)(rr * (((float)col * (float)row)/size) + 0.5f);
  96. g = endColor[1] + (int)(gg * (((float)col * (float)row)/size) + 0.5f);
  97. b = endColor[2] + (int)(bb * (((float)col * (float)row)/size) + 0.5f);
  98. rgbData[index] = ((a & 0xff) << 24) |
  99. ((r & 0xff) << 16)  |
  100. ((g & 0xff) << 8)   |
  101. ((b & 0xff));
  102. index++;
  103. }
  104. }
  105. }
  106. public int[] getStartColor() {
  107. return new int[]{246,53,138};
  108. }
  109. public int[] getEndColor() {
  110. return new int[]{0,255,255};
  111. }
  112. public void setRGB( BufferedImage image, int x, int y, int width, int height, int[] pixels ) {
  113. int type = image.getType();
  114. if ( type == BufferedImage.TYPE_INT_ARGB || type == BufferedImage.TYPE_INT_RGB )
  115. image.getRaster().setDataElements( x, y, width, height, pixels );
  116. else
  117. image.setRGB( x, y, width, height, pixels, 0, width );
  118. }
  119. public static void main(String[] args) {
  120. JFrame frame = new JFrame("Gradient Color Panel");
  121. frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  122. frame.getContentPane().setLayout(new BorderLayout());
  123. // Display the window.
  124. frame.getContentPane().add(new ColorGradientDemo(), BorderLayout.CENTER);
  125. frame.setPreferredSize(new Dimension(280,305));
  126. frame.pack();
  127. frame.setVisible(true);
  128. }
  129. }

图像处理------颜色梯度变化 (Color Gradient)的更多相关文章

  1. 梯度下降(Gradient Descent)小结

    在求解机器学习算法的模型参数,即无约束优化问题时,梯度下降(Gradient Descent)是最常采用的方法之一,另一种常用的方法是最小二乘法.这里就对梯度下降法做一个完整的总结. 1. 梯度 在微 ...

  2. 梯度下降(Gradient Descent)

    在求解机器学习算法的模型参数,即无约束优化问题时,梯度下降(Gradient Descent)是最常采用的方法之一,另一种常用的方法是最小二乘法.这里就对梯度下降法做一个完整的总结. 1. 梯度 在微 ...

  3. [机器学习] ML重要概念:梯度(Gradient)与梯度下降法(Gradient Descent)

    引言 机器学习栏目记录我在学习Machine Learning过程的一些心得笔记,涵盖线性回归.逻辑回归.Softmax回归.神经网络和SVM等等,主要学习资料来自网上的免费课程和一些经典书籍,免费课 ...

  4. Delphi图像处理 -- 颜色矩阵变换

    转载自阿发伯:http://blog.csdn.net/maozefa/article/details/8316430 阅读提示:     <Delphi图像处理>系列以效率为侧重点,一般 ...

  5. 16个基本颜色关键字 Basic color keywords

    16个基本颜色关键字 Basic color keywords Color Color Name HEX RGB   black #000000 0,0,0   silver #C0C0C0 192, ...

  6. 自然梯度(Natural Gradient)

    自然梯度(Natural Gradient)

  7. 机器学习算法的调试---梯度检验(Gradient Checking)

    梯度检验是一种对求导结果进行数值检验的方法,该方法可以验证求导代码是否正确. 1. 数学原理   考虑我们想要最小化以 θ 为自变量的目标函数 J(θ)(θ 可以为标量和可以为矢量,在 Numpy 的 ...

  8. 梯度下降(Gradient Descent)相关概念

    梯度,直观理解: 梯度: 运算的对像是纯量,运算出来的结果会是向量在一个标量场中, 梯度的计算结果会是"在每个位置都算出一个向量,而这个向量的方向会是在任何一点上从其周围(极接近的周围,学过 ...

  9. One-hot 编码/TF-IDF 值来提取特征,LAD/梯度下降法(Gradient Descent),Sigmoid

    1. 多值无序类数据的特征提取: 多值无序类问题(One-hot 编码)把“耐克”编码为[0,1,0],其中“1”代表了“耐克”的中 间位置,而且是唯一标识.同理我们可以把“中国”标识为[1,0],把 ...

随机推荐

  1. 洛谷 [P1129] [ZJOI2007] 矩阵游戏

    这竟然是一道二分图 乍一看,可能是用搜索做,但是这个数据范围,一定会T. 我们观察发现,无论怎样变换,同一行的一定在同一行,同一列的一定还在同一列.所以说,一行只能配一列.这样,我们的目标就是寻找是否 ...

  2. BZOJ 4176: Lucas的数论 [杜教筛]

    4176: Lucas的数论 题意:求\(\sum_{i=1}^n \sum_{j=1}^n \sigma_0(ij)\) \(n \le 10^9\) 代入\(\sigma_0(nm)=\sum_{ ...

  3. BZOJ 2388: 旅行规划 [分块 凸包 等差数列]

    传送门 题意: 区间加和询问一段区间内整体前缀和的最大值 刚才还在想做完这道题做一道区间加等差数列结果发现这道就是.... 唯一的不同在于前缀和一段区间加上等差数列后,区间后面也要加上一个常数!!! ...

  4. python---协程 学习笔记

    协程 协程又称为微线程,协程是一种用户态的轻量级线程 协程拥有自己的寄存器和栈.协程调度切换的时候,将寄存器上下文和栈都保存到其他地方,在切换回来的时候,恢复到先前保存的寄存器上下文和栈,因此:协程能 ...

  5. 【非原创】C++类成员函数的重载、覆盖和隐藏

    链接:https://www.nowcoder.com/questionTerminal/266d3a6d4f1b436aabf1eff3156fed95来源:牛客网 题目:类成员函数的重载.覆盖和隐 ...

  6. 异步请求时有时会让js不起作用,那么重新加载js

    function reloadSmartMenu() { var jsElem = document.createElement('script'); jsElem.src= path+'/syste ...

  7. Linux知识体系之路径属性与目录

    最近在看鸟哥的Linux私房菜,我觉得这本书还是很不错的.这里进行相关的总结. 1.Linux目录权限概念   Linux一般讲目录可存取的方式分为三个类别,分别是owner/group/other, ...

  8. 讲解mybaits的标签语法

    前言:mybatis作为一个持久层的java框架,高度封装了jdbc原始代码,它拥有比Hiberante更便捷.更直接的sql语法和sql灵活配置的特性,比如简单的一个自动映射对象就大大减少了我们的代 ...

  9. 日均数据量千万级,MySQL、TiDB两种存储方案的落地对比

    http://mp.weixin.qq.com/s?__biz=MzIzNjUxMzk2NQ==&mid=2247484743&idx=1&sn=04337e020d268a9 ...

  10. ABP框架源码学习之授权逻辑

    asp.net core的默认的几种授权方法参考"雨夜朦胧"的系列博客,这里要强调的是asp.net core mvc中的授权和asp.net mvc中的授权不一样,建议先看前面& ...