Android_自定义控件(画板)
2019-11-26 本文已影响0人
书虫大王X
使用到的知识点:
1、Canvas总结
2、path总结
3、paint总结
实例框架:
关系图
sliderView类:
public class drawView extends View {
Path mpath;
// 操作的数组
private ArrayList<Grahp> grahps;
// 原始数组
private ArrayList<Grahp> orginaGrahps;
private int lineColor = Color.BLACK;
private int lineWidth = 10;
public drawView(Context context) {
this(context,null);
}
public drawView(Context context, AttributeSet attrs) {
super(context, attrs);
init();
}
private void init() {
grahps = new ArrayList<>();
orginaGrahps = new ArrayList<>();
setBackgroundColor(Color.WHITE);
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
// 遍历数组(遍历器)
Iterator<Grahp> iterator = grahps.iterator();
while (iterator.hasNext()) {
Grahp grahp = iterator.next();
canvas.drawPath(grahp.path,grahp.paint);
}
}
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()){
case MotionEvent.ACTION_DOWN:
Paint mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mPaint.setColor(lineColor);
mPaint.setStrokeWidth(lineWidth);
mPaint.setStyle(Paint.Style.STROKE);
mpath = new Path();
mpath.moveTo(event.getX(),event.getY());
Grahp temp = new Grahp(mpath,mPaint);
grahps.add(new Grahp(mpath,mPaint));
orginaGrahps.add(temp);
break;
case MotionEvent.ACTION_MOVE:
mpath.lineTo(event.getX(),event.getY());
break;
case MotionEvent.ACTION_UP:
break;
}
invalidate();
return true;
}
private class Grahp{
Paint paint;
Path path;
public Grahp(Path path,Paint paint){
this.paint = paint;
this.path = path;
}
}
// 还原上一步
public void retunrnLastStep(){
if (grahps.size() < orginaGrahps.size()) {
int index = grahps.size() - 1 + 1;
grahps.add(orginaGrahps.get(index));
}
invalidate();
}
public void removeLast(){
if (grahps.size() > 0) {
grahps.remove(grahps.size() - 1);
invalidate();
}
}
public void removeAll(){
grahps.clear();
invalidate();
}
public void setLineWidth(int lineWidth) {
this.lineWidth = lineWidth;
}
public int getLineWidth() {
return lineWidth;
}
public void setLineColor(int lineColor) {
this.lineColor = lineColor;
}
public int getLineColor() {
return lineColor;
}
}
drawView类:
public class slideView extends View {
private Path mPath;
private Paint linePaint;
private Paint circlePaitn;
private int circleColor = Color.RED;
private int lineColor = Color.BLACK;
// 线条的粗细
private int lineSize = 10;
// 小圆点
private int cY;
private int cX;
private int radius;
// 半径所除的数
private int thymbScale = 4;
// 记录触摸点的坐标
private float touchPosition = 10;
// 管理进度条的画笔
private Paint progressPaint;
private int progressColor = Color.MAGENTA;
public static int PROGRESS = 0;
public static int SLIDER = 1;
private int style = PROGRESS;
// 进度最大值
private int max = 100;
private float progress;
private onSliderChangedLisenter onSliderChangedLisenter;
public slideView(Context context) {
this(context,null);
}
public slideView(Context context, AttributeSet attrs) {
super(context, attrs);
init();
}
private void init() {
mPath = new Path();
linePaint = new Paint(Paint.ANTI_ALIAS_FLAG);
linePaint.setColor(lineColor);
linePaint.setStrokeWidth(lineSize);
circlePaitn = new Paint(Paint.ANTI_ALIAS_FLAG);
circlePaitn.setColor(circleColor);
progressPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
progressPaint.setColor(progressColor);
progressPaint.setStrokeWidth(lineSize);
setBackgroundColor(Color.WHITE);
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
if (getWidth() > getHeight()){
// 屏幕是横着的
canvas.drawLine(0,(getHeight() )/2,getWidth(),(getHeight())/2, linePaint);
if (touchPosition > 0) {
canvas.drawLine(0,(getHeight() )/2,touchPosition,(getHeight())/2, progressPaint);
}
radius = getHeight()/thymbScale;
cX = (int)touchPosition;
}else {
// 竖屏
canvas.drawLine((getWidth())/2,0,(getWidth())/2,getHeight(), linePaint);
if (touchPosition > 0) {
canvas.drawLine((getWidth())/2,0,(getWidth())/2,touchPosition, progressPaint);
}
radius = getWidth()/thymbScale;
cX = (getWidth())/2;
}
// 画小圆点
if (style == SLIDER) {
canvas.drawCircle(cX,cY,radius,circlePaitn);
}
}
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()){
case MotionEvent.ACTION_DOWN:
// 进度条的小圆点放大
thymbScale = 2;
if (getWidth() > getHeight()) {
// 横屏
touchPosition = event.getX();
}else {
touchPosition = event.getY();
}
callBack();
break;
case MotionEvent.ACTION_MOVE:
// 获取当前触摸点的坐标
if (getWidth() > getHeight()) {
// 横屏
touchPosition = event.getX();
if (touchPosition < 0) {
touchPosition = 0;
} else if (touchPosition > getWidth()) {
touchPosition = getWidth();
}
}else {
touchPosition = event.getY();
if (touchPosition < 0) {
touchPosition = 0;
}else if(touchPosition > getHeight()){
touchPosition = getHeight();
}
}
callBack();
break;
case MotionEvent.ACTION_UP:
thymbScale = 4;
break;
}
if (style == SLIDER) {
invalidate();
}
return true;
}
private void callBack(){
if (onSliderChangedLisenter != null) {
if (getWidth() > getHeight()) {
progress = touchPosition/getWidth();
}else {
progress = touchPosition/getHeight();
}
onSliderChangedLisenter.progerssChanged(progress*max);
}
}
public void setStyle(int style) {
this.style = style;
}
public void setProgress(int progress){
// 计算比例
this.progress = (float)((progress*1.0) / max);
setProgress(this.progress);
}
public void setProgress(float progress) {
this.progress = progress;
if (progress < 1.001) {
// 将进度值转化为控件的尺寸位置
if (getWidth() > getHeight()) {
touchPosition = progress * getWidth();
}else {
touchPosition = progress * getHeight();
}
}
}
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
super.onSizeChanged(w, h, oldw, oldh);
// 将进度值转化为控件的尺寸位置
if (getWidth() > getHeight()) {
touchPosition = progress * getWidth();
}else {
touchPosition = progress * getHeight();
}
invalidate();
}
public float getProgress() {
return progress;
}
public void setMax(int max) {
this.max = max;
}
public int getMax() {
return max;
}
public interface onSliderChangedLisenter{
void progerssChanged(float progress);
}
public void setOnSliderChangedLisenter(onSliderChangedLisenter onSliderChangedLisenter){
this.onSliderChangedLisenter = onSliderChangedLisenter;
}
}
xml布局文件:
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<androidx.constraintlayout.widget.ConstraintLayout
android:layout_width="match_parent"
android:layout_height="0dp"
app:layout_constraintBottom_toTopOf="@+id/text2"
app:layout_constraintTop_toTopOf="parent">
<!--滑动条-->
<xn.ky.a1102_zdy_huaban_1.slideView
android:id="@+id/slider"
android:layout_width="20dp"
android:layout_height="match_parent"
android:layout_marginLeft="10dp"
android:layout_marginBottom="50dp"
android:layout_marginTop="50dp"
app:layout_constraintLeft_toLeftOf="parent"/>
<!--画板-->
<xn.ky.a1102_zdy_huaban_1.drawView
android:id="@+id/dradView"
android:layout_width="0dp"
android:layout_height="match_parent"
app:layout_constraintLeft_toRightOf="@id/slider"
app:layout_constraintRight_toLeftOf="@+id/color" />
<LinearLayout
android:id="@+id/color"
android:layout_width="50dp"
android:layout_height="match_parent"
android:orientation="vertical"
app:layout_constraintRight_toRightOf="parent"
android:layout_marginRight="20dp"
android:gravity="center">
<Button
android:layout_width="match_parent"
android:layout_height="50dp"
android:background="@color/colorAccent"
android:onClick="chioceColor"/>
<Button
android:layout_width="match_parent"
android:layout_height="50dp"
android:background="@color/colorPrimary"
android:onClick="chioceColor"/>
<Button
android:layout_width="match_parent"
android:layout_height="50dp"
android:background="#000000"
android:onClick="chioceColor"/>
<Button
android:layout_width="match_parent"
android:layout_height="50dp"
android:background="#ddde11"
android:onClick="chioceColor"/>
</LinearLayout>
</androidx.constraintlayout.widget.ConstraintLayout>
<LinearLayout
android:id="@+id/text2"
android:orientation="horizontal"
android:layout_width="match_parent"
android:layout_height="60dp"
app:layout_constraintBottom_toBottomOf="parent"
android:gravity="center">
<Button
android:layout_width="70dp"
android:layout_height="wrap_content"
android:text="上一步"
android:onClick="lastStep"/>
<Button
android:layout_width="70dp"
android:layout_height="wrap_content"
android:text="撤销"
android:onClick="goBack"/>
<Button
android:layout_width="70dp"
android:layout_height="wrap_content"
android:text="清空"
android:onClick="clear"/>
<Button
android:layout_width="70dp"
android:layout_height="wrap_content"
android:text="橡皮擦"
android:onClick="eraser"/>
<Button
android:layout_width="70dp"
android:layout_height="wrap_content"
android:text="保存"
android:onClick="save"/>
</LinearLayout>
</androidx.constraintlayout.widget.ConstraintLayout>
MainActivity类:
public class MainActivity extends AppCompatActivity {
drawView drawView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
final slideView slider = findViewById(R.id.slider);
slider.setMax(30);
slider.setStyle(slideView.SLIDER);
// 获取画板对象
drawView = findViewById(R.id.dradView);
slider.setProgress(drawView.getLineWidth());
slider.setOnSliderChangedLisenter(new slideView.onSliderChangedLisenter() {
@Override
public void progerssChanged(float progress) {
// 将滑动条的进度值设置为画笔粗细
drawView.setLineWidth((int)progress);
}
});
}
@Override
public void onWindowFocusChanged(boolean hasFocus) {
super.onWindowFocusChanged(hasFocus);
}
@Override
protected void onResume() {
super.onResume();
// 程序运行起来屏幕就是横屏
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_SENSOR);
}
public void chioceColor(View view) {
// 获取按键的背景颜色
ColorDrawable drawable = (ColorDrawable)view.getBackground();
drawView.setLineColor(drawable.getColor());
}
// 还原到上一步
public void lastStep(View view) {
drawView.retunrnLastStep();
}
public void goBack(View view) {
drawView.removeLast();
}
public void clear(View view) {
drawView.removeAll();
}
public void eraser(View view) {
ColorDrawable drawable = (ColorDrawable)drawView.getBackground();
if (drawable != null){
drawView.setLineColor(drawable.getColor());
}
}
public void save(View view) {
}
}