Android底部导航栏的动态替换方案

Android底部导航栏的动态替换方案,供大家参考,具体内容如下

1、通常来说,一般情况下,我们的app的BottomTab会有下面几种实现方式。

1)、自定义view,然后自己写逻辑去实现互斥。

2)、使用RadioGroup+RadioButton去实现底部的Tab。
自由度比极高,如果想实现搞复杂度的话可以重写 RadioButton。

3)、使用google design包里面的 TabLayout去实现。
可上、可下、可以滑动
偷懒的话可以根据已有api来设置一些资源,也可以 setCustomView()

4)、使用google design包里面的BottomNavigationView去实现。

(1)使用menu设置资源
(2)有默认的动画效果

2.本篇介绍的是日常见到的京东,淘宝类似的根据后台下发实现动态替换底部导航资源图片的方法(基于TabLayout实现)
既然提到了动态替换肯定意味着要下载资源,所以先讲一下IntentService

  • IntentService也是一个service,只不过google帮我们在里面封装并维护了一个HandlerThread,里面的操作都是异步的。
  • 当任务执行完后,IntentService 会自动停止,不需要我们去手动结束。
  • 如果启动 IntentService 多次,那么每一个耗时操作会以工作队列的方式在 IntentService 的 onHandleIntent 回调方法中执行,依次去执行,使用串行的方式,执行完自动结束。

onHandlerIntent(Intent intent) 是最重要的一个方法

@Override
 protected void onHandleIntent(Intent intent) {
  if (intent != null) {
   final String action = intent.getAction();
   if (ACTION_FOO.equals(action)) {
    // 在这里面处理耗时任务,当所有的耗时任务都结束以后,IntentService会自动的finish掉,不需要开发者关心。
   }
  }
 }

选择IntentService的原因是因为下面的这几个操作都是耗时操作,所以我们干脆都封装到这service里面,我们只需要在合适的时机去启动这个Service就ok了

  • 需要下载资源压缩包
  • 因为是动态替换,所以必然涉及到预下载,所以数据格式要先定好(下面是数据格式)。
{
  "currentInfo":{//当前样式
   "id":"111",
   "imageZipUrl":你的下载地址,
   "tabNamesList":[
     "首页1","附近1","发现1","我的1"
   ],
   "tabColorNormal":"B0C4DE",
   "tabColorHighlight":"F7B62D",
   "startTime":开始时间,
   "deadLineTime":结束时间
  },
  "nextInfo":{//下一次要展示的样式
   "id":"111",
   "imageZipUrl":你的下载地址,
   "tabNamesList":[
     "首页2","附近2","发现2","我的2"
   ],
   "tabColorNormal":"B0C4DE",
   "tabColorHighlight":"FE6246",
   "startTime":开始时间,
   "deadLineTime":结束时间
  }
 }
  • 需要存放资源压缩包

下载和存放文件的代码(这里使用的是Retrofit进行下载的)

// 下载文件
  Response<ResponseBody> zipFile = ServiceGenerator.createService(HomeService.class)
   .downloadFileRetrofit(getFileDownLoadUrl(homeTabImageInfoBean, type))
   .execute();

   // 得到文件流
   ResponseBody zipBody = zipFile.body();

   LogUtils.d("DownLoad", "下载完成");

   // 创建一个文件
   File zipDirectory = new File(FilePathUtil.getHuaShengHomeTabZipDirectory(getApplicationContext())
     + createZipFileName(homeTabImageInfoBean, type));

   // 如果文件不存在,则创建文件夹
   if (!zipDirectory.exists()) {
    zipDirectory.createNewFile();
   }

   // 保存文件
   FileUtils.writeFile2Disk(zipBody, zipDirectory);
  • 解压资源并删除文件(解压方法由于过长所以写在了文中底部)
// 解压文件 并删除文件
   if (ZipUtils.unzipFile(zipDirectory.getAbsolutePath(),
     CURRENT.equals(type) ? FilePathUtil.getHuaShengHomeTabImgCurrentDirectory(getApplicationContext())
       : FilePathUtil.getHuaShengHomeTabImgNextDirectory(getApplicationContext()))) {

    // 保存文件解压地址
    saveFileDirPath(homeTabImageInfoBean, type,
      CURRENT.equals(type) ? FilePathUtil.getHuaShengHomeTabImgCurrentDirectory(getApplicationContext())
        : FilePathUtil.getHuaShengHomeTabImgNextDirectory(getApplicationContext()));

    LogUtils.d("HomeTabImageDownLoadInt", "解压完成---");

   }

其实最关键的就是如何创建并获取我们的文件资源

重要的就是资源的两种状态切换(选中 or 不选中),通常我们都是使用drawable来写的

<?xml version="1.0" encoding="utf-8"?>
 <selector xmlns:android="http://schemas.android.com/apk/res/android">
  <item android:drawable="@mipmap/home_tab_financing_selected" android:state_selected="true" />
  <item android:drawable="@mipmap/home_tab_financing_normal" />
 </selector>

现在我们要根据下载下来的图片(存放在sdcard中)去动态创建drawable这样我们便能里面系统控件的互斥特性

下面的三个方法代码很重要

// 构建Drawable选择器
 private StateListDrawable createDrawableSelector(Drawable checked, Drawable unchecked) {
  StateListDrawable stateList = new StateListDrawable();
  int state_selected = android.R.attr.state_selected;
  stateList.addState(new int[]{state_selected}, checked);
  stateList.addState(new int[]{-state_selected}, unchecked);
  return stateList;
 }
// 构建颜色选择器
 private ColorStateList createColorSelector(int checkedColor, int uncheckedColor) {

  return new ColorStateList(
    new int[][]{new int[]{android.R.attr.state_selected},
      new int[]{-android.R.attr.state_selected}},
    new int[]{checkedColor, uncheckedColor});
// 将文件转换成Drawable
 // pathName就是图片存放的绝对路径
 private Drawable getDrawableByFile(String pathName) {
  return Drawable.createFromPath(pathName);
 }

最后就是在TabLayout的tab上设置资源

取出TabLayout的所有的Tab,遍历,然后根据特定条件去设置相应的drawable就可以了

最后在本文结尾附上上文的压缩相关工具类

import com.blankj.utilcode.util.CloseUtils;
import com.blankj.utilcode.util.StringUtils;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Enumeration;
import java.util.List;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;
import java.util.zip.ZipOutputStream;

/**
 * <pre>
 *  author: 程龙
 *  time : 2018/12/14
 *  desc : 压缩相关工具类
 * </pre>
 */
public final class ZipUtils {

 private static final int KB = 1024;

 private ZipUtils() {
  throw new UnsupportedOperationException("u can't instantiate me...");
 }

 /**
  * 批量压缩文件
  *
  * @param resFiles 待压缩文件集合
  * @param zipFilePath 压缩文件路径
  * @return {@code true}: 压缩成功<br>{@code false}: 压缩失败
  * @throws IOException IO错误时抛出
  */
 public static boolean zipFiles(Collection<File> resFiles, String zipFilePath)
   throws IOException {
  return zipFiles(resFiles, zipFilePath, null);
 }

 /**
  * 批量压缩文件
  *
  * @param resFiles 待压缩文件集合
  * @param zipFilePath 压缩文件路径
  * @param comment  压缩文件的注释
  * @return {@code true}: 压缩成功<br>{@code false}: 压缩失败
  * @throws IOException IO错误时抛出
  */
 public static boolean zipFiles(Collection<File> resFiles, String zipFilePath, String comment)
   throws IOException {
  return zipFiles(resFiles, FileUtils.getFileByPath(zipFilePath), comment);
 }

 /**
  * 批量压缩文件
  *
  * @param resFiles 待压缩文件集合
  * @param zipFile 压缩文件
  * @return {@code true}: 压缩成功<br>{@code false}: 压缩失败
  * @throws IOException IO错误时抛出
  */
 public static boolean zipFiles(Collection<File> resFiles, File zipFile)
   throws IOException {
  return zipFiles(resFiles, zipFile, null);
 }

 /**
  * 批量压缩文件
  *
  * @param resFiles 待压缩文件集合
  * @param zipFile 压缩文件
  * @param comment 压缩文件的注释
  * @return {@code true}: 压缩成功<br>{@code false}: 压缩失败
  * @throws IOException IO错误时抛出
  */
 public static boolean zipFiles(Collection<File> resFiles, File zipFile, String comment)
   throws IOException {
  if (resFiles == null || zipFile == null) return false;
  ZipOutputStream zos = null;
  try {
   zos = new ZipOutputStream(new FileOutputStream(zipFile));
   for (File resFile : resFiles) {
    if (!zipFile(resFile, "", zos, comment)) return false;
   }
   return true;
  } finally {
   if (zos != null) {
    zos.finish();
    CloseUtils.closeIO(zos);
   }
  }
 }

 /**
  * 压缩文件
  *
  * @param resFilePath 待压缩文件路径
  * @param zipFilePath 压缩文件路径
  * @return {@code true}: 压缩成功<br>{@code false}: 压缩失败
  * @throws IOException IO错误时抛出
  */
 public static boolean zipFile(String resFilePath, String zipFilePath)
   throws IOException {
  return zipFile(resFilePath, zipFilePath, null);
 }

 /**
  * 压缩文件
  *
  * @param resFilePath 待压缩文件路径
  * @param zipFilePath 压缩文件路径
  * @param comment  压缩文件的注释
  * @return {@code true}: 压缩成功<br>{@code false}: 压缩失败
  * @throws IOException IO错误时抛出
  */
 public static boolean zipFile(String resFilePath, String zipFilePath, String comment)
   throws IOException {
  return zipFile(FileUtils.getFileByPath(resFilePath), FileUtils.getFileByPath(zipFilePath), comment);
 }

 /**
  * 压缩文件
  *
  * @param resFile 待压缩文件
  * @param zipFile 压缩文件
  * @return {@code true}: 压缩成功<br>{@code false}: 压缩失败
  * @throws IOException IO错误时抛出
  */
 public static boolean zipFile(File resFile, File zipFile)
   throws IOException {
  return zipFile(resFile, zipFile, null);
 }

 /**
  * 压缩文件
  *
  * @param resFile 待压缩文件
  * @param zipFile 压缩文件
  * @param comment 压缩文件的注释
  * @return {@code true}: 压缩成功<br>{@code false}: 压缩失败
  * @throws IOException IO错误时抛出
  */
 public static boolean zipFile(File resFile, File zipFile, String comment)
   throws IOException {
  if (resFile == null || zipFile == null) return false;
  ZipOutputStream zos = null;
  try {
   zos = new ZipOutputStream(new FileOutputStream(zipFile));
   return zipFile(resFile, "", zos, comment);
  } finally {
   if (zos != null) {
    CloseUtils.closeIO(zos);
   }
  }
 }

 /**
  * 压缩文件
  *
  * @param resFile 待压缩文件
  * @param rootPath 相对于压缩文件的路径
  * @param zos  压缩文件输出流
  * @param comment 压缩文件的注释
  * @return {@code true}: 压缩成功<br>{@code false}: 压缩失败
  * @throws IOException IO错误时抛出
  */
 private static boolean zipFile(File resFile, String rootPath, ZipOutputStream zos, String comment)
   throws IOException {
  rootPath = rootPath + (isSpace(rootPath) ? "" : File.separator) + resFile.getName();
  if (resFile.isDirectory()) {
   File[] fileList = resFile.listFiles();
   // 如果是空文件夹那么创建它,我把'/'换为File.separator测试就不成功,eggPain
   if (fileList == null || fileList.length <= 0) {
    ZipEntry entry = new ZipEntry(rootPath + '/');
    if (!StringUtils.isEmpty(comment)) entry.setComment(comment);
    zos.putNextEntry(entry);
    zos.closeEntry();
   } else {
    for (File file : fileList) {
     // 如果递归返回false则返回false
     if (!zipFile(file, rootPath, zos, comment)) return false;
    }
   }
  } else {
   InputStream is = null;
   try {
    is = new BufferedInputStream(new FileInputStream(resFile));
    ZipEntry entry = new ZipEntry(rootPath);
    if (!StringUtils.isEmpty(comment)) entry.setComment(comment);
    zos.putNextEntry(entry);
    byte buffer[] = new byte[KB];
    int len;
    while ((len = is.read(buffer, 0, KB)) != -1) {
     zos.write(buffer, 0, len);
    }
    zos.closeEntry();
   } finally {
    CloseUtils.closeIO(is);
   }
  }
  return true;
 }

 /**
  * 批量解压文件
  *
  * @param zipFiles 压缩文件集合
  * @param destDirPath 目标目录路径
  * @return {@code true}: 解压成功<br>{@code false}: 解压失败
  * @throws IOException IO错误时抛出
  */
 public static boolean unzipFiles(Collection<File> zipFiles, String destDirPath)
   throws IOException {
  return unzipFiles(zipFiles, FileUtils.getFileByPath(destDirPath));
 }

 /**
  * 批量解压文件
  *
  * @param zipFiles 压缩文件集合
  * @param destDir 目标目录
  * @return {@code true}: 解压成功<br>{@code false}: 解压失败
  * @throws IOException IO错误时抛出
  */
 public static boolean unzipFiles(Collection<File> zipFiles, File destDir)
   throws IOException {
  if (zipFiles == null || destDir == null) return false;
  for (File zipFile : zipFiles) {
   if (!unzipFile(zipFile, destDir)) return false;
  }
  return true;
 }

 /**
  * 解压文件
  *
  * @param zipFilePath 待解压文件路径
  * @param destDirPath 目标目录路径
  * @return {@code true}: 解压成功<br>{@code false}: 解压失败
  * @throws IOException IO错误时抛出
  */
 public static boolean unzipFile(String zipFilePath, String destDirPath) throws IOException {
  // 判断是否存在这个路径,没有的话就创建这个路径
  File tempDir = new File(destDirPath);
  if (!tempDir.exists()) {
   tempDir.mkdirs();
  }
  return unzipFile(FileUtils.getFileByPath(zipFilePath), FileUtils.getFileByPath(destDirPath));
 }

 /**
  * 解压文件
  *
  * @param zipFile 待解压文件
  * @param destDir 目标目录
  * @return {@code true}: 解压成功<br>{@code false}: 解压失败
  * @throws IOException IO错误时抛出
  */
 public static boolean unzipFile(File zipFile, File destDir)
   throws IOException {
  return unzipFileByKeyword(zipFile, destDir, null) != null;
 }

 /**
  * 解压带有关键字的文件
  *
  * @param zipFilePath 待解压文件路径
  * @param destDirPath 目标目录路径
  * @param keyword  关键字
  * @return 返回带有关键字的文件链表
  * @throws IOException IO错误时抛出
  */
 public static List<File> unzipFileByKeyword(String zipFilePath, String destDirPath, String keyword)
   throws IOException {
  return unzipFileByKeyword(FileUtils.getFileByPath(zipFilePath),
    FileUtils.getFileByPath(destDirPath), keyword);
 }

 /**
  * 解压带有关键字的文件
  *
  * @param zipFile 待解压文件
  * @param destDir 目标目录
  * @param keyword 关键字
  * @return 返回带有关键字的文件链表
  * @throws IOException IO错误时抛出
  */
 public static List<File> unzipFileByKeyword(File zipFile, File destDir, String keyword)
   throws IOException {
  if (zipFile == null || destDir == null) return null;
  List<File> files = new ArrayList<>();
  ZipFile zf = new ZipFile(zipFile);
  Enumeration<?> entries = zf.entries();
  while (entries.hasMoreElements()) {
   ZipEntry entry = ((ZipEntry) entries.nextElement());
   String entryName = entry.getName();
   if (StringUtils.isEmpty(keyword) || FileUtils.getFileName(entryName).toLowerCase().contains(keyword.toLowerCase())) {
    String filePath = destDir + File.separator + entryName;
    File file = new File(filePath);
    files.add(file);
    if (entry.isDirectory()) {
     if (!FileUtils.createOrExistsDir(file)) return null;
    } else {
     if (!FileUtils.createOrExistsFile(file)) return null;
     InputStream in = null;
     OutputStream out = null;
     try {
      in = new BufferedInputStream(zf.getInputStream(entry));
      out = new BufferedOutputStream(new FileOutputStream(file));
      byte buffer[] = new byte[KB];
      int len;
      while ((len = in.read(buffer)) != -1) {
       out.write(buffer, 0, len);
      }
     } finally {
      CloseUtils.closeIO(in, out);
     }
    }
   }
  }
  return files;
 }

 /**
  * 获取压缩文件中的文件路径链表
  *
  * @param zipFilePath 压缩文件路径
  * @return 压缩文件中的文件路径链表
  * @throws IOException IO错误时抛出
  */
 public static List<String> getFilesPath(String zipFilePath)
   throws IOException {
  return getFilesPath(FileUtils.getFileByPath(zipFilePath));
 }

 /**
  * 获取压缩文件中的文件路径链表
  *
  * @param zipFile 压缩文件
  * @return 压缩文件中的文件路径链表
  * @throws IOException IO错误时抛出
  */
 public static List<String> getFilesPath(File zipFile)
   throws IOException {
  if (zipFile == null) return null;
  List<String> paths = new ArrayList<>();
  Enumeration<?> entries = getEntries(zipFile);
  while (entries.hasMoreElements()) {
   paths.add(((ZipEntry) entries.nextElement()).getName());
  }
  return paths;
 }

 /**
  * 获取压缩文件中的注释链表
  *
  * @param zipFilePath 压缩文件路径
  * @return 压缩文件中的注释链表
  * @throws IOException IO错误时抛出
  */
 public static List<String> getComments(String zipFilePath)
   throws IOException {
  return getComments(FileUtils.getFileByPath(zipFilePath));
 }

 /**
  * 获取压缩文件中的注释链表
  *
  * @param zipFile 压缩文件
  * @return 压缩文件中的注释链表
  * @throws IOException IO错误时抛出
  */
 public static List<String> getComments(File zipFile)
   throws IOException {
  if (zipFile == null) return null;
  List<String> comments = new ArrayList<>();
  Enumeration<?> entries = getEntries(zipFile);
  while (entries.hasMoreElements()) {
   ZipEntry entry = ((ZipEntry) entries.nextElement());
   comments.add(entry.getComment());
  }
  return comments;
 }

 /**
  * 获取压缩文件中的文件对象
  *
  * @param zipFilePath 压缩文件路径
  * @return 压缩文件中的文件对象
  * @throws IOException IO错误时抛出
  */
 public static Enumeration<?> getEntries(String zipFilePath)
   throws IOException {
  return getEntries(FileUtils.getFileByPath(zipFilePath));
 }

 /**
  * 获取压缩文件中的文件对象
  *
  * @param zipFile 压缩文件
  * @return 压缩文件中的文件对象
  * @throws IOException IO错误时抛出
  */
 public static Enumeration<?> getEntries(File zipFile)
   throws IOException {
  if (zipFile == null) return null;
  return new ZipFile(zipFile).entries();
 }

 private static boolean isSpace(String s) {
  if (s == null) return true;
  for (int i = 0, len = s.length(); i < len; ++i) {
   if (!Character.isWhitespace(s.charAt(i))) {
    return false;
   }
  }
  return true;
 }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android自定义动态壁纸开发详解

    看到有些手机酷炫的动态壁纸,有没有好奇过他们是如何实现的,其实我们自己也可以实现. 一.动态壁纸原理 如果你了解使用过SurfaceView的话,那么开发一款动态壁纸对你来说其实非常简单. 动态壁纸的本质其实就是一个服务在维护一个动态壁纸引擎Engine,所以我们看到的动态效果其实是通过这个引擎画出来的.而维护这个引擎的服务,就是WallpaperService.本篇文章并不讨论内部实现原理,只是让大家知道如何去实现动态壁纸,所以就不详细说了. 二.实现动态壁纸 大体上可分为三个步骤: 创建自定

  • Android 6.0动态权限及跳转GPS设置界面的方法

    1.动态权限申请 模糊的位置信息android.permission.ACCESS_COARSE_LOCATION权限为例 在AndroidManifest文件中加入权限 <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/> 然后java代码中动态申请 //动态申请权限的测试方法 public void test() { // 要申请的权限 数组 可以同时申请多个权限 Stri

  • Android自定义动态壁纸开发(时钟)

    看到有些手机酷炫的动态壁纸,有没有好奇过他们是如何实现的,其实我们自己也可以实现. 先看效果 上图是动态壁纸钟的一个时钟. 我们先来看看 Livewallpaper(即动态墙纸)的实现,Android的动态墙纸并不是GIF图片,而是一个标准的Android应用程序,也就是APK.既然是应用程序,当然意味着天生具有GIF图片不具备的功能--能与用户发生交互,而且动态的背景变化绝不仅仅局限于GIF图片那般只能是固定的几张图片的循环播放.但是我们在这里没有加入与用户交互的动作,只是加入一个时钟(当然时

  • Android如何动态调整应用字体大小详解

    前言 为什么要动态设置字体大小?由于项目面对的是中老年客户项目中自带的字体无法满足客户需求. Android应用字体大小默认随系统设置的字体大小而变化,但您可能不希望您的应用字体大小随系统设置变化,想要自己控制,例如微信.本文简单介绍一下如何实现应用字体大小动态调整而不是依赖系统设置 字体大小变化是由android.content.res.Configuration.class类中的fontScale控制的,因此,若想我们的应用字体大小变化不随系统变化而是由我们自主控制,就需要我们修改fontS

  • Android实现动态添加标签及其点击事件

    在做Android开发的时候,会遇到动态添加标签让用户选择的功能,所以自己写了个例子,运行效果图如下. 标签可以左右滑动进行选择,点击的时候,会弹出toast提示选择或者取消选择了哪个标签.通过动态添加TextView作为标签,并给TextView设置背景,通过selector选择器改变其背景颜色,来确定是否处于选中状态. 代码如下所示: 1.标签的布局文件,我在标签里只设置了一个TextView <?xml version="1.0" encoding="utf-8&

  • android动态壁纸调用的简单实例

    调用后动态壁纸其实是显示在Activity的后面,而Activity则是透明显示,这样就可以看到下面的动态壁纸,如果Activity不是透明的则什么也看不到. 代码中有用到两个接口 IWallpaperService mService; IWallpaperEngine mEngine; 我们可以看到该目录下面有三个aidl接口,分别是 复制代码 代码如下: interface IWallpaperConnection { void attachEngine(IWallpaperEngine e

  • Android动态修改应用图标与名称的方法实例

    遇到的坑 这里我把做这个功能中遇到的一些问题写在前面,是为了大家能先了解有什么问题存在,遇到这些问题的时候就不慌了,这里我把应用图标和名称先统一使用icon代替进行说明. 1.动态替换icon,只能替换内置的icon,无法从服务器端获取来更新icon: 2.动态替换icon以后,应用内更新的时候必须要切换到原始icon),否则可能导致更新安装失败(AS上表现为adb运行会失败),或者升级后应用图标出现多个甚至应用图标都不显示的情况(这些问题都可以通过下面我推荐的开发规则解决掉,所以这是一个坑点,

  • Android编程之动态壁纸实例分析

    本文实例讲述了Android编程之动态壁纸.分享给大家供大家参考,具体如下: 从android 2.1版本起引入了动态壁纸的概念,熟悉android的人一定不会陌生.这里解释一个动态壁纸是怎么形成又是怎么工作的. 首先动态桌面的动态体现出这个组件是实时变化的,也就是说有一个后台在不停的刷新这个组件.联想到后台组件首先想到的就是service,从代码角度看,果然如此.每一个动态桌面都继承自WallpaperService,其中必须实现的抽象方法onCreateEngine,返回一个Engine对象

  • Android 中使用 dlib+opencv 实现动态人脸检测功能

    1 概述 完成 Android 相机预览功能以后,在此基础上我使用 dlib 与 opencv 库做了一个关于人脸检测的 demo.该 demo 在相机预览过程中对人脸进行实时检测,并将检测到的人脸用矩形框描绘出来.具体实现原理如下: 采用双层 View,底层的 TextureView 用于预览,程序从 TextureView 中获取预览帧数据,然后调用 dlib 库对帧数据进行处理,最后将检测结果绘制在顶层的 SurfaceView 中. 2 项目配置 由于项目中用到了 dlib 与 open

  • Android绘制动态折线图

    所谓动态折线图,就是折线图能随着手指的滑动进行动态绘制,这里很定会产生动画效果.基于这个效果,这里使用SurfaceView进行制图. 实现步奏如下: (1): 这里新建一个绘图ChartView,继承SurfaceView并实现SurfaceHolder.Callback , Runnable接口,主要绘图工作在子线程中完成. (2):现实 SurfaceHolder.Callback接口的三个方法,并在 surfaceCreated中开启子线程进行绘图. (3):重写onTouchEvent

随机推荐