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; } }
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。
相关文章
Android Studio error: Unable to start the daemon process的解决方
这篇文章主要介绍了在 Android Studio 上新建项目,出现 Unable to start the daemon process问题的几种的解决方法,需要的朋友可以参考下2020-10-10Android使用Realm数据库实现App中的收藏功能(代码详解)
这篇文章主要介绍了Android使用Realm数据库实现App中的收藏功能,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下2020-03-03
最新评论