| OLD | NEW |
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "modules/bluetooth/BluetoothRemoteGATTCharacteristic.h" | 5 #include "modules/bluetooth/BluetoothRemoteGATTCharacteristic.h" |
| 6 | 6 |
| 7 #include "bindings/core/v8/ScriptPromise.h" | 7 #include "bindings/core/v8/ScriptPromise.h" |
| 8 #include "bindings/core/v8/ScriptPromiseResolver.h" | 8 #include "bindings/core/v8/ScriptPromiseResolver.h" |
| 9 #include "core/dom/DOMDataView.h" | 9 #include "core/dom/DOMDataView.h" |
| 10 #include "core/dom/DOMException.h" | 10 #include "core/dom/DOMException.h" |
| 11 #include "core/dom/ExceptionCode.h" | 11 #include "core/dom/ExceptionCode.h" |
| 12 #include "core/events/Event.h" | 12 #include "core/events/Event.h" |
| 13 #include "core/inspector/ConsoleMessage.h" | 13 #include "core/inspector/ConsoleMessage.h" |
| 14 #include "modules/bluetooth/Bluetooth.h" |
| 14 #include "modules/bluetooth/BluetoothCharacteristicProperties.h" | 15 #include "modules/bluetooth/BluetoothCharacteristicProperties.h" |
| 16 #include "modules/bluetooth/BluetoothDevice.h" |
| 15 #include "modules/bluetooth/BluetoothError.h" | 17 #include "modules/bluetooth/BluetoothError.h" |
| 16 #include "modules/bluetooth/BluetoothRemoteGATTService.h" | 18 #include "modules/bluetooth/BluetoothRemoteGATTService.h" |
| 17 #include "modules/bluetooth/BluetoothSupplement.h" | |
| 18 #include "public/platform/modules/bluetooth/WebBluetooth.h" | |
| 19 #include <memory> | 19 #include <memory> |
| 20 | 20 |
| 21 namespace blink { | 21 namespace blink { |
| 22 | 22 |
| 23 namespace { | 23 namespace { |
| 24 | 24 |
| 25 const char kGATTServerDisconnected[] = | 25 const char kGATTServerDisconnected[] = |
| 26 "GATT Server disconnected while performing a GATT operation."; | 26 "GATT Server disconnected while performing a GATT operation."; |
| 27 const char kGATTServerNotConnected[] = | 27 const char kGATTServerNotConnected[] = |
| 28 "GATT Server is disconnected. Cannot perform GATT operations."; | 28 "GATT Server is disconnected. Cannot perform GATT operations."; |
| 29 const char kInvalidCharacteristic[] = | 29 const char kInvalidCharacteristic[] = |
| 30 "Characteristic is no longer valid. Remember to retrieve the " | 30 "Characteristic is no longer valid. Remember to retrieve the " |
| 31 "characteristic again after reconnecting."; | 31 "characteristic again after reconnecting."; |
| 32 | 32 |
| 33 DOMDataView* ConvertWebVectorToDataView(const WebVector<uint8_t>& webVector) { | 33 DOMDataView* ConvertWTFVectorToDataView(const Vector<uint8_t>& wtfVector) { |
| 34 static_assert(sizeof(*webVector.data()) == 1, | 34 static_assert(sizeof(*wtfVector.data()) == 1, |
| 35 "uint8_t should be a single byte"); | 35 "uint8_t should be a single byte"); |
| 36 DOMArrayBuffer* domBuffer = | 36 DOMArrayBuffer* domBuffer = |
| 37 DOMArrayBuffer::create(webVector.data(), webVector.size()); | 37 DOMArrayBuffer::create(wtfVector.data(), wtfVector.size()); |
| 38 return DOMDataView::create(domBuffer, 0, webVector.size()); | 38 return DOMDataView::create(domBuffer, 0, wtfVector.size()); |
| 39 } | 39 } |
| 40 | 40 |
| 41 } // anonymous namespace | 41 } // anonymous namespace |
| 42 | 42 |
| 43 BluetoothRemoteGATTCharacteristic::BluetoothRemoteGATTCharacteristic( | 43 BluetoothRemoteGATTCharacteristic::BluetoothRemoteGATTCharacteristic( |
| 44 ExecutionContext* context, | 44 ExecutionContext* context, |
| 45 std::unique_ptr<WebBluetoothRemoteGATTCharacteristicInit> webCharacteristic, | 45 const String& characteristicInstanceId, |
| 46 BluetoothRemoteGATTService* service) | 46 const String& serviceInstanceId, |
| 47 const String& uuid, |
| 48 uint32_t characteristicProperties, |
| 49 BluetoothRemoteGATTService* service, |
| 50 BluetoothDevice* device) |
| 47 : SuspendableObject(context), | 51 : SuspendableObject(context), |
| 48 m_webCharacteristic(std::move(webCharacteristic)), | 52 m_characteristicInstanceId(characteristicInstanceId), |
| 53 m_serviceInstanceId(serviceInstanceId), |
| 54 m_uuid(uuid), |
| 55 m_characteristicProperties(characteristicProperties), |
| 49 m_service(service), | 56 m_service(service), |
| 50 m_stopped(false) { | 57 m_stopped(false), |
| 51 m_properties = BluetoothCharacteristicProperties::create( | 58 m_device(device) { |
| 52 m_webCharacteristic->characteristicProperties); | 59 m_properties = |
| 60 BluetoothCharacteristicProperties::create(m_characteristicProperties); |
| 53 } | 61 } |
| 54 | 62 |
| 55 BluetoothRemoteGATTCharacteristic* BluetoothRemoteGATTCharacteristic::create( | 63 BluetoothRemoteGATTCharacteristic* BluetoothRemoteGATTCharacteristic::create( |
| 56 ExecutionContext* context, | 64 ExecutionContext* context, |
| 57 std::unique_ptr<WebBluetoothRemoteGATTCharacteristicInit> webCharacteristic, | 65 const String& characteristicInstanceId, |
| 58 BluetoothRemoteGATTService* service) { | 66 const String& serviceInstanceId, |
| 59 DCHECK(webCharacteristic); | 67 const String& uuid, |
| 60 | 68 uint32_t characteristicProperties, |
| 69 BluetoothRemoteGATTService* service, |
| 70 BluetoothDevice* device) { |
| 61 BluetoothRemoteGATTCharacteristic* characteristic = | 71 BluetoothRemoteGATTCharacteristic* characteristic = |
| 62 new BluetoothRemoteGATTCharacteristic( | 72 new BluetoothRemoteGATTCharacteristic( |
| 63 context, std::move(webCharacteristic), service); | 73 context, characteristicInstanceId, serviceInstanceId, uuid, |
| 74 characteristicProperties, service, device); |
| 64 // See note in SuspendableObject about suspendIfNeeded. | 75 // See note in SuspendableObject about suspendIfNeeded. |
| 65 characteristic->suspendIfNeeded(); | 76 characteristic->suspendIfNeeded(); |
| 66 return characteristic; | 77 return characteristic; |
| 67 } | 78 } |
| 68 | 79 |
| 69 void BluetoothRemoteGATTCharacteristic::setValue(DOMDataView* domDataView) { | 80 void BluetoothRemoteGATTCharacteristic::setValue(DOMDataView* domDataView) { |
| 70 m_value = domDataView; | 81 m_value = domDataView; |
| 71 } | 82 } |
| 72 | 83 |
| 73 void BluetoothRemoteGATTCharacteristic::dispatchCharacteristicValueChanged( | 84 void BluetoothRemoteGATTCharacteristic::dispatchCharacteristicValueChanged( |
| 74 const WebVector<uint8_t>& value) { | 85 const Vector<uint8_t>& value) { |
| 75 this->setValue(ConvertWebVectorToDataView(value)); | 86 this->setValue(ConvertWTFVectorToDataView(value)); |
| 76 dispatchEvent(Event::create(EventTypeNames::characteristicvaluechanged)); | 87 dispatchEvent(Event::create(EventTypeNames::characteristicvaluechanged)); |
| 77 } | 88 } |
| 78 | 89 |
| 79 void BluetoothRemoteGATTCharacteristic::contextDestroyed() { | 90 void BluetoothRemoteGATTCharacteristic::contextDestroyed() { |
| 80 notifyCharacteristicObjectRemoved(); | 91 notifyCharacteristicObjectRemoved(); |
| 81 } | 92 } |
| 82 | 93 |
| 83 void BluetoothRemoteGATTCharacteristic::dispose() { | 94 void BluetoothRemoteGATTCharacteristic::dispose() { |
| 84 notifyCharacteristicObjectRemoved(); | 95 notifyCharacteristicObjectRemoved(); |
| 85 } | 96 } |
| 86 | 97 |
| 87 void BluetoothRemoteGATTCharacteristic::notifyCharacteristicObjectRemoved() { | 98 void BluetoothRemoteGATTCharacteristic::notifyCharacteristicObjectRemoved() { |
| 88 if (!m_stopped) { | 99 if (!m_stopped) { |
| 89 m_stopped = true; | 100 m_stopped = true; |
| 90 WebBluetooth* webbluetooth = BluetoothSupplement::fromExecutionContext( | 101 m_device->bluetooth()->characteristicObjectRemoved( |
| 91 SuspendableObject::getExecutionContext()); | 102 m_characteristicInstanceId); |
| 92 webbluetooth->characteristicObjectRemoved( | |
| 93 m_webCharacteristic->characteristicInstanceID, this); | |
| 94 } | 103 } |
| 95 } | 104 } |
| 96 | 105 |
| 97 const WTF::AtomicString& BluetoothRemoteGATTCharacteristic::interfaceName() | 106 const WTF::AtomicString& BluetoothRemoteGATTCharacteristic::interfaceName() |
| 98 const { | 107 const { |
| 99 return EventTargetNames::BluetoothRemoteGATTCharacteristic; | 108 return EventTargetNames::BluetoothRemoteGATTCharacteristic; |
| 100 } | 109 } |
| 101 | 110 |
| 102 ExecutionContext* BluetoothRemoteGATTCharacteristic::getExecutionContext() | 111 ExecutionContext* BluetoothRemoteGATTCharacteristic::getExecutionContext() |
| 103 const { | 112 const { |
| 104 return SuspendableObject::getExecutionContext(); | 113 return SuspendableObject::getExecutionContext(); |
| 105 } | 114 } |
| 106 | 115 |
| 107 void BluetoothRemoteGATTCharacteristic::addedEventListener( | 116 void BluetoothRemoteGATTCharacteristic::addedEventListener( |
| 108 const AtomicString& eventType, | 117 const AtomicString& eventType, |
| 109 RegisteredEventListener& registeredListener) { | 118 RegisteredEventListener& registeredListener) { |
| 110 EventTargetWithInlineData::addedEventListener(eventType, registeredListener); | 119 EventTargetWithInlineData::addedEventListener(eventType, registeredListener); |
| 111 // We will also need to unregister a characteristic once all the event | 120 // We will also need to unregister a characteristic once all the event |
| 112 // listeners have been removed. See http://crbug.com/541390 | 121 // listeners have been removed. See http://crbug.com/541390 |
| 113 if (eventType == EventTypeNames::characteristicvaluechanged) { | 122 if (eventType == EventTypeNames::characteristicvaluechanged) { |
| 114 WebBluetooth* webbluetooth = | 123 m_device->bluetooth()->registerCharacteristicObject( |
| 115 BluetoothSupplement::fromExecutionContext(getExecutionContext()); | 124 m_characteristicInstanceId, this); |
| 116 webbluetooth->registerCharacteristicObject( | |
| 117 m_webCharacteristic->characteristicInstanceID, this); | |
| 118 } | 125 } |
| 119 } | 126 } |
| 120 | 127 |
| 121 class ReadValueCallback : public WebBluetoothReadValueCallbacks { | 128 void BluetoothRemoteGATTCharacteristic::ReadValueCallback( |
| 122 public: | 129 ScriptPromiseResolver* resolver, |
| 123 ReadValueCallback(BluetoothRemoteGATTCharacteristic* characteristic, | 130 mojom::blink::WebBluetoothResult result, |
| 124 ScriptPromiseResolver* resolver) | 131 const Optional<Vector<uint8_t>>& value) { |
| 125 : m_characteristic(characteristic), m_resolver(resolver) { | 132 if (!resolver->getExecutionContext() || |
| 126 // We always check that the device is connected before constructing this | 133 resolver->getExecutionContext()->isContextDestroyed()) |
| 127 // object. | 134 return; |
| 128 CHECK(m_characteristic->gatt()->connected()); | 135 |
| 129 m_characteristic->gatt()->AddToActiveAlgorithms(m_resolver.get()); | 136 // If the resolver is not in the set of ActiveAlgorithms then the frame |
| 137 // disconnected so we reject. |
| 138 if (!gatt()->RemoveFromActiveAlgorithms(resolver)) { |
| 139 resolver->reject( |
| 140 DOMException::create(NetworkError, kGATTServerDisconnected)); |
| 141 return; |
| 130 } | 142 } |
| 131 | 143 |
| 132 void onSuccess(const WebVector<uint8_t>& value) override { | 144 if (result == mojom::blink::WebBluetoothResult::SUCCESS) { |
| 133 if (!m_resolver->getExecutionContext() || | 145 DCHECK(value); |
| 134 m_resolver->getExecutionContext()->isContextDestroyed()) | 146 DOMDataView* domDataView = ConvertWTFVectorToDataView(value.value()); |
| 135 return; | 147 setValue(domDataView); |
| 136 | 148 resolver->resolve(domDataView); |
| 137 if (!m_characteristic->gatt()->RemoveFromActiveAlgorithms( | 149 } else { |
| 138 m_resolver.get())) { | 150 resolver->reject(BluetoothError::take(resolver, result)); |
| 139 m_resolver->reject( | |
| 140 DOMException::create(NetworkError, kGATTServerDisconnected)); | |
| 141 return; | |
| 142 } | |
| 143 | |
| 144 DOMDataView* domDataView = ConvertWebVectorToDataView(value); | |
| 145 if (m_characteristic) | |
| 146 m_characteristic->setValue(domDataView); | |
| 147 | |
| 148 m_resolver->resolve(domDataView); | |
| 149 } | 151 } |
| 150 | 152 } |
| 151 void onError( | |
| 152 int32_t | |
| 153 error /* Corresponds to WebBluetoothResult in web_bluetooth.mojom */) | |
| 154 override { | |
| 155 if (!m_resolver->getExecutionContext() || | |
| 156 m_resolver->getExecutionContext()->isContextDestroyed()) | |
| 157 return; | |
| 158 | |
| 159 if (!m_characteristic->gatt()->RemoveFromActiveAlgorithms( | |
| 160 m_resolver.get())) { | |
| 161 m_resolver->reject( | |
| 162 DOMException::create(NetworkError, kGATTServerDisconnected)); | |
| 163 return; | |
| 164 } | |
| 165 | |
| 166 m_resolver->reject(BluetoothError::take(m_resolver, error)); | |
| 167 } | |
| 168 | |
| 169 private: | |
| 170 Persistent<BluetoothRemoteGATTCharacteristic> m_characteristic; | |
| 171 Persistent<ScriptPromiseResolver> m_resolver; | |
| 172 }; | |
| 173 | 153 |
| 174 ScriptPromise BluetoothRemoteGATTCharacteristic::readValue( | 154 ScriptPromise BluetoothRemoteGATTCharacteristic::readValue( |
| 175 ScriptState* scriptState) { | 155 ScriptState* scriptState) { |
| 156 // We always check that the device is connected. |
| 176 if (!gatt()->connected()) { | 157 if (!gatt()->connected()) { |
| 177 return ScriptPromise::rejectWithDOMException( | 158 return ScriptPromise::rejectWithDOMException( |
| 178 scriptState, | 159 scriptState, |
| 179 DOMException::create(NetworkError, kGATTServerNotConnected)); | 160 DOMException::create(NetworkError, kGATTServerNotConnected)); |
| 180 } | 161 } |
| 181 | 162 |
| 182 if (!gatt()->device()->isValidCharacteristic( | 163 if (!gatt()->device()->isValidCharacteristic(m_characteristicInstanceId)) { |
| 183 m_webCharacteristic->characteristicInstanceID)) { | |
| 184 return ScriptPromise::rejectWithDOMException( | 164 return ScriptPromise::rejectWithDOMException( |
| 185 scriptState, | 165 scriptState, |
| 186 DOMException::create(InvalidStateError, kInvalidCharacteristic)); | 166 DOMException::create(InvalidStateError, kInvalidCharacteristic)); |
| 187 } | 167 } |
| 188 | 168 |
| 189 WebBluetooth* webbluetooth = | |
| 190 BluetoothSupplement::fromScriptState(scriptState); | |
| 191 | |
| 192 ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); | 169 ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); |
| 193 ScriptPromise promise = resolver->promise(); | 170 ScriptPromise promise = resolver->promise(); |
| 194 webbluetooth->readValue(m_webCharacteristic->characteristicInstanceID, | 171 gatt()->AddToActiveAlgorithms(resolver); |
| 195 new ReadValueCallback(this, resolver)); | 172 |
| 173 mojom::blink::WebBluetoothService* service = m_device->bluetooth()->service(); |
| 174 service->RemoteCharacteristicReadValue( |
| 175 m_characteristicInstanceId, |
| 176 convertToBaseCallback( |
| 177 WTF::bind(&BluetoothRemoteGATTCharacteristic::ReadValueCallback, |
| 178 wrapPersistent(this), wrapPersistent(resolver)))); |
| 196 | 179 |
| 197 return promise; | 180 return promise; |
| 198 } | 181 } |
| 199 | 182 |
| 200 class WriteValueCallback : public WebBluetoothWriteValueCallbacks { | 183 void BluetoothRemoteGATTCharacteristic::WriteValueCallback( |
| 201 public: | 184 ScriptPromiseResolver* resolver, |
| 202 WriteValueCallback(BluetoothRemoteGATTCharacteristic* characteristic, | 185 const Vector<uint8_t>& value, |
| 203 ScriptPromiseResolver* resolver) | 186 mojom::blink::WebBluetoothResult result) { |
| 204 : m_characteristic(characteristic), m_resolver(resolver) { | 187 if (!resolver->getExecutionContext() || |
| 205 // We always check that the device is connected before constructing this | 188 resolver->getExecutionContext()->isContextDestroyed()) |
| 206 // object. | 189 return; |
| 207 CHECK(m_characteristic->gatt()->connected()); | 190 |
| 208 m_characteristic->gatt()->AddToActiveAlgorithms(m_resolver.get()); | 191 // If the resolver is not in the set of ActiveAlgorithms then the frame |
| 192 // disconnected so we reject. |
| 193 if (!gatt()->RemoveFromActiveAlgorithms(resolver)) { |
| 194 resolver->reject( |
| 195 DOMException::create(NetworkError, kGATTServerDisconnected)); |
| 196 return; |
| 209 } | 197 } |
| 210 | 198 |
| 211 void onSuccess(const WebVector<uint8_t>& value) override { | 199 if (result == mojom::blink::WebBluetoothResult::SUCCESS) { |
| 212 if (!m_resolver->getExecutionContext() || | 200 setValue(ConvertWTFVectorToDataView(value)); |
| 213 m_resolver->getExecutionContext()->isContextDestroyed()) | 201 resolver->resolve(); |
| 214 return; | 202 } else { |
| 215 | 203 resolver->reject(BluetoothError::take(resolver, result)); |
| 216 if (!m_characteristic->gatt()->RemoveFromActiveAlgorithms( | |
| 217 m_resolver.get())) { | |
| 218 m_resolver->reject( | |
| 219 DOMException::create(NetworkError, kGATTServerDisconnected)); | |
| 220 return; | |
| 221 } | |
| 222 | |
| 223 if (m_characteristic) { | |
| 224 m_characteristic->setValue(ConvertWebVectorToDataView(value)); | |
| 225 } | |
| 226 m_resolver->resolve(); | |
| 227 } | 204 } |
| 228 | 205 } |
| 229 void onError( | |
| 230 int32_t | |
| 231 error /* Corresponds to WebBluetoothResult in web_bluetooth.mojom */) | |
| 232 override { | |
| 233 if (!m_resolver->getExecutionContext() || | |
| 234 m_resolver->getExecutionContext()->isContextDestroyed()) | |
| 235 return; | |
| 236 | |
| 237 if (!m_characteristic->gatt()->RemoveFromActiveAlgorithms( | |
| 238 m_resolver.get())) { | |
| 239 m_resolver->reject( | |
| 240 DOMException::create(NetworkError, kGATTServerDisconnected)); | |
| 241 return; | |
| 242 } | |
| 243 | |
| 244 m_resolver->reject(BluetoothError::take(m_resolver, error)); | |
| 245 } | |
| 246 | |
| 247 private: | |
| 248 Persistent<BluetoothRemoteGATTCharacteristic> m_characteristic; | |
| 249 Persistent<ScriptPromiseResolver> m_resolver; | |
| 250 }; | |
| 251 | 206 |
| 252 ScriptPromise BluetoothRemoteGATTCharacteristic::writeValue( | 207 ScriptPromise BluetoothRemoteGATTCharacteristic::writeValue( |
| 253 ScriptState* scriptState, | 208 ScriptState* scriptState, |
| 254 const DOMArrayPiece& value) { | 209 const DOMArrayPiece& value) { |
| 210 // We always check that the device is connected. |
| 255 if (!gatt()->connected()) { | 211 if (!gatt()->connected()) { |
| 256 return ScriptPromise::rejectWithDOMException( | 212 return ScriptPromise::rejectWithDOMException( |
| 257 scriptState, | 213 scriptState, |
| 258 DOMException::create(NetworkError, kGATTServerNotConnected)); | 214 DOMException::create(NetworkError, kGATTServerNotConnected)); |
| 259 } | 215 } |
| 260 | 216 |
| 261 if (!gatt()->device()->isValidCharacteristic( | 217 if (!gatt()->device()->isValidCharacteristic(m_characteristicInstanceId)) { |
| 262 m_webCharacteristic->characteristicInstanceID)) { | |
| 263 return ScriptPromise::rejectWithDOMException( | 218 return ScriptPromise::rejectWithDOMException( |
| 264 scriptState, | 219 scriptState, |
| 265 DOMException::create(InvalidStateError, kInvalidCharacteristic)); | 220 DOMException::create(InvalidStateError, kInvalidCharacteristic)); |
| 266 } | 221 } |
| 267 | 222 |
| 268 WebBluetooth* webbluetooth = | |
| 269 BluetoothSupplement::fromScriptState(scriptState); | |
| 270 // Partial implementation of writeValue algorithm: | 223 // Partial implementation of writeValue algorithm: |
| 271 // https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattchar
acteristic-writevalue | 224 // https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattchar
acteristic-writevalue |
| 272 | 225 |
| 273 // If bytes is more than 512 bytes long (the maximum length of an attribute | 226 // If bytes is more than 512 bytes long (the maximum length of an attribute |
| 274 // value, per Long Attribute Values) return a promise rejected with an | 227 // value, per Long Attribute Values) return a promise rejected with an |
| 275 // InvalidModificationError and abort. | 228 // InvalidModificationError and abort. |
| 276 if (value.byteLength() > 512) | 229 if (value.byteLength() > 512) |
| 277 return ScriptPromise::rejectWithDOMException( | 230 return ScriptPromise::rejectWithDOMException( |
| 278 scriptState, DOMException::create(InvalidModificationError, | 231 scriptState, DOMException::create(InvalidModificationError, |
| 279 "Value can't exceed 512 bytes.")); | 232 "Value can't exceed 512 bytes.")); |
| 280 | 233 |
| 281 // Let valueVector be a copy of the bytes held by value. | 234 // Let valueVector be a copy of the bytes held by value. |
| 282 WebVector<uint8_t> valueVector(value.bytes(), value.byteLength()); | 235 Vector<uint8_t> valueVector; |
| 236 valueVector.append(value.bytes(), value.byteLength()); |
| 283 | 237 |
| 284 ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); | 238 ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); |
| 239 ScriptPromise promise = resolver->promise(); |
| 240 gatt()->AddToActiveAlgorithms(resolver); |
| 285 | 241 |
| 286 ScriptPromise promise = resolver->promise(); | 242 mojom::blink::WebBluetoothService* service = m_device->bluetooth()->service(); |
| 287 webbluetooth->writeValue(m_webCharacteristic->characteristicInstanceID, | 243 service->RemoteCharacteristicWriteValue( |
| 288 valueVector, new WriteValueCallback(this, resolver)); | 244 m_characteristicInstanceId, valueVector, |
| 245 convertToBaseCallback(WTF::bind( |
| 246 &BluetoothRemoteGATTCharacteristic::WriteValueCallback, |
| 247 wrapPersistent(this), wrapPersistent(resolver), valueVector))); |
| 289 | 248 |
| 290 return promise; | 249 return promise; |
| 291 } | 250 } |
| 292 | 251 |
| 293 class NotificationsCallback : public WebBluetoothNotificationsCallbacks { | 252 void BluetoothRemoteGATTCharacteristic::NotificationsCallback( |
| 294 public: | 253 ScriptPromiseResolver* resolver, |
| 295 NotificationsCallback(BluetoothRemoteGATTCharacteristic* characteristic, | 254 mojom::blink::WebBluetoothResult result) { |
| 296 ScriptPromiseResolver* resolver) | 255 if (!resolver->getExecutionContext() || |
| 297 : m_characteristic(characteristic), m_resolver(resolver) { | 256 resolver->getExecutionContext()->isContextDestroyed()) |
| 298 // We always check that the device is connected before constructing this | 257 return; |
| 299 // object. | 258 |
| 300 CHECK(m_characteristic->gatt()->connected()); | 259 // If the resolver is not in the set of ActiveAlgorithms then the frame |
| 301 m_characteristic->gatt()->AddToActiveAlgorithms(m_resolver.get()); | 260 // disconnected so we reject. |
| 261 if (!gatt()->RemoveFromActiveAlgorithms(resolver)) { |
| 262 resolver->reject( |
| 263 DOMException::create(NetworkError, kGATTServerDisconnected)); |
| 264 return; |
| 302 } | 265 } |
| 303 | 266 |
| 304 void onSuccess() override { | 267 if (result == mojom::blink::WebBluetoothResult::SUCCESS) { |
| 305 if (!m_resolver->getExecutionContext() || | 268 resolver->resolve(this); |
| 306 m_resolver->getExecutionContext()->isContextDestroyed()) | 269 } else { |
| 307 return; | 270 resolver->reject(BluetoothError::take(resolver, result)); |
| 308 | |
| 309 if (!m_characteristic->gatt()->RemoveFromActiveAlgorithms( | |
| 310 m_resolver.get())) { | |
| 311 m_resolver->reject( | |
| 312 DOMException::create(NetworkError, kGATTServerDisconnected)); | |
| 313 return; | |
| 314 } | |
| 315 | |
| 316 m_resolver->resolve(m_characteristic); | |
| 317 } | 271 } |
| 318 | 272 } |
| 319 void onError( | |
| 320 int32_t | |
| 321 error /* Corresponds to WebBluetoothResult in web_bluetooth.mojom */) | |
| 322 override { | |
| 323 if (!m_resolver->getExecutionContext() || | |
| 324 m_resolver->getExecutionContext()->isContextDestroyed()) | |
| 325 return; | |
| 326 | |
| 327 if (!m_characteristic->gatt()->RemoveFromActiveAlgorithms( | |
| 328 m_resolver.get())) { | |
| 329 m_resolver->reject( | |
| 330 DOMException::create(NetworkError, kGATTServerDisconnected)); | |
| 331 return; | |
| 332 } | |
| 333 | |
| 334 m_resolver->reject(BluetoothError::take(m_resolver, error)); | |
| 335 } | |
| 336 | |
| 337 private: | |
| 338 Persistent<BluetoothRemoteGATTCharacteristic> m_characteristic; | |
| 339 Persistent<ScriptPromiseResolver> m_resolver; | |
| 340 }; | |
| 341 | 273 |
| 342 ScriptPromise BluetoothRemoteGATTCharacteristic::startNotifications( | 274 ScriptPromise BluetoothRemoteGATTCharacteristic::startNotifications( |
| 343 ScriptState* scriptState) { | 275 ScriptState* scriptState) { |
| 276 // We always check that the device is connected. |
| 344 if (!gatt()->connected()) { | 277 if (!gatt()->connected()) { |
| 345 return ScriptPromise::rejectWithDOMException( | 278 return ScriptPromise::rejectWithDOMException( |
| 346 scriptState, | 279 scriptState, |
| 347 DOMException::create(NetworkError, kGATTServerNotConnected)); | 280 DOMException::create(NetworkError, kGATTServerNotConnected)); |
| 348 } | 281 } |
| 349 | 282 |
| 350 if (!gatt()->device()->isValidCharacteristic( | 283 if (!gatt()->device()->isValidCharacteristic(m_characteristicInstanceId)) { |
| 351 m_webCharacteristic->characteristicInstanceID)) { | |
| 352 return ScriptPromise::rejectWithDOMException( | 284 return ScriptPromise::rejectWithDOMException( |
| 353 scriptState, | 285 scriptState, |
| 354 DOMException::create(InvalidStateError, kInvalidCharacteristic)); | 286 DOMException::create(InvalidStateError, kInvalidCharacteristic)); |
| 355 } | 287 } |
| 356 | 288 |
| 357 WebBluetooth* webbluetooth = | |
| 358 BluetoothSupplement::fromScriptState(scriptState); | |
| 359 ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); | 289 ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); |
| 360 ScriptPromise promise = resolver->promise(); | 290 ScriptPromise promise = resolver->promise(); |
| 361 webbluetooth->startNotifications( | 291 gatt()->AddToActiveAlgorithms(resolver); |
| 362 m_webCharacteristic->characteristicInstanceID, | 292 |
| 363 new NotificationsCallback(this, resolver)); | 293 mojom::blink::WebBluetoothService* service = m_device->bluetooth()->service(); |
| 294 service->RemoteCharacteristicStartNotifications( |
| 295 m_characteristicInstanceId, |
| 296 convertToBaseCallback( |
| 297 WTF::bind(&BluetoothRemoteGATTCharacteristic::NotificationsCallback, |
| 298 wrapPersistent(this), wrapPersistent(resolver)))); |
| 299 |
| 364 return promise; | 300 return promise; |
| 365 } | 301 } |
| 366 | 302 |
| 367 ScriptPromise BluetoothRemoteGATTCharacteristic::stopNotifications( | 303 ScriptPromise BluetoothRemoteGATTCharacteristic::stopNotifications( |
| 368 ScriptState* scriptState) { | 304 ScriptState* scriptState) { |
| 305 // We always check that the device is connected. |
| 369 if (!gatt()->connected()) { | 306 if (!gatt()->connected()) { |
| 370 return ScriptPromise::rejectWithDOMException( | 307 return ScriptPromise::rejectWithDOMException( |
| 371 scriptState, | 308 scriptState, |
| 372 DOMException::create(NetworkError, kGATTServerNotConnected)); | 309 DOMException::create(NetworkError, kGATTServerNotConnected)); |
| 373 } | 310 } |
| 374 | 311 |
| 375 if (!gatt()->device()->isValidCharacteristic( | 312 if (!gatt()->device()->isValidCharacteristic(m_characteristicInstanceId)) { |
| 376 m_webCharacteristic->characteristicInstanceID)) { | |
| 377 return ScriptPromise::rejectWithDOMException( | 313 return ScriptPromise::rejectWithDOMException( |
| 378 scriptState, | 314 scriptState, |
| 379 DOMException::create(InvalidStateError, kInvalidCharacteristic)); | 315 DOMException::create(InvalidStateError, kInvalidCharacteristic)); |
| 380 } | 316 } |
| 381 | 317 |
| 382 WebBluetooth* webbluetooth = | |
| 383 BluetoothSupplement::fromScriptState(scriptState); | |
| 384 ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); | 318 ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); |
| 385 ScriptPromise promise = resolver->promise(); | 319 ScriptPromise promise = resolver->promise(); |
| 386 webbluetooth->stopNotifications(m_webCharacteristic->characteristicInstanceID, | 320 gatt()->AddToActiveAlgorithms(resolver); |
| 387 new NotificationsCallback(this, resolver)); | 321 |
| 322 mojom::blink::WebBluetoothService* service = m_device->bluetooth()->service(); |
| 323 service->RemoteCharacteristicStopNotifications( |
| 324 m_characteristicInstanceId, |
| 325 convertToBaseCallback( |
| 326 WTF::bind(&BluetoothRemoteGATTCharacteristic::NotificationsCallback, |
| 327 wrapPersistent(this), wrapPersistent(resolver), |
| 328 mojom::blink::WebBluetoothResult::SUCCESS))); |
| 388 return promise; | 329 return promise; |
| 389 } | 330 } |
| 390 | 331 |
| 391 DEFINE_TRACE(BluetoothRemoteGATTCharacteristic) { | 332 DEFINE_TRACE(BluetoothRemoteGATTCharacteristic) { |
| 392 visitor->trace(m_service); | 333 visitor->trace(m_service); |
| 393 visitor->trace(m_properties); | 334 visitor->trace(m_properties); |
| 394 visitor->trace(m_value); | 335 visitor->trace(m_value); |
| 336 visitor->trace(m_device); |
| 395 EventTargetWithInlineData::trace(visitor); | 337 EventTargetWithInlineData::trace(visitor); |
| 396 SuspendableObject::trace(visitor); | 338 SuspendableObject::trace(visitor); |
| 397 } | 339 } |
| 398 | 340 |
| 399 } // namespace blink | 341 } // namespace blink |
| OLD | NEW |