Android 仿PhotoShop调色板应用(三) 主体界面绘制

2024-03-10 02:48

本文主要是介绍Android 仿PhotoShop调色板应用(三) 主体界面绘制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Android 仿PhotoShop调色板应用(三) 主体界面绘制

   关于PhotoShop调色板应用的实现我总结了两个最核心的部分:

  1. 主体界面不同区域的绘制

  2. 颜色选择的生成与交互

 这里我讲述一下第一要点,也就是ColorPickerDialog对主体界面的绘制.

   首先还是看一下ColorPickerDialog整体显示的效果(见图1)

                                          图1


对应着效果图我画了一张界面结构分析图,相信看了之后会对该界面的组成很快能够掌握:(见图2)

  图2

一. 界面组成

   可以看到整个显示的部分即为ColorPickerDialog. 这个Dialog根据组件的构成及功能实现上可以分为两大部分:

  1. 红色边框区域 ColorPickerView绘制而成. 主要作为颜色区域的选择,此区域又划分为三个部分:

      (1)  Saturation Area 饱和度选择区域

      (2)  Hue Area 色相选择区域

      (3)  Alpha Area 透明度选择区域  绘制此区域借助了上一篇讲到的AlphaPatternDrawable类

2. 蓝色边框区域 由ColorPickerPanelView绘制. 左边的部分作为初始颜色显示 右边的部分做颜色选择的实时显示区域,点击后可将颜色设置为默认值


该Dialog的布局文件dialog_color_picker.xml:

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="vertical"android:paddingLeft="5dp"android:paddingRight="5dp" ><net.margaritov.preference.colorpicker.ColorPickerViewandroid:id="@+id/color_picker_view"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layerType="software"android:tag="portrait" /><LinearLayoutandroid:id="@+id/text_hex_wrapper"android:layout_width="fill_parent"android:layout_height="wrap_content"android:layout_marginBottom="5dp"android:layout_marginLeft="6dp"android:layout_marginRight="6dp" ><TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:gravity="left"android:text="@string/press_color_to_apply"android:textAppearance="?android:attr/textAppearanceSmall" /><EditTextandroid:id="@+id/hex_val"android:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:hint="HEX"android:imeOptions="actionDone"android:maxLength="7"android:singleLine="true"android:inputType="textCapCharacters"android:visibility="gone" ></EditText></LinearLayout><LinearLayoutandroid:layout_width="wrap_content"android:layout_height="40dp"android:layout_marginBottom="10dp"android:orientation="horizontal" ><net.margaritov.preference.colorpicker.ColorPickerPanelViewandroid:id="@+id/old_color_panel"android:layout_width="0px"android:layout_height="fill_parent"android:layout_weight="0.5" /><TextViewandroid:layout_width="wrap_content"android:layout_height="fill_parent"android:layout_marginLeft="10dp"android:layout_marginRight="10dp"android:gravity="center"android:text="→"android:textSize="20sp" /><net.margaritov.preference.colorpicker.ColorPickerPanelViewandroid:id="@+id/new_color_panel"android:layout_width="0px"android:layout_height="wrap_content"android:layout_weight="0.5" /></LinearLayout></LinearLayout>


二. 不同区域的绘制实现:

  1.     Saturation Area 饱和度选择区域

这里用到了组合渲染(ComposeShader)的方式.

 (1)使用了两个线性渲染器:一个作为亮度的显示渲染,一个作为饱和度的显示渲染,    因为我们可以看到颜色渐变和亮度渐变的综合显示效果

      此区域完成了HSV(也叫HSB)色彩空间之Saturation(饱和度)value(色调)/brightness(明度)的综合显示

(2) 选择圆环绘制:  分为了内外两个圆环分别绘制. 黑色内圆及灰色外圆

      具体实现请看以下代码


/*** 绘制饱和度选择区域* @param canvas*/private void drawSatValPanel(Canvas canvas){final RectF	rect = mSatValRect;if(BORDER_WIDTH_PX > 0){mBorderPaint.setColor(mBorderColor);canvas.drawRect(mDrawingRect.left, mDrawingRect.top, rect.right + BORDER_WIDTH_PX, rect.bottom + BORDER_WIDTH_PX, mBorderPaint);}//明度线性渲染器if (mValShader == null) {mValShader = new LinearGradient(rect.left, rect.top, rect.left, rect.bottom,0xffffffff, 0xff000000, TileMode.CLAMP);}//HSV转化为RGBint rgb = Color.HSVToColor(new float[]{mHue,1f,1f});//饱和线性渲染器mSatShader = new LinearGradient(rect.left, rect.top, rect.right, rect.top,0xffffffff, rgb, TileMode.CLAMP);//组合渲染 = 明度线性渲染器 + 饱和线性渲染器ComposeShader mShader = new ComposeShader(mValShader, mSatShader, PorterDuff.Mode.MULTIPLY);mSatValPaint.setShader(mShader);canvas.drawRect(rect, mSatValPaint);//初始化选择圆块的位置Point p = satValToPoint(mSat, mVal);//绘制黑色内圆mSatValTrackerPaint.setColor(0xff000000);canvas.drawCircle(p.x, p.y, PALETTE_CIRCLE_TRACKER_RADIUS - 1f * mDensity, mSatValTrackerPaint);//绘制外圆mSatValTrackerPaint.setColor(0xffdddddd);canvas.drawCircle(p.x, p.y, PALETTE_CIRCLE_TRACKER_RADIUS, mSatValTrackerPaint);}

2.Hue Area 色相选择区域

   (1) 颜色渲染部分仍为竖直方向的线性渐变.这里可以看一下对应颜色数组的生成:

private int[] buildHueColorArray(){int[] hue = new int[361];int count = 0;for(int i = hue.length -1; i >= 0; i--, count++){hue[count] = Color.HSVToColor(new float[]{i, 1f, 1f});}return hue;}

这里使用了一个大小为 361的int数组, 根据当前位置生成不同的 hsv颜色数组

通过Color.HSVToColor,将HSV转换为ARGB的形式,因为在线性渐变渲染中,我们不能直接使用HSV色彩,而是需要使用ARGB制式的颜色.

因此我们看到了一个数值方向的一个多彩变换效果       

(2) 绘制矩形颜色选择条区域


/*** 绘制右侧色相选择区域* @param canvas*/private void drawHuePanel(Canvas canvas){final RectF rect = mHueRect;if(BORDER_WIDTH_PX > 0){mBorderPaint.setColor(mBorderColor);canvas.drawRect(rect.left - BORDER_WIDTH_PX,rect.top - BORDER_WIDTH_PX,rect.right + BORDER_WIDTH_PX,rect.bottom + BORDER_WIDTH_PX,mBorderPaint);}//初始化色相线性渲染器if (mHueShader == null) {mHueShader = new LinearGradient(rect.left, rect.top, rect.left, rect.bottom, buildHueColorArray(), null, TileMode.CLAMP);mHuePaint.setShader(mHueShader);}canvas.drawRect(rect, mHuePaint);float rectHeight = 4 * mDensity / 2;//初始化色相选择器选择条位置Point p = hueToPoint(mHue);RectF r = new RectF();r.left = rect.left - RECTANGLE_TRACKER_OFFSET;r.right = rect.right + RECTANGLE_TRACKER_OFFSET;r.top = p.y - rectHeight;r.bottom = p.y + rectHeight;//绘制选择条canvas.drawRoundRect(r, 2, 2, mHueTrackerPaint);}

3. Alpha Area 透明度选择区域

  这里主要借助于AlphaPatternDrawable进行绘制,见上一篇博客:

Android 仿PhotoShop调色板应用(二) 透明度绘制之AlphaPatternDrawable

具体请见注释,渲染部分和Hue Area类似

/*** 绘制底部透明度选择区域* @param canvas*/private void drawAlphaPanel(Canvas canvas){if(!mShowAlphaPanel || mAlphaRect == null || mAlphaPattern == null) return;final RectF rect = mAlphaRect;if(BORDER_WIDTH_PX > 0){mBorderPaint.setColor(mBorderColor);canvas.drawRect(rect.left - BORDER_WIDTH_PX,rect.top - BORDER_WIDTH_PX,rect.right + BORDER_WIDTH_PX,rect.bottom + BORDER_WIDTH_PX,mBorderPaint);}mAlphaPattern.draw(canvas);float[] hsv = new float[]{mHue,mSat,mVal};//hsv数组int color = Color.HSVToColor(hsv);int acolor = Color.HSVToColor(0, hsv);//初始化透明度线性渲染器mAlphaShader = new LinearGradient(rect.left, rect.top, rect.right, rect.top,color, acolor, TileMode.CLAMP);mAlphaPaint.setShader(mAlphaShader);canvas.drawRect(rect, mAlphaPaint);if(mAlphaSliderText != null && mAlphaSliderText!= ""){canvas.drawText(mAlphaSliderText, rect.centerX(), rect.centerY() + 4 * mDensity, mAlphaTextPaint);}float rectWidth = 4 * mDensity / 2;//初始化透明度选择器选择条位置Point p = alphaToPoint(mAlpha);RectF r = new RectF();r.left = p.x - rectWidth;r.right = p.x + rectWidth;r.top = rect.top - RECTANGLE_TRACKER_OFFSET;r.bottom = rect.bottom + RECTANGLE_TRACKER_OFFSET;canvas.drawRoundRect(r, 2, 2, mHueTrackerPaint);}


4. ColorPickerPanelView

    由于此区域可做最终颜色的显示,所以也借助了AlphaPatternDrawable的实现

    此矩形区域绘制相对比较简单:

	final RectF	rect = mColorRect;if(BORDER_WIDTH_PX > 0){mBorderPaint.setColor(mBorderColor);canvas.drawRect(mDrawingRect, mBorderPaint);}if(mAlphaPattern != null){mAlphaPattern.draw(canvas);}mColorPaint.setColor(mColor);canvas.drawRect(rect, mColorPaint);

至此,ColorPickerDialog主体面板绘制部分已讲述完毕.下面我会讲述另一大核心部分:颜色选择生成的交互.

如果对颜色渲染方面还是不太清楚的话,可以参照我之前写的颜色渲染系列,关于原理和具体API的讲解.

这篇关于Android 仿PhotoShop调色板应用(三) 主体界面绘制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/792778

相关文章

将Python应用部署到生产环境的小技巧分享

《将Python应用部署到生产环境的小技巧分享》文章主要讲述了在将Python应用程序部署到生产环境之前,需要进行的准备工作和最佳实践,包括心态调整、代码审查、测试覆盖率提升、配置文件优化、日志记录完... 目录部署前夜:从开发到生产的心理准备与检查清单环境搭建:打造稳固的应用运行平台自动化流水线:让部署像

Python中的可视化设计与UI界面实现

《Python中的可视化设计与UI界面实现》本文介绍了如何使用Python创建用户界面(UI),包括使用Tkinter、PyQt、Kivy等库进行基本窗口、动态图表和动画效果的实现,通过示例代码,展示... 目录从像素到界面:python带你玩转UI设计示例:使用Tkinter创建一个简单的窗口绘图魔法:用

Linux中Curl参数详解实践应用

《Linux中Curl参数详解实践应用》在现代网络开发和运维工作中,curl命令是一个不可或缺的工具,它是一个利用URL语法在命令行下工作的文件传输工具,支持多种协议,如HTTP、HTTPS、FTP等... 目录引言一、基础请求参数1. -X 或 --request2. -d 或 --data3. -H 或

使用Python绘制蛇年春节祝福艺术图

《使用Python绘制蛇年春节祝福艺术图》:本文主要介绍如何使用Python的Matplotlib库绘制一幅富有创意的“蛇年有福”艺术图,这幅图结合了数字,蛇形,花朵等装饰,需要的可以参考下... 目录1. 绘图的基本概念2. 准备工作3. 实现代码解析3.1 设置绘图画布3.2 绘制数字“2025”3.3

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

使用Python绘制可爱的招财猫

《使用Python绘制可爱的招财猫》招财猫,也被称为“幸运猫”,是一种象征财富和好运的吉祥物,经常出现在亚洲文化的商店、餐厅和家庭中,今天,我将带你用Python和matplotlib库从零开始绘制一... 目录1. 为什么选择用 python 绘制?2. 绘图的基本概念3. 实现代码解析3.1 设置绘图画

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

Python绘制土地利用和土地覆盖类型图示例详解

《Python绘制土地利用和土地覆盖类型图示例详解》本文介绍了如何使用Python绘制土地利用和土地覆盖类型图,并提供了详细的代码示例,通过安装所需的库,准备地理数据,使用geopandas和matp... 目录一、所需库的安装二、数据准备三、绘制土地利用和土地覆盖类型图四、代码解释五、其他可视化形式1.