Andriod中的两种自定义颜色选择器
第一种:
import android.app.Dialog;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.LinearGradient;
import android.graphics.Paint;
import android.graphics.RectF;
import android.graphics.Shader;
import android.graphics.SweepGradient;
import android.os.Bundle;
import android.view.MotionEvent;
import android.view.View; public class ColorPickerDialog extends Dialog { public interface OnColorChangedListener { void colorChanged(int color); } private OnColorChangedListener mListener; private int mInitialColor; private static class ColorPickerView extends View { private Paint mPaint; private Paint mCenterPaint; private Paint mRadialPaint; private final int[] mRadialColors; private OnColorChangedListener mListener; private Paint mGradientPaint; private int[] mLinearColors; ColorPickerView(Context c, OnColorChangedListener l, int color) { super(c); mListener = l; mRadialColors = new int[] { 0xFFFF0000, 0xFFFF00FF, 0xFF0000FF, 0xFF00FFFF, 0xFF00FF00, 0xFFFFFF00, 0xFFFF0000 }; Shader s = new SweepGradient(0, 0, mRadialColors, null); mPaint = new Paint(Paint.ANTI_ALIAS_FLAG); mPaint.setShader(s); mPaint.setStyle(Paint.Style.STROKE); mPaint.setStrokeWidth(32); mLinearColors = getColors(color); Shader shader = new LinearGradient(0, 0, Center_X * 2, 0, mLinearColors, null, Shader.TileMode.CLAMP); mGradientPaint = new Paint(Paint.ANTI_ALIAS_FLAG); mGradientPaint.setStyle(Paint.Style.STROKE); mGradientPaint.setShader(shader); mGradientPaint.setStrokeWidth(32); mCenterPaint = new Paint(Paint.ANTI_ALIAS_FLAG); mCenterPaint.setColor(color); mCenterPaint.setStrokeWidth(6); mRadialPaint = new Paint(Paint.ANTI_ALIAS_FLAG); mRadialPaint.setColor(color); mRadialPaint.setStrokeWidth(6); } private int[] getColors(int color) { if (color == Color.BLACK || color == Color.WHITE) { return new int[] { Color.BLACK, Color.WHITE }; } return new int[] { Color.BLACK, color, Color.WHITE }; } private boolean mTrackingCenter; private boolean mHighlightCenter; private boolean mTrackingLinGradient; @Override protected void onDraw(Canvas canvas) { float r = COLOR_CIRCLE - mPaint.getStrokeWidth() * 0.5f; canvas.translate(Center_X, COLOR_CIRCLE); canvas.drawOval(new RectF(-r, -r, r, r), mPaint); canvas.drawCircle(0, 0, CENTER_RADIUS, mCenterPaint); if (mTrackingCenter) { int color = mCenterPaint.getColor(); mCenterPaint.setStyle(Paint.Style.STROKE); if (mHighlightCenter) { mCenterPaint.setAlpha(0xFF); } else { mCenterPaint.setAlpha(0x80); } canvas.drawCircle(0, 0, CENTER_RADIUS + mCenterPaint.getStrokeWidth(), mCenterPaint); mCenterPaint.setStyle(Paint.Style.FILL); mCenterPaint.setColor(color); } int color = mRadialPaint.getColor(); mLinearColors = getColors(color); Shader shader = new LinearGradient(0, 0, Center_X * 2, 0, mLinearColors, null, Shader.TileMode.CLAMP); mGradientPaint.setShader(shader); canvas.translate(-Center_X, 0); canvas.drawLine(0, r + 50, Center_X * 2, r + 50, mGradientPaint); } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { /*super.onMeasure(widthMeasureSpec, heightMeasureSpec); int parentWidth = MeasureSpec.getSize(widthMeasureSpec); int parentHeight = MeasureSpec.getSize(heightMeasureSpec); setMeasuredDimension(parentWidth, parentHeight); Center_X = (int) Math.ceil(parentWidth*.5); Center_Y = (int) Math.ceil(parentHeight*.5);*/ setMeasuredDimension(Center_X * 2, Center_Y * 2 + 70); } private static int Center_X = 110; private static int Center_Y = 100; private static final int CENTER_RADIUS = 32; private static final int COLOR_CIRCLE = 100; private int ave(int s, int d, float p) { return s + java.lang.Math.round(p * (d - s)); } private int interpColor(int colors[], float unit) { if (unit <= 0) { return colors[0]; } if (unit >= 1) { return colors[colors.length - 1]; } float p = unit * (colors.length - 1); int i = (int) p; p -= i; // now p is just the fractional part [0...1) and i is the index int c0 = colors[i]; int c1 = colors[i + 1]; int a = ave(Color.alpha(c0), Color.alpha(c1), p); int r = ave(Color.red(c0), Color.red(c1), p); int g = ave(Color.green(c0), Color.green(c1), p); int b = ave(Color.blue(c0), Color.blue(c1), p); return Color.argb(a, r, g, b); } private static final float PI = 3.1415926f; @Override public boolean onTouchEvent(MotionEvent event) { float x = event.getX() - Center_X; float y = event.getY() - COLOR_CIRCLE; boolean inCenter = Math.sqrt(x * x + y * y) <= CENTER_RADIUS; boolean outOfRadialGradient = y > COLOR_CIRCLE; switch (event.getAction()) { case MotionEvent.ACTION_DOWN: mTrackingCenter = inCenter; mTrackingLinGradient = outOfRadialGradient; if (inCenter) { mHighlightCenter = true; invalidate(); break; } case MotionEvent.ACTION_MOVE: if (mTrackingCenter) { if (mHighlightCenter != inCenter) { mHighlightCenter = inCenter; invalidate(); } } else if (mTrackingLinGradient) { float unit = Math.max(0, Math.min(Center_X * 2, x + Center_X)) / (Center_X * 2); mCenterPaint.setColor(interpColor(mLinearColors, unit)); invalidate(); } else { float angle = (float) Math.atan2(y, x); // need to turn angle [-PI ... PI] into unit [0....1] float unit = angle / (2 * PI); if (unit < 0) { unit += 1; } int color = interpColor(mRadialColors, unit); mCenterPaint.setColor(color); mRadialPaint.setColor(color); invalidate(); } break; case MotionEvent.ACTION_UP: if (mTrackingCenter) { if (inCenter) { mListener.colorChanged(mCenterPaint.getColor()); } mTrackingCenter = false; // so we draw w/o halo invalidate(); } break; } return true; } } public ColorPickerDialog(Context context, OnColorChangedListener listener, int initialColor) { super(context); mListener = listener; mInitialColor = initialColor; } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); OnColorChangedListener l = new OnColorChangedListener() { public void colorChanged(int color) { mListener.colorChanged(color); dismiss(); } }; this.setContentView(new ColorPickerView(this.getContext(), l, mInitialColor)); this.setTitle(R.string.color_pick); /*Display display = this.getWindow().getWindowManager().getDefaultDisplay(); if(display.getWidth() < display.getHeight()) { this.getWindow().setLayout( LayoutParams.FILL_PARENT, (int) Math.ceil(display.getHeight()*.7f) ); } else { this.getWindow().setLayout( (int) Math.ceil(display.getWidth()*.7f), LayoutParams.FILL_PARENT ); }*/ }
}
new ColorPickerDialog( context, new ColorPickerDialog.OnColorChangedListener() { public void colorChanged(int color) { XXX.setColor(color); } }, oldColor ).show();
第二种:
package com.example.draw;import android.app.AlertDialog;//透明
import android.app.Dialog;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.LinearGradient;
import android.graphics.Paint;
import android.graphics.RectF;
import android.graphics.Shader;
import android.graphics.SweepGradient;
import android.os.Bundle;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.WindowManager; public class ColorPickerDialog extends Dialog { private final boolean debug = true; private final String TAG = "ColorPicker"; Context context; private String title = "颜色选择器";//标题 private int mInitialColor;//初始颜色 private OnColorChangedListener mListener; /** * 初始颜色黑色 * @param context * @param listener 回调 */ public ColorPickerDialog(Context context, String title, OnColorChangedListener listener) { this(context, Color.BLACK, listener); } /** * * @param context * @param initialColor 初始颜色 * @param listener 回调 */ public ColorPickerDialog(Context context, int initialColor, OnColorChangedListener listener) { super(context); this.context = context; mListener = listener; mInitialColor = initialColor; }@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); WindowManager manager = getWindow().getWindowManager(); int height = (int) (manager.getDefaultDisplay().getHeight() * 0.5f); int width = (int) (manager.getDefaultDisplay().getWidth() * 0.7f); ColorPickerView myView = new ColorPickerView(context, height, width); setContentView(myView); setTitle(title); } private class ColorPickerView extends View { private Paint mPaint;//渐变色环画笔 private Paint mCenterPaint;//中间圆画笔 private Paint mLinePaint;//分隔线画笔 private Paint mRectPaint;//渐变方块画笔 private Shader rectShader;//渐变方块渐变图像 private float rectLeft;//渐变方块左x坐标 private float rectTop;//渐变方块右x坐标 private float rectRight;//渐变方块上y坐标 private float rectBottom;//渐变方块下y坐标 private final int[] mCircleColors;//渐变色环颜色 private final int[] mRectColors;//渐变方块颜色 private int mHeight;//View高 private int mWidth;//View宽 private float r;//色环半径(paint中部) private float centerRadius;//中心圆半径 private boolean downInCircle = true;//按在渐变环上 private boolean downInRect;//按在渐变方块上 private boolean highlightCenter;//高亮 private boolean highlightCenterLittle;//微亮 public ColorPickerView(Context context, int height, int width) { super(context); this.mHeight = height - 36; this.mWidth = width; setMinimumHeight(height - 36); setMinimumWidth(width); //渐变色环参数 mCircleColors = new int[] {0xFFFF0000, 0xFFFF00FF, 0xFF0000FF, 0xFF00FFFF, 0xFF00FF00,0xFFFFFF00, 0xFFFF0000}; Shader s = new SweepGradient(0, 0, mCircleColors, null); mPaint = new Paint(Paint.ANTI_ALIAS_FLAG); mPaint.setShader(s); mPaint.setStyle(Paint.Style.STROKE); mPaint.setStrokeWidth(50); r = width / 2 * 0.7f - mPaint.getStrokeWidth() * 0.5f; //中心圆参数 mCenterPaint = new Paint(Paint.ANTI_ALIAS_FLAG); mCenterPaint.setColor(mInitialColor); mCenterPaint.setStrokeWidth(5); centerRadius = (r - mPaint.getStrokeWidth() / 2 ) * 0.7f; //边框参数 mLinePaint = new Paint(Paint.ANTI_ALIAS_FLAG); mLinePaint.setColor(Color.parseColor("#72A1D1")); mLinePaint.setStrokeWidth(4); //黑白渐变参数 mRectColors = new int[]{0xFF000000, mCenterPaint.getColor(), 0xFFFFFFFF}; mRectPaint = new Paint(Paint.ANTI_ALIAS_FLAG); mRectPaint.setStrokeWidth(5); rectLeft = -r - mPaint.getStrokeWidth() * 0.5f; rectTop = r + mPaint.getStrokeWidth() * 0.5f + mLinePaint.getStrokeMiter() * 0.5f + 15; rectRight = r + mPaint.getStrokeWidth() * 0.5f; rectBottom = rectTop + 50; } @Override protected void onDraw(Canvas canvas) { //移动中心 canvas.translate(mWidth / 2, mHeight / 2 - 50); //画中心圆 canvas.drawCircle(0, 0, centerRadius, mCenterPaint); //是否显示中心圆外的小圆环 if (highlightCenter || highlightCenterLittle) { int c = mCenterPaint.getColor(); mCenterPaint.setStyle(Paint.Style.STROKE); if(highlightCenter) { mCenterPaint.setAlpha(0xFF); }else if(highlightCenterLittle) { mCenterPaint.setAlpha(0x90); } canvas.drawCircle(0, 0, centerRadius + mCenterPaint.getStrokeWidth(), mCenterPaint); mCenterPaint.setStyle(Paint.Style.FILL); mCenterPaint.setColor(c); } //画色环 canvas.drawOval(new RectF(-r, -r, r, r), mPaint); //画黑白渐变块 if(downInCircle) { mRectColors[1] = mCenterPaint.getColor(); } rectShader = new LinearGradient(rectLeft, 0, rectRight, 0, mRectColors, null, Shader.TileMode.MIRROR); mRectPaint.setShader(rectShader); canvas.drawRect(rectLeft, rectTop, rectRight, rectBottom, mRectPaint); float offset = mLinePaint.getStrokeWidth() / 2; canvas.drawLine(rectLeft - offset, rectTop - offset * 2, rectLeft - offset, rectBottom + offset * 2, mLinePaint);//左 canvas.drawLine(rectLeft - offset * 2, rectTop - offset, rectRight + offset * 2, rectTop - offset, mLinePaint);//上 canvas.drawLine(rectRight + offset, rectTop - offset * 2, rectRight + offset, rectBottom + offset * 2, mLinePaint);//右 canvas.drawLine(rectLeft - offset * 2, rectBottom + offset, rectRight + offset * 2, rectBottom + offset, mLinePaint);//下 super.onDraw(canvas); } @Override public boolean onTouchEvent(MotionEvent event) { float x = event.getX() - mWidth / 2; float y = event.getY() - mHeight / 2 + 50; boolean inCircle = inColorCircle(x, y, r + mPaint.getStrokeWidth() / 2, r - mPaint.getStrokeWidth() / 2); boolean inCenter = inCenter(x, y, centerRadius); boolean inRect = inRect(x, y); switch (event.getAction()) { case MotionEvent.ACTION_DOWN: downInCircle = inCircle; downInRect = inRect; highlightCenter = inCenter; case MotionEvent.ACTION_MOVE: if(downInCircle && inCircle) {//down按在渐变色环内, 且move也在渐变色环内 float angle = (float) Math.atan2(y, x); float unit = (float) (angle / (2 * Math.PI)); if (unit < 0) { unit += 1; } mCenterPaint.setColor(interpCircleColor(mCircleColors, unit)); if(debug) Log.v(TAG, "色环内, 坐标: " + x + "," + y); }else if(downInRect && inRect) {//down在渐变方块内, 且move也在渐变方块内 mCenterPaint.setColor(interpRectColor(mRectColors, x)); } if(debug) Log.v(TAG, "[MOVE] 高亮: " + highlightCenter + "微亮: " + highlightCenterLittle + " 中心: " + inCenter); if((highlightCenter && inCenter) || (highlightCenterLittle && inCenter)) {//点击中心圆, 当前移动在中心圆 highlightCenter = true; highlightCenterLittle = false; } else if(highlightCenter || highlightCenterLittle) {//点击在中心圆, 当前移出中心圆 highlightCenter = false; highlightCenterLittle = true; } else { highlightCenter = false; highlightCenterLittle = false; } invalidate(); break; case MotionEvent.ACTION_UP: if(highlightCenter && inCenter) {//点击在中心圆, 且当前启动在中心圆 if(mListener != null) { mListener.colorChanged(mCenterPaint.getColor()); ColorPickerDialog.this.dismiss(); } } if(downInCircle) { downInCircle = false; } if(downInRect) { downInRect = false; } if(highlightCenter) { highlightCenter = false; } if(highlightCenterLittle) { highlightCenterLittle = false; } invalidate(); break; } return true; } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(mWidth, mHeight); } /** * 坐标是否在色环上 * @param x 坐标 * @param y 坐标 * @param outRadius 色环外半径 * @param inRadius 色环内半径 * @return */ private boolean inColorCircle(float x, float y, float outRadius, float inRadius) { double outCircle = Math.PI * outRadius * outRadius; double inCircle = Math.PI * inRadius * inRadius; double fingerCircle = Math.PI * (x * x + y * y); if(fingerCircle < outCircle && fingerCircle > inCircle) { return true; }else { return false; } } /** * 坐标是否在中心圆上 * @param x 坐标 * @param y 坐标 * @param centerRadius 圆半径 * @return */ private boolean inCenter(float x, float y, float centerRadius) { double centerCircle = Math.PI * centerRadius * centerRadius; double fingerCircle = Math.PI * (x * x + y * y); if(fingerCircle < centerCircle) { return true; }else { return false; } } /** * 坐标是否在渐变色中 * @param x * @param y * @return */ private boolean inRect(float x, float y) { if( x <= rectRight && x >=rectLeft && y <= rectBottom && y >=rectTop) { return true; } else { return false; } } /** * 获取圆环上颜色 * @param colors * @param unit * @return */ private int interpCircleColor(int colors[], float unit) { if (unit <= 0) { return colors[0]; } if (unit >= 1) { return colors[colors.length - 1]; } float p = unit * (colors.length - 1); int i = (int)p; p -= i; // now p is just the fractional part [0...1) and i is the index int c0 = colors[i]; int c1 = colors[i+1]; int a = ave(Color.alpha(c0), Color.alpha(c1), p); int r = ave(Color.red(c0), Color.red(c1), p); int g = ave(Color.green(c0), Color.green(c1), p); int b = ave(Color.blue(c0), Color.blue(c1), p); return Color.argb(a, r, g, b); } /** * 获取渐变块上颜色 * @param colors * @param x * @return */ private int interpRectColor(int colors[], float x) { int a, r, g, b, c0, c1; float p; if (x < 0) { c0 = colors[0]; c1 = colors[1]; p = (x + rectRight) / rectRight; } else { c0 = colors[1]; c1 = colors[2]; p = x / rectRight; } a = ave(Color.alpha(c0), Color.alpha(c1), p); r = ave(Color.red(c0), Color.red(c1), p); g = ave(Color.green(c0), Color.green(c1), p); b = ave(Color.blue(c0), Color.blue(c1), p); return Color.argb(a, r, g, b); } private int ave(int s, int d, float p) { return s + Math.round(p * (d - s)); } } /** * 回调接口 * @author <a href="clarkamx@gmail.com">LynK</a> * * Create on 2012-1-6 上午8:21:05 * */ public interface OnColorChangedListener { /** * 回调函数 * @param color 选中的颜色 */ void colorChanged(int color); } public int getmInitialColor() { return mInitialColor; } public void setmInitialColor(int mInitialColor) { this.mInitialColor = mInitialColor; } public OnColorChangedListener getmListener() { return mListener; } public void setmListener(OnColorChangedListener mListener) { this.mListener = mListener; }
}
dialog = new ColorPickerDialog(this, p.getColor(),new ColorPickerDialog.OnColorChangedListener() {@Overridepublic void colorChanged(int color) {p.setColor(color);}});dialog.show();
Andriod中的两种自定义颜色选择器相关推荐
- 批量插入数据库语句java_java相关:MyBatis批量插入数据到Oracle数据库中的两种方式(实例代码)...
java相关:MyBatis批量插入数据到Oracle数据库中的两种方式(实例代码) 发布于 2020-7-22| 复制链接 本文通过实例代码给大家分享了MyBatis批量插入数据到Oracle数据库 ...
- java类型转换答案,在java中支持两种类型的类型转换,自动类型转换和强制类型转换。父类转化为子类需要强制转换。...
在java中支持两种类型的类型转换,自动类型转换和强制类型转换.父类转化为子类需要强制转换. 更多相关问题 计算机病毒通过()传染扩散得极快,危害最大. 当一个现象的数量由小变大,另一个现象的数量相反 ...
- link linux 跨设备,Linux中的两种link方式
Linux系统中包括两种链接方式:硬链接(hard link)和符号链接(symbolic link),其中符合链接就是所谓的软链接(soft link),那么两者之间到底有什么区别呢? inode ...
- Linux中的两种守护进程stand alone和xinetd
Linux中的两种守护进程stand alone和xinetd --http://www.cnblogs.com/itech/archive/2010/12/27/1914846.html#top 一 ...
- Java中的两种异常类型及其区别?
Java中的两种异常类型及其区别? 参考文章: (1)Java中的两种异常类型及其区别? (2)https://www.cnblogs.com/zxfei/p/11182730.html (3)htt ...
- <随笔03>Java中的两种异常类型
<随笔03>Java中的两种异常类型 参考文章: (1)<随笔03>Java中的两种异常类型 (2)https://www.cnblogs.com/newlyfly/p/744 ...
- Allegro中显示两种单位方法
Allegro中显示两种单位方法
- Matlab中的两种除法
Matlab中的两种除法 matlab中的向量运算,试比较: >>a=[1,1,1] a = 1 1 1 >> b=[2,2,2] b = 2 2 ...
- 认识LTE(七):LTE中的两种无反馈模式:发射分集(TM2)和开环空分复用(TM3)
认识LTE(七):LTE中的两种无反馈模式:发射分集(TM2)和开环空分复用(TM3) 文章目录 认识LTE(七):LTE中的两种无反馈模式:发射分集(TM2)和开环空分复用(TM3) 零.代码地址 ...
最新文章
- 使用Caffe基于cifar10进行物体识别
- sql 关联使用id还是code_使用sh格式化nginx访问日志并存入mysql
- 《BI那点儿事》数据流转换——排序
- 深圳SEO外包公司×××,如何选择合适的SEO外包公司?
- SAP中如何给成本要素分配多个默认成本中心
- 20应用统计考研复试要点(part15)--应用多元分析
- 计算机接口教程,运用接口实现计算机各组件信息
- matlab中函数绝对值图像,ex的图像(绝对值的函数图像口诀)
- MFC工作笔记0001---认识MFC
- 手把手教大家如何优化长尾关键词
- 跳过Android6.0+权限的方法
- 星光嵌入式WM8978音频模块发布
- 计算机科学与技术专业认证研讨,CNCC丨一流本科专业建设暨工程认证研讨会
- python对京东评论的爬取_python爬取京东评论(三)
- 【虚拟仿真】Unity3D中如何实现让3D模型显示在UI前面
- iOS开发之strip处理framework的调试符号
- 计算机应用基础知识doc,独家稿件:终稿[定稿]计算机应用基础知识全册教案.docOK版...
- 一个python爬虫工程师参加博主之星之后,必做的事情
- uniapp canvas绘制弧形圆环
- F1意大利站,阿隆索在法拉利的主场夺冠
热门文章
- php调用不存在的方法,php如何调用不存在的方法
- grid++中打印表格时怎么让每页有打印表头_1分钟!学会快速打印标题行,轻松搞定不加班...
- reset.css页面样式初始化
- linux apu 分辨率,编写xorg.conf 简单三行解决Ubuntu分辩率不可调的问题
- Captcha插件后门分析和修复
- 面向对象程序设计 第六次作业
- protobuf-2.5.0安装
- php中 $$str 中 $$ 的解释
- 浅谈JavaScript中的对象和类型(上)
- SPL 关联优化技巧