/* 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::bindings::codegen::BindingDeclarations::HTMLIFrameElementBinding; use dom::bindings::codegen::InheritTypes::{ElementCast, HTMLIFrameElementDerived, HTMLElementCast}; use dom::bindings::error::ErrorResult; use dom::bindings::js::{JSRef, Temporary}; use dom::bindings::trace::Untraceable; use dom::document::Document; use dom::element::{HTMLIFrameElementTypeId, Element}; use dom::element::AttributeHandlers; use dom::eventtarget::{EventTarget, NodeTargetTypeId}; use dom::htmlelement::HTMLElement; use dom::node::{Node, ElementNodeTypeId}; use dom::virtualmethods::VirtualMethods; use dom::window::Window; use servo_util::str::DOMString; use servo_msg::constellation_msg::{PipelineId, SubpageId}; use std::ascii::StrAsciiExt; use url::Url; enum SandboxAllowance { AllowNothing = 0x00, AllowSameOrigin = 0x01, AllowTopNavigation = 0x02, AllowForms = 0x04, AllowScripts = 0x08, AllowPointerLock = 0x10, AllowPopups = 0x20 } #[deriving(Encodable)] pub struct HTMLIFrameElement { pub htmlelement: HTMLElement, frame: Untraceable>, pub size: Option, pub sandbox: Option } impl HTMLIFrameElementDerived for EventTarget { fn is_htmliframeelement(&self) -> bool { match self.type_id { NodeTargetTypeId(ElementNodeTypeId(HTMLIFrameElementTypeId)) => true, _ => false } } } #[deriving(Encodable)] pub struct IFrameSize { pub pipeline_id: PipelineId, pub subpage_id: SubpageId, } pub trait HTMLIFrameElementHelpers { fn is_sandboxed(&self) -> bool; fn set_frame(&mut self, frame: Url); } impl<'a> HTMLIFrameElementHelpers for JSRef<'a, HTMLIFrameElement> { fn is_sandboxed(&self) -> bool { self.sandbox.is_some() } fn set_frame(&mut self, frame: Url) { *self.frame = Some(frame); } } impl HTMLIFrameElement { pub fn new_inherited(localName: DOMString, document: &JSRef) -> HTMLIFrameElement { HTMLIFrameElement { htmlelement: HTMLElement::new_inherited(HTMLIFrameElementTypeId, localName, document), frame: Untraceable::new(None), size: None, sandbox: None, } } pub fn new(localName: DOMString, document: &JSRef) -> Temporary { let element = HTMLIFrameElement::new_inherited(localName, document); Node::reflect_node(~element, document, HTMLIFrameElementBinding::Wrap) } } pub trait HTMLIFrameElementMethods { fn Src(&self) -> DOMString; fn SetSrc(&mut self, _src: DOMString) -> ErrorResult; fn Srcdoc(&self) -> DOMString; fn SetSrcdoc(&mut self, _srcdoc: DOMString) -> ErrorResult; fn Name(&self) -> DOMString; fn SetName(&mut self, _name: DOMString) -> ErrorResult; fn Sandbox(&self) -> DOMString; fn SetSandbox(&mut self, sandbox: DOMString); fn AllowFullscreen(&self) -> bool; fn SetAllowFullscreen(&mut self, _allow: bool) -> ErrorResult; fn Width(&self) -> DOMString; fn SetWidth(&mut self, _width: DOMString) -> ErrorResult; fn Height(&self) -> DOMString; fn SetHeight(&mut self, _height: DOMString) -> ErrorResult; fn GetContentDocument(&self) -> Option>; fn GetContentWindow(&self) -> Option>; fn Align(&self) -> DOMString; fn SetAlign(&mut self, _align: DOMString) -> ErrorResult; fn Scrolling(&self) -> DOMString; fn SetScrolling(&mut self, _scrolling: DOMString) -> ErrorResult; fn FrameBorder(&self) -> DOMString; fn SetFrameBorder(&mut self, _frameborder: DOMString) -> ErrorResult; fn LongDesc(&self) -> DOMString; fn SetLongDesc(&mut self, _longdesc: DOMString) -> ErrorResult; fn MarginHeight(&self) -> DOMString; fn SetMarginHeight(&mut self, _marginheight: DOMString) -> ErrorResult; fn MarginWidth(&self) -> DOMString; fn SetMarginWidth(&mut self, _marginwidth: DOMString) -> ErrorResult; fn GetSVGDocument(&self) -> Option>; } impl<'a> HTMLIFrameElementMethods for JSRef<'a, HTMLIFrameElement> { fn Src(&self) -> DOMString { ~"" } fn SetSrc(&mut self, _src: DOMString) -> ErrorResult { Ok(()) } fn Srcdoc(&self) -> DOMString { ~"" } fn SetSrcdoc(&mut self, _srcdoc: DOMString) -> ErrorResult { Ok(()) } fn Name(&self) -> DOMString { ~"" } fn SetName(&mut self, _name: DOMString) -> ErrorResult { Ok(()) } fn Sandbox(&self) -> DOMString { let element: &JSRef = ElementCast::from_ref(self); element.get_string_attribute("sandbox") } fn SetSandbox(&mut self, sandbox: DOMString) { let element: &mut JSRef = ElementCast::from_mut_ref(self); element.set_string_attribute("sandbox", sandbox); } fn AllowFullscreen(&self) -> bool { false } fn SetAllowFullscreen(&mut self, _allow: bool) -> ErrorResult { Ok(()) } fn Width(&self) -> DOMString { ~"" } fn SetWidth(&mut self, _width: DOMString) -> ErrorResult { Ok(()) } fn Height(&self) -> DOMString { ~"" } fn SetHeight(&mut self, _height: DOMString) -> ErrorResult { Ok(()) } fn GetContentDocument(&self) -> Option> { None } fn GetContentWindow(&self) -> Option> { None } fn Align(&self) -> DOMString { ~"" } fn SetAlign(&mut self, _align: DOMString) -> ErrorResult { Ok(()) } fn Scrolling(&self) -> DOMString { ~"" } fn SetScrolling(&mut self, _scrolling: DOMString) -> ErrorResult { Ok(()) } fn FrameBorder(&self) -> DOMString { ~"" } fn SetFrameBorder(&mut self, _frameborder: DOMString) -> ErrorResult { Ok(()) } fn LongDesc(&self) -> DOMString { ~"" } fn SetLongDesc(&mut self, _longdesc: DOMString) -> ErrorResult { Ok(()) } fn MarginHeight(&self) -> DOMString { ~"" } fn SetMarginHeight(&mut self, _marginheight: DOMString) -> ErrorResult { Ok(()) } fn MarginWidth(&self) -> DOMString { ~"" } fn SetMarginWidth(&mut self, _marginwidth: DOMString) -> ErrorResult { Ok(()) } fn GetSVGDocument(&self) -> Option> { None } } impl<'a> VirtualMethods for JSRef<'a, HTMLIFrameElement> { fn super_type<'a>(&'a mut self) -> Option<&'a mut VirtualMethods:> { let htmlelement: &mut JSRef = HTMLElementCast::from_mut_ref(self); Some(htmlelement as &mut VirtualMethods:) } fn after_set_attr(&mut self, name: DOMString, value: DOMString) { match self.super_type() { Some(ref mut s) => s.after_set_attr(name.clone(), value.clone()), _ => (), } if "sandbox" == name { let mut modes = AllowNothing as u8; for word in value.split(' ') { // FIXME: Workaround for https://github.com/mozilla/rust/issues/10683 let word_lower = word.to_ascii_lower(); modes |= match word_lower.as_slice() { "allow-same-origin" => AllowSameOrigin, "allow-forms" => AllowForms, "allow-pointer-lock" => AllowPointerLock, "allow-popups" => AllowPopups, "allow-scripts" => AllowScripts, "allow-top-navigation" => AllowTopNavigation, _ => AllowNothing } as u8; } self.deref_mut().sandbox = Some(modes); } } fn before_remove_attr(&mut self, name: DOMString, value: DOMString) { match self.super_type() { Some(ref mut s) => s.before_remove_attr(name.clone(), value), _ => (), } if "sandbox" == name { self.deref_mut().sandbox = None; } } }