IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 移动开发 -> Android解决滑动冲突 -> 正文阅读

[移动开发]Android解决滑动冲突

Android解决滑动冲突

滑动冲突的产生

在讨论解决滑动冲突的方法之前,我们需要知道在哪些场景之下会产生滑动冲突。滑动冲突的产生是两个组件之间(或者更多),且这两个组件是包含关系(父控件和子控件),当父控件和子控件都可以对滑动事件进行拦截处理,而我们并没有对他们的滑动冲突进行处理,那么此时只有一个控件可以对滑动事件来进行响应,我们平常称这种情况为父控件和子控件产生了滑动冲突。主要有以下三个场景:

滑动冲突场景一

父控件的滑动方向与子控件的滑动方向不同,这种场景的解决思路比较简单,例如我们平常使用的ViewPager+RecyclerView(假设viewPager是左右滑动,而RecyclerView是上下滑动),这种就是滑动我们滑动冲突的场景一,但是我们使用这种架构没有发生滑动冲突呢,原因是由于在viewPager内部帮助我们实现了解决滑动冲突的方法。而如果我们不使用ViewPager,来使用其他父控件可以滑动并且内部为RecyclerView或者ListView列表,此时就会发生滑动冲突。

对于滑动冲突场景一的具体解决方法为外部拦截法,下面进行具体介绍:

外部拦截法

所谓的外部拦截法指的是对于整个事件序列都要先经过父控件的拦截处理,当我们需要此事件则进行拦截,如果不需要则交由子控件进行处理。(如果不理解事件分发机制的,请学习一下Android的事件分发机制)。我们只需要重写我们父控件的onInterceptTouchEvent()方法来决定是否拦截事件,假设外部为ViewPager(左右滑动,并在此处进行假设ViewPager没有处理滑动冲突,实际ViewPager源码上已解决滑动冲突问题),内部为RecyclerView(上下滑动),则我们重写ViewPager的onInterceptTouchEvent()方法,具体代码逻辑如下所示:

package com.it.test;

import android.content.Context;
import android.view.MotionEvent;

import androidx.annotation.NonNull;
import androidx.viewpager.widget.ViewPager;

public class MyViewPager extends ViewPager {

    private float startX;
    private float startY;
    //使用directionSign来作为拦截的标志,0代表没有初始化,1代表父控件拦截,其他情况代表子控件拦截
    private int directionSign = 0; //只有在滑动开始时需要进行判断由谁进行拦截,之后的这一次事件中无需判断

    public MyViewPager(@NonNull Context context) {
        super(context);
    }

    //onInterceptTouchEvent返回true表示该viewPager拦截事件(不再向子控件中进行分发),返回false代表不拦截,进行分发事件(交由子控件RecyclerView处理)
    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        boolean sign = false; //该标志标示是否拦截事件
        switch (ev.getAction()){
            case MotionEvent.ACTION_DOWN:
                //按下屏幕事件
                sign = false; //此处必须为false,如果为true,则后期滑动的事件将默认会被viewPager拦截,且不在调用onInterceptTouchEvent()方法
                //进行记录当前的坐标
                startX = ev.getX();
                startY = ev.getY();
                break;
            case MotionEvent.ACTION_MOVE:
                //滑动屏幕事件,具体解决滑动冲突的思路在此
                if(checkFatherNeed(ev)){ //检查父控件ViewPager是否需要拦截,如果需要则进行修改sign = true
                    sign = true;
                }else{
                    sign = false;
                }
                break;
            case MotionEvent.ACTION_UP:
                //松开屏幕事件
                sign = false; //此处为true或false意义不大,一般置为false
                break;
        }
        return sign;
    }

    private boolean checkFatherNeed(MotionEvent event) {
        if(directionSign == 0){
            //没有初始化,对directionSign进行初始化
            float currentX = event.getX();
            float currentY = event.getY();
            if(Math.abs(currentX - startX) > Math.abs(currentY-startY)){
                //代表水平滑动内容多,进行拦截事件
                directionSign = 1;
                return true;
            }else{
                //代表水平滑动内容多,交给子控件拦截事件
                directionSign = 2;
                return false;
            }
        }else if(directionSign == 1){
            //父控件拦截
            return true;
        }else{
            //子控件拦截
            return false;
        }
    }
}

一般的解决思路就是判断手指滑动的方向(可以根据水平滑动的距离和竖直滑动距离来进行比较)来判断交给谁来拦截并且处理事件。(左右滑动交给ViewPager,上下滑动交给RecyclerView)

滑动冲突场景二

滑动冲突的场景二是外部滑动方向和内部滑动滑动方向一致,例如上下滑动的ScrollView里面嵌套上下滑动的RecyclerView。该场景的具体解决办法是内部拦截法。

内部拦截法

内部拦截法是指我们需要通过重写子控件的dispatcherTouchEvent分发方法,可以对事件的状态判断来进行分发事件的处理权交给父控件还是子控件。(比如说当处于某一状态时,我们要让子控件进行处理事件,其他状体交给父控件)。

其具体实现思路如下:

我们需要重写子控件的dispatcherTouchEvent方法,重写父控件的onInterceptTouchEvent方法,需要保证当手指按下的一瞬间,即产生ACTION_DOWN事件时,需要保证父控件不对其进行拦截,然后子控件的分发时需要进行parent.requestDisallowInterceptTouchEvent(true)就可以不让父控件来拦截事件。当我们调用了parent.requestDisallowInterceptTouchEvent(false)就可以让父控件进行拦截事件。

具体代码如下:

子控件

package com.it.test;

import android.content.Context;
import android.view.MotionEvent;

import androidx.annotation.NonNull;
import androidx.recyclerview.widget.RecyclerView;

public class MyRecyclerview extends RecyclerView {

    float x;
    float y;
    
    public MyRecyclerview(@NonNull Context context) {
        super(context);
    }

    @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
        switch (ev.getAction()){
            case MotionEvent.ACTION_DOWN:
                float x = ev.getX();
                float y = ev.getY();
                getParent().requestDisallowInterceptTouchEvent(true); //父控件之后不再拦截事件
                break;
            case MotionEvent.ACTION_MOVE:
                if(父控件需要此类事件){
                    getParent().requestDisallowInterceptTouchEvent(false); //父控件可以进行拦截事件  
                } 
                break;
        }
        return super.dispatchTouchEvent(ev);
    }
}


父控件

package com.it.test;

import android.content.Context;
import android.view.MotionEvent;

import androidx.annotation.NonNull;
import androidx.viewpager.widget.ViewPager;

public class MyViewPager extends ViewPager {

    public MyViewPager(@NonNull Context context) {
        super(context);
    }


    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        if(ev.getAction() == MotionEvent.ACTION_DOWN){
            return false; //不拦截
        }else{
            return true; //除了按下事件其他进行全部拦截(前提是子控件的parent.requestDisallowInterceptTouchEvent(false))
        }
    }

}

滑动冲突场景三

滑动冲突的场景三就是以上两种情况的结合或者同时出现,解决办法就是利用内部拦截和外部拦截的联合使用。

你好! 这是你第一次使用 Markdown编辑器 所展示的欢迎页。如果你想学习如何使用Markdown编辑器, 可以仔细阅读这篇文章,了解一下Markdown的基本语法知识。

新的改变

我们对Markdown编辑器进行了一些功能拓展与语法支持,除了标准的Markdown编辑器功能,我们增加了如下几点新功能,帮助你用它写博客:

  1. 全新的界面设计 ,将会带来全新的写作体验;
  2. 在创作中心设置你喜爱的代码高亮样式,Markdown 将代码片显示选择的高亮样式 进行展示;
  3. 增加了 图片拖拽 功能,你可以将本地的图片直接拖拽到编辑区域直接展示;
  4. 全新的 KaTeX数学公式 语法;
  5. 增加了支持甘特图的mermaid语法1 功能;
  6. 增加了 多屏幕编辑 Markdown文章功能;
  7. 增加了 焦点写作模式、预览模式、简洁写作模式、左右区域同步滚轮设置 等功能,功能按钮位于编辑区域与预览区域中间;
  8. 增加了 检查列表 功能。

功能快捷键

撤销:Ctrl/Command + Z
重做:Ctrl/Command + Y
加粗:Ctrl/Command + B
斜体:Ctrl/Command + I
标题:Ctrl/Command + Shift + H
无序列表:Ctrl/Command + Shift + U
有序列表:Ctrl/Command + Shift + O
检查列表:Ctrl/Command + Shift + C
插入代码:Ctrl/Command + Shift + K
插入链接:Ctrl/Command + Shift + L
插入图片:Ctrl/Command + Shift + G
查找:Ctrl/Command + F
替换:Ctrl/Command + G

合理的创建标题,有助于目录的生成

直接输入1次#,并按下space后,将生成1级标题。
输入2次#,并按下space后,将生成2级标题。
以此类推,我们支持6级标题。有助于使用TOC语法后生成一个完美的目录。

如何改变文本的样式

强调文本 强调文本

加粗文本 加粗文本

标记文本

删除文本

引用文本

H2O is是液体。

210 运算结果是 1024.

插入链接与图片

链接: link.

图片: Alt

带尺寸的图片: Alt

居中的图片: Alt

居中并且带尺寸的图片: Alt

当然,我们为了让用户更加便捷,我们增加了图片拖拽功能。

如何插入一段漂亮的代码片

博客设置页面,选择一款你喜欢的代码片高亮样式,下面展示同样高亮的 代码片.

// An highlighted block
var foo = 'bar';

生成一个适合你的列表

  • 项目
    • 项目
      • 项目
  1. 项目1
  2. 项目2
  3. 项目3
  • 计划任务
  • 完成任务

创建一个表格

一个简单的表格是这么创建的:

项目Value
电脑$1600
手机$12
导管$1

设定内容居中、居左、居右

使用:---------:居中
使用:----------居左
使用----------:居右

第一列第二列第三列
第一列文本居中第二列文本居右第三列文本居左

SmartyPants

SmartyPants将ASCII标点字符转换为“智能”印刷标点HTML实体。例如:

TYPEASCIIHTML
Single backticks'Isn't this fun?'‘Isn’t this fun?’
Quotes"Isn't this fun?"“Isn’t this fun?”
Dashes-- is en-dash, --- is em-dash– is en-dash, — is em-dash

创建一个自定义列表

Markdown
Text-to- HTML conversion tool
Authors
John
Luke

如何创建一个注脚

一个具有注脚的文本。2

注释也是必不可少的

Markdown将文本转换为 HTML

KaTeX数学公式

您可以使用渲染LaTeX数学表达式 KaTeX:

Gamma公式展示 Γ ( n ) = ( n ? 1 ) ! ? n ∈ N \Gamma(n) = (n-1)!\quad\forall n\in\mathbb N Γ(n)=(n?1)!?nN 是通过欧拉积分

Γ ( z ) = ∫ 0 ∞ t z ? 1 e ? t d t ? . \Gamma(z) = \int_0^\infty t^{z-1}e^{-t}dt\,. Γ(z)=0?tz?1e?tdt.

你可以找到更多关于的信息 LaTeX 数学表达式here.

新的甘特图功能,丰富你的文章

Mon 06 Mon 13 Mon 20 已完成 进行中 计划一 计划二 现有任务 Adding GANTT diagram functionality to mermaid
  • 关于 甘特图 语法,参考 这儿,

UML 图表

可以使用UML图表进行渲染。 Mermaid. 例如下面产生的一个序列图:

张三 李四 王五 你好!李四, 最近怎么样? 你最近怎么样,王五? 我很好,谢谢! 我很好,谢谢! 李四想了很长时间, 文字太长了 不适合放在一行. 打量着王五... 很好... 王五, 你怎么样? 张三 李四 王五

这将产生一个流程图。:

链接
长方形
圆角长方形
菱形
  • 关于 Mermaid 语法,参考 这儿,

FLowchart流程图

我们依旧会支持flowchart的流程图:

Created with Rapha?l 2.3.0 开始 我的操作 确认? 结束 yes no
  • 关于 Flowchart流程图 语法,参考 这儿.

导出与导入

导出

如果你想尝试使用此编辑器, 你可以在此篇文章任意编辑。当你完成了一篇文章的写作, 在上方工具栏找到 文章导出 ,生成一个.md文件或者.html文件进行本地保存。

导入

如果你想加载一篇你写过的.md文件,在上方工具栏可以选择导入功能进行对应扩展名的文件导入,
继续你的创作。


  1. mermaid语法说明 ??

  2. 注脚的解释 ??

  移动开发 最新文章
Vue3装载axios和element-ui
android adb cmd
【xcode】Xcode常用快捷键与技巧
Android开发中的线程池使用
Java 和 Android 的 Base64
Android 测试文字编码格式
微信小程序支付
安卓权限记录
知乎之自动养号
【Android Jetpack】DataStore
上一篇文章      下一篇文章      查看所有文章
加:2022-05-24 18:20:14  更:2022-05-24 18:21:56 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/25 0:28:57-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码