forked from parcel-bundler/lightningcss
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathrule_list.rs
More file actions
81 lines (71 loc) · 2.63 KB
/
rule_list.rs
File metadata and controls
81 lines (71 loc) · 2.63 KB
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
use napi::{CallContext, sys, JsNumber, JsObject, NapiRaw, NapiValue, JsUndefined, JsUnknown, JsFunction, Property, Ref, Result, Env};
use std::cell::RefCell;
use parcel_css::stylesheet::StyleSheet;
use std::rc::Rc;
use crate::rule;
// TODO: this is probably not safe.
static mut CLASS: RefCell<Option<Ref<()>>> = RefCell::new(None);
struct CSSRuleList {
stylesheet: Rc<RefCell<StyleSheet>>,
rules: Vec<sys::napi_ref>
}
#[js_function(0)]
fn constructor(ctx: CallContext) -> Result<JsUndefined> {
ctx.env.get_undefined()
}
#[js_function(0)]
fn get_length(ctx: CallContext) -> Result<JsNumber> {
let this: JsObject = ctx.this_unchecked();
let list: &mut CSSRuleList = ctx.env.unwrap(&this)?;
ctx.env.create_uint32(list.stylesheet.borrow().rules.0.len() as u32)
}
#[js_function(1)]
fn item(ctx: CallContext) -> Result<JsObject> {
let this: JsObject = ctx.this_unchecked();
let index = ctx.get::<JsNumber>(0)?.get_uint32()? as usize;
let list: &mut CSSRuleList = ctx.env.unwrap(&this)?;
// See if we already have a JS object created for this rule to preserve referential equality.
// e.g. rules.item(0) === rules.item(0)
// This drops down to low level C bindings for napi because there is currently no way
// to create a weak reference in napi-rs.
match &list.rules.get(index) {
Some(r) if !r.is_null() => {
let mut js_value = std::ptr::null_mut();
unsafe {
sys::napi_get_reference_value(ctx.env.raw(), **r, &mut js_value);
if !js_value.is_null() {
return JsObject::from_raw(ctx.env.raw(), js_value)
}
};
}
_ => {}
};
let r = list.stylesheet.borrow().rules.0[index].clone();
let obj = rule::create(ctx.env, r)?;
while list.rules.len() <= index {
list.rules.push(std::ptr::null_mut());
}
unsafe {
sys::napi_create_reference(ctx.env.raw(), obj.raw(), 0, &mut list.rules[index]);
};
Ok(obj)
}
pub fn init(exports: &mut JsObject, env: Env) -> Result<()> {
let stylesheet_class = env
.define_class("CSSRuleList", constructor, &[
Property::new(&env, "length")?.with_getter(get_length),
Property::new(&env, "item")?.with_method(item)
])?;
let mut c = unsafe { CLASS.borrow_mut() };
*c = Some(env.create_reference(&stylesheet_class)?);
exports.set_named_property("CSSRuleList", stylesheet_class)?;
Ok(())
}
pub fn create(env: &Env, stylesheet: Rc<RefCell<StyleSheet>>) -> Result<JsObject> {
let r = unsafe { CLASS.borrow() };
let r = r.as_ref().unwrap();
let c: JsFunction = env.get_reference_value(&r)?;
let mut instance = c.new::<JsUnknown>(&[])?;
env.wrap(&mut instance, CSSRuleList { stylesheet, rules: Vec::new() })?;
Ok(instance)
}