本文出处:http://blog.csdn.net/sk719887916/article/details/40348853,作者:skay
     阅读此文建议先阅读 安卓Tv开发(二)移动智能电视之焦点控制(按键事件)
     接触TV的的开发,操作体验是我们开发者必须要关注的,因此,遥控器做鼠标成了智能设备的开发主流不可缺少的一部分,今天给大家带来怎样实现遥控操作模拟鼠标的功能,觉得ok的请到下方自觉的点个赞。有自动化测试框架也有用此方式实现的。
   实现功能: 遥控器方向键控制鼠标箭头移动
                       遥控器可以操作模拟鼠标进行点击操作。
  效果图:
    实现之前先做个大致分析,记得之前我在一篇安卓实现高仿Ios桌面的博客中就接触过类似此需求的功能,第一眼大家觉得可拖动gridview和实现遥控器拖动鼠标有啥关系,仔细一分析,其实异曲同工,可拖动的gridView我们是用手指进行item移动的,而这次的模拟鼠标我们可以把它想成一个item(只不过展现的UI为一个鼠标箭头的图片),用遥控器进行控制的,当然我们无需拖动,用遥控方向键来控制item的位移即可。至于遥控器点击OK键进行点击执行点击操作的事件,我们可以获取模拟鼠标在屏幕中的绝对坐标,然后模拟一个此处的点击事件即可。

一 自定义鼠标视图

        用于展现鼠标的具体视图,MouseView 继承 FrameLayout ,之后漂浮在控制的activity上,只要由WindowManage控制此视图的展现和移动。

public MouseView(Context context) {super(context);}public MouseView(Context context, MouseManager mMouseMrg) {super(context);init( mMouseMrg);}public OnMouseListener getOnMouseListener() {return mOnMouseListener;}public void setOnMouseListener(OnMouseListener mOnMouseListener) {this.mOnMouseListener = mOnMouseListener;}@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {super.onMeasure(widthMeasureSpec, heightMeasureSpec);if (mMouseView != null && mMouseBitmap != null) {mMouseView.measure(MeasureSpec.makeMeasureSpec(mMouseBitmap.getWidth(), MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(mMouseBitmap.getHeight(), MeasureSpec.EXACTLY));}}private void init(MouseManager manager) {mMouseManager  = manager;Drawable drawable =  getResources().getDrawable(R.drawable.shubiao);mMouseBitmap = drawableToBitamp(drawable);mMouseView = new ImageView(getContext());mMouseView.setImageBitmap(mMouseBitmap);addView(mMouseView, new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));mOffsetX = (int)((mMouseBitmap.getWidth()));mOffsetY = (int)((mMouseBitmap.getHeight()));}@Overrideprotected void onLayout(boolean changed, int left, int top, int right,int bottom) {if(mMouseView != null) {mMouseView.layout(mMouseX, mMouseY, mMouseX + mMouseView.getMeasuredWidth(), mMouseY + mMouseView.getMeasuredHeight());}}

二 鼠标控制器

      用来处理点击事件,和鼠标移动事件。主要是重写事件分发方法 dispatchKeyEvent(KeyEvent event),用于控制鼠标移动和点击。
/*** @param parent* @param type*/public void init(ViewGroup parentView, int type) {mParentView = parentView;mContext = parentView.getContext();mMouseView = new MouseView(mContext, this);mMouseView.setOnMouseListener(this);mCurrentType = type;}/*** @return*/public boolean getMouseType() {return isMouseType;}/*** @return*/public int getCurrentActivityType() {return mCurrentType;}/*** showmouse*/public void showMouseView() {ViewGroup.LayoutParams lp = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,ViewGroup.LayoutParams.MATCH_PARENT);if(mMouseView != null) {mParentView.addView(mMouseView, lp);}}public boolean onDpadClicked(KeyEvent event) {if(!isMouseType) {return false;}if(event.getKeyCode() == KEYCODE_CENTER) {dispatchKeyEventToMouse(event);} else {if(event.getAction() == KeyEvent.ACTION_DOWN) {if(!isKeyEventCousumed) {if(event.getDownTime() - mLastEventTime < defTimes) {if(mSpeed < defMaxSpeed) {mSpeed ++;}} else {mSpeed = 1;}}mLastEventTime = event.getDownTime();dispatchKeyEventToMouse(event);isKeyEventCousumed = true;} else if(event.getAction() == KeyEvent.ACTION_UP) {if(!isKeyEventCousumed){dispatchKeyEventToMouse(event);}isKeyEventCousumed = false;}}return true;}public void sendCenterClickEvent(int x, int y, int action) {sendMotionEvent(x, y, action);}@SuppressLint("InlinedApi")public void sendMouseHoverEvent(int downx, int downy) {sendMotionEvent(downx, downy, MotionEvent.ACTION_HOVER_MOVE);}@TargetApi(Build.VERSION_CODES.HONEYCOMB_MR1)@SuppressLint("NewApi")private void sendMotionEvent(int x, int y, int action) {MotionEvent motionEvent = getMotionEvent( x, y ,action) ;if(action == MotionEvent.ACTION_HOVER_MOVE) {motionEvent.setSource(InputDevice.SOURCE_CLASS_POINTER);mMouseView.dispatchGenericMotionEvent(motionEvent);//mParentView.dispatchGenericMotionEvent(motionEvent);} else {//mParentView.dispatchTouchEvent(motionEvent);mMouseView.dispatchTouchEvent(motionEvent);}}private  MotionEvent getMotionEvent(int downx, int downy, int action) {// TODO Auto-generated method stublong downTime = SystemClock.uptimeMillis();long eventTime = SystemClock.uptimeMillis();int metaState = 0;return MotionEvent.obtain(downTime, eventTime, action, downx, downy, metaState);}@Overridepublic boolean onclick(View v, KeyEvent et) {if (getMouseType()) {return onDpadClicked(et);}return mParentView.dispatchKeyEvent(et);}
   

三 当前页面加入鼠标

         如果某个activity需要加入鼠标,只要初始化 MouseManager,然后设置当先鼠标显示即可,本次为了方便操作和直观的显示,加入一个webView加载网页,用于鼠标控制操作。
public class MainActivity extends Activity {WindowManager wm;WindowManager.LayoutParams params;private MouseManager mMouseManager;public static ViewGroup contentView;private WebView webView;private View mLoginStatusView;private TextView mLoaddingMessageView;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);LayoutInflater inflater = getLayoutInflater();contentView = (ViewGroup) inflater.inflate(R.layout.test, null);setContentView(contentView);init();initMouse();showMouse();}private void init() {webView = (WebView) contentView.findViewById(R.id.web);mLoginStatusView = this.findViewById(R.id.login_status);mLoaddingMessageView = (TextView) this.findViewById(R.id.login_status_message);Button button = (Button) contentView.findViewById(R.id.btn_onclick);button.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {Toast.makeText(MainActivity.this, "onclicked ", 1).show();showProgress(true);webView.setVisibility(View.VISIBLE);webView.loadUrl("https://www.baidu.com/");WebSettings settings = webView.getSettings();settings.setJavaScriptEnabled(true);webView.setWebViewClient(new WebViewClient() {@Overridepublic boolean shouldOverrideUrlLoading(WebView view,String url) {view.loadUrl(url);return true;}@Overridepublic void onPageFinished(WebView view, String url) {super.onPageFinished(view, url);}@Overridepublic void onReceivedError(WebView view, int errorCode,String description, String failingUrl) {Toast.makeText(MainActivity.this, "加载失败 ",Toast.LENGTH_LONG).show();super.onReceivedError(view, errorCode, description,failingUrl);}});webView.setWebChromeClient(new WebChromeClient() {@Overridepublic void onProgressChanged(WebView view, int newProgress) {// TODO Auto-generated method stubif (newProgress == 100) {showProgress(false);} else {}}});}});}@SuppressLint("NewApi")private void showProgress(final boolean show) {// On Honeycomb MR2 we have the ViewPropertyAnimator APIs, which allow// for very easy animations. If available, use these APIs to fade-in// the progress spinner.if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB_MR2) {int shortAnimTime = getResources().getInteger(android.R.integer.config_shortAnimTime);mLoginStatusView.setVisibility(View.VISIBLE);mLoginStatusView.animate().setDuration(shortAnimTime).alpha(show ? 1 : 0).setListener(new AnimatorListenerAdapter() {@Overridepublic void onAnimationEnd(Animator animation) {webView.setVisibility(show ? View.VISIBLE: View.GONE);}});webView.setVisibility(View.VISIBLE);webView.animate().setDuration(shortAnimTime).alpha(show ? 0 : 1).setListener(new AnimatorListenerAdapter() {@Overridepublic void onAnimationEnd(Animator animation) {webView.setVisibility(show ? View.GONE: View.VISIBLE);}});} else {// The ViewPropertyAnimator APIs are not available, so simply show// and hide the relevant UI components.mLoginStatusView.setVisibility(show ? View.VISIBLE : View.GONE);webView.setVisibility(show ? View.GONE : View.VISIBLE);}}private void showMouse() {mMouseManager.showMouseView();}public void initMouse() {initMouseMrg();}public void initMouseMrg() {mMouseManager = new MouseManager();mMouseManager.init(contentView, MouseManager.MOUSE_TYPE);mMouseManager.setshowMouse(true);}}

简单三步,代码就可以实现简单的用于遥控器操作的TV的浏览器,本次demo只是用户手动模拟点击,至于实现自动化模拟点击,以上方式显得极为笨拙,

  如果实现自动化测试或者模拟点击事件 ,采用Instrumentation代理,而Instrumentation不需要一定的activity展现,我们可以将它理解为一种没有图形界面的,具有启动能力的,用于监控其他类(用Target Package声明)的工具类。Instrumentation开发者可以构建一个独立于应用程序的代理程序(Agent),用来监测和协助运行在 JVM 上的程序,甚至能够替换和修改某些类的定义。有了这样的功能,我们就可以实现更为灵活的运行时虚拟机监控和 Java 类操作,代替人为操作,主要用于自动测试框架。

instrumentation发送键盘鼠标事件:Instrumentation提供了丰富的以send开头的函数接口来实现模拟键盘和鼠标,如下所述:

sendCharacterSync(int keyCode)            //用于发送指定KeyCode的按键

sendKeyDownUpSync(int key)                //用于发送指定KeyCode的按键

sendPointerSync(MotionEvent event)     //用于模拟Touch

sendStringSync(String text)                   //用于发送字符串

发发送一条模拟点击事件

  1. Instrumentation inst=new Instrumentation();
  2. inst.sendPointerSync(MotionEvent.obtain(SystemClock.uptimeMillis(),SystemClock.uptimeMillis(), MotionEvent.ACTION_DOWN, 10, 10, 0));
  3. inst.sendPointerSync(MotionEvent.obtain(SystemClock.uptimeMillis(),SystemClock.uptimeMillis(), MotionEvent.ACTION_UP, 10, 10, 0));

具体逻辑是当我的app启动时 。就可以用全局的Instrumentation来监控到底当前在哪个界面,提前将点击的入口定义OK后,接下来的点击事件就如同用户主动点击一样,包括跳转逻辑不再需要考虑。

     当然现在安卓也可以采用辅助功能实现监控某个界面 遍历界面的view元素实现模拟点击操作,有兴趣的朋友可以去看下辅助自动装(http://blog.csdn.net/sk719887916/article/details/46746991)和微信自动枪红包的demo,实现方式比简单,这里就不再分享了。
   尊重原创:本文出处     本文出处:http://blog.csdn.net/sk719887916/article/details/40348853,作者:skay ,欢迎阅读。
源码下载:https://github.com/Tamicer/MouseView_TV
      
    

安卓TV开发(九) Android模拟事件 遥控器变身成鼠标来操作TV相关推荐

  1. STC8H开发(九): STC8H8K64U模拟USB HID外设

    目录 STC8H开发(一): 在Keil5中配置和使用FwLib_STC8封装库(图文详解) STC8H开发(二): 在Linux VSCode中配置和使用FwLib_STC8封装库(图文详解) ST ...

  2. 【安卓开发 】Android初级开发(九)Android中封装View提供接口供点击事件回调的方法及使用

    自古一楼先上图 package com.example.mydialog;import android.app.Dialog; import android.content.Context; impo ...

  3. 安卓多媒体开发!Android高级工程师面试实战,系列篇

    开头 最近,程序员这个圈子可谓热闹非凡,前不久,一个神奇的开源项目996.ICU诞生,几天时间就有超过10w+ star 和 1w+ fork ,可谓是Github上star增长最快的项目.996.i ...

  4. java安卓app开发教程,Android app开发入门 —— your 'Hello, World'

    从这篇可以掌握到 Android app开发环境的搭建 开发工具介绍及安装 创建你的"Hello, World" 工程结构的介绍 工程gradle配置 简单布局 代码sample ...

  5. android+tv+开发+icon,Android TV+HTML5的架构设计和应用开发经验.pdf

    Android TV + HTML5 : 架構設計與App開發經驗 高煥堂 亞太地區Android 技術大會主席 中國電子視像行業協會高級顧問 ★ 講題簡介 HTML5天生麗質,具有天賦的跨端.跨雲. ...

  6. 安卓原生开发!Android面试题集2021版,附答案

    正文 Android行业主要问题是初级Android太多了,会写xml和Activity的程序员太多了,初中级程序员面临很大的竞争,现状也就偏于高级开发者.越来越多的初中级Android程序员找不到满 ...

  7. android tv 开发布局,Android TV开发总结(七)构建一个TV app中的剧集列表控件

    前言:剧集类控件,在TV app中非常常见,今天将介绍构建一个TV app中的剧集列表控件,此控件上传到我的Github:https://github.com/hejunlin2013/Episode ...

  8. Android模拟事件keycode对照表

    1.例子: //这条命令相当于按了设备的Backkey键 adb shell input keyevent 4 //可以解锁屏幕 adb shell input keyevent  82 //在屏幕上 ...

  9. android 模拟crash_Android 收集Crash信息及用户操作步骤

    对于android开发者来说,Crash 可谓是司空见惯的事了,没有一个程序员敢保证自己的程序绝对不会发生crash.开发的时候发生crash还好,可以Logcat来查看log分析出原因,但是在线上, ...

最新文章

  1. Coursera台大机器学习技法课程笔记04-Soft-Margin Support Vector Machine
  2. 机器学习导论(张志华):基本概念
  3. 分布式内存数据库---Redis数据库之(键)key
  4. [Vue源码分析] Virtual DOM
  5. c#精彩编程200例百度云_邂逅百度云智学院:福州理工学院AIOT实训营火热开营!...
  6. P4248-[AHOI2013]差异【SAM or SA】
  7. C#中采用OLEDB方式来读取EXCEL文件
  8. Linux 压缩与解压
  9. sklearn炼丹术之——交叉验证Cross-validation: evaluating estimator performance
  10. ATM系统之问题描述与词汇表
  11. Web移动端常见问题-摘抄
  12. PIM DM技术原理与实验
  13. 服务器配置文档模板,服务器配置模板
  14. 用Java实现24点游戏
  15. 从0写USB摄像头驱动程序
  16. 表格进阶03——简历制作(用表格布局)
  17. 如何开发微信小程序?微信小程序有哪些特点?
  18. 编译型语言和解释型语言的区别总结
  19. 19年6月仔细阅读A篇:游戏界声优
  20. 如何旋转图片方法#ps教程#ps学习#ps修图抠图

热门文章

  1. React Native学习资源汇总
  2. Android MTK系统编译与调试命令
  3. 三相电检测电路c语言,三相缺相检测电路的原理分析
  4. Android OTA releasekey 替换
  5. 使用 CSS 追踪用户
  6. 酷宇宙大学:代币经济学(第四课)如何评估代币的效用?
  7. CNN-PS: CNN-Based Photometric Stereo for General Non-convex Surfaces 2018ECCV
  8. Unity3d绑定键盘弹出UI
  9. 对ABAP程序调优的学习(三)并行并发读取
  10. XSSFWorkbook Excel导出导入