1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85
// ┏━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓
// ┃ ██████ ██████ ██████ █ █ █ █ █ █▄ ▀███ █ ┃
// ┃ ▄▄▄▄▄█ █▄▄▄▄▄ ▄▄▄▄▄█ ▀▀▀▀▀█▀▀▀▀▀ █ ▀▀▀▀▀█ ████████▌▐███ ███▄ ▀█ █ ▀▀▀▀▀ ┃
// ┃ █▀▀▀▀▀ █▀▀▀▀▀ █▀██▀▀ ▄▄▄▄▄ █ ▄▄▄▄▄█ ▄▄▄▄▄█ ████████▌▐███ █████▄ █ ▄▄▄▄▄ ┃
// ┃ █ ██████ █ ▀█▄ █ ██████ █ ███▌▐███ ███████▄ █ ┃
// ┣━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┫
// ┃ Copyright (c) 2017, the Perspective Authors. ┃
// ┃ ╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌ ┃
// ┃ This file is part of the Perspective library, distributed under the terms ┃
// ┃ of the [Apache License 2.0](https://www.apache.org/licenses/LICENSE-2.0). ┃
// ┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛
use wasm_bindgen::prelude::*;
pub trait CustomElementMetadata {
const CUSTOM_ELEMENT_NAME: &'static str;
const STATICS: &'static [&'static str] = [].as_slice();
const TYPE_NAME: &'static str = std::any::type_name::<Self>();
fn struct_name() -> &'static str {
match &Self::TYPE_NAME.rfind(':') {
Some(pos) => &Self::TYPE_NAME[pos + 1..],
None => Self::TYPE_NAME,
}
}
}
#[wasm_bindgen(inline_js = r#"
export function bootstrap(psp, name, clsname, statics) {
const cls = psp[clsname];
const proto = cls.prototype;
class x extends HTMLElement {
constructor() {
super();
this._instance = new cls(this);
}
}
const names = Object.getOwnPropertyNames(proto);
for (const key of names) {
if ('get' in Object.getOwnPropertyDescriptor(proto, key)) {
Object.defineProperty(x.prototype, key, {
get: function() {
return this._instance[key];
}
});
} else {
Object.defineProperty(x.prototype, key, {
value: function(...args) {
return this._instance[key].call(this._instance, ...args);
}
});
}
}
for (const key of statics) {
Object.defineProperty(x, key, {
value: function(...args) {
return psp[key].call(psp, ...args);
}
});
}
Object.defineProperty(x, '__wasm_module__', {
get() {
return psp;
},
});
customElements.define(name, x);
}
"#)]
extern "C" {
#[wasm_bindgen(js_name = "bootstrap")]
fn js_bootstrap(psp: &JsValue, name: &str, cls: &str, statics: js_sys::Array) -> JsValue;
}
pub fn define_web_component<T: CustomElementMetadata>(module: &JsValue) {
js_bootstrap(
module,
T::CUSTOM_ELEMENT_NAME,
T::struct_name(),
T::STATICS.iter().cloned().map(JsValue::from).collect(),
);
}