-
Notifications
You must be signed in to change notification settings - Fork 152
Expand file tree
/
Copy pathScanDrawView.java
More file actions
214 lines (184 loc) · 7.62 KB
/
ScanDrawView.java
File metadata and controls
214 lines (184 loc) · 7.62 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
package com.chavesgu.scan;
import android.animation.Animator;
import android.animation.AnimatorSet;
import android.animation.ObjectAnimator;
import android.animation.TimeInterpolator;
import android.animation.ValueAnimator;
import android.app.Activity;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Matrix;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.PixelFormat;
import android.graphics.Region;
import android.util.DisplayMetrics;
import android.util.Log;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.ViewGroup;
import android.view.animation.LinearInterpolator;
import android.widget.RelativeLayout;
import java.math.BigDecimal;
import java.util.Map;
import androidx.annotation.Nullable;
import static java.lang.Math.min;
import static java.lang.Math.max;
import static java.lang.Math.floor;
public class ScanDrawView extends SurfaceView implements SurfaceHolder.Callback {
private String LOG_TAG = "scan";
private Activity activity;
private double vw;
private double vh;
private double areaX;
private double areaY;
private double areaWidth;
private int scanLineColor;
private boolean transparentScanLine = false;
private double scale = .7;
private float up = 80.0f;
private float dpi;
private boolean running;
private ValueAnimator positionAnimator;
private float scanLinePositionValue;
private Matrix matrix;
public ScanDrawView(Context context, Activity activity, @Nullable Map<String, Object> args) {
super(context);
scale = (double) args.get("scale");
final int r = (int) args.get("r");
final int g = (int) args.get("g");
final int b = (int) args.get("b");
final double alpha = (double) args.get("a");
up = BigDecimal.valueOf((double) args.get("up")).floatValue();
RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);
final int a = max(0, min(255, (int)floor(alpha * 256.0)));
if (a==0) transparentScanLine = true;
scanLineColor = Color.argb(a, r, g, b);
this.activity = activity;
getHolder().addCallback(this);
}
@Override
public void surfaceCreated(SurfaceHolder surfaceHolder) {
setWillNotDraw(false);
setZOrderOnTop(true);
surfaceHolder.setFormat(PixelFormat.TRANSPARENT);
running = true;
}
@Override
public void surfaceChanged(SurfaceHolder surfaceHolder, int i, int width, int height) {
vw = width;
vh = height;
areaWidth = min(vw, vh) * scale;
areaX = (vw - areaWidth) / 2 ;
areaY = (vh - areaWidth) / 2 ;
DisplayMetrics dm = new DisplayMetrics();
activity.getWindowManager().getDefaultDisplay().getMetrics(dm);
dpi = dm.density;
// init animate
final float scanLineWidth = (float) (areaWidth * 0.8);
final long duration = (long) (areaWidth/175/dpi*1.5*1000);
positionAnimator = ValueAnimator.ofFloat(0, scanLineWidth);
positionAnimator.setDuration(duration);
positionAnimator.setInterpolator(null);
positionAnimator.setRepeatMode(ValueAnimator.RESTART);
positionAnimator.setRepeatCount(ValueAnimator.INFINITE);
positionAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
@Override
public void onAnimationUpdate(ValueAnimator valueAnimator) {
scanLinePositionValue = (float) valueAnimator.getAnimatedValue();
// Log.i(LOG_TAG, "scanLinePositionValue:"+scanLinePositionValue);
invalidate();
}
});
}
@Override
public void surfaceDestroyed(SurfaceHolder surfaceHolder) {
if (positionAnimator!=null) {
positionAnimator.removeAllUpdateListeners();
positionAnimator = null;
}
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
if (positionAnimator!=null && !positionAnimator.isStarted())positionAnimator.start();
drawing(canvas);
}
private void tryDraw(final SurfaceHolder holder) {
Canvas canvas = holder.lockCanvas();
if (canvas == null) {
Log.e(LOG_TAG, "Cannot draw onto the canvas as it's null");
} else {
drawing(canvas);
holder.unlockCanvasAndPost(canvas);
}
}
private void drawing(Canvas canvas) {
final float x = (float) areaX;
final float y = (float) areaY - up;
final float width = (float) areaWidth;
final float shortWidth = (float) (areaWidth * 0.1);
final float scanLineWidth = (float) (areaWidth * 0.8);
final float scanLineX = (float) (vw - scanLineWidth)/2;
final float scanLineY = (float) (vh - scanLineWidth)/2 - up;
if (scale < 1) {
Paint paint = new Paint();
paint.setColor(scanLineColor);
paint.setStrokeWidth(2*dpi);
paint.setStrokeCap(Paint.Cap.ROUND);
paint.setStrokeJoin(Paint.Join.ROUND);
paint.setStyle(Paint.Style.STROKE);
canvas.drawLine(x, y, x+shortWidth, y, paint);
canvas.drawLine(x, y, x, y+shortWidth, paint);
canvas.drawLine(x+width, y, x+width-shortWidth, y, paint);
canvas.drawLine(x+width, y, x+width, y+shortWidth, paint);
canvas.drawLine(x+width, y+width, x+width-shortWidth, y+width, paint);
canvas.drawLine(x+width, y+width, x+width, y+width-shortWidth, paint);
canvas.drawLine(x, y+width, x+shortWidth, y+width, paint);
canvas.drawLine(x, y+width, x, y+width-shortWidth, paint);
// mask
canvas.save();
Path clipPath = new Path();
clipPath.addRect(x-2,y-2,(float)(x+areaWidth+2),(float)(y+areaWidth+2),Path.Direction.CCW);
canvas.clipPath(clipPath, Region.Op.DIFFERENCE);
Paint maskPaint = new Paint();
final int a = max(0, min(255, (int)floor(0.5 * 256.0)));
maskPaint.setColor(Color.argb(a, 0, 0, 0));
maskPaint.setStyle(Paint.Style.FILL);
canvas.drawRect(0, 0, (float) vw, (float)vh, maskPaint);
canvas.restore();
}
if (running && !transparentScanLine) {
Paint scanPaint = new Paint();
scanPaint.setColor(scanLineColor);
scanPaint.setStrokeWidth(2*dpi);
scanPaint.setStrokeCap(Paint.Cap.ROUND);
scanPaint.setStrokeJoin(Paint.Join.ROUND);
scanPaint.setStyle(Paint.Style.STROKE);
if (scanLinePositionValue/scanLineWidth < (float) (2.0/3.0)) {
scanPaint.setAlpha(255);
} else {
final float a = 1 - (scanLinePositionValue/scanLineWidth - (float) (2.0/3.0))*3;
final int alpha = max(0, min(255, (int)floor(a * 256.0)));
scanPaint.setAlpha(alpha);
}
Path scanPath = new Path();
scanPath.moveTo(scanLineX, scanLineY + scanLinePositionValue);
scanPath.lineTo(scanLineX + scanLineWidth, scanLineY + scanLinePositionValue);
scanPath.close();
canvas.drawPath(scanPath, scanPaint);
}
// invalidate();
}
public void resume() {
running = true;
if (positionAnimator!=null)positionAnimator.resume();
invalidate();
}
public void pause() {
running = false;
if (positionAnimator!=null)positionAnimator.pause();
invalidate();
}
}