mirror of
https://github.com/servo/servo.git
synced 2025-06-06 16:45:39 +00:00
Derive DomObject with a proc macro
This commit is contained in:
parent
1dfdd0bab8
commit
c84cea995b
9 changed files with 92 additions and 91 deletions
9
Cargo.lock
generated
9
Cargo.lock
generated
|
@ -650,6 +650,14 @@ dependencies = [
|
||||||
"time 0.1.36 (registry+https://github.com/rust-lang/crates.io-index)",
|
"time 0.1.36 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "domobject_derive"
|
||||||
|
version = "0.0.1"
|
||||||
|
dependencies = [
|
||||||
|
"quote 0.3.12 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"syn 0.10.7 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "dtoa"
|
name = "dtoa"
|
||||||
version = "0.2.2"
|
version = "0.2.2"
|
||||||
|
@ -2255,6 +2263,7 @@ dependencies = [
|
||||||
"cookie 0.2.5 (registry+https://github.com/rust-lang/crates.io-index)",
|
"cookie 0.2.5 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"cssparser 0.7.4 (registry+https://github.com/rust-lang/crates.io-index)",
|
"cssparser 0.7.4 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"devtools_traits 0.0.1",
|
"devtools_traits 0.0.1",
|
||||||
|
"domobject_derive 0.0.1",
|
||||||
"encoding 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)",
|
"encoding 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"euclid 0.10.5 (registry+https://github.com/rust-lang/crates.io-index)",
|
"euclid 0.10.5 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"fnv 1.0.5 (registry+https://github.com/rust-lang/crates.io-index)",
|
"fnv 1.0.5 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
|
14
components/domobject_derive/Cargo.toml
Normal file
14
components/domobject_derive/Cargo.toml
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
[package]
|
||||||
|
name = "domobject_derive"
|
||||||
|
version = "0.0.1"
|
||||||
|
authors = ["The Servo Project Developers"]
|
||||||
|
license = "MPL-2.0"
|
||||||
|
publish = false
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
path = "lib.rs"
|
||||||
|
proc-macro = true
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
syn = "0.10"
|
||||||
|
quote = "0.3"
|
53
components/domobject_derive/lib.rs
Normal file
53
components/domobject_derive/lib.rs
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
/* 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/. */
|
||||||
|
|
||||||
|
extern crate proc_macro;
|
||||||
|
#[macro_use] extern crate quote;
|
||||||
|
extern crate syn;
|
||||||
|
|
||||||
|
#[proc_macro_derive(DomObject)]
|
||||||
|
pub fn expand_token_stream(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
|
||||||
|
expand_string(&input.to_string()).parse().unwrap()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn expand_string(input: &str) -> String {
|
||||||
|
let type_ = syn::parse_macro_input(input).unwrap();
|
||||||
|
|
||||||
|
let first_field_name = if let syn::Body::Struct(syn::VariantData::Struct(ref fields)) = type_.body {
|
||||||
|
let first_field = fields.first().expect("#[derive(DomObject)] should not be applied on empty structs");
|
||||||
|
first_field.ident.as_ref().unwrap()
|
||||||
|
} else {
|
||||||
|
panic!("#[derive(DomObject)] should only be applied on proper structs")
|
||||||
|
};
|
||||||
|
|
||||||
|
let name = &type_.ident;
|
||||||
|
let (impl_generics, ty_generics, where_clause) = type_.generics.split_for_impl();
|
||||||
|
|
||||||
|
let tokens = quote! {
|
||||||
|
impl #impl_generics ::js::conversions::ToJSValConvertible for #name #ty_generics #where_clause {
|
||||||
|
#[allow(unsafe_code)]
|
||||||
|
unsafe fn to_jsval(&self,
|
||||||
|
cx: *mut ::js::jsapi::JSContext,
|
||||||
|
rval: ::js::jsapi::MutableHandleValue) {
|
||||||
|
let object = ::dom::bindings::reflector::DomObject::reflector(self).get_jsobject();
|
||||||
|
object.to_jsval(cx, rval)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl #impl_generics ::dom::bindings::reflector::DomObject for #name #ty_generics #where_clause {
|
||||||
|
#[inline]
|
||||||
|
fn reflector(&self) -> &::dom::bindings::reflector::Reflector {
|
||||||
|
self.#first_field_name.reflector()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl #impl_generics ::dom::bindings::reflector::MutDomObject for #name #ty_generics #where_clause {
|
||||||
|
fn init_reflector(&mut self, obj: *mut ::js::jsapi::JSObject) {
|
||||||
|
self.#first_field_name.init_reflector(obj);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
tokens.to_string()
|
||||||
|
}
|
|
@ -15,9 +15,7 @@ pub fn expand_dom_struct(cx: &mut ExtCtxt, sp: Span, _: &MetaItem, anno: Annotat
|
||||||
item2.attrs.push(quote_attr!(cx, #[repr(C)]));
|
item2.attrs.push(quote_attr!(cx, #[repr(C)]));
|
||||||
item2.attrs.push(quote_attr!(cx, #[derive(JSTraceable)]));
|
item2.attrs.push(quote_attr!(cx, #[derive(JSTraceable)]));
|
||||||
item2.attrs.push(quote_attr!(cx, #[derive(HeapSizeOf)]));
|
item2.attrs.push(quote_attr!(cx, #[derive(HeapSizeOf)]));
|
||||||
|
item2.attrs.push(quote_attr!(cx, #[derive(DomObject)]));
|
||||||
// The following attributes are only for internal usage
|
|
||||||
item2.attrs.push(quote_attr!(cx, #[_generate_reflector]));
|
|
||||||
// #[dom_struct] gets consumed, so this lets us keep around a residue
|
// #[dom_struct] gets consumed, so this lets us keep around a residue
|
||||||
// Do NOT register a modifier/decorator on this attribute
|
// Do NOT register a modifier/decorator on this attribute
|
||||||
item2.attrs.push(quote_attr!(cx, #[_dom_struct_marker]));
|
item2.attrs.push(quote_attr!(cx, #[_dom_struct_marker]));
|
||||||
|
|
|
@ -34,8 +34,6 @@ use syntax::symbol::Symbol;
|
||||||
/// Handles the auto-deriving for `#[derive(JSTraceable)]`
|
/// Handles the auto-deriving for `#[derive(JSTraceable)]`
|
||||||
pub mod jstraceable;
|
pub mod jstraceable;
|
||||||
pub mod lints;
|
pub mod lints;
|
||||||
/// Autogenerates implementations of DomObject on DOM structs
|
|
||||||
pub mod reflector;
|
|
||||||
/// Utilities for writing plugins
|
/// Utilities for writing plugins
|
||||||
mod utils;
|
mod utils;
|
||||||
|
|
||||||
|
@ -45,10 +43,6 @@ pub fn plugin_registrar(reg: &mut Registry) {
|
||||||
Symbol::intern("dom_struct"),
|
Symbol::intern("dom_struct"),
|
||||||
MultiModifier(box jstraceable::expand_dom_struct));
|
MultiModifier(box jstraceable::expand_dom_struct));
|
||||||
|
|
||||||
reg.register_syntax_extension(
|
|
||||||
Symbol::intern("_generate_reflector"),
|
|
||||||
MultiDecorator(box reflector::expand_reflector));
|
|
||||||
|
|
||||||
reg.register_late_lint_pass(box lints::unrooted_must_root::UnrootedPass::new());
|
reg.register_late_lint_pass(box lints::unrooted_must_root::UnrootedPass::new());
|
||||||
reg.register_late_lint_pass(box lints::privatize::PrivatizePass);
|
reg.register_late_lint_pass(box lints::privatize::PrivatizePass);
|
||||||
reg.register_late_lint_pass(box lints::inheritance_integrity::InheritancePass);
|
reg.register_late_lint_pass(box lints::inheritance_integrity::InheritancePass);
|
||||||
|
|
|
@ -1,82 +0,0 @@
|
||||||
/* 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/. */
|
|
||||||
|
|
||||||
use syntax::ast::{ItemKind, MetaItem};
|
|
||||||
use syntax::codemap::Span;
|
|
||||||
use syntax::ext::base::{Annotatable, ExtCtxt};
|
|
||||||
use utils::match_ty_unwrap;
|
|
||||||
|
|
||||||
|
|
||||||
pub fn expand_reflector(cx: &mut ExtCtxt, span: Span, _: &MetaItem, annotatable: &Annotatable,
|
|
||||||
push: &mut FnMut(Annotatable)) {
|
|
||||||
if let &Annotatable::Item(ref item) = annotatable {
|
|
||||||
if let ItemKind::Struct(ref def, _) = item.node {
|
|
||||||
if def.fields().is_empty() {
|
|
||||||
cx.span_err(span, "#[dom_struct] should have a reflector or \
|
|
||||||
parent dom struct as its first field");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
let struct_name = item.ident;
|
|
||||||
// This path has to be hardcoded, unfortunately, since we can't resolve paths at expansion time
|
|
||||||
match def.fields().iter().find(
|
|
||||||
|f| match_ty_unwrap(&*f.ty, &["dom", "bindings", "reflector", "Reflector"]).is_some()) {
|
|
||||||
// If it has a field that is a Reflector, use that
|
|
||||||
Some(f) => {
|
|
||||||
let field_name = f.ident;
|
|
||||||
let impl_item = quote_item!(cx,
|
|
||||||
impl ::dom::bindings::reflector::DomObject for $struct_name {
|
|
||||||
fn reflector<'a>(&'a self) -> &'a ::dom::bindings::reflector::Reflector {
|
|
||||||
&self.$field_name
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
let impl_item_mut = quote_item!(cx,
|
|
||||||
impl ::dom::bindings::reflector::MutDomObject for $struct_name {
|
|
||||||
fn init_reflector(&mut self, obj: *mut ::js::jsapi::JSObject) {
|
|
||||||
self.$field_name.set_jsobject(obj);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
impl_item.map(|it| push(Annotatable::Item(it)));
|
|
||||||
impl_item_mut.map(|it| push(Annotatable::Item(it)))
|
|
||||||
},
|
|
||||||
// Or just call it on the first field (supertype).
|
|
||||||
None => {
|
|
||||||
let field_name = def.fields()[0].ident;
|
|
||||||
let impl_item = quote_item!(cx,
|
|
||||||
impl ::dom::bindings::reflector::DomObject for $struct_name {
|
|
||||||
fn reflector<'a>(&'a self) -> &'a ::dom::bindings::reflector::Reflector {
|
|
||||||
self.$field_name.reflector()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
let impl_item_mut = quote_item!(cx,
|
|
||||||
impl ::dom::bindings::reflector::MutDomObject for $struct_name {
|
|
||||||
fn init_reflector(&mut self, obj: *mut ::js::jsapi::JSObject) {
|
|
||||||
self.$field_name.init_reflector(obj);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
impl_item.map(|it| push(Annotatable::Item(it)));
|
|
||||||
impl_item_mut.map(|it| push(Annotatable::Item(it)))
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let impl_item = quote_item!(cx,
|
|
||||||
impl ::js::conversions::ToJSValConvertible for $struct_name {
|
|
||||||
#[allow(unsafe_code)]
|
|
||||||
unsafe fn to_jsval(&self,
|
|
||||||
cx: *mut ::js::jsapi::JSContext,
|
|
||||||
rval: ::js::jsapi::MutableHandleValue) {
|
|
||||||
let object = ::dom::bindings::reflector::DomObject::reflector(self).get_jsobject();
|
|
||||||
object.to_jsval(cx, rval)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
impl_item.map(|it| push(Annotatable::Item(it)));
|
|
||||||
} else {
|
|
||||||
cx.span_err(span, "#[dom_struct] seems to have been applied to a non-struct");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -36,6 +36,7 @@ caseless = "0.1.0"
|
||||||
cookie = {version = "0.2.5", features = ["serialize-rustc"]}
|
cookie = {version = "0.2.5", features = ["serialize-rustc"]}
|
||||||
cssparser = {version = "0.7", features = ["heap_size", "serde-serialization"]}
|
cssparser = {version = "0.7", features = ["heap_size", "serde-serialization"]}
|
||||||
devtools_traits = {path = "../devtools_traits"}
|
devtools_traits = {path = "../devtools_traits"}
|
||||||
|
domobject_derive = {path = "../domobject_derive"}
|
||||||
encoding = "0.2"
|
encoding = "0.2"
|
||||||
euclid = "0.10.1"
|
euclid = "0.10.1"
|
||||||
fnv = "1.0"
|
fnv = "1.0"
|
||||||
|
|
|
@ -83,8 +83,20 @@ pub trait DomObject {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl DomObject for Reflector {
|
||||||
|
fn reflector(&self) -> &Self {
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// A trait to initialize the `Reflector` for a DOM object.
|
/// A trait to initialize the `Reflector` for a DOM object.
|
||||||
pub trait MutDomObject: DomObject {
|
pub trait MutDomObject: DomObject {
|
||||||
/// Initializes the Reflector
|
/// Initializes the Reflector
|
||||||
fn init_reflector(&mut self, obj: *mut JSObject);
|
fn init_reflector(&mut self, obj: *mut JSObject);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl MutDomObject for Reflector {
|
||||||
|
fn init_reflector(&mut self, obj: *mut JSObject) {
|
||||||
|
self.set_jsobject(obj)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -39,6 +39,8 @@ extern crate core;
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
extern crate cssparser;
|
extern crate cssparser;
|
||||||
extern crate devtools_traits;
|
extern crate devtools_traits;
|
||||||
|
#[macro_use]
|
||||||
|
extern crate domobject_derive;
|
||||||
extern crate encoding;
|
extern crate encoding;
|
||||||
extern crate euclid;
|
extern crate euclid;
|
||||||
extern crate fnv;
|
extern crate fnv;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue