Support [NoInterfaceObject] in web-sys

This commit enables `[NoInterfaceObject]` annotated interfaces in
`web-sys`. The `NoInterfaceObject` attribute means that there's not
actually a JS class for the object, but all of its properties and such
can still be accessed structually and invoked. This should help provide
more bindings for some more common types on the web!

Note that this builds on recent features to ensure that `dyn_into` and
friends always fail for `NoInterfaceObject` objects because they don't
actually have a class.

Closes #893
Closes #1257
Closes #1315
This commit is contained in:
Alex Crichton 2019-04-12 11:08:05 -07:00
parent c4776becbb
commit 01a52233d3
7 changed files with 121 additions and 17 deletions

View File

@ -56,6 +56,7 @@ Algorithm = []
AlignSetting = [] AlignSetting = []
AnalyserNode = [] AnalyserNode = []
AnalyserOptions = [] AnalyserOptions = []
AngleInstancedArrays = []
Animation = [] Animation = []
AnimationEffect = [] AnimationEffect = []
AnimationEvent = [] AnimationEvent = []
@ -63,6 +64,7 @@ AnimationEventInit = []
AnimationPlayState = [] AnimationPlayState = []
AnimationPlaybackEvent = [] AnimationPlaybackEvent = []
AnimationPlaybackEventInit = [] AnimationPlaybackEventInit = []
AnimationPropertyDetails = []
AnimationPropertyValueDetails = [] AnimationPropertyValueDetails = []
AnimationTimeline = [] AnimationTimeline = []
AssignedNodesOptions = [] AssignedNodesOptions = []
@ -159,6 +161,7 @@ ChromeFilePropertyBag = []
ChromeWorker = [] ChromeWorker = []
Client = [] Client = []
ClientQueryOptions = [] ClientQueryOptions = []
ClientRectsAndTexts = []
ClientType = [] ClientType = []
Clients = [] Clients = []
ClipboardEvent = [] ClipboardEvent = []
@ -195,6 +198,7 @@ ContextAttributes2d = []
ConvertCoordinateOptions = [] ConvertCoordinateOptions = []
ConvolverNode = [] ConvolverNode = []
ConvolverOptions = [] ConvolverOptions = []
Coordinates = []
Credential = [] Credential = []
CredentialCreationOptions = [] CredentialCreationOptions = []
CredentialRequestOptions = [] CredentialRequestOptions = []
@ -240,6 +244,7 @@ DecoderDoctorNotificationType = []
DedicatedWorkerGlobalScope = [] DedicatedWorkerGlobalScope = []
DelayNode = [] DelayNode = []
DelayOptions = [] DelayOptions = []
DeviceAcceleration = []
DeviceAccelerationInit = [] DeviceAccelerationInit = []
DeviceLightEvent = [] DeviceLightEvent = []
DeviceLightEventInit = [] DeviceLightEventInit = []
@ -249,6 +254,7 @@ DeviceOrientationEvent = []
DeviceOrientationEventInit = [] DeviceOrientationEventInit = []
DeviceProximityEvent = [] DeviceProximityEvent = []
DeviceProximityEventInit = [] DeviceProximityEventInit = []
DeviceRotationRate = []
DeviceRotationRateInit = [] DeviceRotationRateInit = []
DhKeyDeriveParams = [] DhKeyDeriveParams = []
DirectionSetting = [] DirectionSetting = []
@ -311,11 +317,22 @@ EventModifierInit = []
EventSource = [] EventSource = []
EventSourceInit = [] EventSourceInit = []
EventTarget = [] EventTarget = []
Exception = []
ExtBlendMinmax = []
ExtColorBufferFloat = []
ExtColorBufferHalfFloat = []
ExtDisjointTimerQuery = []
ExtFragDepth = []
ExtSRgb = []
ExtShaderTextureLod = []
ExtTextureFilterAnisotropic = []
ExtendableEvent = [] ExtendableEvent = []
ExtendableEventInit = [] ExtendableEventInit = []
ExtendableMessageEvent = [] ExtendableMessageEvent = []
ExtendableMessageEventInit = [] ExtendableMessageEventInit = []
External = []
FakePluginMimeEntry = [] FakePluginMimeEntry = []
FakePluginTagInit = []
FetchEvent = [] FetchEvent = []
FetchEventInit = [] FetchEventInit = []
FetchObserver = [] FetchObserver = []
@ -345,6 +362,7 @@ FontFace = []
FontFaceDescriptors = [] FontFaceDescriptors = []
FontFaceLoadStatus = [] FontFaceLoadStatus = []
FontFaceSet = [] FontFaceSet = []
FontFaceSetIterator = []
FontFaceSetIteratorResult = [] FontFaceSetIteratorResult = []
FontFaceSetLoadEvent = [] FontFaceSetLoadEvent = []
FontFaceSetLoadEventInit = [] FontFaceSetLoadEventInit = []
@ -368,8 +386,10 @@ GamepadHapticActuatorType = []
GamepadMappingType = [] GamepadMappingType = []
GamepadPose = [] GamepadPose = []
GamepadServiceTest = [] GamepadServiceTest = []
Geolocation = []
GetNotificationOptions = [] GetNotificationOptions = []
GetRootNodeOptions = [] GetRootNodeOptions = []
GetUserMediaRequest = []
GridDeclaration = [] GridDeclaration = []
GridTrackState = [] GridTrackState = []
GroupedHistoryEventInit = [] GroupedHistoryEventInit = []
@ -416,6 +436,7 @@ HtmlHeadElement = []
HtmlHeadingElement = [] HtmlHeadingElement = []
HtmlHrElement = [] HtmlHrElement = []
HtmlHtmlElement = [] HtmlHtmlElement = []
HtmlHyperlinkElementUtils = []
HtmlIFrameElement = [] HtmlIFrameElement = []
HtmlImageElement = [] HtmlImageElement = []
HtmlInputElement = [] HtmlInputElement = []
@ -491,10 +512,12 @@ IdbVersionChangeEventInit = []
IdleDeadline = [] IdleDeadline = []
IdleRequestOptions = [] IdleRequestOptions = []
IirFilterNode = [] IirFilterNode = []
IirFilterOptions = []
ImageBitmap = [] ImageBitmap = []
ImageBitmapFormat = [] ImageBitmapFormat = []
ImageBitmapRenderingContext = [] ImageBitmapRenderingContext = []
ImageCapture = [] ImageCapture = []
ImageCaptureError = []
ImageCaptureErrorEvent = [] ImageCaptureErrorEvent = []
ImageCaptureErrorEventInit = [] ImageCaptureErrorEventInit = []
ImageData = [] ImageData = []
@ -505,12 +528,14 @@ IntersectionObserver = []
IntersectionObserverEntry = [] IntersectionObserverEntry = []
IntersectionObserverEntryInit = [] IntersectionObserverEntryInit = []
IntersectionObserverInit = [] IntersectionObserverInit = []
IntlUtils = []
IterableKeyAndValueResult = [] IterableKeyAndValueResult = []
IterableKeyOrValueResult = [] IterableKeyOrValueResult = []
IterationCompositeOperation = [] IterationCompositeOperation = []
JsonWebKey = [] JsonWebKey = []
KeyAlgorithm = [] KeyAlgorithm = []
KeyEvent = [] KeyEvent = []
KeyIdsInitData = []
KeyboardEvent = [] KeyboardEvent = []
KeyboardEventInit = [] KeyboardEventInit = []
KeyframeEffect = [] KeyframeEffect = []
@ -519,6 +544,7 @@ L10nElement = []
L10nValue = [] L10nValue = []
LifecycleCallbacks = [] LifecycleCallbacks = []
LineAlignSetting = [] LineAlignSetting = []
ListBoxObject = []
LocalMediaStream = [] LocalMediaStream = []
LocaleInfo = [] LocaleInfo = []
Location = [] Location = []
@ -569,6 +595,7 @@ MediaStreamAudioDestinationNode = []
MediaStreamAudioSourceNode = [] MediaStreamAudioSourceNode = []
MediaStreamAudioSourceOptions = [] MediaStreamAudioSourceOptions = []
MediaStreamConstraints = [] MediaStreamConstraints = []
MediaStreamError = []
MediaStreamEvent = [] MediaStreamEvent = []
MediaStreamEventInit = [] MediaStreamEventInit = []
MediaStreamTrack = [] MediaStreamTrack = []
@ -602,6 +629,7 @@ MimeTypeArray = []
MouseEvent = [] MouseEvent = []
MouseEventInit = [] MouseEventInit = []
MouseScrollEvent = [] MouseScrollEvent = []
MozDebug = []
MutationEvent = [] MutationEvent = []
MutationObserver = [] MutationObserver = []
MutationObserverInit = [] MutationObserverInit = []
@ -612,6 +640,7 @@ NativeOsFileReadOptions = []
NativeOsFileWriteAtomicOptions = [] NativeOsFileWriteAtomicOptions = []
NavigationType = [] NavigationType = []
Navigator = [] Navigator = []
NavigatorAutomationInformation = []
NetworkCommandOptions = [] NetworkCommandOptions = []
NetworkInformation = [] NetworkInformation = []
NetworkResultOptions = [] NetworkResultOptions = []
@ -627,6 +656,13 @@ NotificationEventInit = []
NotificationOptions = [] NotificationOptions = []
NotificationPermission = [] NotificationPermission = []
ObserverCallback = [] ObserverCallback = []
OesElementIndexUint = []
OesStandardDerivatives = []
OesTextureFloat = []
OesTextureFloatLinear = []
OesTextureHalfFloat = []
OesTextureHalfFloatLinear = []
OesVertexArrayObject = []
OfflineAudioCompletionEvent = [] OfflineAudioCompletionEvent = []
OfflineAudioCompletionEventInit = [] OfflineAudioCompletionEventInit = []
OfflineAudioContext = [] OfflineAudioContext = []
@ -672,6 +708,7 @@ PerformanceNavigation = []
PerformanceNavigationTiming = [] PerformanceNavigationTiming = []
PerformanceObserver = [] PerformanceObserver = []
PerformanceObserverEntryList = [] PerformanceObserverEntryList = []
PerformanceObserverInit = []
PerformanceResourceTiming = [] PerformanceResourceTiming = []
PerformanceServerTiming = [] PerformanceServerTiming = []
PerformanceTiming = [] PerformanceTiming = []
@ -693,7 +730,9 @@ PopStateEvent = []
PopStateEventInit = [] PopStateEventInit = []
PopupBlockedEvent = [] PopupBlockedEvent = []
PopupBlockedEventInit = [] PopupBlockedEventInit = []
Position = []
PositionAlignSetting = [] PositionAlignSetting = []
PositionError = []
PositionOptions = [] PositionOptions = []
Presentation = [] Presentation = []
PresentationAvailability = [] PresentationAvailability = []
@ -716,9 +755,11 @@ ProfileTimelineStackFrame = []
ProfileTimelineWorkerOperationType = [] ProfileTimelineWorkerOperationType = []
ProgressEvent = [] ProgressEvent = []
ProgressEventInit = [] ProgressEventInit = []
PromiseNativeHandler = []
PromiseRejectionEvent = [] PromiseRejectionEvent = []
PromiseRejectionEventInit = [] PromiseRejectionEventInit = []
PublicKeyCredential = [] PublicKeyCredential = []
PublicKeyCredentialCreationOptions = []
PublicKeyCredentialDescriptor = [] PublicKeyCredentialDescriptor = []
PublicKeyCredentialEntity = [] PublicKeyCredentialEntity = []
PublicKeyCredentialParameters = [] PublicKeyCredentialParameters = []
@ -792,6 +833,7 @@ RtcIdentityAssertionResult = []
RtcIdentityProvider = [] RtcIdentityProvider = []
RtcIdentityProviderDetails = [] RtcIdentityProviderDetails = []
RtcIdentityProviderOptions = [] RtcIdentityProviderOptions = []
RtcIdentityProviderRegistrar = []
RtcIdentityValidationResult = [] RtcIdentityValidationResult = []
RtcInboundRtpStreamStats = [] RtcInboundRtpStreamStats = []
RtcLifecycleEvent = [] RtcLifecycleEvent = []
@ -844,6 +886,7 @@ ScreenOrientation = []
ScriptProcessorNode = [] ScriptProcessorNode = []
ScrollAreaEvent = [] ScrollAreaEvent = []
ScrollBehavior = [] ScrollBehavior = []
ScrollBoxObject = []
ScrollIntoViewOptions = [] ScrollIntoViewOptions = []
ScrollLogicalPosition = [] ScrollLogicalPosition = []
ScrollOptions = [] ScrollOptions = []
@ -952,7 +995,27 @@ SvgMetadataElement = []
SvgNumber = [] SvgNumber = []
SvgNumberList = [] SvgNumberList = []
SvgPathElement = [] SvgPathElement = []
SvgPathSeg = []
SvgPathSegArcAbs = []
SvgPathSegArcRel = []
SvgPathSegClosePath = []
SvgPathSegCurvetoCubicAbs = []
SvgPathSegCurvetoCubicRel = []
SvgPathSegCurvetoCubicSmoothAbs = []
SvgPathSegCurvetoCubicSmoothRel = []
SvgPathSegCurvetoQuadraticAbs = []
SvgPathSegCurvetoQuadraticRel = []
SvgPathSegCurvetoQuadraticSmoothAbs = []
SvgPathSegCurvetoQuadraticSmoothRel = []
SvgPathSegLinetoAbs = []
SvgPathSegLinetoHorizontalAbs = []
SvgPathSegLinetoHorizontalRel = []
SvgPathSegLinetoRel = []
SvgPathSegLinetoVerticalAbs = []
SvgPathSegLinetoVerticalRel = []
SvgPathSegList = [] SvgPathSegList = []
SvgPathSegMovetoAbs = []
SvgPathSegMovetoRel = []
SvgPatternElement = [] SvgPatternElement = []
SvgPoint = [] SvgPoint = []
SvgPointList = [] SvgPointList = []
@ -1044,7 +1107,9 @@ TrackEventInit = []
TransitionEvent = [] TransitionEvent = []
TransitionEventInit = [] TransitionEventInit = []
Transport = [] Transport = []
TreeBoxObject = []
TreeCellInfo = [] TreeCellInfo = []
TreeView = []
TreeWalker = [] TreeWalker = []
U2f = [] U2f = []
U2fClientData = [] U2fClientData = []
@ -1183,6 +1248,19 @@ WebKitCssMatrix = []
WebSocket = [] WebSocket = []
WebSocketDict = [] WebSocketDict = []
WebSocketElement = [] WebSocketElement = []
WebglColorBufferFloat = []
WebglCompressedTextureAstc = []
WebglCompressedTextureAtc = []
WebglCompressedTextureEtc = []
WebglCompressedTextureEtc1 = []
WebglCompressedTexturePvrtc = []
WebglCompressedTextureS3tc = []
WebglCompressedTextureS3tcSrgb = []
WebglDebugRendererInfo = []
WebglDebugShaders = []
WebglDepthTexture = []
WebglDrawBuffers = []
WebglLoseContext = []
WebrtcGlobalStatisticsReport = [] WebrtcGlobalStatisticsReport = []
WheelEvent = [] WheelEvent = []
WheelEventInit = [] WheelEventInit = []

View File

@ -12,3 +12,4 @@ pub mod global;
pub mod namespace; pub mod namespace;
pub mod simple; pub mod simple;
pub mod throws; pub mod throws;
pub mod no_interface;

View File

@ -0,0 +1,8 @@
global.GetNoInterfaceObject = class {
static get() {
return {
number: 3,
foo: () => {},
}
}
};

View File

@ -0,0 +1,11 @@
use js_sys::Object;
use wasm_bindgen_test::*;
include!(concat!(env!("OUT_DIR"), "/no_interface.rs"));
#[wasm_bindgen_test]
fn smoke() {
let obj = GetNoInterfaceObject::get();
assert_eq!(obj.number(), 3.0);
obj.foo();
}

View File

@ -0,0 +1,9 @@
[NoInterfaceObject]
interface NoInterfaceObject {
readonly attribute double number;
void foo();
};
interface GetNoInterfaceObject {
static NoInterfaceObject get();
};

View File

@ -45,6 +45,7 @@ pub(crate) struct FirstPassRecord<'src> {
pub(crate) struct InterfaceData<'src> { pub(crate) struct InterfaceData<'src> {
/// Whether only partial interfaces were encountered /// Whether only partial interfaces were encountered
pub(crate) partial: bool, pub(crate) partial: bool,
pub(crate) has_interface: bool,
pub(crate) deprecated: Option<String>, pub(crate) deprecated: Option<String>,
pub(crate) attributes: Vec<&'src AttributeInterfaceMember<'src>>, pub(crate) attributes: Vec<&'src AttributeInterfaceMember<'src>>,
pub(crate) consts: Vec<&'src ConstMember<'src>>, pub(crate) consts: Vec<&'src ConstMember<'src>>,
@ -303,22 +304,14 @@ impl<'src> FirstPass<'src, ()> for weedle::InterfaceDefinition<'src> {
return Ok(()); return Ok(());
} }
if util::is_no_interface_object(&self.attributes) {
log::info!(
"Skipping because of `NoInterfaceObject` attribute: {:?}",
self.identifier.0
);
return Ok(());
}
{
let interface_data = record.interfaces.entry(self.identifier.0).or_default(); let interface_data = record.interfaces.entry(self.identifier.0).or_default();
interface_data.partial = false; interface_data.partial = false;
interface_data.superclass = self.inheritance.map(|s| s.identifier.0); interface_data.superclass = self.inheritance.map(|s| s.identifier.0);
interface_data.definition_attributes = self.attributes.as_ref(); interface_data.definition_attributes = self.attributes.as_ref();
interface_data.deprecated = interface_data.deprecated =
util::get_rust_deprecated(&self.attributes).map(|s| s.to_string()); util::get_rust_deprecated(&self.attributes).map(|s| s.to_string());
} interface_data.has_interface =
!util::is_no_interface_object(&self.attributes);
if let Some(attrs) = &self.attributes { if let Some(attrs) = &self.attributes {
for attr in attrs.body.list.iter() { for attr in attrs.body.list.iter() {
process_interface_attribute(record, self.identifier.0, attr); process_interface_attribute(record, self.identifier.0, attr);

View File

@ -514,7 +514,11 @@ impl<'src> FirstPassRecord<'src> {
attrs, attrs,
doc_comment: None, doc_comment: None,
instanceof_shim: format!("__widl_instanceof_{}", name), instanceof_shim: format!("__widl_instanceof_{}", name),
is_type_of: None, is_type_of: if data.has_interface {
None
} else {
Some(syn::parse_quote!{ |_| false })
},
extends: Vec::new(), extends: Vec::new(),
vendor_prefixes: Vec::new(), vendor_prefixes: Vec::new(),
}; };