Chrome Extension how to send data from content script to popup.html

100,849

Solution 1

Although you are definitely in the right direction (and actually pretty close to the end), there are several (imo) bad practises in your code (e.g. injecting a whole library (jquery) for such a trivial task, declaring unnecessary permissions, making superflous calls to API methods etc).
I did not test your code myself, but from a quick overview I believe that correcting the following could result in a working solution (although not very close to optimal):

  1. In manifest.json: Change the order of the content scripts, putting jquery first. According to the relevant docs:

"js" [...] The list of JavaScript files to be injected into matching pages. These are injected in the order they appear in this array.

(emphasis mine)

  1. In contentscript.js: Move the chrome.runtime.sendMessage({...}) block inside the onMessage listener callback.

That said, here is my proposed approach:

Control flow:

  1. A content script is injected into each page matching some criteria.
  2. Once injected, the content scripts send a message to the event page (a.k.a. non-persistent background page) and the event page attaches a page-action to the tab.
  3. As soon as the page-action popup is loaded, it sends a message to the content script, asking for the info it needs.
  4. The content script processes the request, and responds so the page-action popup can display the info.

Directory structure:

          root-directory/
           |__img
               |__icon19.png
               |__icon38.png
           |__manifest.json
           |__background.js
           |__content.js
           |__popup.js
           |__popup.html

manifest.json:

{
  "manifest_version": 2,
  "name": "Test Extension",
  "version": "0.0",
  "offline_enabled": true,

  "background": {
    "persistent": false,
    "scripts": ["background.js"]
  },

  "content_scripts": [{
    "matches": ["*://*.stackoverflow.com/*"],
    "js": ["content.js"],
    "run_at": "document_idle",
    "all_frames": false
  }],

  "page_action": {
    "default_title": "Test Extension",
    //"default_icon": {
    //  "19": "img/icon19.png",
    //  "38": "img/icon38.png"
    //},
    "default_popup": "popup.html"
  }

  // No special permissions required...
  //"permissions": []
}

background.js:

chrome.runtime.onMessage.addListener((msg, sender) => {
  // First, validate the message's structure.
  if ((msg.from === 'content') && (msg.subject === 'showPageAction')) {
    // Enable the page-action for the requesting tab.
    chrome.pageAction.show(sender.tab.id);
  }
});

content.js:

// Inform the background page that 
// this tab should have a page-action.
chrome.runtime.sendMessage({
  from: 'content',
  subject: 'showPageAction',
});

// Listen for messages from the popup.
chrome.runtime.onMessage.addListener((msg, sender, response) => {
  // First, validate the message's structure.
  if ((msg.from === 'popup') && (msg.subject === 'DOMInfo')) {
    // Collect the necessary data. 
    // (For your specific requirements `document.querySelectorAll(...)`
    //  should be equivalent to jquery's `$(...)`.)
    var domInfo = {
      total: document.querySelectorAll('*').length,
      inputs: document.querySelectorAll('input').length,
      buttons: document.querySelectorAll('button').length,
    };

    // Directly respond to the sender (popup), 
    // through the specified callback.
    response(domInfo);
  }
});

popup.js:

// Update the relevant fields with the new data.
const setDOMInfo = info => {
  document.getElementById('total').textContent = info.total;
  document.getElementById('inputs').textContent = info.inputs;
  document.getElementById('buttons').textContent = info.buttons;
};

// Once the DOM is ready...
window.addEventListener('DOMContentLoaded', () => {
  // ...query for the active tab...
  chrome.tabs.query({
    active: true,
    currentWindow: true
  }, tabs => {
    // ...and send a request for the DOM info...
    chrome.tabs.sendMessage(
        tabs[0].id,
        {from: 'popup', subject: 'DOMInfo'},
        // ...also specifying a callback to be called 
        //    from the receiving end (content script).
        setDOMInfo);
  });
});

popup.html:

<!DOCTYPE html>
<html>
  <head>
    <script type="text/javascript" src="popup.js"></script>
  </head>
  <body>
    <h3 style="font-weight:bold; text-align:center;">DOM Info</h3>
    <table border="1" cellpadding="3" style="border-collapse:collapse;">
      <tr>
        <td nowrap>Total number of elements:</td>
        <td align="right"><span id="total">N/A</span></td>
      </tr>
      <tr>
        <td nowrap>Number of input elements:</td>
        <td align="right"><span id="inputs">N/A</span></td>
      </tr>
      <tr>
        <td nowrap>Number of button elements:</td>
        <td align="right"><span id="buttons">N/A</span></td>
      </tr>
    </table>
  </body>
</html>

Solution 2

You can use localStorage for that. You can store any data in a hash table format in browser memory and then access it any time. I'm not sure if we can access localStorage from the content script (it was blocked before), try to do it by yourself. Here's how to do it through you background page (I pass data from content script to background page first, then save it in localStorage):

in contentScript.js:

chrome.runtime.sendMessage({
  total_elements: totalElements // or whatever you want to send
});

in eventPage.js (your background page):

chrome.runtime.onMessage.addListener(
    function(request, sender, sendResponse){
       localStorage["total_elements"] = request.total_elements;
    }
);

Then you can access that variable in popup.js with localStorage["total_elements"].

Maybe you can access localStorage directly from the content script in modern browsers. Then you don't need to pass the data through your background page.

Nice reading about localStorage: http://diveintohtml5.info/storage.html

Share:
100,849

Related videos on Youtube

Sumair Baloch
Author by

Sumair Baloch

I am a student studying Software Engineering from Mehran University of Engineering &amp; Technology. I've been into programming for quite sometime, personally love to work on games but I casually work on Web development or designing. Contact me if you need any help.

Updated on December 21, 2021

Comments

  • Sumair Baloch
    Sumair Baloch over 2 years

    I know this this has been asked in numerous posts but honestly I don't get them. I am new to JavaScript, Chrome Extensions and everything and I have this class assignment. So I need to make a plugin that would count DOM objects on any given page using Cross Domain Requests. I've been able to achieve this so far using Chrome Extension API's. Now the problem is I need to show the data on my popup.html page from the contentScript.js file. I don't know how to do that I've tried reading the documentation but messaging in chrome I just can't understand what to do.

    following is the code so far.

    manifest.json

    {
    "manifest_version":2,
    
    "name":"Dom Reader",
    "description":"Counts Dom Objects",
    "version":"1.0",
    
    "page_action": {
        "default_icon":"icon.png",
        "default_title":"Dom Reader",
        "default_popup":"popup.html"
    },
    
    "background":{
        "scripts":["eventPage.js"],
        "persistent":false
    },
    
    "content_scripts":[
        {
            "matches":["http://pluralsight.com/training/Courses/*", "http://pluralsight.com/training/Authors/Details/*",                                          "https://www.youtube.com/user/*", "https://sites.google.com/site/*", "http://127.0.0.1:3667/popup.html"],
            "js":["domReader_cs.js","jquery-1.10.2.js"]
            //"css":["pluralsight_cs.css"]
        }
    ],
    
    "permissions":[
        "tabs",
        "http://pluralsight.com/*",
        "http://youtube.com/*",
        "https://sites.google.com/*",
        "http://127.0.0.1:3667/*"
    ]
    

    popup.html

    <!doctype html>
    <html>
    
        <title> Dom Reader </title>    
        <script src="jquery-1.10.2.js" type="text/javascript"></script>
        <script src="popup.js" type="text/javascript"></script>
    
    <body>
        <H1> Dom Reader </H1>
        <input type="submit" id="readDom" value="Read DOM Objects" />
    
       <div id="domInfo">
    
        </div>
    </body>
    </html>
    

    eventPage.js

    var value1,value2,value3;
    
    chrome.runtime.onMessage.addListener(function (request, sender, sendResponse) {
    if (request.action == "show") {
        chrome.tabs.query({ active: true, currentWindow: true }, function (tabs) {
            chrome.pageAction.show(tabs[0].id);
        });
    }
    
    value1 = request.tElements;
    });
    

    popup.js

    $(function (){
    $('#readDom').click(function(){
    chrome.tabs.query({active: true, currentWindow: true}, function (tabs){
        chrome.tabs.sendMessage(tabs[0].id, {action: "readDom"});
    
     });
    });
    });
    

    contentScript

    var totalElements;
    var inputFields;
    var buttonElement;
    
    chrome.runtime.onMessage.addListener(function (request, sender, sendResponse){
    if(request.action == "readDom"){
    
        totalElements = $("*").length;
        inputFields = $("input").length;
        buttonElement = $("button").length;
    
    
    }
    })
    
    chrome.runtime.sendMessage({ 
    action: "show", 
    tElements: totalElements, 
    Ifields: inputFields, 
    bElements: buttonElement 
    
    });
    

    Any help would be appreciated and please avoid any noobness I did :)

  • gkalpak
    gkalpak over 10 years
    Please, do not promote the use of the deprecated chrome.extension.onRequest/sendRequest (which btw do not load a non-persistent background page before executing). Use chrome.runtime.* instead.
  • Sumair Baloch
    Sumair Baloch over 10 years
    Wow ! thanks brother, checking your approach now I feel how many problems I created with my code. Thanks alot this works like a charm. :)
  • Sumair Baloch
    Sumair Baloch over 10 years
    hello, Sorry I haven't been on for some time, just checked on your comment. I can't accept any answer. it says you need 15 reputation points to do so.
  • Xan
    Xan about 10 years
    @ExpertSystem Please, if you see such outdated information, suggest/make an edit.
  • gkalpak
    gkalpak about 10 years
    @Xan: So, you are the successor in [Google-Chrome-Extension] :) I don't like editing people's posts (I find it too intrusive). Besides, with so many outdated tutorials and examples (at least back then), I found it better to have an explicit comment on why it is bad to use .extension.xxx, instead of just showing the right way (which some people might think of as an "equally OK alternative"). It's more a matter of style I guess. Keep up the good work !
  • wuno
    wuno over 8 years
    Would you mind helping me solve a very similar problem? <a href="stackoverflow.com/questions/34467627/…> My current shot at this is pretty much your code from this answer. But I still cant get it to work and cant seem to find any good help on the matter .
  • Habib Kazemi
    Habib Kazemi over 8 years
    why you used chrome.tabs.sendMessage in popupjs and chrome.runtime.onMessage.addListener in content.js why .tabs for popupjs and .runtime in content.js
  • gkalpak
    gkalpak over 8 years
    @hkm: Because that is how you send and receive messages. It's all in the docs.
  • DanGordon
    DanGordon almost 8 years
    So one thing - does the background.js/content.js parse the DOM before you click the extension? It seems to me that it does. I want to do the same thing here - but only parse the DOM if the user clicks the extension - but I still want the extension to be grayed out when not at the correct site. Would that require some extra message passing - or does your code already do this? It seems that the popup.js will send a message any time the DOM is loaded - but really I only want popup.js to send a message when the user clicks.
  • Farbod Aprin
    Farbod Aprin over 4 years
    The popup didn't came up in my case! I need help :D