javascript - 仅当右键单击以 "Story"开头的类时才显示上下文菜单按钮

标签 javascript google-chrome google-chrome-extension contextmenu

有没有一种方法可以仅在用户右键单击以“故事”开头的类时显示上下文菜单操作。

例如:如果用户右键单击类“story ...”页面中的一个对象,上下文菜单按钮应该出现,否则什么也不会发生。

这是我的代码(虽然它不起作用):

var divs = document.querySelectorAll("[class^=story]"); //get all classes that start with "Story"

    window.oncontextmenu = function() {

        for(var i=0; i < divs.length; i++)
        {
            divs[i].onclick = function() { 
            chrome.contextMenus.create
            (
                {"id": "butto1", "title": "1", "contexts":["all"], "onclick": genericOnClick}
            );
            chrome.contextMenus.create
            (
                {"id": "button2", "title": "2", "contexts":["all"], "onclick": genericOnClick}
            );
            chrome.contextMenus.create
            (
                {"id": "button3", "title": "3", "contexts":["all"], "onclick": genericOnClick}
            );

            };
        }

        return true; 
    };


function genericOnClick(info, tab) {
  //do some crap here
  chrome.contextMenus.removeAll();
}

最佳答案

this related answer ,我解释说不能即时创建上下文菜单项,因为 contextmenu 事件和上下文菜单项出现之间的时间不足以获得 chrome.contextMenus.create。中间调用。

另一个答案解释了如何确保上下文菜单条目显示所选文本。这是通过监听 selectionchange 事件完成的。为了您的目的,我们希望使用具有所需时间的事件。

我将使用 mouseovermouseout 事件。根据鼠标事件,当您使用键盘时,上下文菜单将不起作用,例如通过使用 JavaScript 或 Tab 键聚焦元素,然后按上下文菜单键。我没有在下面的解决方案中实现它。

演示包含三个文件(加上一个用于测试的 HTML 页面)。我将所有文件放在一个 zip 文件中,可在 https://robwu.nl/contextmenu-dom.zip 获得.

list .json

每个 Chrome 扩展都需要这个文件才能工作。对于此应用程序,使用了背景页面和内容脚本。此外,还需要 contextMenus 权限。

{
    "name": "Contextmenu based on activated element",
    "description": "Demo for https://stackoverflow.com/q/14829677",
    "version": "1",
    "manifest_version": 2,
    "background": {
        "scripts": ["background.js"]
    },
    "content_scripts": [{
        "run_at": "document_idle",
        "js": ["contentscript.js"],
        "matches": ["<all_urls>"]
    }],
    "permissions": [
        "contextMenus"
    ]
}

background.js

后台页面将代表内容脚本创建上下文菜单。这是通过将事件监听器绑定(bind)到 chrome.runtime.onConnect 来实现的。 ,它提供了一个简单的界面来替换上下文菜单条目。

chrome.contextMenus.create每当通过此端口(来自内容脚本)收到消息时调用。除了 onclick 之外的所有属性都是 JSON 可序列化的,因此只有“onclick”处理程序需要特殊处理。我使用字符串来标识字典中的预定义函数 (var clickHandlers)。

var lastTabId;
// Remove context menus for a given tab, if needed
function removeContextMenus(tabId) {
    if (lastTabId === tabId) chrome.contextMenus.removeAll();
}
// chrome.contextMenus onclick handlers:
var clickHandlers = {
    'example': function(info, tab) {
        // This event handler receives two arguments, as defined at
        // https://developer.chrome.com/extensions/contextMenus#property-onClicked-callback

        // Example: Notify the tab's content script of something
        // chrome.tabs.sendMessage(tab.id, ...some JSON-serializable data... );

        // Example: Remove contextmenus for context
        removeContextMenus(tab.id);
    }
};

chrome.runtime.onConnect.addListener(function(port) {
    if (!port.sender.tab || port.name != 'contextMenus') {
        // Unexpected / unknown port, do not interfere with it
        return;
    }
    var tabId = port.sender.tab.id;
    port.onDisconnect.addListener(function() {
        removeContextMenus(tabId);
    });
    // Whenever a message is posted, expect that it's identical to type
    // createProperties of chrome.contextMenus.create, except for onclick.
    // "onclick" should be a string which maps to a predefined function
    port.onMessage.addListener(function(newEntries) {
        chrome.contextMenus.removeAll(function() {
            for (var i=0; i<newEntries.length; i++) {
                var createProperties = newEntries[i];
                createProperties.onclick = clickHandlers[createProperties.onclick];
                chrome.contextMenus.create(createProperties);
            }
        });
    });
});

// When a tab is removed, check if it added any context menu entries. If so, remove it
chrome.tabs.onRemoved.addListener(removeContextMenus);

contentscript.js

此脚本的第一部分创建用于创建上下文菜单的简单方法。
在最后 5 行中,上下文菜单条目被定义并绑定(bind)到与给定选择器匹配的所有当前和 future 元素。正如我在上一节中所说,参数类型与 createProperties argument of chrome.contextMenus.create 相同。除了“onclick”,它是一个字符串,映射到后台页面中的一个函数。

// Port management
var _port;
var getPort = function() {
    if (_port) return _port;
    _port = chrome.runtime.connect({name: 'contextMenus'});
    _port.onDisconnect.addListener(function() {
        _port = null;
    });
    return _port;   
}

// listOfCreateProperties is an array of createProperties, which is defined at
// https://developer.chrome.com/extensions/contextMenus#method-create
// with a single exception: "onclick" is a string which corresponds to a function
// at the background page. (Functions are not JSON-serializable, hence this approach)
function addContextMenuTo(selector, listOfCreateProperties) {
    // Selector used to match an element. Match if an element, or its child is hovered
    selector = selector + ', ' + selector + ' *';
    var matches;
    ['matches', 'webkitMatchesSelector', 'webkitMatches', 'matchesSelector'].some(function(m) {
        if (m in document.documentElement) {
            matches = m;
            return true;
        }
    });
    // Bind a single mouseover+mouseout event to catch hovers over all current and future elements.
    var isHovering = false;
    document.addEventListener('mouseover', function(event) {
        if (event.target && event.target[matches](selector)) {
            getPort().postMessage(listOfCreateProperties);
            isHovering = true;
        } else if(isHovering) {
            getPort().postMessage([]);
            isHovering = false;
        }
    });
    document.addEventListener('mouseout', function(event) {
        if (isHovering && (!event.target || !event.target[matches](selector))) {
            getPort().postMessage([]);
            isHovering = false;
        }
    });
}

// Example: Bind the context menus to the elements which contain a class attribute starts with "story"
addContextMenuTo('[class^=story]', [
    {"id": "butto1", "title": "1", "contexts":["all"], "onclick": 'example'},
    {"id": "button2", "title": "2", "contexts":["all"], "onclick": 'example'},
    {"id": "button3", "title": "3", "contexts":["all"], "onclick": 'example'}
]);

前面的代码假定所有上下文菜单点击都由后台页面处理。如果您想改为处理内容脚本中的逻辑,则需要在内容脚本中绑定(bind)消息事件。我已经展示了 chrome.tabs.sendMessage 的一个(注释)实例在 background.js 示例中,显示应在何处触发此事件。

如果您需要确定哪个元素触发了事件,请不要使用我示例中所示的预定义函数(在字典中),而是使用内联函数或工厂函数。要识别元素,消息需要与唯一标识符配对。我会将创建此实现的任务留给读者(这并不困难)。

关于javascript - 仅当右键单击以 "Story"开头的类时才显示上下文菜单按钮,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14829677/

相关文章:

asp.net - 使用最新的 Google Chrome 浏览器时未加载 WebResource.axd

java - Struts 2 和 Chrome 扩展之间的不兼容性

html - 即使使用 meyer 重置,Chrome 和 IE9 之间的 CSS 不一致?

javascript - 如何通过单击按钮将从内容脚本检索到的值存储到文本框中

css - 如何将 CSS 添加到页面到 <head> 自定义 chrome 扩展的底部

javascript - RegEx 在某些情况下不起作用

javascript - 如何从异步调用返回响应?

javascript - 如何配置 TinyMCE 编辑器事件仅触发一次?

javascript - 使用 js 修复两个 fer 程序中的错误

javascript - 源 map 根本不加载