зеркало из https://github.com/mozilla/gecko-dev.git
103 строки
3.2 KiB
C++
103 строки
3.2 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim:set ts=2 sw=2 sts=2 et cindent: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "mozilla/dom/MIDIInput.h"
|
|
#include "mozilla/dom/Document.h"
|
|
#include "mozilla/dom/MIDIPortChild.h"
|
|
#include "mozilla/dom/MIDIInputBinding.h"
|
|
#include "mozilla/dom/MIDIMessageEvent.h"
|
|
#include "mozilla/dom/MIDIMessageEventBinding.h"
|
|
#include "nsDOMNavigationTiming.h"
|
|
|
|
#include "MIDILog.h"
|
|
|
|
namespace mozilla::dom {
|
|
|
|
MIDIInput::MIDIInput(nsPIDOMWindowInner* aWindow, MIDIAccess* aMIDIAccessParent)
|
|
: MIDIPort(aWindow, aMIDIAccessParent), mKeepAlive(false) {}
|
|
|
|
// static
|
|
MIDIInput* MIDIInput::Create(nsPIDOMWindowInner* aWindow,
|
|
MIDIAccess* aMIDIAccessParent,
|
|
const MIDIPortInfo& aPortInfo,
|
|
const bool aSysexEnabled) {
|
|
MOZ_ASSERT(static_cast<MIDIPortType>(aPortInfo.type()) ==
|
|
MIDIPortType::Input);
|
|
auto* port = new MIDIInput(aWindow, aMIDIAccessParent);
|
|
if (!port->Initialize(aPortInfo, aSysexEnabled)) {
|
|
return nullptr;
|
|
}
|
|
return port;
|
|
}
|
|
|
|
JSObject* MIDIInput::WrapObject(JSContext* aCx,
|
|
JS::Handle<JSObject*> aGivenProto) {
|
|
return MIDIInput_Binding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
void MIDIInput::Receive(const nsTArray<MIDIMessage>& aMsgs) {
|
|
if (!GetOwner()) {
|
|
return; // Ignore messages once we've been disconnected from the owner
|
|
}
|
|
|
|
nsCOMPtr<Document> doc = GetOwner()->GetDoc();
|
|
if (!doc) {
|
|
NS_WARNING("No document available to send MIDIMessageEvent to!");
|
|
return;
|
|
}
|
|
for (const auto& msg : aMsgs) {
|
|
RefPtr<MIDIMessageEvent> event(
|
|
MIDIMessageEvent::Constructor(this, msg.timestamp(), msg.data()));
|
|
DispatchTrustedEvent(event);
|
|
}
|
|
}
|
|
|
|
void MIDIInput::StateChange() {
|
|
if (Port()->ConnectionState() == MIDIPortConnectionState::Open ||
|
|
(Port()->DeviceState() == MIDIPortDeviceState::Connected &&
|
|
Port()->ConnectionState() == MIDIPortConnectionState::Pending)) {
|
|
KeepAliveOnMidimessage();
|
|
} else {
|
|
DontKeepAliveOnMidimessage();
|
|
}
|
|
}
|
|
|
|
void MIDIInput::EventListenerAdded(nsAtom* aType) {
|
|
if (aType == nsGkAtoms::onmidimessage) {
|
|
// HACK: the Web MIDI spec states that we should open a port only when
|
|
// setting the midimessage event handler but Chrome does it even when
|
|
// adding event listeners hence this.
|
|
if (Port()->ConnectionState() != MIDIPortConnectionState::Open) {
|
|
LOG("onmidimessage event listener added, sending implicit Open");
|
|
Port()->SendOpen();
|
|
}
|
|
}
|
|
|
|
DOMEventTargetHelper::EventListenerAdded(aType);
|
|
}
|
|
|
|
void MIDIInput::DisconnectFromOwner() {
|
|
DontKeepAliveOnMidimessage();
|
|
|
|
MIDIPort::DisconnectFromOwner();
|
|
}
|
|
|
|
void MIDIInput::KeepAliveOnMidimessage() {
|
|
if (!mKeepAlive) {
|
|
mKeepAlive = true;
|
|
KeepAliveIfHasListenersFor(nsGkAtoms::onmidimessage);
|
|
}
|
|
}
|
|
|
|
void MIDIInput::DontKeepAliveOnMidimessage() {
|
|
if (mKeepAlive) {
|
|
IgnoreKeepAliveIfHasListenersFor(nsGkAtoms::onmidimessage);
|
|
mKeepAlive = false;
|
|
}
|
|
}
|
|
|
|
} // namespace mozilla::dom
|