WebView自定義長按彈出菜單

ActionMode

ActionMode上下文操作菜單模式,Android3.0之后出現(xiàn)的一種菜單選擇模式。

ActionMode.Callback

startActionMode方法調(diào)用時啟動,用來配置和處理用戶與動作模式的交互引發(fā)的事件。

  • onCreateActionMode(ActionMode mode, Menu menu)
    • 首次創(chuàng)建操作模式時調(diào)用。 提供的菜單將用于生成操作模式的操作按鈕。
  • onPrepareActionMode(ActionMode mode, Menu menu)
    • 在操作模式無效時刷新操作模式的操作菜單,此時被調(diào)用。
  • onActionItemClicked(ActionMode mode, MenuItem item)
    • 用戶點擊操作按鈕時被調(diào)用。
  • onDestroyActionMode(ActionMode mode)
    • 在即將退出并銷毀動作模式時調(diào)用。

ActionMode.Callback2

繼承ActionMode.Callback。擴展ActionMode.Callback以提供內(nèi)容rect信息。對于具有動態(tài)定位的ActionMode是必需的,例如類型為ActionMode.TYPE_FLOATING的ActionModes,以確保定位不會遮蓋應(yīng)用內(nèi)容。如果應(yīng)用程序無法提供此類的子類,則將使用默認(rèn)實現(xiàn)。

  • onGetContentRect
    • 當(dāng)ActionMode需要定位在屏幕上時調(diào)用,可能會遮擋視圖內(nèi)容。注意,這可以基于每幀調(diào)用

ActionMode.TYPE...

ActionMode模式類型,可以通過setType設(shè)置。

  • TYPE_PRIMARY
    • 設(shè)置ActionMode為基礎(chǔ)模式,為默認(rèn)值
  • TYPE_FLOATING
    • 設(shè)置ActionMode為浮動工具欄模式

實現(xiàn)步驟

  1. WebView實現(xiàn)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) {
                // 獲取網(wǎng)頁中選擇的文本
                getSelectedData((String) item.getTitle());
                // 釋放ActionMode
                releaseAction();
                return true;
            }
        });
    }
    return actionMode;
}
  1. 獲取網(wǎng)頁中選擇的文本,通過JS回傳給原生
/**
 * 獲取網(wǎng)頁中選擇的文本
 * 
 * @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注冊接口,并實現(xiàn)回調(diào)

/**
 * 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回傳給系統(tǒng)
    startActionModereturn修改后的ActionMode。

其他

  1. Activity中相關(guān)回調(diào)
  • onActionModeStarted
    • 會在startActionMode后被調(diào)用
  • onActionModeFinished
    • 菜單消失后被調(diào)用
各系統(tǒng)測試情況
  • 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手機菜單都未被攔截

  • 部分網(wǎng)頁會有長按后只彈出復(fù)制菜單的情況

  • 部分網(wǎng)頁獲取不到選中文本

問題
  • 獲取選中文本是否可以通過系統(tǒng)API拿到?
    • 通過反射WebView.emulateShiftHeld方法,將選中文本復(fù)制到剪貼板;--未成功
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

友情鏈接更多精彩內(nèi)容