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)步驟
- 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);
}
- 重定義
ActionMode中MenuItem
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;
}
- 獲取網(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");
- 釋放
ActionMode
private void releaseAction() {
if (mActionMode != null) {
mActionMode.finish();
mActionMode = null;
}
}
- 將修改后的
ActionMode回傳給系統(tǒng)
startActionMode中return修改后的ActionMode。
其他
-
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ù)制到剪貼板;--未成功
- 通過反射