mirror of
https://github.com/rustwasm/wasm-bindgen.git
synced 2025-01-05 11:11:52 +03:00
Merge pull request #266 from ohanar/webidl_attributes
WebIDL: add support for typedefs and interface attributes
This commit is contained in:
commit
0cd767c9d4
@ -10,6 +10,7 @@ pub struct Program {
|
|||||||
pub imports: Vec<Import>,
|
pub imports: Vec<Import>,
|
||||||
pub enums: Vec<Enum>,
|
pub enums: Vec<Enum>,
|
||||||
pub structs: Vec<Struct>,
|
pub structs: Vec<Struct>,
|
||||||
|
pub type_aliases: Vec<TypeAlias>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))]
|
#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))]
|
||||||
@ -120,6 +121,13 @@ pub enum TypeLocation {
|
|||||||
ExportRet,
|
ExportRet,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))]
|
||||||
|
pub struct TypeAlias {
|
||||||
|
pub vis: syn::Visibility,
|
||||||
|
pub dest: Ident,
|
||||||
|
pub src: syn::Type,
|
||||||
|
}
|
||||||
|
|
||||||
impl Program {
|
impl Program {
|
||||||
pub fn push_item(
|
pub fn push_item(
|
||||||
&mut self,
|
&mut self,
|
||||||
|
@ -59,6 +59,9 @@ impl ToTokens for ast::Program {
|
|||||||
for e in self.enums.iter() {
|
for e in self.enums.iter() {
|
||||||
e.to_tokens(tokens);
|
e.to_tokens(tokens);
|
||||||
}
|
}
|
||||||
|
for a in self.type_aliases.iter() {
|
||||||
|
a.to_tokens(tokens);
|
||||||
|
}
|
||||||
|
|
||||||
// Generate a static which will eventually be what lives in a custom section
|
// Generate a static which will eventually be what lives in a custom section
|
||||||
// of the wasm executable. For now it's just a plain old static, but we'll
|
// of the wasm executable. For now it's just a plain old static, but we'll
|
||||||
@ -823,3 +826,15 @@ impl ToTokens for ast::ImportStatic {
|
|||||||
}).to_tokens(into);
|
}).to_tokens(into);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ToTokens for ast::TypeAlias {
|
||||||
|
fn to_tokens(&self, into: &mut TokenStream) {
|
||||||
|
let vis = &self.vis;
|
||||||
|
let dest = &self.dest;
|
||||||
|
let src = &self.src;
|
||||||
|
(quote! {
|
||||||
|
#[allow(non_camel_case_types)]
|
||||||
|
#vis type #dest = #src;
|
||||||
|
}).to_tokens(into);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -123,6 +123,7 @@ impl<'a> WebidlParse<'a> for webidl::ast::Definition {
|
|||||||
webidl::ast::Definition::Interface(ref interface) => {
|
webidl::ast::Definition::Interface(ref interface) => {
|
||||||
interface.webidl_parse(program, ())
|
interface.webidl_parse(program, ())
|
||||||
}
|
}
|
||||||
|
webidl::ast::Definition::Typedef(ref typedef) => typedef.webidl_parse(program, ()),
|
||||||
// TODO
|
// TODO
|
||||||
webidl::ast::Definition::Callback(..)
|
webidl::ast::Definition::Callback(..)
|
||||||
| webidl::ast::Definition::Dictionary(..)
|
| webidl::ast::Definition::Dictionary(..)
|
||||||
@ -130,8 +131,7 @@ impl<'a> WebidlParse<'a> for webidl::ast::Definition {
|
|||||||
| webidl::ast::Definition::Implements(..)
|
| webidl::ast::Definition::Implements(..)
|
||||||
| webidl::ast::Definition::Includes(..)
|
| webidl::ast::Definition::Includes(..)
|
||||||
| webidl::ast::Definition::Mixin(..)
|
| webidl::ast::Definition::Mixin(..)
|
||||||
| webidl::ast::Definition::Namespace(..)
|
| webidl::ast::Definition::Namespace(..) => {
|
||||||
| webidl::ast::Definition::Typedef(..) => {
|
|
||||||
warn!("Unsupported WebIDL definition: {:?}", self);
|
warn!("Unsupported WebIDL definition: {:?}", self);
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -156,6 +156,34 @@ impl<'a> WebidlParse<'a> for webidl::ast::Interface {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<'a> WebidlParse<'a> for webidl::ast::Typedef {
|
||||||
|
type Extra = ();
|
||||||
|
|
||||||
|
fn webidl_parse(&self, program: &mut backend::ast::Program, _: ()) -> Result<()> {
|
||||||
|
let dest = rust_ident(&self.name);
|
||||||
|
let src = match webidl_ty_to_syn_ty(&self.type_, TypePosition::Return) {
|
||||||
|
Some(src) => src,
|
||||||
|
None => {
|
||||||
|
warn!(
|
||||||
|
"typedef's source type is not yet supported: {:?}. Skipping typedef {:?}",
|
||||||
|
*self.type_, self
|
||||||
|
);
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
program.type_aliases.push(backend::ast::TypeAlias {
|
||||||
|
vis: syn::Visibility::Public(syn::VisPublic {
|
||||||
|
pub_token: Default::default(),
|
||||||
|
}),
|
||||||
|
dest,
|
||||||
|
src,
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<'a> WebidlParse<'a> for webidl::ast::NonPartialInterface {
|
impl<'a> WebidlParse<'a> for webidl::ast::NonPartialInterface {
|
||||||
type Extra = ();
|
type Extra = ();
|
||||||
|
|
||||||
@ -185,10 +213,12 @@ impl<'a> WebidlParse<'a> for webidl::ast::InterfaceMember {
|
|||||||
|
|
||||||
fn webidl_parse(&self, program: &mut backend::ast::Program, self_name: &'a str) -> Result<()> {
|
fn webidl_parse(&self, program: &mut backend::ast::Program, self_name: &'a str) -> Result<()> {
|
||||||
match *self {
|
match *self {
|
||||||
|
webidl::ast::InterfaceMember::Attribute(ref attr) => {
|
||||||
|
attr.webidl_parse(program, self_name)
|
||||||
|
}
|
||||||
webidl::ast::InterfaceMember::Operation(ref op) => op.webidl_parse(program, self_name),
|
webidl::ast::InterfaceMember::Operation(ref op) => op.webidl_parse(program, self_name),
|
||||||
// TODO
|
// TODO
|
||||||
webidl::ast::InterfaceMember::Attribute(_)
|
webidl::ast::InterfaceMember::Const(_)
|
||||||
| webidl::ast::InterfaceMember::Const(_)
|
|
||||||
| webidl::ast::InterfaceMember::Iterable(_)
|
| webidl::ast::InterfaceMember::Iterable(_)
|
||||||
| webidl::ast::InterfaceMember::Maplike(_)
|
| webidl::ast::InterfaceMember::Maplike(_)
|
||||||
| webidl::ast::InterfaceMember::Setlike(_) => {
|
| webidl::ast::InterfaceMember::Setlike(_) => {
|
||||||
@ -199,6 +229,21 @@ impl<'a> WebidlParse<'a> for webidl::ast::InterfaceMember {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<'a> WebidlParse<'a> for webidl::ast::Attribute {
|
||||||
|
type Extra = &'a str;
|
||||||
|
|
||||||
|
fn webidl_parse(&self, program: &mut backend::ast::Program, self_name: &'a str) -> Result<()> {
|
||||||
|
match *self {
|
||||||
|
webidl::ast::Attribute::Regular(ref attr) => attr.webidl_parse(program, self_name),
|
||||||
|
// TODO
|
||||||
|
webidl::ast::Attribute::Static(_) | webidl::ast::Attribute::Stringifier(_) => {
|
||||||
|
warn!("Unsupported WebIDL attribute: {:?}", self);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<'a> WebidlParse<'a> for webidl::ast::Operation {
|
impl<'a> WebidlParse<'a> for webidl::ast::Operation {
|
||||||
type Extra = &'a str;
|
type Extra = &'a str;
|
||||||
|
|
||||||
@ -257,6 +302,10 @@ enum TypePosition {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn webidl_ty_to_syn_ty(ty: &webidl::ast::Type, pos: TypePosition) -> Option<syn::Type> {
|
fn webidl_ty_to_syn_ty(ty: &webidl::ast::Type, pos: TypePosition) -> Option<syn::Type> {
|
||||||
|
// nullable types are not yet supported (see issue #14)
|
||||||
|
if ty.nullable {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
Some(match ty.kind {
|
Some(match ty.kind {
|
||||||
// `any` becomes `::wasm_bindgen::JsValue`.
|
// `any` becomes `::wasm_bindgen::JsValue`.
|
||||||
webidl::ast::TypeKind::Any => {
|
webidl::ast::TypeKind::Any => {
|
||||||
@ -332,6 +381,151 @@ fn simple_fn_arg(ident: Ident, ty: syn::Type) -> syn::FnArg {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<'a> WebidlParse<'a> for webidl::ast::RegularAttribute {
|
||||||
|
type Extra = &'a str;
|
||||||
|
|
||||||
|
fn webidl_parse(&self, program: &mut backend::ast::Program, self_name: &'a str) -> Result<()> {
|
||||||
|
fn create_getter(
|
||||||
|
this: &webidl::ast::RegularAttribute,
|
||||||
|
self_name: &str,
|
||||||
|
) -> Option<backend::ast::Import> {
|
||||||
|
let name = raw_ident(&this.name);
|
||||||
|
let rust_name = rust_ident(&this.name.to_snake_case());
|
||||||
|
|
||||||
|
let (output, ret) = match webidl_ty_to_syn_ty(&this.type_, TypePosition::Return) {
|
||||||
|
None => {
|
||||||
|
warn!("Attribute's type does not yet support reading: {:?}. Skipping getter binding for {:?}",
|
||||||
|
this.type_, this);
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
Some(ty) => (
|
||||||
|
syn::ReturnType::Type(Default::default(), Box::new(ty.clone())),
|
||||||
|
Some(ty),
|
||||||
|
),
|
||||||
|
};
|
||||||
|
|
||||||
|
let self_ty = ident_ty(rust_ident(self_name));
|
||||||
|
let self_ref_ty = shared_ref(self_ty.clone());
|
||||||
|
|
||||||
|
let shim = rust_ident(&format!("__wbg_f_{}_{}_{}", name, rust_name, self_name));
|
||||||
|
|
||||||
|
Some(backend::ast::Import {
|
||||||
|
module: None,
|
||||||
|
version: None,
|
||||||
|
js_namespace: None,
|
||||||
|
kind: backend::ast::ImportKind::Function(backend::ast::ImportFunction {
|
||||||
|
function: backend::ast::Function {
|
||||||
|
name: name.clone(),
|
||||||
|
arguments: vec![self_ref_ty.clone()],
|
||||||
|
ret,
|
||||||
|
opts: backend::ast::BindgenAttrs {
|
||||||
|
attrs: vec![
|
||||||
|
backend::ast::BindgenAttr::Method,
|
||||||
|
backend::ast::BindgenAttr::Getter(Some(name.clone())),
|
||||||
|
],
|
||||||
|
},
|
||||||
|
rust_attrs: vec![],
|
||||||
|
rust_decl: Box::new(syn::FnDecl {
|
||||||
|
fn_token: Default::default(),
|
||||||
|
generics: Default::default(),
|
||||||
|
paren_token: Default::default(),
|
||||||
|
inputs: syn::punctuated::Punctuated::from_iter(vec![simple_fn_arg(
|
||||||
|
raw_ident("self_"),
|
||||||
|
self_ref_ty,
|
||||||
|
)]),
|
||||||
|
variadic: None,
|
||||||
|
output,
|
||||||
|
}),
|
||||||
|
rust_vis: syn::Visibility::Public(syn::VisPublic {
|
||||||
|
pub_token: Default::default(),
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
rust_name,
|
||||||
|
kind: backend::ast::ImportFunctionKind::Method {
|
||||||
|
class: self_name.to_string(),
|
||||||
|
ty: self_ty,
|
||||||
|
},
|
||||||
|
shim,
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_setter(
|
||||||
|
this: &webidl::ast::RegularAttribute,
|
||||||
|
self_name: &str,
|
||||||
|
) -> Option<backend::ast::Import> {
|
||||||
|
let name = raw_ident(&this.name);
|
||||||
|
let rust_attr_name = rust_ident(&this.name.to_snake_case());
|
||||||
|
let rust_name = rust_ident(&format!("set_{}", rust_attr_name));
|
||||||
|
|
||||||
|
let self_ty = ident_ty(rust_ident(self_name));
|
||||||
|
|
||||||
|
let (inputs, arguments) = match webidl_ty_to_syn_ty(&this.type_, TypePosition::Argument)
|
||||||
|
{
|
||||||
|
None => {
|
||||||
|
warn!("Attribute's type does not yet support writing: {:?}. Skipping setter binding for {:?}",
|
||||||
|
this.type_, this);
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
Some(ty) => {
|
||||||
|
let self_ref_ty = shared_ref(self_ty.clone());
|
||||||
|
let mut inputs = syn::punctuated::Punctuated::new();
|
||||||
|
inputs.push(simple_fn_arg(raw_ident("self_"), self_ref_ty.clone()));
|
||||||
|
inputs.push(simple_fn_arg(rust_attr_name, ty.clone()));
|
||||||
|
(inputs, vec![self_ref_ty, ty])
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let shim = rust_ident(&format!("__wbg_f_{}_{}_{}", name, rust_name, self_name));
|
||||||
|
|
||||||
|
Some(backend::ast::Import {
|
||||||
|
module: None,
|
||||||
|
version: None,
|
||||||
|
js_namespace: None,
|
||||||
|
kind: backend::ast::ImportKind::Function(backend::ast::ImportFunction {
|
||||||
|
function: backend::ast::Function {
|
||||||
|
name: name.clone(),
|
||||||
|
arguments,
|
||||||
|
ret: None,
|
||||||
|
opts: backend::ast::BindgenAttrs {
|
||||||
|
attrs: vec![
|
||||||
|
backend::ast::BindgenAttr::Method,
|
||||||
|
backend::ast::BindgenAttr::Setter(Some(name)),
|
||||||
|
],
|
||||||
|
},
|
||||||
|
rust_attrs: vec![],
|
||||||
|
rust_decl: Box::new(syn::FnDecl {
|
||||||
|
fn_token: Default::default(),
|
||||||
|
generics: Default::default(),
|
||||||
|
paren_token: Default::default(),
|
||||||
|
inputs,
|
||||||
|
variadic: None,
|
||||||
|
output: syn::ReturnType::Default,
|
||||||
|
}),
|
||||||
|
rust_vis: syn::Visibility::Public(syn::VisPublic {
|
||||||
|
pub_token: Default::default(),
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
rust_name,
|
||||||
|
kind: backend::ast::ImportFunctionKind::Method {
|
||||||
|
class: self_name.to_string(),
|
||||||
|
ty: self_ty,
|
||||||
|
},
|
||||||
|
shim,
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
create_getter(self, self_name).map(|import| program.imports.push(import));
|
||||||
|
|
||||||
|
if !self.read_only {
|
||||||
|
create_setter(self, self_name).map(|import| program.imports.push(import));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<'a> WebidlParse<'a> for webidl::ast::RegularOperation {
|
impl<'a> WebidlParse<'a> for webidl::ast::RegularOperation {
|
||||||
type Extra = &'a str;
|
type Extra = &'a str;
|
||||||
|
|
||||||
|
File diff suppressed because one or more lines are too long
3
crates/webidl/tests/fixtures/Event.webidl
vendored
3
crates/webidl/tests/fixtures/Event.webidl
vendored
@ -10,6 +10,9 @@
|
|||||||
* liability, trademark and document use rules apply.
|
* liability, trademark and document use rules apply.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// TODO: don't include this here, use Performance.webidl instead
|
||||||
|
typedef double DOMHighResTimeStamp;
|
||||||
|
|
||||||
[Constructor(DOMString type, optional EventInit eventInitDict),
|
[Constructor(DOMString type, optional EventInit eventInitDict),
|
||||||
Exposed=(Window,Worker,System), ProbablyShortLivingWrapper]
|
Exposed=(Window,Worker,System), ProbablyShortLivingWrapper]
|
||||||
interface Event {
|
interface Event {
|
||||||
|
Loading…
Reference in New Issue
Block a user