在清单v3扩展中正确使用chrome.tabCapture

ev7lccsx  于 2022-12-06  发布在  Go
关注(0)|答案(3)|浏览(378)

Edit:

As the end of the year and the end of Manifest V2 is approaching I did a bit more research on this and found the following workarounds:

The problem with this approach is that it requires the user to select a capture source via some UI which can be disruptive. The --auto-select-desktop-capture-source command line switch can apparently be used to bypass this but I haven't been able to use it with success.

  • The example extension here that works around tabCapture not working in service workers by creating its own inactive tab from which to access the tabCapture API and record the currently active tab:

https://github.com/zhw2590582/chrome-audio-capture
So far this seems to be the best solution I've found so far in terms of UX. The background page provided in Manifest V2 is essentially replaced with a phantom tab.
The roundaboutedness of the second solution also seems to suggest that the tabCapture API is essentially not intended for use in Manifest V3, or else there would have been a more straightforward way to use it. I am disappointed that Manifest V3 is being enforced while essentially leaving behind Manifest V2 features such as this one.

Original Post:

I'm trying to write a manifest v3 Chrome extension that captures tab audio. However as far as I can tell, with manifest v3 there are some changes that make this a bit difficult:

  • Background scripts are replaced by service workers.
  • Service workers do not have access to the chrome.tabCapture API.

Despite this I managed to get something that nearly works as popup scripts still have access to chrome.tabCapture . However, there is a drawback - the audio of the tab is muted and there doesn't seem to be a way to unmute it. This is what I have so far:

  1. Query the service worker current tab from the popup script.
let tabId;

// Fetch tab immediately
chrome.runtime.sendMessage({command: 'query-active-tab'}, (response) => {
    tabId = response.id;
});

This is the service worker, which response with the current tab ID.

chrome.runtime.onMessage.addListener(
    (request, sender, sendResponse) => {
        // Popup asks for current tab
        if (request.command === 'query-active-tab') {
            chrome.tabs.query({active: true}, (tabs) => {
                if (tabs.length > 0) {
                    sendResponse({id: tabs[0].id});
                }
            });

            return true;
        }
        ...
  1. Again in the popup script, from a keyboard shortcut command, use chrome.tabCapture.getMediaStreamId to get a media stream ID to be consumed by the current tab, and send that stream ID back to the service worker.
// On command, get the stream ID and forward it back to the service worker
chrome.commands.onCommand.addListener((command) => {
    chrome.tabCapture.getMediaStreamId({consumerTabId: tabId}, (streamId) => {
        chrome.runtime.sendMessage({
            command: 'tab-media-stream',
            tabId: tabId,
            streamId: streamId
        })
    });
});
  1. The service worker forwards that stream ID to the content script.
chrome.runtime.onMessage.addListener(
    (request, sender, sendResponse) => {
        ...
        // Popup sent back media stream ID, forward it to the content script
        if (request.command === 'tab-media-stream') {
            chrome.tabs.sendMessage(request.tabId, {
                command: 'tab-media-stream',
                streamId: request.streamId
            });
        }
    }
);
  1. The content script uses navigator.mediaDevices.getUserMedia to get the stream.
// Service worker sent us the stream ID, use it to get the stream
chrome.runtime.onMessage.addListener((request, sender, sendResponse) => {
    navigator.mediaDevices.getUserMedia({
        video: false,
        audio: true,
        audio: {
            mandatory: {
                chromeMediaSource: 'tab',
                chromeMediaSourceId: request.streamId
            }
        }
    })
    .then((stream) => {
        // Once we're here, the audio in the tab is muted
        // However, recording the audio works!
        const recorder = new MediaRecorder(stream);
        const chunks = [];
        recorder.ondataavailable = (e) => {
            chunks.push(e.data);
        };
        recorder.onstop = (e) => saveToFile(new Blob(chunks), "test.wav");
        recorder.start();
        setTimeout(() => recorder.stop(), 5000);
    });
});

Here is the code that implements the above: https://github.com/killergerbah/-test-tab-capture-extension
This actually does produce a MediaStream , but the drawback is that the sound of the tab is muted. I've tried playing the stream through an audio element, but that seems to do nothing.

  • Is there a way to obtain a stream of the tab audio in a manifest v3 extension without muting the audio in the tab?*

I suspect that this approach might be completely wrong as it's so roundabout, but this is the best I could come up with after reading through the docs and various StackOverflow posts. I've also read that the tabCapture API is going to be moved for manifest v3 at some point, so maybe the question doesn't even make sense to ask - however if there is a way to still properly use it I would like to know.

lokaqttq

lokaqttq1#

我发现你的帖子在我的音频标签录音机的实现进展非常有用。
关于您遇到的特定静音问题,我通过查看以下内容解决了该问题:Original audio of tab gets muted while using chrome.tabCapture.capture() and MediaRecorder()

// Service worker sent us the stream ID, use it to get the stream
chrome.runtime.onMessage.addListener((request, sender, sendResponse) => {
    navigator.mediaDevices.getUserMedia({
        video: false,
        audio: true,
        audio: {
            mandatory: {
                chromeMediaSource: 'tab',
                chromeMediaSourceId: request.streamId
            }
        }
    })
    .then((stream) => {
        // To resolve original audio muting
        context = new AudioContext();
        var audio = context.createMediaStreamSource(stream);
        audio.connect(context.destination);

        const recorder = new MediaRecorder(stream);
        const chunks = [];
        recorder.ondataavailable = (e) => {
            chunks.push(e.data);
        };
        recorder.onstop = (e) => saveToFile(new Blob(chunks), "test.wav");
        recorder.start();
        setTimeout(() => recorder.stop(), 5000);
    });
});
ruyhziif

ruyhziif2#

这可能不完全是你正在寻找的,但也许它可以提供一些见解。
我尝试过通过音频元素播放流,但似乎没有任何效果。
具有讽刺意味的是,这就是我如何设法绕过这个问题;在弹出窗口脚本中使用tabCapture时,它返回流,我将音频srcObject设置为该流。
HTML:<audio id="audioObject" autoplay> No source detected </audio>
JS:

chrome.tabCapture.capture({audio: true, video: false}, function(stream) {
    var audio = document.getElementById("audioObject");
    audio.srcObject = stream
})

根据Manifest V3上的this post,chrome.capture将成为tabCapture等的新名称空间,但除此之外我还没有看到任何东西。

bnl4lu3b

bnl4lu3b3#

我也遇到过这个问题,我通过使用Web音频API解决了它。只需创建一个新的上下文,并使用捕获MediaStream将其连接到媒体流源,这是一个示例:

avoidSilenceInTab: (desktopStream: MediaStream) => {
 var contextTab = new AudioContext();
 contextTab
   .createMediaStreamSource(desktopStream)
   .connect(contextTab.destination);

}

相关问题