qq_26437603 2016-07-17 15:17 采纳率: 100%
浏览 1483
已采纳

安卓中 surfaceview中显示三个折线图,可左右拖动查看未显示的图形

我在做安卓surfaceview的时候,要加载显示3条折线,但只有一条折线的时候,正常拖动都没问题,一旦显示3条折线,拖动就不能显示预期的折线,拖动后显示的是一片空白,除了最右边一点折线

自定义类

 public class MultiLineChartSurfaceView extends SurfaceView implements SurfaceHolder.Callback, Runnable {
    /** surface持有者 */
    private SurfaceHolder mHolder;
    /** 当前画布 */
    private Canvas mCanvas;
    /** 是否开始绘画 */
    private boolean mIsDrawing;

    /** 坐标轴举例view边框的距离 */
    private static final int DEFAULT_PADDING = 20;
    /** 默认宽度设置为400dp */
    private static final int DEFAULT_WIDTH = 450;
    /** 默认高度设置为400dp */
    private static final int DEFAULT_HEIGHT = 300;
    /** 默认折线的颜色 */
    private static final int DEFAULT_LINE_COLOR = Color.GREEN;
    /** 默认x轴的颜色 */
    private static final int DEFAULT_X_LINE_COLOR = 0xffEBEBEB;
    /** 默认Y轴的颜色 */
    private static final int DEFAULT_Y_LINE_COLOR = 0xffEBEBEB;
    /** 默认折线的宽度 */
    private static final int DEFAULT_LINE_WIDTH = 1;
    /** 默认x轴的量程 */
    private static final int DEFAULT_X_RANGE = 1000;
    /** 默认y轴的量程 */
    private static final int DEFAULT_Y_RANGE = 1000;
    /** 默认x轴每个刻度代表的单位 */
    private static final int DEFAULT_X_UNIT = 50;
    /** 默认y轴每个刻度代表的单位 */
    private static final int DEFAULT_Y_UNIT = 50;
    /** 每个刻度的高度 */
    private static final int DEFAULT_SCALE_HEIGHT = 4;
    /** 默认x轴尽头箭头的长度 */
    private static final int DEFAULT_X_BLANK = 20;
    /** 默认y轴尽头箭头的长度 */
    private static final int DEFAULT_Y_BLANK = 20;
    /** x轴的单位 */
    private int mXUnit;
    /** 每个y轴的单位的高度 */
    private int mYUnit;
    /** 折线的颜色 */
    private int mLineColor;
    /** x轴的颜色 */
    private int mXLineColor;
    /** y轴的颜色 */
    private int mYLineColor;
    /** 折线的宽度 */
    private int mLineWidth;
    /** x轴的量程 */
    private int mXRange;
    /** y轴的量程 */
    private int mYRange;
    /** 绘制折线的画笔 */
    private Paint mLinePaint;
    /** 绘制坐标的画笔 */
    private Paint mCoordinatePaint;
    /** view的宽度 */
    private int mWidth;
    /** View的高度 */
    private int mHeight;
    /** 坐标轴距离view边距的间距 */
    private int mPaddint;
    /** 刻度的高度 */
    private int mScaleHeight;
    /** y轴的实际高度 */
    private int mAxisHeight;
    /** x轴的实际高度 */
    private int mAxisWidth;
    /** X轴最后一个竖线与箭头的空格 */
    private int mXBlank;
    /** Y轴最后一个竖线与箭头的空格 */
    private int mYBlank;
    /** 每个x轴刻度的宽度 */
    private int mXScaleWidth;
    /** 每个y轴刻度的宽度 */
    private int mYScaleWidth;

    /** 外部的list,用来存放折线上的值 */
    private List<List<Integer>> mLines;
    /** 最后一次点击的x坐标 */
    private int lastX;
    /** 偏移量,用来实现平滑移动 */
    private int mOffset = 0;
    /** 移动速度,用来实现速度递减 */
    private int mSpeed = 0;
    /** 是否触摸屏幕 */
    private boolean mIsTouch = false;
    /** 时间计数器,用来快速滚动时候减速 */
    private int time = 0;
    /** 移动时候X方向上的速度 */
    private double xVelocity = 0;
    /** 是否可以滚动,当不在范围时候不可以滚动 */
    private boolean isScroll = true;
    /** x轴上的格子数量 */
    private int mXScaleNum;
    /** y轴上的格子数量 */
    private int mYScaleNum;

    public MultiLineChartSurfaceView(Context context) {
        this(context, null);
    }

    public MultiLineChartSurfaceView(Context context, AttributeSet attrs) {
        this(context, attrs, -1);
    }

    public MultiLineChartSurfaceView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        init();
    }

    /** 设置一些变量的尺寸 */
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        int widthMode = MeasureSpec.getMode(widthMeasureSpec);
        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        int heightMode = MeasureSpec.getMode(heightMeasureSpec);
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);
        if (widthMode == MeasureSpec.AT_MOST) {
            mWidth = dp2px(getContext(), DEFAULT_WIDTH);
        } else {
            mWidth = Math.max(widthSize, dp2px(getContext(), DEFAULT_WIDTH));
        }
        if (heightMode == MeasureSpec.AT_MOST) {
            mHeight = dp2px(getContext(), DEFAULT_HEIGHT);
        } else {
            mHeight = Math.max(heightSize, dp2px(getContext(), DEFAULT_HEIGHT));
        }
        setMeasuredDimension(mWidth, mHeight);
    }

    /** 初始化surfaceView操作 */
    private void init() {
        mHolder = getHolder();
        mHolder.addCallback(this);
        // 设置可以获取焦点
        setFocusable(true);
        // 进入触摸输入模式后,该控件是否还有获得焦点的能力
        setFocusableInTouchMode(true);
        // 是否保持屏幕常亮
        this.setKeepScreenOn(true);
        // 为部分成员变量赋初值,可以写成自定义属性
        mLineColor = DEFAULT_LINE_COLOR;
        mLineWidth = dp2px(getContext(), DEFAULT_LINE_WIDTH);
        mXLineColor = DEFAULT_X_LINE_COLOR;
        mYLineColor = DEFAULT_Y_LINE_COLOR;
        mXRange = DEFAULT_X_RANGE;
        mYRange = DEFAULT_Y_RANGE;
        mXUnit = DEFAULT_X_UNIT;
        mYUnit = DEFAULT_Y_UNIT;
        mPaddint = dp2px(getContext(), DEFAULT_PADDING);
        mScaleHeight = dp2px(getContext(), DEFAULT_SCALE_HEIGHT);

        mLinePaint = new Paint();
        mCoordinatePaint = new Paint();
    }

    /** 子线程,循环绘制折线 */
    @Override
    public void run() {
        // 如果处于绘画状态,那么就开始绘制
        while (mIsDrawing) {
            // 判断是否处于边缘状态
            setSpeedCut();
            // 绘制方法
            draw();
        }
    }

    /** 具体的绘制方法 */
    private void draw() {
        try {
            long start = System.currentTimeMillis();
            // 获取并锁定画布
            mCanvas = mHolder.lockCanvas();
            // 设置画布背景为白色
            mCanvas.drawColor(0xff000000);
            // 绘制坐标轴
            drawAxis();
            // 绘制曲线
            drawLine();
            long end = System.currentTimeMillis();
            if (end - start < 50) {
                Thread.sleep(50 - (end - start));
            }

        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            if (mCanvas != null) {
                // 保证每次都将绘制的内容提交到服务器
                mHolder.unlockCanvasAndPost(mCanvas);
            }
        }
    }

    /** 绘制坐标系 */
    private void drawAxis() {
        mCanvas.save();
        // 移动坐标原点到左下角
        mCanvas.translate(mPaddint, mHeight - mPaddint);
        // 画线
        drawXYLine();
        // 绘制坐标轴上的刻度
        // 绘制xy轴末端的箭头
        drawXYScale();
        mCanvas.restore();
    }

    /** 绘制折线 */
    private void drawLine() {
        mCanvas.save();
        mCanvas.translate(mPaddint, mHeight - mPaddint);
        // 设置画笔属性
        mLinePaint.setAntiAlias(true);
        mLinePaint.setStrokeWidth(3);
        mLinePaint.setColor(mLineColor);
        mLinePaint.setStyle(Style.STROKE);
        //x轴每次的步进
        int xInterval=(int)mXScaleWidth/5;
        // 如果折线集合不为空
        if (mLines != null && mLines.size() > 0) {
            for(int k=0;k<mLines.size();k++) {
                // 循环绘制所有线条
                for (int i = 1; i < mLines.get(k).size(); i++) {
                    // 上一个点的xy坐标
                    int previousY = (int) (mLines.get(k).get(i - 1) * 1.0 / mYRange * (mAxisHeight - mYBlank));
                    int previousX = (i - 1) * mXScaleWidth + mOffset;
                    // 当前点的xy坐标
                    int thisY = (int) (mLines.get(k).get(i) * 1.0 / mYRange * (mAxisHeight - mYBlank));
                    int thisX = i * xInterval + mOffset;
                    // 保证只绘制坐标轴范围内的部分
                    if (previousX > 0 && previousX < mAxisWidth - mXBlank && thisX > 0 && thisX < mAxisWidth - mXBlank)
                        // 两个坐标连线
                        mCanvas.drawLine(previousX, -previousY, thisX, -thisY, mLinePaint);
                }
            }
        }
        mCanvas.restore();
    }

    /**
     * 设置快速滚动时,末尾的减速
     */
    private void setSpeedCut() {
        if (!mIsTouch && isScroll) {
            // 通过当前速度计算所对应的偏移量
            mOffset = mOffset + mSpeed;
            setOffsetRange();
        }
        // 每次偏移量的计算
        if (mSpeed != 0) {
            time++;
            mSpeed = (int) (xVelocity + time * time * (xVelocity / 1600.0) - (xVelocity / 20.0) * time);
        } else {
            time = 0;
            mSpeed = 0;
        }

    }

    /**
     * 绘制x、y轴
     */
    private void drawXYLine() {
        mAxisWidth = mWidth -  mPaddint;
        mAxisHeight = mHeight - 2 * mPaddint;
        mCoordinatePaint.setStrokeWidth(mLineWidth);
        mCoordinatePaint.setAntiAlias(true);
        // 绘制x轴
        mCoordinatePaint.setColor(mXLineColor);
        mCanvas.drawLine(0, 0, mAxisWidth, 0, mCoordinatePaint);
        // 绘制y轴
        mCoordinatePaint.setColor(mYLineColor);
        mCanvas.drawLine(0, 0, 0, -mAxisHeight, mCoordinatePaint);
    }

    /** 绘制刻度线和箭头 */
    private void drawXYScale() {
        // 画y轴的刻度
        mYScaleNum = mYRange / mYUnit;
        mYScaleWidth = (int) (mAxisHeight * 1.0 / mYScaleNum);
        mCoordinatePaint.setTextSize(30);
        for (int i = 0; i < mYScaleNum; i++) {
            mCanvas.drawLine(0, -mYScaleWidth * (i + 1), mAxisWidth, -mYScaleWidth * (i + 1), mCoordinatePaint);
            String yText = "";
            if (i== 3||i==9||i==15) {
                yText = "0";
                mCanvas.drawText(yText,-30,-mYScaleWidth * (i + 1),mCoordinatePaint);
            } else if (i  == 1||i==7||i==13) {
                yText = "-1";
                mCanvas.drawText(yText,-30,-mYScaleWidth * (i + 1),mCoordinatePaint);
            }
            else if(i  == 5||i==11||i==17){
                yText = "1";
                mCanvas.drawText(yText,-30,-mYScaleWidth * (i + 1),mCoordinatePaint);
            }
        }

        // 画x轴的刻度
        // x轴的分割线数量
        mXScaleNum=mAxisWidth/mYScaleWidth;
        // 每个刻度的宽度
        mXScaleWidth =mYScaleWidth;// (int) ((mAxisWidth - mXBlank) * 1.0 / mXScaleNum);
        for (int i = 0; i < mXScaleNum; i++) {
            mCanvas.drawLine(mXScaleWidth * (i + 1), 0, mXScaleWidth * (i + 1), -mAxisHeight, mCoordinatePaint);
            if(i!=0&&(i+1)%5==0) {
                mCanvas.drawText(String.valueOf((i+1) / 5), mXScaleWidth * (i + 1), 30, mCoordinatePaint);
            }
        }
    }

    /** 对偏移量进行边界值判定 */
    private void setOffsetRange() {
        int offsetMax = -mXScaleWidth * (mLines.size()) + (mXScaleWidth * mXScaleNum + 2);
        int offsetMin = 2 * mXScaleWidth;
        if (mOffset > offsetMin) {
            isScroll = false;
            mOffset = offsetMin;
        } else if (mOffset < offsetMax) {// 如果划出最大值范围
            isScroll = false;
            mOffset = offsetMax;
        } else {
            isScroll = true;
        }
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        int rawX = (int) (event.getRawX());
        // 计算当前速度
        VelocityTracker velocityTracker = VelocityTracker.obtain();
        velocityTracker.addMovement(event);
        // 计算速度的单位时间
        velocityTracker.computeCurrentVelocity(50);
        switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN:
            // 记录触摸点坐标
            lastX = rawX;
            mIsTouch = true;
            break;
        case MotionEvent.ACTION_MOVE:
            // 计算便宜量
            int offsetX = rawX - lastX;
            // 在当前偏移量的基础上增加偏移量
            mOffset = mOffset + offsetX;
            setOffsetRange();
            // 偏移量修改后下次重绘会有变化
            lastX = rawX;
            // 获取X方向上的速度
            xVelocity = velocityTracker.getXVelocity();
            mSpeed = (int) xVelocity;
            break;
        case MotionEvent.ACTION_UP:
            mIsTouch = false;
            break;
        }
        // 计算完成后回收内存
        velocityTracker.clear();
        velocityTracker.recycle();
        return true;
    }

    /**
     * 画布创建时候执行的方法
     * 
     * @param holder
     */
    @Override
    public void surfaceCreated(SurfaceHolder holder) {
        // 开始绘画
        mIsDrawing = true;
        // 启动绘画线程
        new Thread(this).start();

    }

    @Override
    public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {

    }

    /**
     * 画布销毁时候执行的方法
     */
    @Override
    public void surfaceDestroyed(SurfaceHolder holder) {
        // 停止绘画
        mIsDrawing = false;
    }

    /**
     * 外部用来设置折线数据的方法
     */
    public void setSingleLine(List<Integer> lines) {
        if(this.mLines==null)
            mLines=new ArrayList<List<Integer>>() ;
        this.mLines.add(lines);
        invalidate();
    }

    /**
     * 外部用来设置折线数据的方法
     */
    public void setMultiLines(List<List<Integer>> lines) {
        if(this.mLines==null)
            mLines=new ArrayList<List<Integer>>() ;
        this.mLines=lines;
        invalidate();
    }

    /** dp转化为px工具 */
    private int dp2px(Context context, int dp) {
        return (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, dp,
                context.getResources().getDisplayMetrics());
    }   
}

activity中

 public class MainActivity extends Activity {

    private MultiLineChartSurfaceView mllcv;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        List<List<Integer>> mLinesList = getData();        
        mllcv = (MultiLineChartSurfaceView) findViewById(R.id.mllcv);
        mllcv.setMultiLines(mLinesList);
    }

    private List<List<Integer>> getData() {
        List<List<Integer>> mLineList = new ArrayList<List<Integer>>();
        List<Integer> line1 = new ArrayList<Integer>();
        List<Integer> line2 = new ArrayList<Integer>();
        for (int i = 0; i < 600; i++) {
            line1.add(550 + (int) (Math.random() * 200));
            line2.add(550 - (int) (Math.random() * 200));
        }
        mLineList.add(line1);
        mLineList.add(line2);
        return mLineList;
    }
}

Layout

 <?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context="com.bisa.linetest.MainActivity" >
    <com.bisa.linetest.MultiLineChartSurfaceView
        android:id="@+id/mllcv"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />
</RelativeLayout>
  • 写回答

1条回答 默认 最新

  • devmiao 2016-07-17 15:58
    关注
    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论

报告相同问题?

悬赏问题

  • ¥15 时间序列预测算法 预测结果出现负值
  • ¥15 在win系统Django虚拟环境下载mysqlclient报错
  • ¥15 pg数据库导入数据序列重复
  • ¥15 三分类机器学习模型可视化分析
  • ¥15 本地测试网站127.0.0.1 已拒绝连接,如何解决?(标签-ubuntu)
  • ¥50 Qt在release捕获异常并跟踪堆栈(有Demo,跑一下环境再回答)
  • ¥30 python,LLM 文本提炼
  • ¥15 关于将inet引入的相关问题
  • ¥15 关于一个倒计时的操作和显示设计
  • ¥15 提问STK的问题,哪位航天领域的同学会啊