亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Android > 正文

Android自定義控件實現帶數值和動畫的圓形進度條

2019-10-21 21:31:48
字體:
來源:轉載
供稿:網友

本文實例實現一個如下圖所示的Android自定義控件,可以直觀地展示某個球隊在某個賽季的積分數和勝場、負場、平局數

Android,進度條

首先對畫布進行區域劃分,整個控件分上下兩部分

上邊是個大的圓環,圓環中間兩行文字,沒什么難度,選好圓心坐標和半徑后直接繪制即可,繪制文字也是如此。

下部分是三個小的圓弧進度條,弧的末端繪制一個小的實心圓

首先選好坐標和半徑,然后先繪制三個圓環作為弧形進度條的背景

之后從12點鐘開始繪制進度弧,知道了圓環的圓心和半徑,也知道了弧對應于12點鐘和圓環圓心的偏移角度

通過三角函數可以計算出進度弧終點坐標,以進度弧終點坐標為圓心繪制一個小的實心圓即可

動畫效果通過Handler的postDelayed方法觸發重繪即可實現

在項目中的效果如圖所示:

Android,進度條

測試代碼如下:

final Random random=new Random();final ScoreBoardView myView=(ScoreBoardView)findViewById(R.id.custom_view);myView.setOnClickListener(new View.OnClickListener(){ @Override public void onClick(View v){  myView.setColor(Color.BLUE);  myView.setScore(random.nextInt(28));  myView.setWinDrawLose(random.nextInt(12),random.nextInt(15),random.nextInt(26)); }});

完整代碼如下:

public class ScoreBoardView extends View { private Context context; /*弧的顏色*/ private int mColor; /*積分數,勝場數,平局數,負場數*/ private int mScore, mWinNumber, mDrawNumber, mLoseNumber; /*傳入數字的最大值*/ private final int FULL_SCORE = 30; /*動畫插值器*/ DecelerateInterpolator mDecelerateInterpolator = new DecelerateInterpolator(); /*動畫持續時間(刷新次數)*/ private int mDuration = 10; /*動畫刷新過程中的當前值*/ private int mCurrentTime = 0; private TypedValue typedValue; private TypedValue typedValue1; private Handler mHandler = new Handler(); private Runnable mAnimation = new Runnable() {  @Override  public void run() {   if (mCurrentTime < mDuration) {    mCurrentTime++;    /*導致重繪調用onDraw,onDraw最后再次postDelay執行此動畫,直到達到指定的次數*/    ScoreBoardView.this.invalidate();   }  } }; /*繪制圖形*/ private Paint paint = new Paint(); /*繪制文字*/ private Paint paintText = new Paint();  public ScoreBoardView(Context context) {  super(context);  this.context=context;  init(); }  public ScoreBoardView(Context context, AttributeSet attrs) {  super(context, attrs);  this.context=context;  init(); }  public ScoreBoardView(Context context, AttributeSet attrs, int defStyleAttr) {  super(context, attrs, defStyleAttr);  this.context=context;  init(); }  private void init() {  /*數據初始化,默認屬性*/  mColor = Color.rgb(95, 112, 72);  mScore = 0;  mWinNumber = 0;  mDrawNumber = 0;  mLoseNumber = 0;  typedValue=new TypedValue();  typedValue1=new TypedValue();  context.getTheme().resolveAttribute(R.attr.maintextclor, typedValue, true);  context.getTheme().resolveAttribute(R.attr.maintextclor_reverse,typedValue1,true); }  @Override protected void onDraw(Canvas canvas) {  super.onDraw(canvas);  /*獲取控件總的寬高*/  float totalWidth = getWidth();  float totalHeight = getHeight();  /*  * DecelerateInterpolator:動畫從開始到結束,變化率是一個減速的過程。  * AccelerateInterpolator:動畫從開始到結束,變化率是一個加速的過程。  * CycleInterpolator:動畫從開始到結束,變化率是循環給定次數的正弦曲線  * AccelerateDecelerateInterpolator:動畫從開始到結束,變化率是先加速后減速的過程。  * LinearInterpolator:動畫從開始到結束,變化率是線性變化。  * */  /*計算當前時刻動畫進度值*/  float AnimCurrentValue =mDecelerateInterpolator.getInterpolation(1.0f * mCurrentTime / mDuration);   /*圖形畫筆設置*/  paint.setAntiAlias(true);  paint.setStyle(Paint.Style.STROKE);   /*積分數,上邊的大圓*/  paint.setStrokeWidth(4);  paint.setColor(mColor);  /*積分大圓的中心坐標和半徑*/  float score_radius = totalHeight * 1 / 5, score_circle_x = totalWidth / 2, score_circle_y = totalHeight / 3;  /*繪制圓弧*/  canvas.drawCircle(score_circle_x, score_circle_y, score_radius, paint);  /*文字畫筆基本設置*/  paintText.setAntiAlias(true);  paintText.setStyle(Paint.Style.STROKE);  /*文字從中間開始繪制*/  /*Paint.Align.CENTER:The text is drawn centered horizontally on the x,y origin*/  paintText.setTextAlign(Paint.Align.CENTER);  /*文字畫筆大小和顏色設置*/  paintText.setTextSize(score_radius * 3 / 4);  paintText.setColor(getResources().getColor(typedValue.resourceId));  /*圓心位置繪制積分數值*/  canvas.drawText("" + mScore, score_circle_x, score_circle_y, paintText);  /*縮小字體繪制文本信息*/  paintText.setTextSize(score_radius * 1 / 4);  paintText.setAlpha(80);  /*圓心下邊繪制文本*/  canvas.drawText("積分", score_circle_x, score_circle_y + score_radius / 2, paintText);   /*設置畫筆,畫下邊的三個小圓*/  paint.setStrokeWidth(4);  paint.setAlpha(255);  /*下邊三個小圓的半徑*/  float small_radius = totalHeight / 10;  /*三個小圓的圓心的x坐標*/  float[] circleXs = new float[]{totalWidth / 2 - score_radius * 3 / 2,          totalWidth / 2,          totalWidth / 2 + score_radius * 3 / 2};  /*三個小圓的圓心的y坐標*/  float circleY = totalHeight * 3 / 4;  /*計算三個小圓弧掃過的角度*/  float[] theta_values = new float[]{360 * mWinNumber / FULL_SCORE* AnimCurrentValue,           360 * mDrawNumber / FULL_SCORE* AnimCurrentValue,           360 * mLoseNumber / FULL_SCORE* AnimCurrentValue};  /*設置畫筆顏色,繪制外圍圓環*/  paint.setColor(getResources().getColor(typedValue1.resourceId));  /*分別繪制三個外圍圓環*/  canvas.drawCircle(circleXs[0], circleY, small_radius, paint);//畫WIN背景圓  canvas.drawCircle(circleXs[1], circleY, small_radius, paint);//畫DRAW背景圓  canvas.drawCircle(circleXs[2], circleY, small_radius, paint);//畫LOSE背景圓  /*更改畫筆顏色,繪制圓弧進度條*/  paint.setColor(mColor);  /*drawArc的起始角度是3點鐘方向,因此要從12點鐘方向開始繪制,起始角度為270度*/  canvas.drawArc(new RectF(circleXs[0] - small_radius,         circleY - small_radius,         circleXs[0] + small_radius,         circleY + small_radius),      270, theta_values[0], false, paint);//畫WIN圓形進度條  canvas.drawArc(new RectF(circleXs[1] - small_radius,         circleY - small_radius,         circleXs[1] + small_radius,         circleY + small_radius),      270, theta_values[1], false, paint);//畫DRAW圓形進度條  canvas.drawArc(new RectF(circleXs[2] - small_radius,         circleY - small_radius,         circleXs[2] + small_radius,         circleY + small_radius),      270, theta_values[2], false, paint);//畫LOSE圓形進度條  /*繪制圓弧結束處的小圓點,實心圓*/  paint.setStyle(Paint.Style.FILL);  /*已知半徑、圓心位置、便宜角度,根據三角函數很方便計算出小實心圓圓心坐標*/  canvas.drawCircle(circleXs[0] + small_radius * (float) Math.sin(theta_values[0] * Math.PI / 180),    circleY - small_radius * (float) Math.cos(theta_values[0] * Math.PI / 180), 6, paint);//畫WIN末尾小圓點  canvas.drawCircle(circleXs[1] + small_radius * (float) Math.sin(theta_values[1] * Math.PI / 180),    circleY - small_radius * (float) Math.cos(theta_values[1] * Math.PI / 180), 6, paint);//畫DRAW末尾小圓點  canvas.drawCircle(circleXs[2] + small_radius * (float) Math.sin(theta_values[2] * Math.PI / 180),    circleY - small_radius * (float) Math.cos(theta_values[2] * Math.PI / 180), 6, paint);//畫LOSE末尾小圓點   /*繪制文字*/  paintText.setColor(getResources().getColor(typedValue.resourceId));  paintText.setTextSize(small_radius * 2 / 3);  /*測量文字大小,確定繪制文字時垂直方向的位置*/  Paint.FontMetrics fontMetrics = paint.getFontMetrics();  float textBaseLineOffset = (fontMetrics.bottom - fontMetrics.top) / 2 - fontMetrics.bottom;  /*在三個小圓的正中心位置繪制相應的數字*/  canvas.drawText("" + (int)(mWinNumber * AnimCurrentValue), circleXs[0], circleY + textBaseLineOffset, paintText);  canvas.drawText("" + (int)(mDrawNumber * AnimCurrentValue), circleXs[1], circleY + textBaseLineOffset, paintText);  canvas.drawText("" + (int)(mLoseNumber * AnimCurrentValue), circleXs[2], circleY + textBaseLineOffset, paintText);  /*調整字體大小,繪制文本信息*/  paintText.setTextSize(small_radius * 4 / 9);  canvas.drawText("勝場", circleXs[0], circleY - small_radius*4/3, paintText);  canvas.drawText("平局", circleXs[1], circleY - small_radius*4/3, paintText);  canvas.drawText("負場", circleXs[2], circleY - small_radius*4/3, paintText);  /*20ms刷新一次數據*/  mHandler.postDelayed(mAnimation, 20);//啟動動畫 }  public void setColor(int mColor) {  this.mColor = mColor;  invalidate(); }  public void setScore(int score) {  this.mScore = score;  invalidate(); }  public void setWinDrawLose(int win,int draw,int lose) {  this.mWinNumber = win;  this.mDrawNumber = draw;  this.mLoseNumber = lose;  mCurrentTime =0;  invalidate(); }}

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。


注:相關教程知識閱讀請移步到Android開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美黄色免费网站| 久久久91精品国产一区不卡| 国产精品亚洲精品| 久久理论片午夜琪琪电影网| 欧美性猛交丰臀xxxxx网站| 色yeye香蕉凹凸一区二区av| 日韩电视剧在线观看免费网站| 欧美xxxwww| 久久深夜福利免费观看| 久久免费视频网站| 91在线视频免费| 国产精品第8页| 国内精品久久久久久影视8| 欧美午夜片在线免费观看| 日韩欧美中文字幕在线观看| 中文字幕久热精品在线视频| 亚洲加勒比久久88色综合| 亚洲综合中文字幕68页| 欧美黑人xxxⅹ高潮交| 亚洲丁香久久久| 一本色道久久综合狠狠躁篇的优点| 伊人男人综合视频网| 久热爱精品视频线路一| 国产99久久精品一区二区| 成人有码视频在线播放| 原创国产精品91| 亚洲成人激情视频| 91久久嫩草影院一区二区| 国产精品美女www| 亚洲精品v欧美精品v日韩精品| 欧美一区亚洲一区| 亚洲欧洲一区二区三区在线观看| 欧美视频在线观看免费网址| 欧美日韩国产精品一区二区不卡中文| 日韩黄色高清视频| 另类图片亚洲另类| 亚洲精品福利免费在线观看| 欧美日韩中国免费专区在线看| 日韩在线观看免费高清| 热久久美女精品天天吊色| 亚洲精品资源美女情侣酒店| 午夜精品一区二区三区在线播放| 91精品国产自产在线| 亚洲美女视频网站| 在线精品视频视频中文字幕| 久久久国产精品免费| 久久福利网址导航| 高潮白浆女日韩av免费看| 欧美一级视频在线观看| 欧美最猛性xxxxx亚洲精品| 精品爽片免费看久久| 日本一区二区三区在线播放| 97人人爽人人喊人人模波多| 亚洲xxxxx性| 亚洲一区二区福利| 热久久这里只有| 亚洲欧美中文在线视频| 国产精品无码专区在线观看| 久久久免费观看视频| 2019国产精品自在线拍国产不卡| 日韩欧美国产视频| 日韩最新免费不卡| 亚洲欧美日韩视频一区| 国产精品视频网站| 日韩精品在线观看一区二区| 欧美不卡视频一区发布| 日韩电影中文 亚洲精品乱码| 久久久av一区| 欧美国产精品va在线观看| 国产精品免费小视频| 日韩在线观看免费高清完整版| 在线观看日韩专区| 亚洲区在线播放| 92看片淫黄大片欧美看国产片| 欧美日本黄视频| 亚洲一级黄色片| 性欧美xxxx视频在线观看| 亚洲人成亚洲人成在线观看| 国产精品久久久久久婷婷天堂| 美日韩精品免费观看视频| 国产欧美日韩91| 亚洲xxxx妇黄裸体| 日韩一区二区av| 国产精品色悠悠| 久久久久久97| 国产噜噜噜噜久久久久久久久| 欧美日韩国产丝袜另类| 精品亚洲男同gayvideo网站| 亚洲自拍高清视频网站| 国产福利成人在线| 亚洲综合第一页| 欧美一级电影免费在线观看| 欧美激情精品久久久久久免费印度| 国产欧美欧洲在线观看| 亚洲最大福利网站| 日韩中文字幕精品视频| 久久成年人视频| 欧美放荡办公室videos4k| 91中文字幕一区| 亚洲精品欧美一区二区三区| 亚洲春色另类小说| 欧美日韩国产第一页| 欧美裸体xxxxx| 亚洲免费成人av电影| 亚洲综合大片69999| 成人黄色片在线| 97超级碰碰人国产在线观看| 亚洲国产97在线精品一区| 久久久国产精品x99av| 欧美日韩午夜激情| 日韩精品视频在线免费观看| 国产日韩欧美影视| 亚洲国产日韩欧美在线99| 久久亚洲电影天堂| 久久深夜福利免费观看| 国产精品流白浆视频| 日韩**中文字幕毛片| 国产精品自拍视频| 久久亚洲一区二区三区四区五区高| 午夜精品久久久99热福利| 亚洲高清av在线| 精品在线观看国产| 午夜精品在线视频| 精品久久久久久久大神国产| 国产精品自拍偷拍视频| 91福利视频在线观看| 欧美色播在线播放| 亚洲精品自在久久| 欧美香蕉大胸在线视频观看| 91色中文字幕| 成人激情视频网| 中文字幕久久久| 97久久精品人搡人人玩| 久久国产精品久久国产精品| 欧美一区二区三区四区在线| 日韩中文娱乐网| 亚洲国产精品视频在线观看| 国产欧美在线视频| 久久av中文字幕| 成人免费视频在线观看超级碰| 日韩精品www| 最新国产成人av网站网址麻豆| 激情懂色av一区av二区av| 国模gogo一区二区大胆私拍| 色樱桃影院亚洲精品影院| 国产91精品不卡视频| 国产一区视频在线播放| 国产日韩在线免费| 亚洲国产精品成人精品| 视频一区视频二区国产精品| 国内精品久久久久久影视8| 亚洲色在线视频| 欧美大尺度在线观看| 日本欧美在线视频| 这里只有精品视频| 午夜精品美女自拍福到在线| 亚洲精品白浆高清久久久久久| 狠狠干狠狠久久| 精品一区二区亚洲| 91九色综合久久| 亚洲国产成人久久| 久久精品这里热有精品| 亚洲毛片在线观看.| www.日本久久久久com.|