Chrome扩展名:背景JS偶尔无法从弹出js接收消息



弹出窗口上有一个按钮。单击时,popup.js将向background.js发送消息。它几乎一直都很好。但是很长一段时间没有操作chrome后,当我单击按钮时,发送消息时(警报("开始发送消息")执行),但无法通过Background.js( alter)收到("启动开始渲染")未执行)。当我再单击一个时,有时会起作用。如果它仍然不起作用,我将再次单击,第三次有更多成功的机会。这很奇怪。

我在popup.html中使用jQuery,但我认为这并不重要。popup.js

中的JavaScript代码
$('#create).html(chrome.i18n.getMessage('create'))
.on('click', function(){
    _current = null;
    chrome.runtime.sendMessage({
        cmd: 'SITEMAP_DEFINE'
    });
    alert("Start to send message");
    window.close();

})

背景中的JavaScript代码

var curTabId; //current tab in which scraper is going to scraping
var devtoolPort;
var coord;
var sitemap;
var finish = false;

function sendMessage(msg) {
  chrome.tabs.query({active: true, currentWindow: true}, function(tabs) {
    chrome.tabs.sendMessage(tabs[0].id, msg);
  });
}
//detective devtools
chrome.runtime.onConnect.addListener(function (port) {
    if (port.name == "devtools-papa") {
        devtoolPort = port;
    }
});
//tab update
chrome.tabs.onUpdated.addListener(function(tabId, changeInfo, tab) {
    if(finish === false && coord && changeInfo.status == 'loading'){
        coord.onUpdate(tab);
    }
})
chrome.tabs.onRemoved.addListener(function(tabId, removeInfo) {
    if(finish === false && coord && removeInfo.isWindowClosing === false){
        coord.onRemove(tabId);
    }
})
chrome.runtime.onMessage.addListener(function(msg, sender, callback){
    var p = msg.params;
    //scrape
    if(msg.cmd == 'START_SCRAPE'){
        sitemap = p.sitemap;
        if(sitemap){
            finish = false;
            coord = new Coordinator(sitemap, 0);
            var page = coord.sitemap.pages[0];
            coord.openTab(page.id, page.link);
        }
    }
    else if(msg.cmd == 'TEST_SCRAPE'){
        sitemap = p.sitemap;
        if(sitemap){
            chrome.tabs.query({active: true, currentWindow: true}, function(tabs) {
                var tab = tabs[0];
                sitemap.pages[0].link = tab.url;
                finish = false;
                coord = new Coordinator(sitemap, 20);
                var page = coord.sitemap.pages[0];
                coord.openTab(page.id, page.link);
            })
        }
    }
    else if(msg.cmd == 'SCRAPE_READY'){
        print(sender);
        coord.readyForScrape(sender.tab.id);
    }
    else if(msg.cmd == 'TAB_CLICK'){
        coord.currentPage = sitemap.getPage(p.page);
    }
    //define
    else if(msg.cmd == 'START_SELECT'){
        if(msg.tabId){
            chrome.tabs.sendMessage(msg.tabId, msg);
        }else{
            sendMessage(msg);
        }
    }
    else if(msg.cmd == 'END_SELECT'){
        if(devtoolPort){
            devtoolPort.postMessage(msg);
        }
        sendMessage(msg);
    }
    else if(msg.cmd == 'SITEMAP_DEFINE'){
        alert("Start to render");
        var url = p && p.url;
        sitemap = p && p.sitemap;
        chrome.tabs.query({active: true, currentWindow: true}, function(tabs) {
            var tab = tabs[0];
            if(url && url != tab.url){
                chrome.tabs.create({url: url}, function(newTab){
                    doInjectScript(newTab.id);
                })
            }else{
                injectScript(tab.id);
            }
        });
    }
    else if(msg.cmd == 'SITEMAP_DEFINE_END'){
        var _sm = p.sitemap;
        chrome.tabs.query({active: true, currentWindow: true}, function(tabs) {
            var tab = tabs[0];
            _sm.pages[0].link = tab.url;
            saveSitemap(_sm);
        })
    }
    else if(msg.cmd == 'SITEMAP_DELETE'){
        deleteSitemap(p.sitemap);
    }
    else if(msg.cmd == 'DEFINE_PAGE_LOADED'){
        callback(sitemap);
    }
    else{
        sendMessage(msg);
    }
});
var print = function(msg){
    if(curTabId){
        chrome.tabs.sendMessage(curTabId, {
            cmd: 'LOG',
            params: msg
        });
    }else{
        sendMessage(msg)
    }
}
function saveSitemap(_sm){
    var id = _sm.id;
    chrome.storage.local.get("sitemaps", function(db){
        var _sitemaps = db["sitemaps"];
        if(!_sitemaps || !_sitemaps.length){
            _sitemaps = [_sm];
        }else{
            var _idx = _sitemaps.findIndex(function(s){return s.id == id});
            if(_idx == -1){
                _sitemaps.push(_sm);
            }else{
                _sitemaps.splice(_idx, 1, _sm);
            }
        }
        var _db = {'sitemaps':_sitemaps}
        chrome.storage.local.set(_db, function(){
            alert(chrome.i18n.getMessage('saved'));
        });
    })
}
function deleteSitemap(id){
    chrome.storage.local.get("sitemaps", function(db){
        var _sitemaps = db["sitemaps"];
        var index = _sitemaps.findIndex(function(s){return s.id == id})
        _sitemaps.splice(index, 1);
        var _db = {'sitemaps':_sitemaps}
        chrome.storage.local.set(_db, function(){
            alert(chrome.i18n.getMessage('deleted'));
        });
    })
}
function injectScript(tabId){
    var flag;
    chrome.tabs.sendMessage(tabId, {
        cmd: 'SELF_CHECK'
    }, function(r){
        flag = r;
    });
    setTimeout(function(){
        if(flag){
            chrome.tabs.executeScript(tabId, {file: "injected/define/initUI.js"}, function() {});
        }else{
            doInjectScript(tabId);
        }
    }, 200)
}
//Injected scripts will be removed once the tab reload again.
function doInjectScript(tabId){
    chrome.tabs.executeScript(tabId, {file: "injected/listener.js"});
    chrome.tabs.executeScript(tabId, {file: "lib/jquery/jquery-2.1.1.min.js"}, function() {
        chrome.tabs.executeScript(tabId, {file: "lib/model.js"});
        chrome.tabs.executeScript(tabId, {file: "lib/css-selector-generator.js"}, function() {});
        chrome.tabs.executeScript(tabId, {file: "injected/SelectTool.js"}, function() {});
        chrome.tabs.executeScript(tabId, {file: "injected/define/initUI.js"}, function() {});
        chrome.tabs.insertCSS(tabId, {file: "injected/injected.css", runAt: "document_end"});
    });
}

function createResultTab(title, fields, data){
    chrome.tabs.create({url: chrome.extension.getURL('result/table.html')}, function(newTab){
        var tabId = newTab.id;
        setTimeout(function(){
            chrome.tabs.sendMessage(tabId, {
                cmd: 'RESULT',
                params:{
                    data: data,
                    title: title,
                    fields: fields
                }
            });
        }, 200)
    })
}

事先感谢您的帮助。

我在官方文档中找到答案。如果我们设置持久 = false,则背景JS会在闲置时卸载。

{
  "name": "My extension",
  ...
  "background": {
    "scripts": ["eventPage.js"],
    "persistent": false
  },
  ...
}

该文档告诉从内容脚本发送消息将导致背景JS重新加载。但是我想弹出脚本的消息有问题。

因此,要么更改持续 true chrome.rome.runtime.getBackgroundpage 在弹出窗口中引起背景JS Reload将解决此问题。

最新更新