JOJO is one of the most imaginative anime I have ever seen
Recently, I am going to make a resume, and I want to customize a capability analysis chart, so I thought of this first: no more nonsense, go, oh, oh, oh, oh…
1. Drawing of static diagram
1. Draw the outer ring
In order to reduce the value of the variable and make the size have good linkage (equal scaling), the length and width of the small black bar will depend on the maximum radius mRadius
Then: small black bar length :mRadius*0.08 small black bar width :mRadius*0.05 so R2 = mradius-mradius *0.08
public class AbilityView extends View { private float mRadius = dp(100); Private float mLineWidth = dp(1); private float mLineWidth = dp(1); // Private Paint mLinePaint; // Private Paint mFillPaint; Public AbilityView(Context Context) {this(Context, null); } public AbilityView(Context context, @Nullable AttributeSet attrs) { this(context, attrs, 0); } public AbilityView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); init(); } private void init() { mLinePaint = new Paint(Paint.ANTI_ALIAS_FLAG); mLinePaint.setStrokeWidth(mLineWidth); mLinePaint.setStyle(Paint.Style.STROKE); mFillPaint = new Paint(Paint.ANTI_ALIAS_FLAG); MFillPaint. SetStrokeWidth (0.05 f * mRadius); } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); canvas.translate(mRadius, mRadius); DrawOutCircle (canvas); } /** * draw outer circle * @param canvas */ private void drawOutCircle(canvas canvas) {canvas.save(); canvas.drawCircle(0, 0, mRadius, mLinePaint); Float r2 = mradius-0.08f * mRadius; float R2 = mradius-0.08f * mRadius; DrawCircle (0, 0, R2, mLinePaint); for (int i = 0; i < 22; I++) {// loop to draw the little black bar canvas.save(); canvas.rotate(360 / 22f * i); canvas.drawLine(0, -mRadius, 0, -r2, mFillPaint); canvas.restore(); } canvas.restore(); } protected float dp(float dp) { return TypedValue.applyDimension( TypedValue.COMPLEX_UNIT_DIP, dp, getResources().getDisplayMetrics()); }}Copy the code
2. Draw the inner circle
The same size is in line with the outermost circle. Drawing here is a bit more complicated. You need to understand the use of Canvas and path
If you don’t understand it, you can go to Canvas and Path. If you read these two articles and ask what are the techniques for drawing, you can go here and it will tell you what are the techniques
/** * drawInnerCircle * @param canvas */ private void drawInnerCircle(canvas canvas) {canvas. Save (); Float innerRadius = 0.6f * mRadius; float innerRadius = 0.6f * mRadius; canvas.drawCircle(0, 0, innerRadius, mLinePaint); canvas.save(); for (int i = 0; i < 6; I++) {// iterate over 6 lines canvas.save(); canvas.rotate(60 * i); MPath. MoveTo (0, -innerradius); mPath.rLineTo(0, innerRadius); For (int j = 1; j < 6; {math. MoveTo (-mradius * 0.02f, innerRadius / 6 * j); MPath. RLineTo (mRadius * 0.02f * 2, 0); }// Add 5 lines to canvas. DrawPath (mPath, mLinePaint); Canvas. Restore (); } canvas.restore(); }Copy the code
3. Text drawing
The text is in the same direction, it feels better that way, no matter how you turn it
/ / define test data mAbilityInfo = new String [] {" destruction "and" speed ", "range distance", "staying power", "precision", "growth"}; mAbilityMark = new int[]{100, 100, 60, 100, 100, 100}; mMarkMapper = new String[]{"A", "B", "C", "D", "E"};Copy the code
Private void drawInfoText(canvas canvas) {float r2 = mradius-0.08f * mRadius; For (int I = 0; i < 6; i++) { canvas.save(); canvas.rotate(60 * i + 180); MTextPaint. SetTextSize (mRadius * 0.1 f); Canvas. DrawText (mAbilityInfo[I], 0, r2-0.06 f * mRadius, mTextPaint); MTextPaint. SetTextSize (mRadius * 0.15 f); DrawText (mabilitymark2str (mAbilityMark[I]), 0, r2-0.18f * mRadius, mTextPaint); canvas.restore(); } mTextPaint. SetTextSize (mRadius * 0.07 f); for (int k = 0; k < 5; K ++) {canvas. DrawText (mMarkMapper[k], mRadius * 0.06f, mInnerRadius / 6 * (k + 1) + mRadius * 0.02f-minnerradius, mTextPaint); @param mark mark 100~0 @return */ private String abilityMark2Str(int mark) {if (mark <= 100 && mark > 80) { return mMarkMapper[0]; } else if (mark <= 80 && mark > 60) { return mMarkMapper[1]; } else if (mark <= 60 && mark > 40) { return mMarkMapper[2]; } else if (mark <= 40 && mark > 20) { return mMarkMapper[3]; } else if (mark <= 20 && mark > 0) { return mMarkMapper[4]; } return "up"; }Copy the code
4. The final step: Draw the content
Thought even a point, did not expect… Typed half a page of my scratch paper (manual emoji – horrible)
What you see here is a for loop, which is actually a little bit of analysis and testing and finding patterns. What’s the trick? Take out the draft paper to draw, calculate + analyze… Eyes alone will not do
MAbilityPaint = new Paint(paint.anti_alias_flag); mAbilityPaint.setColor(0x8897C5FE); mAbilityPath = new Path();Copy the code
/** * @param canvas */ private void drawAbility(canvas canvas) {float step = mInnerRadius / 6; MAbilityPath. MoveTo (0, -mabilitymark [0] / 20.f * step); For (int I = 1; i < 6; i++) { float mark = mAbilityMark[i] / 20.f; mAbilityPath.lineTo( (float) (mark * step * Math.cos(Math.PI/180*(-30+60*(i-1)))), (float) (mark * step * Math.sin(Math.PI/180*(-30+60*(i-1))))); } mAbilityPath.close(); canvas.drawPath(mAbilityPath, mAbilityPaint); }Copy the code
That’s it. You think that’s the end? It’s just the beginning!
Second, data extraction and encapsulation
I’m just using test data, and it’s all in the View, which is definitely not going to work
Now encapsulate the data, expose the interface methods, and open the View and external access
1. Size limit of View
Use width as diameter, ignore height, dimensions for circular areas
As shown below, you can see that all dimensions are determined in terms of mRadius, so scaling is also proportional
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
mRadius = MeasureSpec.getSize(widthMeasureSpec) / 2;
mInnerRadius = 0.6f * mRadius;
setMeasuredDimension(MeasureSpec.getSize(widthMeasureSpec), MeasureSpec.getSize(widthMeasureSpec));
}
Copy the code
2. Data processing
To make it easy to see the relationships between the data, use a Map to load the capabilities and values
private HashMap<String, Integer> mData; MData = new HashMap<>(); Mdata. put(" destructive force ", 100); Mdata. put(" speed ", 100); Mdata. put(" range ", 60); Mdata. put(" endurance ", 100); Data.put(" precision ", 100); Data.put(" growth ", 100); mAbilityInfo = mData.keySet().toArray(new String[mData.size()]); mAbilityMark = mData.values().toArray(new Integer[mData.size()]);Copy the code
3. Mapping between data and characters:DataMapper
In other words, the string between 100 and 80 can be customized, such as “1”, “I”, and “☆”
This is one of the decoupling methods THAT I just realized, which is a policy design pattern (only five levels). If you want to customize the classification, override the abilityMark2Str method
/** * Author: Zhang Feng Jiete Lie <br/> * Time: 2018/12/280028:12:21 <br/> * Email: [email protected]<br/> * Description: Public class DataMapper {protected String[] mapper; public DataMapper(String[] mapper) { if (mapper.length ! = 5) { throw new IllegalArgumentException("the length of mapper must be 5"); } this.mapper = mapper; } public String[] getMapper() { return mapper; } public String abilityMark2Str(int mark) {if (mark <= 100 &&) mark > 80) { return mapper[0]; } else if (mark <= 80 && mark > 60) { return mapper[1]; } else if (mark <= 60 && mark > 40) { return mapper[2]; } else if (mark <= 40 && mark > 20) { return mapper[3]; } else if (mark <= 20 && mark > 0) { return mapper[4]; } return "up"; }}Copy the code
Give a default mapping class: WordMapper
That’s the same method that I just wrote in the View
/** * Author: Zhang Feng Jiete Lie <br/> * Time: 2018/12/280028:12:24 <br/> * Email: [email protected]<br/> * Description: Public WordMapper extends DataMapper {public WordMapper() {super(new String[]{"A", "B", "C", "D", "E"}); }Copy the code
So how do you modify it in a View?
Private DataMapper mDataMapper; MDataMapper = new WordMapper(); Private void drawInfoText(Canvas Canvas) {float R2 = mRadius - 0.08 f * mRadius; For (int I = 0; i < 6; i++) { canvas.save(); canvas.rotate(60 * i + 180); MTextPaint. SetTextSize (mRadius * 0.1 f); Canvas. DrawText (mAbilityInfo[I], 0, r2-0.06 f * mRadius, mTextPaint); MTextPaint. SetTextSize (mRadius * 0.15 f); Canvas. DrawText (mDataMapper. AbilityMark2Str (mAbilityMark [I]), and 0, r2-0.18 f * mRadius mTextPaint); canvas.restore(); } mTextPaint. SetTextSize (mRadius * 0.07 f); for (int k = 0; k < 5; K ++) {canvas. DrawText (mdatamapper.getMapper ()[k], mRadius * 0.06f, MInnerRadius / 6 * (k + 1) + mRadius * 0.02f-minnerradius, mTextPaint); }} public DataMapper getDataMapper() {return mDataMapper; } public void setDataMapper(DataMapper dataMapper) { mDataMapper = dataMapper; Public HashMap<String, Integer> getData() {return mData; } public void setData(HashMap<String, Integer> data) { mData = data; mAbilityInfo = mData.keySet().toArray(new String[mData.size()]); mAbilityMark = mData.values().toArray(new Integer[mData.size()]); invalidate(); }Copy the code
4. Usage:
Use DataMapper to pull strings out, and you can also return strings based on numeric values
AbilityView abilityView = findViewById(R.id.id_ability_view); mData = new HashMap<>(); mData.put("Java", 100); mData.put("Kotlin", 70); mData.put("JavaScript", 100); mData.put("Python", 60); mData.put("Dart", 50); mData.put("C++", 60); AbilityView. SetDataMapper (new DataMapper (new String [] {" god ", "high", "general", "new", "into"})); abilityView.setData(mData);Copy the code
Okay, done. You think it’s over? No, continue
Three, n attributes you than
Got six of them. Can’t we stop? You can see that there is also a dead thing, that is the number of data
This is trouble, if just is 0->1 creation, filling data is 1->2 accumulation, then the next is 2->n life well, I played half a draft paper, finally finished! View is less than 200 lines of code in total, which feels very elegant. If you are interested in doing your own research (drawing pictures, making drafts), if you are not interested in using it directly,
/**
* 作者:张风捷特烈<br/>
* 时间:2018/12/28 0028:7:40<br/>
* 邮箱:[email protected]<br/>
* 说明:能力对比图
*/
public class AbilityView extends View {
private static final String TAG = "AbilityView";
private float mRadius = dp(100);//外圆半径
private float mLineWidth = dp(1);//线宽
private Paint mLinePaint;//线画笔
private Paint mFillPaint;//填充画笔
private Path mPath;
private HashMap<String, Integer> mData;//核心数据
private Paint mTextPaint;
String[] mAbilityInfo;
Integer[] mAbilityMark;
private float mInnerRadius;
private Path mAbilityPath;
private Paint mAbilityPaint;
private DataMapper mDataMapper;//数据与字符串映射规则
public AbilityView(Context context) {
this(context, null);
}
public AbilityView(Context context, @Nullable AttributeSet attrs) {
this(context, attrs, 0);
}
public AbilityView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init();
}
private void init() {
mLinePaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mLinePaint.setStrokeWidth(mLineWidth);
mLinePaint.setStyle(Paint.Style.STROKE);
mFillPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mFillPaint.setStrokeWidth(0.05f * mRadius);
mTextPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mTextPaint.setTextSize(mRadius * 0.1f);
mTextPaint.setTextAlign(Paint.Align.CENTER);
mAbilityPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mAbilityPaint.setColor(0x8897C5FE);
mAbilityPath = new Path();
mPath = new Path();
mData = new HashMap<>();
mDataMapper = new WordMapper();//初始化DataMapper--默认WordMapper
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
mRadius = MeasureSpec.getSize(widthMeasureSpec) / 2;
mInnerRadius = 0.6f * mRadius;
setMeasuredDimension(MeasureSpec.getSize(widthMeasureSpec), MeasureSpec.getSize(widthMeasureSpec));
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
if (mAbilityInfo == null) {
return;
}
canvas.translate(mRadius, mRadius);//移动坐标系
drawOutCircle(canvas);
drawInnerCircle(canvas);
drawInfoText(canvas);
drawAbility(canvas);
}
/**
* 绘制能力面
*
* @param canvas
*/
private void drawAbility(Canvas canvas) {
float step = mInnerRadius / (mDataMapper.getMapper().length + 1);//每小段的长度
mAbilityPath.moveTo(0, -mAbilityMark[0] / 20.f * step);//起点
for (int i = 1; i < mData.size(); i++) {
float mark = mAbilityMark[i] / 20.f;
mAbilityPath.lineTo(
(float) (mark * step * Math.cos(Math.PI / 180 * (360.f / mData.size() * i - 90))),
(float) (mark * step * Math.sin(Math.PI / 180 * (360.f / mData.size() * i - 90))));
}
mAbilityPath.close();
canvas.drawPath(mAbilityPath, mAbilityPaint);
}
/**
* 绘制文字
*
* @param canvas 画布
*/
private void drawInfoText(Canvas canvas) {
float r2 = mRadius - 0.08f * mRadius;//下圆半径
for (int i = 0; i < mData.size(); i++) {
canvas.save();
canvas.rotate(360.f / mData.size() * i + 180);
mTextPaint.setTextSize(mRadius * 0.1f);
canvas.drawText(mAbilityInfo[i], 0, r2 - 0.06f * mRadius, mTextPaint);
mTextPaint.setTextSize(mRadius * 0.15f);
canvas.drawText(
mDataMapper.abilityMark2Str(mAbilityMark[i]), 0, r2 - 0.18f * mRadius, mTextPaint);
canvas.restore();
}
mTextPaint.setTextSize(mRadius * 0.07f);
for (int k = 0; k < mDataMapper.getMapper().length; k++) {
canvas.drawText(mDataMapper.getMapper()[k], mRadius * 0.06f,
mInnerRadius / (mDataMapper.getMapper().length + 1) * (k + 1) + mRadius * 0.02f - mInnerRadius, mTextPaint);
}
}
/**
* 绘制内圈圆
*
* @param canvas 画布
*/
private void drawInnerCircle(Canvas canvas) {
canvas.save();
canvas.drawCircle(0, 0, mInnerRadius, mLinePaint);
canvas.save();
for (int i = 0; i < mData.size(); i++) {//遍历6条线
canvas.save();
canvas.rotate(360.f / mData.size() * i);//每次旋转60°
mPath.moveTo(0, -mInnerRadius);
mPath.rLineTo(0, mInnerRadius);//线的路径
for (int j = 1; j <= mDataMapper.getMapper().length; j++) {
mPath.moveTo(-mRadius * 0.02f, -mInnerRadius / (mDataMapper.getMapper().length + 1) * j);
mPath.rLineTo(mRadius * 0.02f * 2, 0);
}//加5条小线
canvas.drawPath(mPath, mLinePaint);//绘制线
canvas.restore();
}
canvas.restore();
}
/**
* 绘制外圈
*
* @param canvas 画布
*/
private void drawOutCircle(Canvas canvas) {
canvas.save();
canvas.drawCircle(0, 0, mRadius, mLinePaint);
float r2 = mRadius - 0.08f * mRadius;//下圆半径
canvas.drawCircle(0, 0, r2, mLinePaint);
for (int i = 0; i < 22; i++) {//循环画出小黑条
canvas.save();
canvas.rotate(360 / 22f * i);
canvas.drawLine(0, -mRadius, 0, -r2, mFillPaint);
canvas.restore();
}
canvas.restore();
}
protected float dp(float dp) {
return TypedValue.applyDimension(
TypedValue.COMPLEX_UNIT_DIP, dp, getResources().getDisplayMetrics());
}
/////////////////////////////---------------------
public float getRadius() {
return mRadius;
}
public void setRadius(float radius) {
mRadius = radius;
}
public DataMapper getDataMapper() {
return mDataMapper;
}
public void setDataMapper(DataMapper dataMapper) {
mDataMapper = dataMapper;
}
public HashMap<String, Integer> getData() {
return mData;
}
public void setData(HashMap<String, Integer> data) {
mData = data;
mAbilityInfo = mData.keySet().toArray(new String[mData.size()]);
mAbilityMark = mData.values().toArray(new Integer[mData.size()]);
invalidate();
}
}
Copy the code
Well, this is really over
Postscript: Jie wen standard
1. Growth record and Errata of this paper
Program source code | The date of | note |
---|---|---|
V0.1 – making | 2018-12-28 | Android custom control (master level)- JOJO with the ability analysis diagram |
2. More about me
Pen name | hobby | ||
---|---|---|---|
Zhang Feng Jie te Li | 1981462002 | zdl1994328 | language |
My lot | My Jane books | I’m the nuggets | Personal website |
3. The statement
1—- This article is originally written by Zhang Fengjie, please note if reproduced
2—- welcome the majority of programming enthusiasts to communicate with each other 3—- personal ability is limited, if there is something wrong welcome to criticize and testify, must be humble to correct 4—- see here, I thank you here for your love and support