我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Android仿UC浏览器左右上下滚动功能

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Android仿UC浏览器左右上下滚动功能

本文要解决在侧滑菜单右边加个文本框,并能实现文本的上下滑动和菜单的左右滚动。这里推荐可以好好看看android的触摸事件的分发机制,这里我就不详细讲了,我只讲讲这个应用。要实现的功能就像UC浏览器(或其它手机浏览器)的左右滚动,切换网页,上下滚动,拖动内容。
本文的效果:

 

一、功能要求与实现
1、功能要求:
(1)手指一开始按着屏幕左右移动时,只能左右滚动菜单,如果这时手指一直按着,而且上下移动了,那么菜单显示部分保持不变,但文本框也不上下移动!                      
(2)手指一开始按着屏幕上下移动时,只能上下滚动文本框,如果这时手指一直按着,而且左右移动了,那么文本框显示部分保持不变,但菜单也不左右移动!
2、初步实现:
      左边的菜单项增加一个listview,为右边的内容项添加一个textview,并且为了能让它实现上下滚动的功能,给textview加了个scrollview。
       这种效果肯定是不对的,你看,我们手指上下禾移动文本时,如果还左右移动了,菜单也显示出来了。

 

3、修改实现   
     这时我就想从触摸事件的分发入手,这里因为我是把ScrollView的触摸事件注册到LinearLayout。(LinearLayout中包含了ScrollView,不懂看下面的布局)中去,所以触摸事件会先传递给LinearLayout。
分以下两种情况:
(1)如果是手指左右移动,则把触摸事件传给LinearLayout。函数onTouch返回true,表示触摸事件不再传递下去,那么ScrollView就动不了了
(2)如果是手指上下移动,触摸事件先传给LinearLayout,但LinearLayout不做任何处理,直接传递给ScrollView,ScrollView来处理触摸事件。
这是修改后的效果:

                                             

二、布局与代码
1、布局


<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
 xmlns:tools="http://schemas.android.com/tools" 
 android:id="@+id/layout" 
 android:layout_width="match_parent" 
 android:layout_height="match_parent" 
 android:orientation="horizontal" 
 tools:context=".MainActivity" > 
 <LinearLayout 
  android:id="@+id/menu" 
  android:layout_width="match_parent" 
  android:layout_height="match_parent" 
  android:orientation="vertical" 
  android:background="@drawable/menu" > 
  <!-- 添加一个ListView控件 --> 
   <ListView 
   android:id="@+id/menuList"  
  android:layout_width="fill_parent"  
  android:layout_height="fill_parent"/>   
 </LinearLayout> 
 <LinearLayout 
  android:id="@+id/content" 
  android:layout_width="match_parent" 
  android:layout_height="match_parent" 
  android:orientation="vertical"> 
<ScrollView 
 android:id="@+id/scrollview" 
 android:layout_width="fill_parent" 
 android:layout_height="wrap_content" > 
  <TextView android:id="@+id/content_text" 
   android:layout_width="wrap_content" 
   android:layout_height="wrap_content" 
   android:text="@string/text1" 
    android:textSize="22px" /> 
 </ScrollView> 
 </LinearLayout> 
</LinearLayout> 

2、代码


package com.example.learningjava; 
import java.util.ArrayList; 
import java.util.HashMap; 
import java.util.Map; 
import com.example.learningjava.R.string; 
import android.R.integer; 
import android.R.menu; 
import android.os.AsyncTask; 
import android.os.Build; 
import android.os.Bundle; 
import android.annotation.SuppressLint; 
import android.annotation.TargetApi; 
import android.widget.AdapterView; 
import android.widget.AdapterView.OnItemClickListener; 
import android.widget.ArrayAdapter; 
import android.widget.LinearLayout.LayoutParams; 
import android.widget.ListView; 
import android.widget.ScrollView; 
import android.widget.Toast; 
import android.app.Activity; 
import android.content.Context; 
import android.util.AttributeSet; 
import android.util.DisplayMetrics; 
import android.util.Log; 
import android.view.GestureDetector; 
import android.view.Menu; 
import android.view.MotionEvent; 
import android.view.VelocityTracker; 
import android.view.View; 
import android.view.View.OnTouchListener; 
import android.view.Window; 
import android.widget.LinearLayout; 
public class MainActivity extends Activity implements OnTouchListener{ 
 private LinearLayout menuLayout;//菜单项 
 private LinearLayout contentLayout;//内容项 
 private LayoutParams menuParams;//菜单项目的参数 
 private LayoutParams contentParams;//内容项目的参数contentLayout的宽度值 
 private int disPlayWidth;//手机屏幕分辨率 
 private float xDown;//手指点下去的横坐标 
 private float xMove;//手指移动的横坐标 
 private float xUp;//记录手指上抬后的横坐标 
 private float yDown;//手指点下去的纵坐标 
 private float yMove;//手指移动的纵坐标 
 private VelocityTracker mVelocityTracker; // 用于计算手指滑动的速度。 
 private float velocityX;//手指左右移动的速度 
 public static final int SNAP_VELOCITY = 400; //滚动显示和隐藏menu时,手指滑动需要达到的速度。 
 private boolean menuIsShow = false;//初始化菜单项不可翙 
 private static final int menuPadding=160;//menu完成显示,留给content的宽度 
 private ListView menuListView;//菜单列表的内容 
 private ScrollView scrollView;// 文本框的滚动条 
 private boolean wantToScrollText=false;//想要下下滚动文本内容 
 private boolean wantToScrollTextMenu=false; 
 private boolean oneFucction=false;//确保函数只被调用一次 
 protected void onCreate(Bundle savedInstanceState) { 
  super.onCreate(savedInstanceState); 
  requestWindowFeature(Window.FEATURE_NO_TITLE); 
  setContentView(R.layout.activity_main); 
  initLayoutParams(); 
  initMenuList(); 
  initScrollView(); 
 } 
  
 private void initLayoutParams() 
 { 
 //得到屏幕的大小 
  DisplayMetrics dm = new DisplayMetrics(); 
  getWindowManager().getDefaultDisplay().getMetrics(dm); 
  disPlayWidth =dm.widthPixels; 
  //获得控件 
  menuLayout = (LinearLayout) findViewById(R.id.menu); 
  contentLayout = (LinearLayout) findViewById(R.id.content); 
  findViewById(R.id.layout).setOnTouchListener(this); 
  //获得控件参数 
  menuParams=(LinearLayout.LayoutParams)menuLayout.getLayoutParams(); 
  contentParams = (LinearLayout.LayoutParams) contentLayout.getLayoutParams(); 
  //初始化菜单和内容的宽和边距 
  menuParams.width = disPlayWidth - menuPadding; 
  menuParams.leftMargin = 0 - menuParams.width; 
  contentParams.width = disPlayWidth; 
  contentParams.leftMargin=0; 
  //设置参数 
  menuLayout.setLayoutParams(menuParams); 
  contentLayout.setLayoutParams(contentParams); 
 } 
  
 private void initMenuList() 
 { 
 final String[] strs = new String[] { "第1章 Java概述 ", "第2章 理解面向对象", "第3章 数据类型和运算符", "第4章 流程控制和数组", "第5章 面向对象(上)"}; 
  menuListView = (ListView) findViewById(R.id.menuList); 
  menuListView.setAdapter(new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1, strs));//为ListView绑定适配器 
  //启动列表点击监听事件 
  menuListView.setOnItemClickListener(new OnItemClickListener() { 
   @Override 
   public void onItemClick(AdapterView<?> arg0, View arg1, int arg2,long arg3) { 
     Toast.makeText(getApplicationContext(),"您选择了" + strs[arg2], Toast.LENGTH_SHORT).show(); 
   } 
  }); 
 } 
  
 public void initScrollView(){ 
  scrollView = (ScrollView)this.findViewById(R.id.scrollview); 
  scrollView.setOnTouchListener(this);//绑定监听侧滑事件的View,即在绑定的View进行滑动才可以显示和隐藏左侧布局。 这句非常重要,不要设置它的触摸事件 了,要不会吞掉布局的触摸事件 
 } 
 @Override 
 public boolean onTouch(View v, MotionEvent event) 
 { 
  acquireVelocityTracker(event); 
  if (event.getAction()==MotionEvent.ACTION_DOWN) 
  { 
   xDown=event.getRawX(); 
   yDown=event.getRawY(); 
   return false; 
  } 
  else if(event.getAction()==MotionEvent.ACTION_MOVE) 
  { 
   if(wantToScrollText)//当前想滚动显示文本 
    return false; 
   xMove=event.getRawX(); 
   yMove=event.getRawY(); 
   if(menuIsShow){ 
    isScrollToShowMenu(); 
    return true; 
   } 
   if(!oneFucction) 
   { 
   oneFucction=true; 
   //这个if只能被调用一次 
   if(Math.abs(xDown-xMove)<Math.abs(yDown-yMove)) 
    { 
    wantToScrollText=true; 
    return false; 
    } 
   }  
   isScrollToShowMenu(); 
  } 
  else if(event.getAction()==MotionEvent.ACTION_UP)  
  { 
   oneFucction=false; 
   if(wantToScrollText){ 
   wantToScrollText=false; 
   return false; 
   }  
   xUp=event.getRawX(); 
   isShowMenu(); 
   releaseVelocityTracker(); 
  } 
  else if (event.getAction()==MotionEvent.ACTION_CANCEL) 
  {  
   releaseVelocityTracker(); 
   return false; 
  } 
  return true;//false时才能把触摸事件再传给scroll 
 } 
  
 private void isScrollToShowMenu() 
 { 
  int distanceX = (int) (xMove - xDown);  
  if (!menuIsShow) { 
   scrollToShowMenu(distanceX); 
  }else{ 
   scrollToHideMenu(distanceX); 
  } 
 } 
  
 private void isShowMenu() 
 { 
  velocityX =getScrollVelocity(); 
  if(wantToShowMenu()){ 
   if(shouldShowMenu()){ 
    showMenu(); 
   }else{ 
    hideMenu(); 
   } 
  } 
  else if(wantToHideMenu()){ 
   if(shouldHideMenu()){ 
    hideMenu(); 
   }else{ 
    showMenu(); 
   } 
  }  
 } 
  
 private boolean wantToShowMenu(){ 
  return !menuIsShow&&xUp-xDown>0; 
 } 
  
 private boolean wantToHideMenu(){ 
  return menuIsShow&&xDown-xUp>0; 
 } 
  
 private boolean shouldShowMenu(){ 
  return xUp-xDown>menuParams.width/2||velocityX>SNAP_VELOCITY; 
 } 
  
 private boolean shouldHideMenu(){ 
  return xDown-xUp>menuParams.width/2||velocityX>SNAP_VELOCITY; 
 } 
  
 private void showMenu() 
 { 
  new showMenuAsyncTask().execute(50); 
  menuIsShow=true; 
 } 
  
 private void hideMenu() 
 { 
  new showMenuAsyncTask().execute(-50); 
  menuIsShow=false; 
 } 
  
 private void scrollToShowMenu(int scrollX) 
 { 
  if(scrollX>0&&scrollX<= menuParams.width) 
  menuParams.leftMargin =-menuParams.width+scrollX; 
  menuLayout.setLayoutParams(menuParams); 
 } 
  
 private void scrollToHideMenu(int scrollX) 
 { 
  if(scrollX>=-menuParams.width&&scrollX<0) 
  menuParams.leftMargin=scrollX; 
  menuLayout.setLayoutParams(menuParams); 
 } 
  
 private void acquireVelocityTracker(final MotionEvent event) { 
  if(null == mVelocityTracker) { 
   mVelocityTracker = VelocityTracker.obtain(); 
  } 
  mVelocityTracker.addMovement(event); 
 } 
  
 private int getScrollVelocity() { 
  mVelocityTracker.computeCurrentVelocity(1000); 
  int velocity = (int) mVelocityTracker.getXVelocity(); 
  return Math.abs(velocity); 
 } 
  
 private void releaseVelocityTracker() { 
  if(null != mVelocityTracker) { 
   mVelocityTracker.clear(); 
   mVelocityTracker.recycle(); 
   mVelocityTracker = null; 
  } 
 } 
  
 class showMenuAsyncTask extends AsyncTask<Integer, Integer, Integer> 
 { 
  @Override 
  protected Integer doInBackground(Integer... params) 
  { 
   int leftMargin = menuParams.leftMargin; 
   while (true) 
   {// 根据传入的速度来滚动界面,当滚动到达左边界或右边界时,跳出循环。 
    leftMargin += params[0]; 
    if (params[0] > 0 && leftMargin > 0) 
    { 
     leftMargin= 0; 
     break; 
    } else if (params[0] < 0 && leftMargin <-menuParams.width) 
    { 
     leftMargin=-menuParams.width; 
     break; 
    } 
    publishProgress(leftMargin); 
    try 
    { 
     Thread.sleep(40);//休眠一下,肉眼才能看到滚动效果 
    } catch (InterruptedException e) 
    { 
     e.printStackTrace(); 
    } 
   } 
   return leftMargin; 
  } 
  @Override 
  protected void onProgressUpdate(Integer... value) 
  { 
   menuParams.leftMargin = value[0]; 
   menuLayout.setLayoutParams(menuParams); 
  } 
  @Override 
  protected void onPostExecute(Integer result) 
  { 
   menuParams.leftMargin = result; 
   menuLayout.setLayoutParams(menuParams); 
  } 
 } 
} 

三、原理与说明
原理 :
1、将ScrollView的触摸事件注册到LinearLayout中去。(LinearLayout中包含了ScrollView,不懂看布局)
2、首先判断手势是想要左右运动还是上下运动,如果是左右运动,那么LinearLayout得到触摸事件,即函数OnTouch返回true;如果想上下运动,即函数OnTouch返回false;
这里要注意的是,手势判断只一次,什么意思呢?就是说你第1次按下,到你一直按着,这中间只判断一次你的手势想要做的运动。
3、手指离开屏幕后,再来恢复所有的参数。

这是为大家分享的源码,请下载:Android仿UC浏览器左右上下滚动功能,希望本文所述对大家学习Android软件编程有所帮助。

您可能感兴趣的文章:Android自定义View实现广告信息上下滚动效果Android仿天天动听歌曲自动滚动viewAndroid高仿京东垂直循环滚动新闻栏Android PickerView滚动选择器的使用方法Android仿淘宝商品浏览界面图片滚动效果Android程序开发ListView+Json+异步网络图片加载+滚动翻页的例子(图片能缓存,图片不错乱)android开发之横向滚动/竖向滚动的ListView(固定列头)android实现上下滚动的TextViewAndroid中实现多行、水平滚动的分页的Gridview实例源码Android新闻广告条滚动效果


免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Android仿UC浏览器左右上下滚动功能

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android仿UC浏览器左右上下滚动功能

本文要解决在侧滑菜单右边加个文本框,并能实现文本的上下滑动和菜单的左右滚动。这里推荐可以好好看看android的触摸事件的分发机制,这里我就不详细讲了,我只讲讲这个应用。要实现的功能就像UC浏览器(或其它手机浏览器)的左右滚动,切换网页,上
2022-06-06

Android视频播放器屏幕左侧边随手指上下滑动亮度调节功能的原理实现

本文给大家分享Android视频播放器屏幕左侧边随手指上下滑动亮度调节功能的原理实现,具体代码如下所示:import android.app.Activity; import android.os.Bundle; import an
2022-06-06

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录