Android修改6.0系统文本选择菜单

96
简名
0.5 2017.02.22 22:47* 字数 1063

因为我身患流行性绝症(拖延症),不知不觉中博客已经有半年没更新了 ( ⊙ o ⊙ )。

正好最近在写文本选择菜单的功能,就整理出了一篇文章,算是我与该死的拖延症展开的殊死搏斗吧。

本文主要写了在Android6.0+版本上,修改EditText、TextView的文本选择菜单内容和为其他APP提供自定义文本操作这两个功能。

1.修改EditText和TextView的文本选择菜单内容

Android官方控件中,EditText中的文字默认长按呼出文本选择菜单,而TextView需要设置android:textIsSelectable="true"。

我们修改文本选择菜单内容,只需要为TextView或者EditText设置setCustomSelectionActionModeCallback()方法,并且在方法里实现ActionMode.Callback()或ActionMode.Callback2()接口。

ActionMode.Callback()和ActionMode.Callback2()接口的主要内容相同,只是Callback2中多了一个onGetContentRect()方法,重写可以改变弹出菜单的位置。另外,Callback2需要判断sdk23及以上版本,Callback()不用,但是在6.0以下系统中实际是无效的。

我们先在res下的menu文件夹里新建一个菜单文件,我把它命名为selection_action_menu.xml,内容如下

    <?xml version="1.0" encoding="utf-8"?>
    <menu
 xmlns:android="http://schemas.android.com/apk/res/android">
    <item
        android:id="@+id/Informal22"
        android:title="自定义22" />
    <item
        android:id="@+id/Informal33"
        android:title="自定义33" />
 </menu>

然后我们完成文本选择菜单的修改

    ActionMode.Callback2 textSelectionActionModeCallback;
    if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.M) {
        textSelectionActionModeCallback = new ActionMode.Callback2() {
            @Override
            public boolean onCreateActionMode(ActionMode actionMode, Menu menu) {
                MenuInflater menuInflater = actionMode.getMenuInflater();
                menuInflater.inflate(R.menu.selection_action_menu,menu);
                return true;//返回false则不会显示弹窗
            }

            @Override
            public boolean onPrepareActionMode(ActionMode actionMode, Menu menu) {
                return false;
            }

            @Override
            public boolean onActionItemClicked(ActionMode actionMode, MenuItem menuItem) {
                //根据item的ID处理点击事件
                switch (menuItem.getItemId()){
                    case R.id.Informal22:
                        Toast.makeText(MainActivity.this, "点击的是22", Toast.LENGTH_SHORT).show();
                        actionMode.finish();//收起操作菜单
                        break;
                    case R.id.Informal33:
                        Toast.makeText(MainActivity.this, "点击的是33", Toast.LENGTH_SHORT).show();
                        actionMode.finish();
                        break;
                }
                return false;//返回true则系统的"复制"、"搜索"之类的item将无效,只有自定义item有响应
            }

            @Override
            public void onDestroyActionMode(ActionMode actionMode) {

            }

            @Override
            public void onGetContentRect(ActionMode mode, View view, Rect outRect) {
                //可选  用于改变弹出菜单的位置
                super.onGetContentRect(mode, view, outRect);
            }
        };
    }
增加自定义item
增加自定义item

上面这样写的话会有系统的“复制”、“全选”、“搜索”等item,以及其他APP提供的操作item,如果你(或者你们产品经理)傲娇的想要屏蔽所有非本APP自定义item,可以改成这样

  @Override
  public boolean onCreateActionMode(ActionMode actionMode, Menu menu) {
        return true;//返回false则不会显示弹窗
  }

  @Override
  public boolean onPrepareActionMode(ActionMode actionMode, Menu menu) {
         MenuInflater menuInflater = actionMode.getMenuInflater();
         menu.clear();
         menuInflater.inflate(R.menu.selection_action_menu,menu);
         return true;
  }
只有自定义item

然后我们为TextView和EditText设置文本选择操作回调

  textView.setCustomSelectionActionModeCallback(textSelectionActionModeCallback);
  editText.setCustomSelectionActionModeCallback(textSelectionActionModeCallback);

EditText在还有一个插入操作菜单可以设置

  edittext.setCustomInsertionActionModeCallback(textSelectionActionModeCallback);
EditText插入操作菜单

MIUI上小米重写了EditText和TextView中的文本选择菜单,上面这些代码修改无效。好吧,算你狠。 = =、

2.为其他APP提供文本选择操作

一般剪切板、翻译、搜索、记事本类APP会提供这种功能,为用户提供便捷操作,同时也刷下自己的存在感。

我先建了一个继承自Activity的CustomTextProcessingActivity,简单写了下布局,我这因为拖延症的常见并发症——懒癌发作,代码就不贴出来了。

然后按照基本法到清单文件里去注册一下

<activity android:name=".CustomTextProcessingActivity">
    <intent-filter>
        <action android:name="android.intent.action.PROCESS_TEXT"/>
        <category android:name="android.intent.category.DEFAULT"/>
        <data android:mimeType="text/plain"/>
    </intent-filter>
</activity>

这里可以为activity自定义主题和标签,标签值就是文本选择操作菜单中出现的item名。默认就是APP名称"MY APPLICATION"。如果你想让这个操作只出现在本APP中,可以设置android:exported=”false”。

回到CustomTextProcessingActivity中,获取用户选择的文本只需要这样

 CharSequence text = getIntent().getCharSequenceExtra(Intent.EXTRA_PROCESS_TEXT);

我们还可以判断文本是否可编辑

 boolean isReadonly = getIntent().getBooleanExtra(Intent.EXTRA_PROCESS_TEXT_READONLY, false);

我们点击“MY APPLICATION”的时候,其实系统是通过是 startActivityForResult() 来启动的CustomTextProcessingActivity,所以我们可以返回一个文本。当选中的文本可编辑时(如EditText中的文字),系统会替换文字;不可编辑时,系统会通过toast的方式展示出来。

   //返回一个文本
   Intent intent = new Intent();
   intent.putExtra(Intent.EXTRA_PROCESS_TEXT, "这是替换的文字");
   setResult(RESULT_OK, intent);

这个界面写成弹窗的样式感觉更好,看大家的具体设计了。

在我一加的氢OS上,EditText在setCustomSelectionActionModeCallback()之后就只有第一部分中自定义的item以及系统的“复制”等基础item,本APP和其他的三方APP提供的操作item都无法显示。华为上是不显示第二部分中APP提供的操作item,其他的正常显示。而一加和华为上的TextView都是正常的。

至于小米,EditText和TextView不会显示三方APP提供的操作item。但是WebView中选择文字弹出来的还是原生文本选择操作菜单,有三方APP的操作item,估计是小米程序猿漏掉了


这篇文章周末就开始写了,现在的时间是周三。嗯...还行,拖得时间不算太长...

欢迎拖延症病友交流病情。。。。

Android
Web note ad 1