2016-05-30 18:08:09 +05:30
|
|
|
// Copyright (c) 2016 GitHub, Inc.
|
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2019-06-19 13:46:59 -07:00
|
|
|
#include "shell/browser/lib/bluetooth_chooser.h"
|
2016-05-30 18:08:09 +05:30
|
|
|
|
2019-10-25 22:03:28 +09:00
|
|
|
#include "shell/common/gin_converters/callback_converter.h"
|
|
|
|
#include "shell/common/gin_helper/dictionary.h"
|
|
|
|
|
|
|
|
namespace gin {
|
2016-05-30 18:08:09 +05:30
|
|
|
|
2018-04-17 21:55:30 -04:00
|
|
|
template <>
|
2019-06-19 14:23:04 -07:00
|
|
|
struct Converter<electron::BluetoothChooser::DeviceInfo> {
|
2016-05-30 18:08:09 +05:30
|
|
|
static v8::Local<v8::Value> ToV8(
|
2018-04-17 21:55:30 -04:00
|
|
|
v8::Isolate* isolate,
|
2019-06-19 14:23:04 -07:00
|
|
|
const electron::BluetoothChooser::DeviceInfo& val) {
|
2023-08-21 03:43:41 +02:00
|
|
|
auto dict = gin_helper::Dictionary::CreateEmpty(isolate);
|
2016-05-30 18:08:09 +05:30
|
|
|
dict.Set("deviceName", val.device_name);
|
|
|
|
dict.Set("deviceId", val.device_id);
|
2019-10-25 22:03:28 +09:00
|
|
|
return gin::ConvertToV8(isolate, dict);
|
2016-05-30 18:08:09 +05:30
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-10-25 22:03:28 +09:00
|
|
|
} // namespace gin
|
2016-05-30 18:08:09 +05:30
|
|
|
|
2019-06-19 14:23:04 -07:00
|
|
|
namespace electron {
|
2016-05-30 18:08:09 +05:30
|
|
|
|
2018-04-17 21:55:30 -04:00
|
|
|
BluetoothChooser::BluetoothChooser(api::WebContents* contents,
|
|
|
|
const EventHandler& event_handler)
|
2018-05-22 00:18:38 +02:00
|
|
|
: api_web_contents_(contents), event_handler_(event_handler) {}
|
2016-05-30 18:08:09 +05:30
|
|
|
|
2021-02-26 14:10:27 -05:00
|
|
|
BluetoothChooser::~BluetoothChooser() {
|
|
|
|
event_handler_.Reset();
|
|
|
|
}
|
2016-05-30 18:08:09 +05:30
|
|
|
|
|
|
|
void BluetoothChooser::SetAdapterPresence(AdapterPresence presence) {
|
|
|
|
switch (presence) {
|
|
|
|
case AdapterPresence::ABSENT:
|
2025-04-25 15:01:12 -05:00
|
|
|
NOTREACHED();
|
2016-05-30 18:08:09 +05:30
|
|
|
case AdapterPresence::POWERED_OFF:
|
2020-10-15 18:30:41 -07:00
|
|
|
event_handler_.Run(content::BluetoothChooserEvent::CANCELLED, "");
|
2016-05-30 18:08:09 +05:30
|
|
|
break;
|
2025-04-25 15:01:12 -05:00
|
|
|
case AdapterPresence::UNAUTHORIZED:
|
|
|
|
event_handler_.Run(content::BluetoothChooserEvent::DENIED_PERMISSION, "");
|
|
|
|
break;
|
2016-05-30 18:08:09 +05:30
|
|
|
case AdapterPresence::POWERED_ON:
|
2021-06-09 10:48:18 -04:00
|
|
|
rescan_ = true;
|
2016-05-30 18:08:09 +05:30
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void BluetoothChooser::ShowDiscoveryState(DiscoveryState state) {
|
2023-03-27 09:31:15 -04:00
|
|
|
bool idle_state = false;
|
2016-05-30 18:08:09 +05:30
|
|
|
switch (state) {
|
|
|
|
case DiscoveryState::FAILED_TO_START:
|
2021-02-26 14:10:27 -05:00
|
|
|
refreshing_ = false;
|
2020-10-15 18:30:41 -07:00
|
|
|
event_handler_.Run(content::BluetoothChooserEvent::CANCELLED, "");
|
2023-03-27 09:31:15 -04:00
|
|
|
return;
|
2016-05-30 18:08:09 +05:30
|
|
|
case DiscoveryState::IDLE:
|
2021-02-26 14:10:27 -05:00
|
|
|
refreshing_ = false;
|
2023-03-27 09:31:15 -04:00
|
|
|
idle_state = true;
|
2016-05-30 18:08:09 +05:30
|
|
|
break;
|
2025-04-25 15:01:12 -05:00
|
|
|
// The first time this state fires is due to a rescan triggering so we
|
|
|
|
// set a flag to ignore devices - the second time this state fires
|
|
|
|
// we are now safe to pick a device.
|
2016-05-30 18:08:09 +05:30
|
|
|
case DiscoveryState::DISCOVERING:
|
2021-06-09 10:48:18 -04:00
|
|
|
if (rescan_ && !refreshing_) {
|
2021-02-26 14:10:27 -05:00
|
|
|
refreshing_ = true;
|
|
|
|
} else {
|
|
|
|
refreshing_ = false;
|
|
|
|
}
|
2016-05-30 18:08:09 +05:30
|
|
|
break;
|
|
|
|
}
|
2025-04-25 15:01:12 -05:00
|
|
|
|
2023-03-27 09:31:15 -04:00
|
|
|
bool prevent_default =
|
|
|
|
api_web_contents_->Emit("select-bluetooth-device", GetDeviceList(),
|
2025-04-25 15:01:12 -05:00
|
|
|
base::BindOnce(&BluetoothChooser::OnDeviceChosen,
|
|
|
|
weak_ptr_factory_.GetWeakPtr()));
|
2023-03-27 09:31:15 -04:00
|
|
|
if (!prevent_default && idle_state) {
|
2025-04-25 15:01:12 -05:00
|
|
|
if (device_id_to_name_map_.empty()) {
|
2023-03-27 09:31:15 -04:00
|
|
|
event_handler_.Run(content::BluetoothChooserEvent::CANCELLED, "");
|
|
|
|
} else {
|
2025-04-25 15:01:12 -05:00
|
|
|
auto it = device_id_to_name_map_.begin();
|
2023-03-27 09:31:15 -04:00
|
|
|
auto device_id = it->first;
|
|
|
|
event_handler_.Run(content::BluetoothChooserEvent::SELECTED, device_id);
|
|
|
|
}
|
|
|
|
}
|
2016-05-30 18:08:09 +05:30
|
|
|
}
|
|
|
|
|
2016-11-30 16:30:03 +09:00
|
|
|
void BluetoothChooser::AddOrUpdateDevice(const std::string& device_id,
|
|
|
|
bool should_update_name,
|
2021-03-16 12:18:45 -04:00
|
|
|
const std::u16string& device_name,
|
2016-11-30 16:30:03 +09:00
|
|
|
bool is_gatt_connected,
|
|
|
|
bool is_paired,
|
|
|
|
int signal_strength_level) {
|
2025-04-25 15:01:12 -05:00
|
|
|
// Don't fire an event during refresh.
|
|
|
|
if (refreshing_)
|
2021-02-26 14:10:27 -05:00
|
|
|
return;
|
2023-07-13 03:16:25 -05:00
|
|
|
|
2025-04-25 15:01:12 -05:00
|
|
|
// Emit a select-bluetooth-device handler to allow for user to listen for
|
|
|
|
// bluetooth device found. If there's no listener in place, then select the
|
|
|
|
// first device that matches the filters provided.
|
|
|
|
auto [iter, changed] =
|
|
|
|
device_id_to_name_map_.try_emplace(device_id, device_name);
|
2023-07-13 03:16:25 -05:00
|
|
|
if (!changed && should_update_name) {
|
|
|
|
iter->second = device_name;
|
2018-11-28 08:36:00 -08:00
|
|
|
changed = true;
|
2017-12-18 20:46:06 +10:00
|
|
|
}
|
2016-05-30 18:08:09 +05:30
|
|
|
|
2018-11-28 08:36:00 -08:00
|
|
|
if (changed) {
|
2019-05-02 16:49:27 -07:00
|
|
|
bool prevent_default = api_web_contents_->Emit(
|
|
|
|
"select-bluetooth-device", GetDeviceList(),
|
2025-04-25 15:01:12 -05:00
|
|
|
base::BindOnce(&BluetoothChooser::OnDeviceChosen,
|
|
|
|
weak_ptr_factory_.GetWeakPtr()));
|
2018-11-28 08:36:00 -08:00
|
|
|
|
2025-04-25 15:01:12 -05:00
|
|
|
if (!prevent_default)
|
2020-10-15 18:30:41 -07:00
|
|
|
event_handler_.Run(content::BluetoothChooserEvent::SELECTED, device_id);
|
2025-04-25 15:01:12 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void BluetoothChooser::OnDeviceChosen(const std::string& device_id) {
|
|
|
|
if (event_handler_.is_null())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (device_id.empty()) {
|
|
|
|
event_handler_.Run(content::BluetoothChooserEvent::CANCELLED, device_id);
|
|
|
|
} else {
|
|
|
|
event_handler_.Run(content::BluetoothChooserEvent::SELECTED, device_id);
|
2016-05-30 18:08:09 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-19 14:23:04 -07:00
|
|
|
std::vector<electron::BluetoothChooser::DeviceInfo>
|
2018-11-28 08:36:00 -08:00
|
|
|
BluetoothChooser::GetDeviceList() {
|
2019-06-19 14:23:04 -07:00
|
|
|
std::vector<electron::BluetoothChooser::DeviceInfo> vec;
|
2025-04-25 15:01:12 -05:00
|
|
|
vec.reserve(device_id_to_name_map_.size());
|
|
|
|
for (const auto& [device_id, device_name] : device_id_to_name_map_)
|
2023-07-13 03:16:25 -05:00
|
|
|
vec.emplace_back(device_id, device_name);
|
2018-11-28 08:36:00 -08:00
|
|
|
return vec;
|
|
|
|
}
|
|
|
|
|
2019-06-19 14:23:04 -07:00
|
|
|
} // namespace electron
|