如何在 Chrome 扩展中使用 promises?

How do I use promises in a Chrome extension?

我想做的是创建一个 chrome 扩展,使用 promises.

创建新的嵌套书签文件夹

执行此操作的函数是 chrome.bookmarks.create()。但是我不能只是 循环此函数,因为 chrome.bookmarks.create 是异步的。我需要等到创建文件夹并获取其新 ID,然后再继续其子项。

承诺似乎是必经之路。不幸的是,我找不到使用带有自己回调的异步调用的最小工作示例,例如 chrome.bookmarks.create.

我已经阅读了一些教程1, 2, 3, 4。我搜索了 Whosebug,但所有问题似乎都不是关于 chrome 扩展库的普通承诺。

我不想使用插件或库:没有 node.js 或 jquery 或 Q 或其他。

我尝试按照教程中的示例进行操作,但很多事情都没有意义。例如,教程指出:

The promise constructor takes one argument—a callback with two parameters: resolve and reject.

但后来我看到了这样的例子:

const wait = ms => new Promise(resolve => setTimeout(resolve, ms));

这对我来说是个谜。

此外,当 resolve() 从未被定义时,您如何调用它?教程中的示例似乎与现实生活中的代码不符。另一个例子是:

function isUserTooYoung(id) {
   return openDatabase() // returns a promise   
.then(function(col) {return find(col, {'id': id});})

如何传入 col 或得到任何结果!

因此,如果有人能给我一个带有自己的回调的异步函数的承诺的最小工作示例,将不胜感激。

SO 想要代码,所以这是我的无效尝试:

//loop through all
function createBookmarks(nodes, parentid){  

  var jlen = nodes.length;
  var i;
  var node;
  for(var i = 0; i < nodes.length; i++){
        var node = nodes[i];

        createBookmark(node, parentid);
  }
}

//singular create
function createBookmark(node, parentid){  
      var bookmark = { 
    parentId : parentid,
    index : node['index'],
    title : node['title'],
    url : node['url']
  }

  var callback = function(result){
    console.log("creation callback happened.");
    return result.id;  //pass ID to the callback, too
  }

    var promise = new Promise(function(resolve, reject) {

        var newid = chrome.bookmarks.create(bookmark, callback)

      if (newid){
        console.log("Creating children with new id: " + newid);
        resolve( createBookmarks(bookmark.children, newid));
      }

    });
}

//allnodes already exists
createBookmarks(allnodes[0],"0");

只是行不通。回调的结果始终是未定义的,它应该是未定义的,而且我看不出 promise 对象如何改变任何东西。当我尝试使用 promise.then() 时,我同样感到困惑。

var newid = promise.then(  //wait for a response?
            function(result){
            return chrome.bookmarks.create(bookmark, callback); 
            }
        ).catch(function(error){
            console.log("error " + error);
        });  

        if (node.children) createBookmarks(node.children, newid);

同样,newid 总是未定义的,因为当然 bookmarks.create() 是异步的。

感谢您提供的任何帮助。

老实说,你应该只使用 web extension polyfill。手动承诺 chrome API 既浪费时间又容易出错。

如果您绝对坚持,这就是您如何承诺 chrome.bookmarks.create 的示例。对于其他 chrome.* API,您还必须拒绝回调的错误参数。

function createBookmark(bookmark) {
  return new Promise(function(resolve, reject) {
    try {
      chrome.bookmarks.create(bookmark, function (result) {
        if (chrome.runtime.lastError) reject(chrome.runtime.lastError)
        else resolve(result)
      })
    } catch (error) {
      reject(error)
    }
  })
}


createBookmark({})
  .then(function (result) {
     console.log(result)
   }).catch(function (error) {
     console.log(error)
   })

要创建多个书签,您可以:

function createBookmarks(bookmarks) {
  return Promise.all(
    bookmarks.map(function (bookmark) {
      return createBookmark(bookmark)
    })
  )
}

createBookmarks([{}, {}, {}, {}])
  .catch(function (error) {
    console.log(error)
  })

利用回调函数总是最后一个参数的惯例,我使用一个简单的辅助函数来 promisify chrome API:

function toPromise(api) {
  return (...args) => {
    return new Promise((resolve) => {
      api(...args, resolve);
    });
  };
}

并像这样使用它:

toPromise(chrome.bookmarks.create)(bookmark).then(...);

在我的用例中,它大部分时间都能正常工作。