aboutsummaryrefslogtreecommitdiffstats
path: root/components/script/dom/namednodemap.rs
blob: 75ea2bcfeb3a816c2a406c8ab1206169ac754abb (plain) (blame)
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
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
/* 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 dom::attr::Attr;
use dom::bindings::codegen::Bindings::NamedNodeMapBinding;
use dom::bindings::codegen::Bindings::NamedNodeMapBinding::NamedNodeMapMethods;
use dom::bindings::error::{Error, Fallible};
use dom::bindings::global::GlobalRef;
use dom::bindings::js::{JS, JSRef, Rootable, Temporary};
use dom::bindings::utils::{Reflector, reflect_dom_object};
use dom::element::{AttributeHandlers, Element, ElementHelpers};
use dom::window::Window;
use util::namespace;
use util::str::DOMString;

use string_cache::Atom;

#[dom_struct]
pub struct NamedNodeMap {
    reflector_: Reflector,
    owner: JS<Element>,
}

impl NamedNodeMap {
    fn new_inherited(elem: JSRef<Element>) -> NamedNodeMap {
        NamedNodeMap {
            reflector_: Reflector::new(),
            owner: JS::from_rooted(elem),
        }
    }

    pub fn new(window: JSRef<Window>, elem: JSRef<Element>) -> Temporary<NamedNodeMap> {
        reflect_dom_object(box NamedNodeMap::new_inherited(elem),
                           GlobalRef::Window(window), NamedNodeMapBinding::Wrap)
    }
}

impl<'a> NamedNodeMapMethods for JSRef<'a, NamedNodeMap> {
    // https://dom.spec.whatwg.org/#dom-namednodemap-length
    fn Length(self) -> u32 {
        let owner = self.owner.root();
        // FIXME(https://github.com/rust-lang/rust/issues/23338)
        let owner = owner.r();
        let attrs = owner.attrs();
        attrs.len() as u32
    }

    // https://dom.spec.whatwg.org/#dom-namednodemap-item
    fn Item(self, index: u32) -> Option<Temporary<Attr>> {
        let owner = self.owner.root();
        // FIXME(https://github.com/rust-lang/rust/issues/23338)
        let owner = owner.r();
        let attrs = owner.attrs();
        attrs.get(index as usize).map(|x| Temporary::from_rooted(x.clone()))
    }

    // https://dom.spec.whatwg.org/#dom-namednodemap-getnameditem
    fn GetNamedItem(self, name: DOMString) -> Option<Temporary<Attr>> {
        let owner = self.owner.root();
        // FIXME(https://github.com/rust-lang/rust/issues/23338)
        let owner = owner.r();
        owner.get_attribute_by_name(name)
    }

    // https://dom.spec.whatwg.org/#dom-namednodemap-getnameditemns
    fn GetNamedItemNS(self, namespace: Option<DOMString>, local_name: DOMString)
                     -> Option<Temporary<Attr>> {
        let owner = self.owner.root();
        // FIXME(https://github.com/rust-lang/rust/issues/23338)
        let owner = owner.r();
        let ns = namespace::from_domstring(namespace);
        owner.get_attribute(&ns, &Atom::from_slice(&local_name))
    }

    // https://dom.spec.whatwg.org/#dom-namednodemap-removenameditem
    fn RemoveNamedItem(self, name: DOMString) -> Fallible<Temporary<Attr>> {
        let owner = self.owner.root();
        // FIXME(https://github.com/rust-lang/rust/issues/23338)
        let owner = owner.r();
        let name = owner.parsed_name(name);
        owner.remove_attribute_by_name(&Atom::from_slice(&name)).ok_or(Error::NotFound)
    }

    // https://dom.spec.whatwg.org/#dom-namednodemap-removenameditemns
    fn RemoveNamedItemNS(self, namespace: Option<DOMString>, local_name: DOMString)
                      -> Fallible<Temporary<Attr>> {
        let owner = self.owner.root();
        // FIXME(https://github.com/rust-lang/rust/issues/23338)
        let owner = owner.r();
        let ns = namespace::from_domstring(namespace);
        owner.remove_attribute(&ns, &Atom::from_slice(&local_name)).ok_or(Error::NotFound)
    }

    fn IndexedGetter(self, index: u32, found: &mut bool) -> Option<Temporary<Attr>> {
        let item = self.Item(index);
        *found = item.is_some();
        item
    }

    fn NamedGetter(self, name: DOMString, found: &mut bool) -> Option<Temporary<Attr>> {
        let item = self.GetNamedItem(name);
        *found = item.is_some();
        item
    }
}