日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Android自定義控件之刻度尺控件

瀏覽:28日期:2022-09-23 15:29:19

今天我做的是一個自定義刻度尺控件,由于項目需求需要使用刻度尺那樣滑動選擇,由于對自定義控件的認識還不夠深入,于是花了一周多時間才把這個控件給整出來,也是嘔心瀝血的經(jīng)歷啊,也讓我對自定義控件有了自己的認識,廢話不多說,先上一個簡單的效果圖,大家可以看看,如有需求可以直接拿去使用

效果圖如下:只是我的一個簡單Demo,效果有點丑陋了點,希望海涵!

Android自定義控件之刻度尺控件

效果已經(jīng)出來接下來就是代碼部分了,一看就只是一般的控件很難實現(xiàn),于是就開始了我的自定義View之旅,每次自定義完后總是會收獲很多東西,如下是我的代碼:

package android.tst.com.myapplication; import android.annotation.SuppressLint;import android.content.Context;import android.graphics.Canvas;import android.graphics.Color;import android.graphics.Paint;import android.text.Layout;import android.text.TextPaint;import android.util.AttributeSet;import android.view.MotionEvent;import android.view.VelocityTracker;import android.view.View;import android.view.ViewConfiguration;import android.widget.Scroller; /** * 卷尺控件類。由于時間比較緊,只有下班后有時間,因此只實現(xiàn)了基本功能。<br> * 細節(jié)問題包括滑動過程中widget邊緣的刻度顯示問題等<br> * @version create:2014年8月26日 */@SuppressLint('ClickableViewAccessibility')public class RulerView extends View { public interface OnValueChangeListener { public void onValueChange(float value); } public static final int MOD_TYPE_HALF = 2; public static final int MOD_TYPE_ONE = 10; private static final int ITEM_HALF_DIVIDER = 10; private static final int ITEM_ONE_DIVIDER = 10; private static final int ITEM_MAX_HEIGHT = 20; private static final int ITEM_MIN_HEIGHT = 10; private static final int TEXT_SIZE = 7; private float mDensity; private int mValue = 50, mMaxValue = 100, mModType = MOD_TYPE_HALF, mLineDivider = ITEM_HALF_DIVIDER; // private int mValue = 50, mMaxValue = 500, mModType = MOD_TYPE_ONE, // mLineDivider = ITEM_ONE_DIVIDER; private int mLastX, mMove; private int mWidth, mHeight; private int mMinVelocity; private Scroller mScroller; private VelocityTracker mVelocityTracker; private OnValueChangeListener mListener; @SuppressWarnings('deprecation') public RulerView(Context context, AttributeSet attrs) { super(context, attrs); mScroller = new Scroller(getContext()); mDensity = getContext().getResources().getDisplayMetrics().density; mMinVelocity = ViewConfiguration.get(getContext()) .getScaledMinimumFlingVelocity(); } /** * * 考慮可擴展,但是時間緊迫,只可以支持兩種類型效果圖中兩種類型 * * @param value * 初始值 * @param maxValue * 最大值 * @param model * 刻度盤精度:<br> * {@link MOD_TYPE_HALF}<br> * {@link MOD_TYPE_ONE}<br> */ public void initViewParam(int defaultValue, int maxValue, int model) { switch (model) { case MOD_TYPE_HALF: mModType = MOD_TYPE_HALF; mLineDivider = ITEM_HALF_DIVIDER; mValue = defaultValue * 2; mMaxValue = maxValue * 2; break; case MOD_TYPE_ONE: mModType = MOD_TYPE_ONE; mLineDivider = ITEM_ONE_DIVIDER; mValue = defaultValue; mMaxValue = maxValue; break; default: break; } invalidate(); mLastX = 0; mMove = 0; notifyValueChange(); } /** * 設置用于接收結(jié)果的監(jiān)聽器 * * @param listener */ public void setValueChangeListener(OnValueChangeListener listener) { mListener = listener; } /** * 獲取當前刻度值 * * @return */ public float getValue() { return mValue; } public void setValue(int value){ mValue = value; notifyValueChange(); postInvalidate(); } public void setValueToChange(int what) { mValue += what; notifyValueChange(); postInvalidate(); } @Override protected void onLayout(boolean changed, int left, int top, int right, int bottom) { mWidth = getWidth(); mHeight = getHeight(); super.onLayout(changed, left, top, right, bottom); } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); drawScaleLine(canvas); // drawWheel(canvas); drawMiddleLine(canvas); } /** * 從中間往兩邊開始畫刻度線 * * @param canvas */ private void drawScaleLine(Canvas canvas) { canvas.save(); Paint linePaint = new Paint(); linePaint.setStrokeWidth(2); linePaint.setColor(Color.rgb(141, 189, 225)); TextPaint textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); textPaint.setColor(Color.rgb(68, 135, 188)); textPaint.setTextSize(TEXT_SIZE * mDensity); int width = mWidth, drawCount = 0; float xPosition = 0, textWidth = Layout.getDesiredWidth('0', textPaint); for (int i = 0; drawCount <= 4 * width; i++) { int numSize = String.valueOf(mValue + i).length(); // 前 xPosition = (width / 2 - mMove) + i * mLineDivider * mDensity; if (xPosition + getPaddingRight() < mWidth) { if ((mValue + i) % mModType == 0) { linePaint.setColor(Color.rgb(68, 135, 188)); canvas.drawLine(xPosition, getPaddingTop(), xPosition, mDensity * ITEM_MAX_HEIGHT, linePaint); if (mValue + i <= mMaxValue) { switch (mModType) { case MOD_TYPE_HALF: canvas.drawText( String.valueOf((mValue + i) / 2), countLeftStart(mValue + i, xPosition, textWidth), getHeight() - textWidth, textPaint); break; case MOD_TYPE_ONE: canvas.drawText(String.valueOf(mValue + i), xPosition - (textWidth * numSize / 2), getHeight() - textWidth, textPaint); break; default: break; } } } else { linePaint.setColor(Color.rgb(141, 189, 225)); // linePaint.setColor(Color.rgb(68, 135, 188)); canvas.drawLine(xPosition, getPaddingTop(), xPosition, mDensity * ITEM_MIN_HEIGHT, linePaint); } } // 后 xPosition = (width / 2 - mMove) - i * mLineDivider * mDensity; if (xPosition > getPaddingLeft()) { if ((mValue - i) % mModType == 0) { linePaint.setColor(Color.rgb(68, 135, 188)); canvas.drawLine(xPosition, getPaddingTop(), xPosition, mDensity * ITEM_MAX_HEIGHT, linePaint); if (mValue - i >= 0) { switch (mModType) { case MOD_TYPE_HALF: canvas.drawText( String.valueOf((mValue - i) / 2), countLeftStart(mValue - i, xPosition, textWidth), getHeight() - textWidth, textPaint); break; case MOD_TYPE_ONE: canvas.drawText(String.valueOf(mValue - i), xPosition - (textWidth * numSize / 2), getHeight() - textWidth, textPaint); break; default: break; } } } else { linePaint.setColor(Color.rgb(141, 189, 225)); canvas.drawLine(xPosition, getPaddingTop(), xPosition, mDensity * ITEM_MIN_HEIGHT, linePaint); } } drawCount += 2 * mLineDivider * mDensity; } canvas.restore(); } /** * 計算沒有數(shù)字顯示位置的輔助方法 * * @param value * @param xPosition * @param textWidth * @return */ private float countLeftStart(int value, float xPosition, float textWidth) { float xp = 0f; if (value < 20) { xp = xPosition - (textWidth * 1 / 2); } else { xp = xPosition - (textWidth * 2 / 2); } return xp; } /** * 畫中間的紅色指示線、陰影等。指示線兩端簡單的用了兩個矩形代替 * * @param canvas */ private void drawMiddleLine(Canvas canvas) { // TOOD 常量太多,暫時放這,最終會放在類的開始,放遠了怕很快忘記 int gap = 12, indexWidth = 2, indexTitleWidth = 24, indexTitleHight = 10, shadow = 6; String color = '#66999999'; canvas.save(); Paint redPaint = new Paint(); redPaint.setStrokeWidth(indexWidth); redPaint.setColor(Color.RED); canvas.drawLine(mWidth / 2, 0, mWidth / 2, mHeight, redPaint); canvas.restore(); } @Override public boolean onTouchEvent(MotionEvent event) { int action = event.getAction(); int xPosition = (int) event.getX(); if (mVelocityTracker == null) { mVelocityTracker = VelocityTracker.obtain(); } mVelocityTracker.addMovement(event); switch (action) { case MotionEvent.ACTION_DOWN: mScroller.forceFinished(true); mLastX = xPosition; mMove = 0; break; case MotionEvent.ACTION_MOVE: getParent().requestDisallowInterceptTouchEvent(true); mMove += (mLastX - xPosition); changeMoveAndValue(); break; case MotionEvent.ACTION_UP: case MotionEvent.ACTION_CANCEL: countMoveEnd(); countVelocityTracker(event); getParent().requestDisallowInterceptTouchEvent(false); return false; // break; default: break; } mLastX = xPosition; return true; } private void countVelocityTracker(MotionEvent event) { mVelocityTracker.computeCurrentVelocity(1000); float xVelocity = mVelocityTracker.getXVelocity(); if (Math.abs(xVelocity) > mMinVelocity) { mScroller.fling(0, 0, (int) xVelocity, 0, Integer.MIN_VALUE, Integer.MAX_VALUE, 0, 0); } } private void changeMoveAndValue() { int tValue = (int) (mMove / (mLineDivider * mDensity)); if (Math.abs(tValue) > 0) { mValue += tValue; mMove -= tValue * mLineDivider * mDensity; if (mValue <= 0 || mValue > mMaxValue) { mValue = mValue <= 0 ? 0 : mMaxValue; mMove = 0; mScroller.forceFinished(true); } notifyValueChange(); } postInvalidate(); } private void countMoveEnd() { int roundMove = Math.round(mMove / (mLineDivider * mDensity)); mValue = mValue + roundMove; mValue = mValue <= 0 ? 0 : mValue; mValue = mValue > mMaxValue ? mMaxValue : mValue; mLastX = 0; mMove = 0; notifyValueChange(); postInvalidate(); } private void notifyValueChange() { if (null != mListener) { if (mModType == MOD_TYPE_ONE) { mListener.onValueChange(mValue); } if (mModType == MOD_TYPE_HALF) { mListener.onValueChange(mValue / 2f); } } } @Override public void computeScroll() { super.computeScroll(); if (mScroller.computeScrollOffset()) { if (mScroller.getCurrX() == mScroller.getFinalX()) { // over countMoveEnd(); } else { int xPosition = mScroller.getCurrX(); mMove += (mLastX - xPosition); changeMoveAndValue(); mLastX = xPosition; } } } @Override public boolean dispatchTouchEvent(MotionEvent event) { getParent().requestDisallowInterceptTouchEvent(true); return super.dispatchTouchEvent(event); }}

這是我的自定義View部分的代碼,下面就是在布局中的使用了

<TextView android: android:layout_width='match_parent' android:layout_height='wrap_content' android:padding='10dp' android:gravity='center' android:textColor='@android:color/holo_red_dark'/> <android.tst.com.myapplication.RulerView android: android:layout_width='match_parent' android:layout_height='60dp'/> <LinearLayout android:layout_width='match_parent' android:orientation='horizontal' android:layout_height='wrap_content'> <Button android: android:layout_width='0dp' android:layout_height='wrap_content' android:text='+' android:textSize='25sp' android:gravity='center' android:layout_marginRight='15dp' android:layout_weight='1'/> <Button android: android:layout_width='0dp' android:layout_height='wrap_content' android:text='-' android:layout_marginLeft='15dp' android:textSize='25sp' android:gravity='center' android:layout_weight='1'/></LinearLayout>

如上根據(jù)效果圖,我需要一個TextView進行顯示,還有就是我的自定義刻度尺控件了,接下來就是兩個Button控件加減。

接下來就是在Activity中的使用了

首先需要一個Handler進行更新TextView中的值

Handler handler = new Handler() { @Override public void handleMessage(Message msg) { tv_values.setText(rv_view.getValue() + 'Kg'); }; };

其次就是初始化相關的操作了

private void deployRulerView(){ rv_view= (RulerView) findViewById(R.id.rv_view); btn_jia= (Button) findViewById(R.id.btn_jia); btn_jian= (Button) findViewById(R.id.btn_jian); tv_values= (TextView) findViewById(R.id.tv_values); //設置RulerView的初始值 rv_view.setValue(60); //初始化刻度尺范圍 rv_view.initViewParam(60, 200, RulerView.MOD_TYPE_ONE); rv_view.setValueChangeListener(new RulerView.OnValueChangeListener() { @Override public void onValueChange(float value) { handler.sendMessage(new Message()); } }); tv_values.setText(60+'KG'); //給兩個控件添加監(jiān)聽事件 btn_jia.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { rv_view.setValueToChange(1); } }); btn_jian.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { rv_view.setValueToChange(-1); } }); }

到這里整個過程已經(jīng)完成了,如果不好的地方盡情吐槽,整個過程,最復雜的莫過于自定義中的繪制過程,但是一切的問題當你靜下心好好去實現(xiàn)時,那一切的問題都就不存在了。

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網(wǎng)。

標簽: Android
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
制服诱惑一区二区| 久久精品国产网站| 日韩欧美1区| 欧美www视频在线观看| 国产精品激情| 日韩和欧美的一区| 日本综合精品一区| 国产精品一在线观看| 国产精一区二区| 国产欧美另类| 麻豆国产精品| 成人在线视频区| www在线观看黄色| 在线一区av| 影视先锋久久| 性色一区二区| 日本一区二区三区中文字幕| 亚洲免费专区| 欧美一区影院| 久久av中文| 精品久久在线| 伊人精品一区| 蜜臀av亚洲一区中文字幕| 日韩黄色av| 国产一区二区三区黄网站 | 精品一区毛片| 美女日韩在线中文字幕| 日韩精品一区二区三区中文字幕| 国产欧美另类| 日韩精品永久网址| 国产一级一区二区| 午夜亚洲福利| 欧美成人精品一级| 日本欧美不卡| 欧美69视频| 亚洲色图综合| 久久99影视| 亚洲国产专区校园欧美| 免费人成精品欧美精品| 国产精品流白浆在线观看| 日本少妇一区| 日韩三级精品| 国产中文字幕一区二区三区| 亚洲二区视频| 婷婷综合福利| 国产成人精品一区二区三区免费 | 国产精品2023| 美女网站视频一区| 亚洲性视频在线| 国产中文欧美日韩在线 | 国内精品福利| 国产三级精品三级在线观看国产| 亚洲永久av| 四虎国产精品免费久久| 黄色aa久久| 中文字幕亚洲精品乱码| 成人在线免费观看91| 国产高清一区| 欧美激情精品| 香蕉精品999视频一区二区| 精品亚洲精品| 蜜臀av在线播放一区二区三区 | 日韩影院二区| 日本综合精品一区| 成人看片网站| 欧美亚洲三区| 亚洲午夜在线| 麻豆免费精品视频| 水野朝阳av一区二区三区| 久久三级毛片| 亚洲一区久久| 中文在线资源| 久久国产婷婷国产香蕉| 不卡在线一区二区| 国产成人久久精品一区二区三区| 一区二区三区网站| 在线观看精品| 国产精品主播| 综合国产在线| 欧美精选一区二区三区| 日本久久精品| 清纯唯美亚洲综合一区| 免费欧美一区| 蜜臀国产一区| 国产欧美日韩视频在线| 国产精品美女| av在线日韩| 精品黄色一级片| 日韩精品久久理论片| 亚洲女同中文字幕| 波多视频一区| 国产欧美一区二区三区米奇| 老牛影视一区二区三区| 久久裸体视频| 91日韩免费| 免费看久久久| 国产毛片久久久| 亚洲精品一区二区在线播放∴| 999久久久91| 国产a久久精品一区二区三区| 亚洲理论在线| 91久久在线| 久久久精品网| а√天堂8资源中文在线| 国产激情综合| 欧美一级一区| 婷婷五月色综合香五月| 亚洲字幕久久| 麻豆精品91| 亚洲综合精品| 狠狠干综合网| 欧美影院三区| 美女毛片一区二区三区四区| 香蕉成人av| 成人日韩在线| 伊人久久在线| 成人片免费看| 国产拍在线视频| av中文字幕在线观看第一页| 久久亚洲精精品中文字幕| 美女性感视频久久| 国产精品网站在线看| 国产欧美一区二区三区精品观看| 日本少妇一区二区| 日韩av电影一区| 国产视频网站一区二区三区| 国产伦乱精品| 美日韩一区二区三区| 国产精品xxxav免费视频| 国产欧美综合一区二区三区| 久久国内精品视频| 久久99精品久久久野外观看| 开心激情综合| 精品午夜av| 97精品国产| 丝袜美腿诱惑一区二区三区| 久久久久.com| 不卡在线一区二区| 蜜臀久久99精品久久久久久9| 亚洲精品黄色| 日韩1区2区日韩1区2区| 国产精品久久久久久模特| 国产精品对白| sm捆绑调教国产免费网站在线观看 | 日韩欧美视频专区| 桃色一区二区| 亚洲精品小说| 日韩黄色在线观看| 精品久久影院| 日韩毛片在线| 一区在线视频观看| 日韩一区欧美二区| 日本a口亚洲| 免费视频一区二区三区在线观看 | 亚洲主播在线| 天堂久久av| 国产欧美日韩一级| 亚洲精品88| 国产一区91| 久久狠狠久久| 中文字幕在线高清| 黄色在线一区| 日韩av中文字幕一区二区三区| 久久精品国内一区二区三区| 激情五月综合| 日本成人在线视频网站| 麻豆精品视频在线| 久久尤物视频| 日韩精品麻豆| 一区二区三区网站| 精品一区二区三区亚洲| 日韩88av| 热久久久久久久| 久久av网址| 女人天堂亚洲aⅴ在线观看| 日本特黄久久久高潮| 国产999精品在线观看| 亚洲一区黄色| 国产在线视频欧美一区| 国产精品人人爽人人做我的可爱| 日韩二区在线观看| 色婷婷综合网| 亚洲欧美日韩国产一区| 麻豆精品av| 视频一区视频二区中文字幕| 久久精品国产成人一区二区三区| 日韩欧美精品综合| 日韩黄色av| 激情婷婷欧美| 欧美中文高清| 久久三级福利| 日本免费新一区视频| 欧美 日韩 国产精品免费观看| 国产精品porn| 99国产精品99久久久久久粉嫩| 精品视频在线观看网站| 免费观看日韩电影| 亚洲成人精品| 国产成人精品亚洲线观看| 只有精品亚洲|