Browse Source

fix: `desktopCapturer` and `screen` display ids should match (#42891)

* fix: `desktopCapturer` and `screen` display IDs should match

Co-authored-by: clavin <[email protected]>

* simplify wide-to-utf8 conversion

Co-authored-by: clavin <[email protected]>

* remove unnecessary include

Co-authored-by: clavin <[email protected]>

---------

Co-authored-by: trop[bot] <37223003+trop[bot]@users.noreply.github.com>
Co-authored-by: clavin <[email protected]>
trop[bot] 9 months ago
parent
commit
c0e69d0fc6
2 changed files with 24 additions and 16 deletions
  1. 21 12
      shell/browser/api/electron_api_desktop_capturer.cc
  2. 3 4
      spec/api-screen-spec.ts

+ 21 - 12
shell/browser/api/electron_api_desktop_capturer.cc

@@ -399,28 +399,37 @@ void DesktopCapturer::UpdateSourcesList(DesktopMediaList* list) {
     if (using_directx_capturer_) {
       std::vector<std::string> device_names;
       // Crucially, this list of device names will be in the same order as
-      // |media_list_sources|.
+      // |screen_sources|.
       if (!webrtc::DxgiDuplicatorController::Instance()->GetDeviceNames(
               &device_names)) {
         HandleFailure();
         return;
       }
+      DCHECK_EQ(device_names.size(), screen_sources.size());
 
       std::vector<HMONITOR> monitors;
       EnumDisplayMonitors(nullptr, nullptr, EnumDisplayMonitorsCallback,
                           reinterpret_cast<LPARAM>(&monitors));
 
-      std::vector<std::pair<std::string, MONITORINFOEX>> pairs;
-      for (const auto& device_name : device_names) {
-        std::wstring wide_device_name;
-        base::UTF8ToWide(device_name.c_str(), device_name.size(),
-                         &wide_device_name);
-        for (const auto monitor : monitors) {
-          MONITORINFOEX monitorInfo{{sizeof(MONITORINFOEX)}};
-          if (GetMonitorInfo(monitor, &monitorInfo)) {
-            if (wide_device_name == monitorInfo.szDevice)
-              pairs.push_back(std::make_pair(device_name, monitorInfo));
-          }
+      base::flat_map<std::string, int64_t> device_name_to_id;
+      device_name_to_id.reserve(monitors.size());
+      for (auto* monitor : monitors) {
+        MONITORINFOEX monitorInfo{{sizeof(MONITORINFOEX)}};
+        if (!GetMonitorInfo(monitor, &monitorInfo)) {
+          continue;
+        }
+
+        device_name_to_id[base::WideToUTF8(monitorInfo.szDevice)] =
+            display::win::internal::DisplayInfo::DisplayIdFromMonitorInfo(
+                monitorInfo);
+      }
+
+      int device_name_index = 0;
+      for (auto& source : screen_sources) {
+        const auto& device_name = device_names[device_name_index++];
+        if (auto id_iter = device_name_to_id.find(device_name);
+            id_iter != device_name_to_id.end()) {
+          source.display_id = base::NumberToString(id_iter->second);
         }
       }
     }

+ 3 - 4
spec/api-screen-spec.ts

@@ -1,6 +1,5 @@
 import { expect } from 'chai';
 import { Display, screen, desktopCapturer } from 'electron/main';
-import { ifit } from './lib/spec-helpers';
 
 describe('screen module', () => {
   describe('methods reassignment', () => {
@@ -24,14 +23,14 @@ describe('screen module', () => {
       }
     });
 
-    // desktopCapturer.getSources does not work as expected in Windows CI.
-    ifit(process.platform !== 'win32')('returns displays with IDs matching desktopCapturer source display IDs', async () => {
+    it('returns displays with IDs matching desktopCapturer source display IDs', async () => {
       const displayIds = screen.getAllDisplays().map(d => `${d.id}`);
 
       const sources = await desktopCapturer.getSources({ types: ['screen'] });
       const sourceIds = sources.map(s => s.display_id);
 
-      expect(displayIds).to.have.members(sourceIds);
+      expect(displayIds).to.have.length(sources.length);
+      expect(displayIds).to.have.same.members(sourceIds);
     });
   });