mirror of
https://github.com/servo/servo.git
synced 2025-06-08 00:23:30 +00:00
This trait is used to hold onto the downcast and upcast functions of all castable IDL interfaces. A castable IDL interface is one which either derives from or is derived by other interfaces. The deriving relation is represented by implementations of marker trait DerivedFrom<T: Castable> generated in InheritTypes. /^[ ]*use dom::bindings::codegen::InheritTypes::.*(Base|Cast|Derived)/ { /::[a-zA-Z]+(Base|Cast|Derived);/d s/([{ ])[a-zA-Z]+(Base|Cast|Derived), /\1/g s/([{ ])[a-zA-Z]+(Base|Cast|Derived), /\1/g s/, [a-zA-Z]+(Base|Cast|Derived)([},])/\2/g s/, [a-zA-Z]+(Base|Cast|Derived)([},])/\2/g /\{([a-zA-Z]+(Base|Cast|Derived))?\};$/d s/\{([a-zA-Z_]+)\};$/\1;/ } s/([a-zA-Z]+)Cast::from_ref\(\&?\**([a-zA-Z_]+)(\.r\(\))?\)/\2.upcast::<\1>()/g s/([a-zA-Z]+)Cast::from_ref\(\&?\**([a-zA-Z_]+)(\.[a-zA-Z_]+\(\))?\)/\2\3.upcast::<\1>()/g s/\(([a-zA-Z]+)Cast::from_ref\)/\(Castable::upcast::<\1>\)/g s/([a-zA-Z]+)Cast::from_root/Root::upcast::<\1>/g s/([a-zA-Z]+)Cast::from_layout_js\(\&([a-zA-Z_.]+)\)/\2.upcast::<\1>()/g s/([a-zA-Z]+)Cast::to_ref\(\&?\**([a-zA-Z_]+)(\.r\(\))?\)/\2.downcast::<\1>()/g s/([a-zA-Z]+)Cast::to_ref\(\&?\**([a-zA-Z_]+)(\.[a-zA-Z_]+\(\))?\)/\2\3.downcast::<\1>()/g s/\(([a-zA-Z]+)Cast::to_ref\)/\(Castable::downcast::<\1>\)/g s/([a-zA-Z]+)Cast::to_root/Root::downcast::<\1>/g s/([a-zA-Z]+)Cast::to_layout_js\(&?([a-zA-Z_.]+(\(\))?)\)/\2.downcast::<\1>()/g s/\.is_document\(\)/.is::<Document>()/g s/\.is_htmlanchorelement\(\)/.is::<HTMLAnchorElement>()/g s/\.is_htmlappletelement\(\)/.is::<HTMLAppletElement>()/g s/\.is_htmlareaelement\(\)/.is::<HTMLAreaElement>()/g s/\.is_htmlbodyelement\(\)/.is::<HTMLBodyElement>()/g s/\.is_htmlembedelement\(\)/.is::<HTMLEmbedElement>()/g s/\.is_htmlfieldsetelement\(\)/.is::<HTMLFieldSetElement>()/g s/\.is_htmlformelement\(\)/.is::<HTMLFormElement>()/g s/\.is_htmlframesetelement\(\)/.is::<HTMLFrameSetElement>()/g s/\.is_htmlhtmlelement\(\)/.is::<HTMLHtmlElement>()/g s/\.is_htmlimageelement\(\)/.is::<HTMLImageElement>()/g s/\.is_htmllegendelement\(\)/.is::<HTMLLegendElement>()/g s/\.is_htmloptgroupelement\(\)/.is::<HTMLOptGroupElement>()/g s/\.is_htmloptionelement\(\)/.is::<HTMLOptionElement>()/g s/\.is_htmlscriptelement\(\)/.is::<HTMLScriptElement>()/g s/\.is_htmltabledatacellelement\(\)/.is::<HTMLTableDataCellElement>()/g s/\.is_htmltableheadercellelement\(\)/.is::<HTMLTableHeaderCellElement>()/g s/\.is_htmltablerowelement\(\)/.is::<HTMLTableRowElement>()/g s/\.is_htmltablesectionelement\(\)/.is::<HTMLTableSectionElement>()/g s/\.is_htmltitleelement\(\)/.is::<HTMLTitleElement>()/g
193 lines
7.6 KiB
Rust
193 lines
7.6 KiB
Rust
/* 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 devtools_traits::{CONSOLE_API, CachedConsoleMessage, CachedConsoleMessageTypes, PAGE_ERROR};
|
|
use devtools_traits::{ComputedNodeLayout, ConsoleAPI, PageError, ScriptToDevtoolsControlMsg};
|
|
use devtools_traits::{EvaluateJSReply, Modification, NodeInfo, TimelineMarker, TimelineMarkerType};
|
|
use dom::bindings::codegen::Bindings::DOMRectBinding::{DOMRectMethods};
|
|
use dom::bindings::codegen::Bindings::DocumentBinding::DocumentMethods;
|
|
use dom::bindings::codegen::Bindings::ElementBinding::{ElementMethods};
|
|
use dom::bindings::conversions::{Castable, FromJSValConvertible, jsstring_to_str};
|
|
use dom::bindings::global::GlobalRef;
|
|
use dom::bindings::js::Root;
|
|
use dom::element::Element;
|
|
use dom::node::Node;
|
|
use ipc_channel::ipc::IpcSender;
|
|
use js::jsapi::{ObjectClassName, RootedObject, RootedValue};
|
|
use js::jsval::UndefinedValue;
|
|
use msg::constellation_msg::PipelineId;
|
|
use page::{IterablePage, Page};
|
|
use script_task::get_page;
|
|
use std::ffi::CStr;
|
|
use std::rc::Rc;
|
|
use std::str;
|
|
use uuid::Uuid;
|
|
|
|
#[allow(unsafe_code)]
|
|
pub fn handle_evaluate_js(global: &GlobalRef, eval: String, reply: IpcSender<EvaluateJSReply>) {
|
|
let cx = global.get_cx();
|
|
let mut rval = RootedValue::new(cx, UndefinedValue());
|
|
global.evaluate_js_on_global_with_result(&eval, rval.handle_mut());
|
|
|
|
reply.send(if rval.ptr.is_undefined() {
|
|
EvaluateJSReply::VoidValue
|
|
} else if rval.ptr.is_boolean() {
|
|
EvaluateJSReply::BooleanValue(rval.ptr.to_boolean())
|
|
} else if rval.ptr.is_double() || rval.ptr.is_int32() {
|
|
EvaluateJSReply::NumberValue(
|
|
FromJSValConvertible::from_jsval(cx, rval.handle(), ()).unwrap())
|
|
} else if rval.ptr.is_string() {
|
|
EvaluateJSReply::StringValue(jsstring_to_str(cx, rval.ptr.to_string()))
|
|
} else if rval.ptr.is_null() {
|
|
EvaluateJSReply::NullValue
|
|
} else {
|
|
assert!(rval.ptr.is_object());
|
|
|
|
let obj = RootedObject::new(cx, rval.ptr.to_object());
|
|
let class_name = unsafe { CStr::from_ptr(ObjectClassName(cx, obj.handle())) };
|
|
let class_name = str::from_utf8(class_name.to_bytes()).unwrap();
|
|
|
|
EvaluateJSReply::ActorValue {
|
|
class: class_name.to_owned(),
|
|
uuid: Uuid::new_v4().to_string(),
|
|
}
|
|
}).unwrap();
|
|
}
|
|
|
|
pub fn handle_get_root_node(page: &Rc<Page>, pipeline: PipelineId, reply: IpcSender<NodeInfo>) {
|
|
let page = get_page(&*page, pipeline);
|
|
let document = page.document();
|
|
|
|
let node = document.upcast::<Node>();
|
|
reply.send(node.summarize()).unwrap();
|
|
}
|
|
|
|
pub fn handle_get_document_element(page: &Rc<Page>, pipeline: PipelineId, reply: IpcSender<NodeInfo>) {
|
|
let page = get_page(&*page, pipeline);
|
|
let document = page.document();
|
|
let document_element = document.r().GetDocumentElement().unwrap();
|
|
|
|
let node = document_element.upcast::<Node>();
|
|
reply.send(node.summarize()).unwrap();
|
|
}
|
|
|
|
fn find_node_by_unique_id(page: &Rc<Page>, pipeline: PipelineId, node_id: String) -> Root<Node> {
|
|
let page = get_page(&*page, pipeline);
|
|
let document = page.document();
|
|
let node = document.upcast::<Node>();
|
|
|
|
for candidate in node.traverse_preorder() {
|
|
if candidate.r().get_unique_id() == node_id {
|
|
return candidate;
|
|
}
|
|
}
|
|
|
|
panic!("couldn't find node with unique id {}", node_id)
|
|
}
|
|
|
|
pub fn handle_get_children(page: &Rc<Page>, pipeline: PipelineId, node_id: String, reply: IpcSender<Vec<NodeInfo>>) {
|
|
let parent = find_node_by_unique_id(&*page, pipeline, node_id);
|
|
let children = parent.r().children().map(|child| {
|
|
child.r().summarize()
|
|
}).collect();
|
|
reply.send(children).unwrap();
|
|
}
|
|
|
|
pub fn handle_get_layout(page: &Rc<Page>,
|
|
pipeline: PipelineId,
|
|
node_id: String,
|
|
reply: IpcSender<ComputedNodeLayout>) {
|
|
let node = find_node_by_unique_id(&*page, pipeline, node_id);
|
|
let elem = node.downcast::<Element>().expect("should be getting layout of element");
|
|
let rect = elem.GetBoundingClientRect();
|
|
let width = rect.Width() as f32;
|
|
let height = rect.Height() as f32;
|
|
reply.send(ComputedNodeLayout { width: width, height: height }).unwrap();
|
|
}
|
|
|
|
pub fn handle_get_cached_messages(_pipeline_id: PipelineId,
|
|
message_types: CachedConsoleMessageTypes,
|
|
reply: IpcSender<Vec<CachedConsoleMessage>>) {
|
|
//TODO: check the messageTypes against a global Cache for console messages and page exceptions
|
|
let mut messages = Vec::new();
|
|
if message_types.contains(PAGE_ERROR) {
|
|
//TODO: make script error reporter pass all reported errors
|
|
// to devtools and cache them for returning here.
|
|
let msg = PageError {
|
|
_type: "PageError".to_owned(),
|
|
errorMessage: "page error test".to_owned(),
|
|
sourceName: String::new(),
|
|
lineText: String::new(),
|
|
lineNumber: 0,
|
|
columnNumber: 0,
|
|
category: String::new(),
|
|
timeStamp: 0,
|
|
error: false,
|
|
warning: false,
|
|
exception: false,
|
|
strict: false,
|
|
private: false,
|
|
};
|
|
messages.push(CachedConsoleMessage::PageError(msg));
|
|
}
|
|
if message_types.contains(CONSOLE_API) {
|
|
//TODO: do for real
|
|
let msg = ConsoleAPI {
|
|
_type: "ConsoleAPI".to_owned(),
|
|
level: "error".to_owned(),
|
|
filename: "http://localhost/~mihai/mozilla/test.html".to_owned(),
|
|
lineNumber: 0,
|
|
functionName: String::new(),
|
|
timeStamp: 0,
|
|
private: false,
|
|
arguments: vec!["console error test".to_owned()],
|
|
};
|
|
messages.push(CachedConsoleMessage::ConsoleAPI(msg));
|
|
}
|
|
reply.send(messages).unwrap();
|
|
}
|
|
|
|
pub fn handle_modify_attribute(page: &Rc<Page>,
|
|
pipeline: PipelineId,
|
|
node_id: String,
|
|
modifications: Vec<Modification>) {
|
|
let node = find_node_by_unique_id(&*page, pipeline, node_id);
|
|
let elem = node.downcast::<Element>().expect("should be getting layout of element");
|
|
|
|
for modification in &modifications {
|
|
match modification.newValue {
|
|
Some(ref string) => {
|
|
let _ = elem.SetAttribute(modification.attributeName.clone(), string.clone());
|
|
},
|
|
None => elem.RemoveAttribute(modification.attributeName.clone()),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn handle_wants_live_notifications(global: &GlobalRef, send_notifications: bool) {
|
|
global.set_devtools_wants_updates(send_notifications);
|
|
}
|
|
|
|
pub fn handle_set_timeline_markers(page: &Rc<Page>,
|
|
marker_types: Vec<TimelineMarkerType>,
|
|
reply: IpcSender<TimelineMarker>) {
|
|
let window = page.window();
|
|
window.set_devtools_timeline_markers(marker_types, reply);
|
|
}
|
|
|
|
pub fn handle_drop_timeline_markers(page: &Rc<Page>,
|
|
marker_types: Vec<TimelineMarkerType>) {
|
|
let window = page.window();
|
|
window.drop_devtools_timeline_markers(marker_types);
|
|
}
|
|
|
|
pub fn handle_request_animation_frame(page: &Rc<Page>, id: PipelineId, actor_name: String) {
|
|
let page = page.find(id).expect("There is no such page");
|
|
let doc = page.document();
|
|
let devtools_sender = page.window().devtools_chan().unwrap();
|
|
doc.r().request_animation_frame(box move |time| {
|
|
let msg = ScriptToDevtoolsControlMsg::FramerateTick(actor_name, time);
|
|
devtools_sender.send(msg).unwrap();
|
|
});
|
|
}
|