WebView自定义长按弹出菜单

ActionMode

ActionMode上下文操作菜单模式,Android3.0之后出现的一种菜单选择模式。

ActionMode.Callback

startActionMode方法调用时启动,用来配置和处理用户与动作模式的交互引发的事件。

  • onCreateActionMode(ActionMode mode, Menu menu)
    • 首次创建操作模式时调用。 提供的菜单将用于生成操作模式的操作按钮。
  • onPrepareActionMode(ActionMode mode, Menu menu)
    • 在操作模式无效时刷新操作模式的操作菜单,此时被调用。
  • onActionItemClicked(ActionMode mode, MenuItem item)
    • 用户点击操作按钮时被调用。
  • onDestroyActionMode(ActionMode mode)
    • 在即将退出并销毁动作模式时调用。

ActionMode.Callback2

继承ActionMode.Callback。扩展ActionMode.Callback以提供内容rect信息。对于具有动态定位的ActionMode是必需的,例如类型为ActionMode.TYPE_FLOATING的ActionModes,以确保定位不会遮盖应用内容。如果应用程序无法提供此类的子类,则将使用默认实现。

  • onGetContentRect
    • 当ActionMode需要定位在屏幕上时调用,可能会遮挡视图内容。注意,这可以基于每帧调用

ActionMode.TYPE...

ActionMode模式类型,可以通过setType设置。

  • TYPE_PRIMARY
    • 设置ActionMode为基础模式,为默认值
  • TYPE_FLOATING
    • 设置ActionMode为浮动工具栏模式

实现步骤

  1. WebView实现startActionMode方法,拦截ActionMode
@Override
public ActionMode startActionMode(ActionMode.Callback callback) {
    ActionMode actionMode = super.startActionMode(callback);
    return resolveActionMode(actionMode);
}

@Override
public ActionMode startActionMode(ActionMode.Callback callback, int type) 
    ActionMode actionMode =  super.startActionMode(callback, type);
    return resolveActionMode(actionMode);
}
  1. 重定义ActionModeMenuItem
private ActionMode mActionMode;
private List<String> mActionList = new ArrayList<String>() {
    {
        add("菜单1");
        add("菜单2");
        add("菜单3");
    }
};

/**
 * 重定义ActionMode中的MenuItem
 * 
 * @return 拥有新MenuItem的ActionMode 
 */
private ActionMode resolveActionMode(ActionMode actionMode) {
    if (actionMode == null) {
        mActionMode = null;
        return;
    }
    // 获取并清除原菜单
    final Menu menu = actionMode.getMenu();
    mActionMode = actionMode;
    menu.clear();
    // 添加新菜单项
    for (int i = 0; i < mActionList.size(); i++) {
        menu.add(mActionList.get(i));
    }
    // 为新菜单项注册点击事件
    for (int i = 0; i < menu.size(); i++) {
        MenuItem menuItem = menu.getItem(i);
        menuItem.setOnMenuItemClickListener(new MenuItem.OnMenuItemClickListener() {
            @Override
            public boolean onMenuItemClick(MenuItem item) {
                // 获取网页中选择的文本
                getSelectedData((String) item.getTitle());
                // 释放ActionMode
                releaseAction();
                return true;
            }
        });
    }
    return actionMode;
}
  1. 获取网页中选择的文本,通过JS回传给原生
/**
 * 获取网页中选择的文本
 * 
 * @param title 传入点击的item文本,通过js返回传给原生
 */
private void getSelectedData(String title) {
    String js = "(function getSelectedText() {" +
            "var txt;" +
            "var title = \"" + title + "\";" +
            "if (window.getSelection) {" +
            "txt = window.getSelection().toString();" +
            "} else if (window.document.getSelection) {" +
            "txt = window.document.getSelection().toString();" +
            "} else if (window.document.selection) {" +
            "txt = window.document.selection.createRange().text;" +
            "}" +
            "JSInterface.callback(txt,title);" +
            "})()";
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        evaluateJavascript("javascript:" + js, null);
    } else {
        loadUrl("javascript:" + js);
    }
}

通过addJavascriptInterface()WebView注册接口,并实现回调

/**
 * ActionMode原生与JS交互接口
 */
public class ActionModeWebBridge implements INoProGuard {

    public ActionModeWebBridge() {
        
    }

    @JavascriptInterface
    public void callback(final String value, final String title) {
        // do something...
    }

}
webView.addJavascriptInterface(new ActionModeWebBridge(), "JSInterface");
  1. 释放ActionMode
private void releaseAction() {
    if (mActionMode != null) {
        mActionMode.finish();
        mActionMode = null;
    }
}
  1. 将修改后的ActionMode回传给系统
    startActionModereturn修改后的ActionMode

其他

  1. Activity中相关回调
  • onActionModeStarted
    • 会在startActionMode后被调用
  • onActionModeFinished
    • 菜单消失后被调用
各系统测试情况
  • HUAWEI CRR-UL00 6.0 正常

  • TCL P588L 5.0.2 正常

  • vivo V3Max A 5.1.1 X

  • Honor 8 Lite 8.0.0 正常

  • vivo Y51 5.0.2 X

  • OPPO A57 6.0.1 正常

  • 两台vivo手机菜单都未被拦截

  • 部分网页会有长按后只弹出复制菜单的情况

  • 部分网页获取不到选中文本

问题
  • 获取选中文本是否可以通过系统API拿到?
    • 通过反射WebView.emulateShiftHeld方法,将选中文本复制到剪贴板;--未成功
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,233评论 6 495
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,357评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,831评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,313评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,417评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,470评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,482评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,265评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,708评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,997评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,176评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,503评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,150评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,391评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,034评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,063评论 2 352