pbootcms网站模板|日韩1区2区|织梦模板||网站源码|日韩1区2区|jquery建站特效-html5模板网

Android EditText長按菜單中分享功能的隱藏方法

Android EditText控件是經常使用的控件,下面這篇文章主要給大家介紹了關于Android中EditText長按菜單中分享功能的隱藏方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有

常見的EditText長按菜單如下

oppo

小米

需求是隱藏掉其中的分享/搜索功能,禁止將內容分享到其他應用。

最終解決方案

這里先說下最終解決方案

像華為/oppo等手機,該菜單實際是谷歌系統的即沒有改過源代碼,像小米的菜單則是自定義,該部分的源代碼改動過。
兩方面修改:

1.谷歌系統自帶的 通過 EditText.setCustomSelectionActionModeCallback()方法設置自定義的選中后動作模式接口,只保留需要的菜單項

代碼如下


 editText.customSelectionActionModeCallback = object : ActionMode.Callback {
 override fun onCreateActionMode(
 mode: ActionMode?,
 menu: Menu?
 ): Boolean {
 menu?.let {
 val size = menu.size()
 for (i in size - 1 downTo 0) {
 val item = menu.getItem(i)
 val itemId = item.itemId
 //只保留需要的菜單項 
 if (itemId != android.R.id.cut
 && itemId != android.R.id.copy
 && itemId != android.R.id.selectAll
 && itemId != android.R.id.paste
 ) {
 menu.removeItem(itemId)
 }
 }
 }
 return true
 }

 override fun onActionItemClicked(
 mode: ActionMode?,
 item: MenuItem?
 ): Boolean {
 return false
 }

 override fun onPrepareActionMode(
 mode: ActionMode?,
 menu: Menu?
 ): Boolean {
 return false
 }

 override fun onDestroyActionMode(mode: ActionMode?) {
 }
 }

2.小米等手機自定義菜單無法進行隱藏,可以是分享、搜索等功能失效,即在BaseActivity的startActivityForResult中進行跳轉攔截,如果是調用系統的分享/搜索功能,則不允許跳轉


 override fun startActivityForResult(
 intent: Intent?,
 requestCode: Int
 ) {
 if (!canStart(intent)) return
 super.startActivityForResult(intent, requestCode)
 }

 @SuppressLint("RestrictedApi")
 @RequiresApi(Build.VERSION_CODES.JELLY_BEAN)
 override fun startActivityForResult(
 intent: Intent?,
 requestCode: Int,
 options: Bundle?
 ) {
 if (!canStart(intent)) return
 super.startActivityForResult(intent, requestCode, options)
 }

 private fun canStart(intent: Intent?): Boolean {
 return intent?.let {
 val action = it.action
 action != Intent.ACTION_CHOOSER//分享
 && action != Intent.ACTION_VIEW//跳轉到瀏覽器
 && action != Intent.ACTION_SEARCH//搜索
 } ?: false
 }

如果以上不滿足要求,只能通過自定義長按菜單來實現自定義的菜單欄。

解決思路(RTFSC)

分析源碼菜單的創建和點擊事件

既然是長按松手后彈出的,應該在onTouchEvent中的ACTION_UP事件或者在performLongClick中,從兩方面著手
先看perfomLongEvent EditText沒有實現 去它的父類TextView中查找


TextView.java
 public boolean performLongClick() {
 ···省略部分代碼
 if (mEditor != null) {
 handled |= mEditor.performLongClick(handled);
 mEditor.mIsBeingLongClicked = false;
 }

 ···省略部分代碼
 return handled;
 }

可看到調用了 mEditor.performLongClick(handled)方法


Editor.java

 public boolean performLongClick(boolean handled) {
 if (!handled && !isPositionOnText(mLastDownPositionX, mLastDownPositionY)
 && mInsertionControllerEnabled) {
 final int offset = mTextView.getOffsetForPosition(mLastDownPositionX,
 mLastDownPositionY);//獲取當前松手時的偏移量
 Selection.setSelection((Spannable) mTextView.getText(), offset);//設置選中的內容
 getInsertionController().show();//插入控制器展示
 mIsInsertionActionModeStartPending = true;
 handled = true;
 ···
 }
 if (!handled && mTextActionMode != null) {
 if (touchPositionIsInSelection()) {
 startDragAndDrop();//開始拖動
 ···
 } else {
 stopTextActionMode();
 selectCurrentWordAndStartDrag();//選中當前單詞并且開始拖動
 ···
 }
 handled = true;
 }
 if (!handled) {
 handled = selectCurrentWordAndStartDrag();//選中當前單詞并且開始拖動
 ···
 }
 }

 return handled;
 }

從上面代碼分析

1.長按時會先選中內容 Selection.setSelection((Spannable) mTextView.getText(), offset)

2.顯示插入控制器  getInsertionController().show()

3.開始拖動/選中單詞后拖動 startDragAndDrop()/ selectCurrentWordAndStartDrag()

看著很像了

看下第二步中展示的內容


Editor.java -> InsertionPointCursorController

 public void show() {
 getHandle().show();
 if (mSelectionModifierCursorController != null) {
 mSelectionModifierCursorController.hide();
 }
 }

 ···
 private InsertionHandleView getHandle() {
 if (mSelectHandleCenter == null) {
 mSelectHandleCenter = mTextView.getContext().getDrawable(
 mTextView.mTextSelectHandleRes);
 }
 if (mHandle == null) {
 mHandle = new InsertionHandleView(mSelectHandleCenter);
 }
 return mHandle;
 }

實際是InsertionHandleView 執行了show方法。  查看其父類HandlerView的構造方法


 private HandleView(Drawable drawableLtr, Drawable drawableRtl, final int id) {
 super(mTextView.getContext());
 ···
 mContainer = new PopupWindow(mTextView.getContext(), null,
 com.android.internal.R.attr.textSelectHandleWindowStyle);
 ···
 mContainer.setContentView(this);
 ···
 }

由源碼可看出 HandlerView實際上是PopWindow的View。 即選中的圖標實際上是popwidow
看源碼可看出HandleView有兩個實現類 InsertionHandleView  和SelectionHandleView 由名字可看出一個是插入的,一個選擇的 看下HandleView的show方法


Editor.java ->HandleView

 public void show() {
 if (isShowing()) return;
 getPositionListener().addSubscriber(this, true );
 // Make sure the offset is always considered new, even when focusing at same position
 mPreviousOffset = -1;
 positionAtCursorOffset(getCurrentCursorOffset(), false, false);
 }

看下positionAtCursorOffset方法


Editor.java ->HandleView 

 protected void positionAtCursorOffset(int offset, boolean forceUpdatePosition,
 boolean fromTouchScreen) {
 ···
 if (offsetChanged || forceUpdatePosition) {
 if (offsetChanged) {
 updateSelection(offset);
 ···
 }
 ···
 }
 }

里面有一個updateSelection更新選中的位置,該方法會導致EditText重繪,再看show方法的getPositionListener().addSubscriber(this, true )

getPositionListener()返回的實際上是ViewTreeObserver.OnPreDrawListener的實現類PositionListener
重繪會調用onPreDraw的方法


Editor.java-> PositionListener 

 @Override
 public boolean onPreDraw() {
 ···
 for (int i = 0; i < MAXIMUM_NUMBER_OF_LISTENERS; i++) {
 ···
 positionListener.updatePosition(mPositionX, mPositionY,
 mPositionHasChanged, mScrollHasChanged);
 ···
 }
 ···
 return true;
 }

調用了positionListener.updatePosition方法, positionListener這個實現類對應的是HandlerView

重點在HandleView的updatePosition方法,該方法進行popWindow的顯示和更新位置

看一下該方法的實現


Editor.java ->HandleView

 @Override
 public void updatePosition(int parentPositionX, int parentPositionY,
 boolean parentPositionChanged, boolean parentScrolled) {
 ···
 if (isShowing()) {
 mContainer.update(pts[0], pts[1], -1, -1);
 } else {
 mContainer.showAtLocation(mTextView, Gravity.NO_GRAVITY, pts[0], pts[1]);
 }
 } 
 ···
 }
 }

到此我們知道選中的圖標即下面紅框內的實際上popWindow展示

點擊選中的圖標可以展示菜單,看下HandleView的onTouchEvent方法


Editor.java ->HandleView
 @Override
 public boolean onTouchEvent(MotionEvent ev) {
 updateFloatingToolbarVisibility(ev);
 ···
 }

updateFloatingToolbarVisibility(ev)真相在這里,該方法進行懸浮菜單欄的展示
經過進一步查找,可以看到會調用下面SelectionActionModeHelper的這個方法


SelectionActionModeHelper.java

 public void invalidateActionModeAsync() {
 cancelAsyncTask();
 if (skipTextClassification()) {
 invalidateActionMode(null);
 } else {
 resetTextClassificationHelper();
 mTextClassificationAsyncTask = new TextClassificationAsyncTask(
  mTextView,
  mTextClassificationHelper.getTimeoutDuration(),
  mTextClassificationHelper::classifyText,
  this::invalidateActionMode)
  .execute();
 }
 }

會啟動一個叫TextClassificationAsyncTask的異步任務,該異步任務最后會執行mEditor.getTextActionMode().invalidate()


 private void invalidateActionMode(@Nullable SelectionResult result) {
 ···
 final ActionMode actionMode = mEditor.getTextActionMode();
 if (actionMode != null) {
 actionMode.invalidate();
 }
 ···
 }

最后看下mTextActionMode 如何在Editor中賦值


Editor.java

 void startInsertionActionMode() {
 ···
 ActionMode.Callback actionModeCallback =
 new TextActionModeCallback(false /* hasSelection */);
 mTextActionMode = mTextView.startActionMode(
 actionModeCallback, ActionMode.TYPE_FLOATING);
 ···
 }

看下mTextView.startActionMode的注釋,在View類中,Start an action mode with the given type. 根據給的類型,開啟一個動作模式,該模式是一個TYPE_FLOATING模式,菜單的生成就在TextActionModeCallback類中
在TextActionModeCallback的onCreateActionMode方法中


Editor.java ->TextActionModeCallback

 @Override
 public boolean onCreateActionMode(ActionMode mode, Menu menu) {
 mode.setTitle(null);
 mode.setSubtitle(null);
 mode.setTitleOptionalHint(true);
 //生成菜單
 populateMenuWithItems(menu);

 Callback customCallback = getCustomCallback();
 if (customCallback != null) {
 if (!customCallback.onCreateActionMode(mode, menu)) {
  // The custom mode can choose to cancel the action mode, dismiss selection.
  Selection.setSelection((Spannable) mTextView.getText(),
  mTextView.getSelectionEnd());
  return false;
 }
 }
 ···
 }

生成的菜單的方法populateMenuWithItems(menu)中,生成完菜單會執行自定義的回調getCustomCallback() , 看下該回調如何賦值。

在TextView中


TextView.java
 public void setCustomSelectionActionModeCallback(ActionMode.Callback actionModeCallback) {
 createEditorIfNeeded();
 mEditor.mCustomSelectionActionModeCallback = actionModeCallback;
 }

因此我們可以在自定義回調的onCreateActionMode方法中,刪除不需要的菜單項。

但該方法對小米手機無效,小米手機的菜單展示,不是通過startActionMode來展示的。不過可以對菜單中的分享等功能進行禁止跳轉,解決方法看最上面

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對html5模板網的支持。

【網站聲明】本站部分內容來源于互聯網,旨在幫助大家更快的解決問題,如果有圖片或者內容侵犯了您的權益,請聯系我們刪除處理,感謝您的支持!

相關文檔推薦

這篇文章主要介紹了Android開發之TabHost選項卡及相關疑難解決方法,結合實例形式較為詳細的分析了Android開發中TabHost選項卡的常見用法以及相關疑難問題解決方法,需要的朋友可以參考下
這篇文章主要介紹了Android TabHost選項卡標簽圖標始終不出現的解決方法,涉及Android界面布局相關屬性與狀態設置操作技巧,需要的朋友可以參考下
這篇文章主要介紹了Android開發之Notification手機狀態欄通知用法,結合實例形式分析了Android Notification手機狀態欄通知的常見函數、功能及使用技巧,需要的朋友可以參考下
這篇文章主要介紹了Android開發實現模仿微信小窗口功能,結合實例形式分析了Android實現微信風格Dialog對話框窗口相關功能與布局操作技巧,需要的朋友可以參考下
這篇文章主要介紹了Android開發之PopupWindow創建彈窗、對話框的方法,結合實例形式詳細分析了Android使用PopupWindow創建對話框相關操作技巧,需要的朋友可以參考下
這篇文章主要介紹了Android開發之DatePickerDialog、TimePickerDialog時間日期對話框用法,結合實例形式分析了Android使用DatePickerDialog、TimePickerDialog顯示日期時間相關操作技巧,需要的朋友可以參考
主站蜘蛛池模板: 穿线管|波纹穿线管|包塑金属软管|蛇皮管?闵彬专注弱电工程? | 防堵吹扫装置-防堵风压测量装置-电动操作显示器-兴洲仪器 | 泰州物流公司_泰州货运公司_泰州物流专线-东鑫物流公司 | 中视电广_短视频拍摄_短视频推广_短视频代运营_宣传片拍摄_影视广告制作_中视电广 | 小型高低温循环试验箱-可程式高低温湿热交变试验箱-东莞市拓德环境测试设备有限公司 | 防勒索软件_数据防泄密_Trellix(原McAfee)核心代理商_Trellix(原Fireeye)售后-广州文智信息科技有限公司 | 纸布|钩编布|钩针布|纸草布-莱州佳源工艺纸布厂 | 复合土工膜厂家|hdpe防渗土工膜|复合防渗土工布|玻璃纤维|双向塑料土工格栅-安徽路建新材料有限公司 | hdpe土工膜-防渗膜-复合土工膜-长丝土工布价格-厂家直销「恒阳新材料」-山东恒阳新材料有限公司 ETFE膜结构_PTFE膜结构_空间钢结构_膜结构_张拉膜_浙江萬豪空间结构集团有限公司 | 热工多功能信号校验仪-热电阻热电偶校验仿真仪-金湖虹润仪表 | 滚筒烘干机_转筒烘干机_滚筒干燥机_转筒干燥机_回转烘干机_回转干燥机-设备生产厂家 | 线材成型机,线材折弯机,线材成型机厂家,贝朗自动化设备有限公司1 | 耐高温风管_耐高温软管_食品级软管_吸尘管_钢丝软管_卫生级软管_塑料波纹管-东莞市鑫翔宇软管有限公司 | 丹尼克尔拧紧枪_自动送钉机_智能电批_柔性振动盘_螺丝供料器品牌 | 便携式表面粗糙度仪-彩屏硬度计-分体式粗糙度仪-北京凯达科仪科技有限公司 | 西安微信朋友圈广告投放_微信朋友圈推广_西安度娘网络科技有限公司 | 液压油缸-液压缸厂家价格,液压站系统-山东国立液压制造有限公司 液压油缸生产厂家-山东液压站-济南捷兴液压机电设备有限公司 | 水稻烘干机,小麦烘干机,大豆烘干机,玉米烘干机,粮食烘干机_巩义市锦华粮食烘干机械制造有限公司 水环真空泵厂家,2bv真空泵,2be真空泵-淄博真空设备厂 | 山东臭氧发生器,臭氧发生器厂家-山东瑞华环保设备 | 苏商学院官网 - 江苏地区唯一一家企业家自办的前瞻型、实操型商学院 | 法钢特种钢材(上海)有限公司 - 耐磨钢板、高强度钢板销售加工 阀门智能定位器_电液动执行器_气动执行机构-赫尔法流体技术(北京)有限公司 | 防水套管|柔性防水套管|伸缩器|伸缩接头|传力接头-河南伟创管道 防水套管_柔性防水套管_刚性防水套管-巩义市润达管道设备制造有限公司 | 南京技嘉环保科技有限公司-杀菌除臭剂|污水|垃圾|厕所|橡胶厂|化工厂|铸造厂除臭剂 | 无菌水质袋-NASCO食品无菌袋-Whirl-Pak无菌采样袋-深圳市慧普德贸易有限公司 | 九爱图纸|机械CAD图纸下载交流中心 | 砂尘试验箱_淋雨试验房_冰水冲击试验箱_IPX9K淋雨试验箱_广州岳信试验设备有限公司 | 耐酸泵,耐酸泵厂家-淄博华舜耐腐蚀真空泵 | 应急灯_消防应急灯_应急照明灯_应急灯厂家-大成智慧官网 | 户外环保不锈钢垃圾桶_标识标牌制作_园林公园椅厂家_花箱定制-北京汇众环艺 | 工作心得_读书心得_学习心得_找心得体会范文就上学道文库 | YT保温材料_YT无机保温砂浆_外墙保温材料_南阳银通节能建材高新技术开发有限公司 | 全屋整木定制-橱柜,家具定制-四川峨眉山龙马木业有限公司 | 综合管廊模具_生态,阶梯护坡模具_检查井模具制造-致宏模具厂家 | 洗瓶机厂家-酒瓶玻璃瓶冲瓶机-瓶子烘干机-封口旋盖压盖打塞机_青州惠联灌装机械 | 中天寰创-内蒙古钢结构厂家|门式刚架|钢结构桁架|钢结构框架|包头钢结构煤棚 | 拉力测试机|材料拉伸试验机|电子拉力机价格|万能试验机厂家|苏州皖仪实验仪器有限公司 | 南昌旅行社_南昌国际旅行社_南昌国旅在线 | 食品无尘净化车间,食品罐装净化车间,净化车间配套风淋室-青岛旭恒洁净技术有限公司 | 杭州月嫂技术培训服务公司-催乳师培训中心报名费用-产后康复师培训机构-杭州优贝姆健康管理有限公司 | 仿真茅草_人造茅草瓦价格_仿真茅草厂家_仿真茅草供应-深圳市科佰工贸有限公司 | 导电银胶_LED封装导电银胶_半导体封装导电胶厂家-上海腾烁 |