先上效果图

移除频道的动画效果模仿得不是很像,如果有更好的实现方法可以在下面留言告诉我。

页面效果拆解

部分已选频道位置固定,不能拖动
已选频道可以长按拖拽改变顺序,长按后背景色改变,删除按钮隐藏,原本的位置会出现一个虚线的矩形
点击已选频道或删除按钮会移除该频道,这里会分两种情况,若频道属于下方显示的频道,则移动到下方第一个,隐藏删除按钮;不属于则出现一个动画进行移除
点击推荐频道或地方新闻下方内容会进行切换
点击推荐频道或地方新闻下的频道,频道会移动到已选频道的最后一个,同时出现删除按钮
既然它是一个列表,可以拖动,又需要动画,那这里就使用recyclerView+ItemTouchHelper来实现整个页面的效果

拖拽实现

  • 这里需要新建一个类继承ItemTouchHelper.Callback,并重写其中的几个方法
public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder)

这个方法返回的是拖拽和滑动的方向。一般使用makeMovementFlags(int,int)或者makeFlag(int,int)构造返回值。我们需要的所有方向的拖拽,不需要滑动,所以可以这么写:

public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN | ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT;int swipeFlags = 0;return makeMovementFlags(dragFlags, swipeFlags);}

但已选频道这里是有一个或多个固定频道的,而且我们是一个recyclerView去实现整个页面,tab及tab下面的item都是不能拖动的,所以还需要处理一下:

public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {//固定位置及tab下面的channel不能拖动if (viewHolder.getLayoutPosition() < mAdapter.getFixSize() + 1 || viewHolder.getLayoutPosition() > mAdapter.getSelectedSize()) {return makeMovementFlags(0, 0);}int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN | ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT;int swipeFlags = 0;return makeMovementFlags(dragFlags, swipeFlags);}

public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target)
当用户拖动item进行移动时会回调这个方法,如果item移动到新的位置,返回true;否则返回false。我们可以在这里实现item的位置交换。要注意的是,某些item是不能改变位置的,所以也要进行处理:

public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) {int fromPosition = viewHolder.getAdapterPosition();   //拖动的positionint toPosition = target.getAdapterPosition();     //释放的position//固定位置及tab下面的channel不能拖动if (toPosition < mAdapter.getFixSize() + 1 || toPosition > mAdapter.getSelectedSize())return false;mAdapter.itemMove(fromPosition, toPosition);return true;}
  • public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction)
    当用户滑动item的时候会回调此方法,这里我们不需要滑动,不需要重写这个方法
    重写完以上两个方法就可以实现拖拽滑动了,但距离我们想要的效果还是有点差距的。

长按拖拽的时候item的颜色会改变,出现阴影,删除按钮会隐藏,原本的位置还会出现一个虚线的方框。

吐槽完了还得继续做啊……
emmmmm…看一下ItemTouchHelper.Callback还有哪些方法能用得上吧

public void onSelectedChanged(ViewHolder viewHolder, int actionState)
当viewHolder被拖拽或滑动时回调(感觉这么翻译有点怪…)。这里有个actionState参数,它的值共有3个:ACTION_STATE_IDLE、ACTION_STATE_SWIPE、ACTION_STATE_DRAG。这里可以根据actionState改变item的样式。

public void onSelectedChanged(RecyclerView.ViewHolder viewHolder, int actionState) {super.onSelectedChanged(viewHolder, actionState);if(actionState==ACTION_STATE_DRAG){//长按时调用ChannelAdapter.ChannelHolder holder= (ChannelAdapter.ChannelHolder) viewHolder;holder.name.setBackgroundColor(Color.parseColor("#FDFDFE"));holder.delete.setVisibility(View.GONE);holder.name.setElevation(5f);}}

这里需要注意一下,不能当actionState为ACTION_STATE_IDLE时重置item的状态,viewHolder有可能为空指针

public void clearView(RecyclerView recyclerView, ViewHolder viewHolder)
当交互完成后会回调此方法。重写这个方法重置item的样式。

public void clearView(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {super.clearView(recyclerView, viewHolder);ChannelAdapter.ChannelHolder holder= (ChannelAdapter.ChannelHolder) viewHolder;holder.name.setBackgroundColor(Color.parseColor("#f0f0f0"));holder.name.setElevation(0f);holder.delete.setVisibility(View.VISIBLE);}

虚线方框需要绘制,只能找跟draw相关的方法罗

  • onChildDraw
  • onChildDrawOver
    区别想必大家都懂的,这里不多说,重写一下onChildDrawOver方法绘制虚线方框。
public void onChildDrawOver(Canvas c, RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, float dX, float dY, int actionState, boolean isCurrentlyActive) {super.onChildDrawOver(c, recyclerView, viewHolder, dX, dY, actionState, isCurrentlyActive);if (dX != 0 && dY != 0 || isCurrentlyActive) {//长按拖拽时底部绘制一个虚线矩形c.drawRect(viewHolder.itemView.getLeft(),viewHolder.itemView.getTop()-mPadding,viewHolder.itemView.getRight(),viewHolder.itemView.getBottom(),mPaint);}}

适配器

多布局什么的我想大家都懂的,就不细说了。tab这里为了方便直接使用了两个textView实现。
着重说一下要注意的几个地方吧:

  • 频道的长按事件需要返回true,否则长按频道触发拖动但却不移动的话会触发后面的点击事件
  • 如果使用的ItemDecoration,频道添加和删除后需要调用recyclerView.invalidateItemDecorations()刷新ItemDecoration
  • 频道的移除分两种:移除的频道属于当前下方显示的频道,直接移动item,利用系统动画完成;不属于则移动到tab的位置并逐渐消失。
  • 这里我的做法是用ObjectAnimator同时实现平移和透明度动画。但是会出现下面的问题…

这里(应该是吧,其实我也不太清楚啊…)recyclerView的复用引起的问题,所以在动画结束后需要重置view的属性
emmmm…… 贴一下部分代码吧

private void setChannel(final ChannelHolder holder, ChannelBean bean) {final int position = holder.getLayoutPosition();holder.name.setText(bean.getName());holder.name.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {if (holder.getLayoutPosition() < selectedSize + 1) {//tab上面的 点击移除removeFromSelected(holder);} else {//tab下面的 点击添加到已选频道selectedSize++;itemMove(holder.getLayoutPosition(), selectedSize);notifyItemChanged(selectedSize);if (onItemRangeChangeListener != null) {onItemRangeChangeListener.refreshItemDecoration();}}}});holder.name.setOnLongClickListener(new View.OnLongClickListener() {@Overridepublic boolean onLongClick(View v) {//返回true 防止长按拖拽事件跟点击事件冲突return true;}});holder.delete.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {removeFromSelected(holder);}});}private void removeFromSelected(ChannelHolder holder) {int position = holder.getLayoutPosition();holder.delete.setVisibility(View.GONE);ChannelBean bean = mList.get(position);if ((isRecommend && bean.isRecommend()) || (!isRecommend && !bean.isRecommend())) {//移除的频道属于当前tab显示的频道,直接调用系统的移除动画itemMove(position, selectedSize + 1);notifyItemRangeChanged(selectedSize + 1, 1);if (onItemRangeChangeListener != null) {//如果设置了itemDecoration,必须调用recyclerView.invalidateItemDecorations(),否则间距会不对onItemRangeChangeListener.refreshItemDecoration();}} else {//不属于当前tab显示的频道removeAnimation(holder.itemView, isRecommend ? mRight : mLeft, mTabY, position);}selectedSize--;}void itemMove(int fromPosition, int toPosition) {if (fromPosition < toPosition) {for (int i = fromPosition; i < toPosition; i++) {Collections.swap(mList, i, i + 1);}} else {for (int i = fromPosition; i > toPosition; i--) {Collections.swap(mList, i, i - 1);}}notifyItemMoved(fromPosition, toPosition);}}

由于使用了textView代替tab,所以会有一些计算用于蓝色线条的位置改变。

源码地址

Android 高仿腾讯新闻频道定制页面相关推荐

  1. 仿腾讯新闻频道定制界面效果

    转载请注明出处 http://blog.csdn.net/oddshou/article/details/73467589 更新: 仿腾讯新闻频道定制界面效果2 这个效果一直想做,最近项目无事决定着手 ...

  2. 仿腾讯新闻频道定制界面效果2

    转载请注明出处 http://blog.csdn.net/oddshou/article/details/73558553 上一篇文章仿腾讯新闻频道定制界面效果1中在实现过程中有很多不如意的地方,可以 ...

  3. android高仿今日头条 --新闻阅读器

    摘要: 开发流程 第一篇:(android高仿系列)今日头条 --新闻阅读器 (一) 涉及到的知识点有 1.slidingmenu.lib  (侧拉菜单包)   使用方法配置以及下载:点击这里   实 ...

  4. 高仿腾讯新闻科技频道页面

    HTML代码: <!DOCTYPE html> <html lang="zh-CN"><head><meta charset=" ...

  5. (android高仿系列)今日头条 --新闻阅读器 (三) 完结 、总结 篇

    从写第一篇今日头条高仿系列开始,到现在已经过去了1个多月了,其实大体都做好了,就是迟迟没有放出来,因为我觉得,做这个东西也是有个过程的,我想把这个模仿中一步一步学习的过程,按照自己的思路写下来,在根据 ...

  6. 【Android UI设计与开发】7.底部菜单栏(四)PopupWindow 实现显示仿腾讯新闻底部弹出菜单...

    前一篇文章中有用到 PopupWindow 来实现弹窗的功能.简单介绍以下吧. 官方文档是这样解释的:这就是一个弹出窗口,可以用来显示一个任意视图.出现的弹出窗口是一个浮动容器的当前活动. 1.首先来 ...

  7. android高仿京东快报(垂直循环滚动新闻栏)

    的android高仿京东快报(垂直循环滚动新闻栏) 标签: 机器人 2016年3月20日03:08 2676阅读人 评论(15)收藏举报    分类: 机器人(71)  版权声明:本文为博主原创文章, ...

  8. Android高仿网易新闻客户端之动态添加标签

    承接上一篇文章:Android高仿网易新闻客户端之首页,今天来实现动态添加标签效果. 动态标签页是一个流式布局,实现了宽度自动换行高度自动分配的功能,代码如下: FlowLayout.java pac ...

  9. (android高仿系列)今日头条 --新闻阅读器 (一)

    在模仿中循序渐进,以程序员角度去看待每一个APP是如何实现的,它有什么优缺点,并从中提升自己. 之前发现很多人在群里面.论坛上求网易新闻客户端的源码,之后我就去下了个网易新闻客户端和今日头条新闻客户端 ...

最新文章

  1. 使用Python中的Turtle库绘制简单的图形
  2. hibernate中持久化对象的生命周期(三态:自由态,持久态,游离态 之间的转换)
  3. c/c++与java------之JNI学习(一)
  4. linux降低eigen3版本_玩转 Windows 自带的 Linux 子系统 (图文指南)
  5. 你可能不需要 jQuery!使用原生 JavaScript 进行开发
  6. 从芯开始,解读华为云的全栈全场景与全球化布局
  7. wgan 不理解 损失函数_GAN:「太难的部分我就不生成了,在下告退」
  8. ubuntu gedit 工具菜单下没有 Manage external tools
  9. python实现arxiv论文数据解析处理
  10. SQL 比较时间大小
  11. 文本特征提取专题_以python为工具【Python机器学习系列(十二)】
  12. qq 登陆界面怎么测试
  13. 2020年西北工业大学 J- 不讲武德
  14. 地球的3D模型制作教程【3DsMax】
  15. 查询给定区域内曲面平均高程
  16. HAL库配置STM32F1系列定时器驱动步进电机(三)
  17. 我在 GitHub 上找到了这些游戏项目,刺激!
  18. 靠谱的IT人力外包企业有哪些?
  19. 在Chrome谷歌浏览器安装插件!
  20. MATLAB randint替换为randi函数的操作

热门文章

  1. 【Drools规则引擎】基础入门案例五(Kie-Server+WorkBench)
  2. HDU 1848 SG函数
  3. qq邮箱 SMTP connect() failed
  4. 2017最佳泛娱乐企业排行榜 最佳泛娱乐短视频平台排行榜
  5. 【LeetCode-面试算法经典-Java实现】【198-House Robber(抢劫犯)】
  6. mysql 反弹shell_反弹shell升级成交互式shell
  7. 远程网络教学系统用例图练习
  8. python中dtype的用法
  9. eclipse oxgen的tomcat三只小猫安装
  10. 1823196912