OLD | NEW |
1 // Copyright 2016 The Chromium Authors. All rights reserved. | 1 // Copyright 2016 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "ash/devtools/ash_devtools_dom_agent.h" | 5 #include "ash/devtools/ash_devtools_dom_agent.h" |
6 | 6 |
| 7 #include "ash/devtools/ui_element.h" |
| 8 #include "ash/devtools/view_element.h" |
| 9 #include "ash/devtools/widget_element.h" |
| 10 #include "ash/devtools/window_element.h" |
7 #include "ash/public/cpp/shell_window_ids.h" | 11 #include "ash/public/cpp/shell_window_ids.h" |
8 #include "ash/root_window_controller.h" | 12 #include "ash/root_window_controller.h" |
9 #include "ash/shell.h" | 13 #include "ash/shell.h" |
10 #include "components/ui_devtools/devtools_server.h" | 14 #include "components/ui_devtools/devtools_server.h" |
11 #include "third_party/skia/include/core/SkColor.h" | 15 #include "third_party/skia/include/core/SkColor.h" |
12 #include "ui/aura/window.h" | 16 #include "ui/aura/window.h" |
13 #include "ui/display/display.h" | 17 #include "ui/display/display.h" |
14 #include "ui/views/background.h" | 18 #include "ui/views/background.h" |
15 #include "ui/views/border.h" | 19 #include "ui/views/border.h" |
| 20 #include "ui/views/view.h" |
| 21 #include "ui/views/widget/widget.h" |
16 #include "ui/wm/core/window_util.h" | 22 #include "ui/wm/core/window_util.h" |
17 | 23 |
18 namespace ash { | 24 namespace ash { |
19 namespace devtools { | 25 namespace devtools { |
| 26 namespace { |
20 | 27 |
21 namespace { | |
22 using namespace ui::devtools::protocol; | 28 using namespace ui::devtools::protocol; |
23 // TODO(mhashmi): Make ids reusable | 29 // TODO(mhashmi): Make ids reusable |
24 DOM::NodeId node_ids = 1; | |
25 | 30 |
26 std::unique_ptr<DOM::Node> BuildNode( | 31 std::unique_ptr<DOM::Node> BuildNode( |
27 const std::string& name, | 32 const std::string& name, |
28 std::unique_ptr<Array<std::string>> attributes, | 33 std::unique_ptr<Array<std::string>> attributes, |
29 std::unique_ptr<Array<DOM::Node>> children) { | 34 std::unique_ptr<Array<DOM::Node>> children, |
| 35 int node_ids) { |
30 constexpr int kDomElementNodeType = 1; | 36 constexpr int kDomElementNodeType = 1; |
31 std::unique_ptr<DOM::Node> node = DOM::Node::create() | 37 std::unique_ptr<DOM::Node> node = DOM::Node::create() |
32 .setNodeId(node_ids++) | 38 .setNodeId(node_ids) |
33 .setNodeName(name) | 39 .setNodeName(name) |
34 .setNodeType(kDomElementNodeType) | 40 .setNodeType(kDomElementNodeType) |
35 .setAttributes(std::move(attributes)) | 41 .setAttributes(std::move(attributes)) |
36 .build(); | 42 .build(); |
37 node->setChildNodeCount(children->length()); | 43 node->setChildNodeCount(children->length()); |
38 node->setChildren(std::move(children)); | 44 node->setChildren(std::move(children)); |
39 return node; | 45 return node; |
40 } | 46 } |
41 | 47 |
42 std::unique_ptr<Array<std::string>> GetAttributes(const aura::Window* window) { | 48 std::unique_ptr<Array<std::string>> GetAttributes(UIElement* ui_element) { |
43 std::unique_ptr<Array<std::string>> attributes = Array<std::string>::create(); | 49 std::unique_ptr<Array<std::string>> attributes = Array<std::string>::create(); |
44 attributes->addItem("name"); | 50 attributes->addItem("name"); |
45 attributes->addItem(window->GetName()); | 51 switch (ui_element->type()) { |
46 attributes->addItem("active"); | 52 case UIElementType::WINDOW: { |
47 attributes->addItem(::wm::IsActiveWindow(window) ? "true" : "false"); | 53 aura::Window* window = |
| 54 UIElement::GetBackingElement<aura::Window, WindowElement>(ui_element); |
| 55 attributes->addItem(window->GetName()); |
| 56 attributes->addItem("active"); |
| 57 attributes->addItem(::wm::IsActiveWindow(window) ? "true" : "false"); |
| 58 break; |
| 59 } |
| 60 case UIElementType::WIDGET: { |
| 61 views::Widget* widget = |
| 62 UIElement::GetBackingElement<views::Widget, WidgetElement>( |
| 63 ui_element); |
| 64 attributes->addItem(widget->GetName()); |
| 65 attributes->addItem("active"); |
| 66 attributes->addItem(widget->IsActive() ? "true" : "false"); |
| 67 break; |
| 68 } |
| 69 case UIElementType::VIEW: { |
| 70 attributes->addItem( |
| 71 UIElement::GetBackingElement<views::View, ViewElement>(ui_element) |
| 72 ->GetClassName()); |
| 73 break; |
| 74 } |
| 75 default: |
| 76 DCHECK(false); |
| 77 } |
48 return attributes; | 78 return attributes; |
49 } | 79 } |
50 | 80 |
51 std::unique_ptr<Array<std::string>> GetAttributes(const views::Widget* widget) { | |
52 std::unique_ptr<Array<std::string>> attributes = Array<std::string>::create(); | |
53 attributes->addItem("name"); | |
54 attributes->addItem(widget->GetName()); | |
55 attributes->addItem("active"); | |
56 attributes->addItem(widget->IsActive() ? "true" : "false"); | |
57 return attributes; | |
58 } | |
59 | |
60 std::unique_ptr<Array<std::string>> GetAttributes(const views::View* view) { | |
61 std::unique_ptr<Array<std::string>> attributes = Array<std::string>::create(); | |
62 attributes->addItem("name"); | |
63 attributes->addItem(view->GetClassName()); | |
64 return attributes; | |
65 } | |
66 | |
67 aura::Window* FindPreviousSibling(aura::Window* window) { | |
68 const aura::Window::Windows& siblings = window->parent()->children(); | |
69 auto it = std::find(siblings.begin(), siblings.end(), window); | |
70 DCHECK(it != siblings.end()); | |
71 // If this is the first child of its parent, the previous sibling is null | |
72 return it == siblings.begin() ? nullptr : *std::prev(it); | |
73 } | |
74 | |
75 views::View* FindPreviousSibling(views::View* view) { | |
76 views::View* parent = view->parent(); | |
77 int view_index = -1; | |
78 for (int i = 0, count = parent->child_count(); i < count; i++) { | |
79 if (view == parent->child_at(i)) { | |
80 view_index = i; | |
81 break; | |
82 } | |
83 } | |
84 DCHECK_GE(view_index, 0); | |
85 return view_index == 0 ? nullptr : parent->child_at(view_index - 1); | |
86 } | |
87 | |
88 int MaskColor(int value) { | 81 int MaskColor(int value) { |
89 return value & 0xff; | 82 return value & 0xff; |
90 } | 83 } |
91 | 84 |
92 SkColor RGBAToSkColor(DOM::RGBA* rgba) { | 85 SkColor RGBAToSkColor(DOM::RGBA* rgba) { |
93 if (!rgba) | 86 if (!rgba) |
94 return SkColorSetARGB(0, 0, 0, 0); | 87 return SkColorSetARGB(0, 0, 0, 0); |
95 // Default alpha value is 0 (not visible) and need to convert alpha decimal | 88 // Default alpha value is 0 (not visible) and need to convert alpha decimal |
96 // percentage value to hex | 89 // percentage value to hex |
97 return SkColorSetARGB(MaskColor(static_cast<int>(rgba->getA(0) * 255)), | 90 return SkColorSetARGB(MaskColor(static_cast<int>(rgba->getA(0) * 255)), |
98 MaskColor(rgba->getR()), MaskColor(rgba->getG()), | 91 MaskColor(rgba->getR()), MaskColor(rgba->getG()), |
99 MaskColor(rgba->getB())); | 92 MaskColor(rgba->getB())); |
100 } | 93 } |
101 | 94 |
102 views::Widget* GetWidgetFromWindow(aura::Window* window) { | 95 views::Widget* GetWidgetFromWindow(aura::Window* window) { |
103 return views::Widget::GetWidgetForNativeView(window); | 96 return views::Widget::GetWidgetForNativeView(window); |
104 } | 97 } |
105 | 98 |
106 } // namespace | 99 } // namespace |
107 | 100 |
108 AshDevToolsDOMAgent::AshDevToolsDOMAgent() {} | 101 AshDevToolsDOMAgent::AshDevToolsDOMAgent() { |
| 102 is_building_tree_ = false; |
| 103 } |
109 | 104 |
110 AshDevToolsDOMAgent::~AshDevToolsDOMAgent() { | 105 AshDevToolsDOMAgent::~AshDevToolsDOMAgent() { |
111 RemoveObservers(); | 106 Reset(); |
112 } | 107 } |
113 | 108 |
114 ui::devtools::protocol::Response AshDevToolsDOMAgent::disable() { | 109 ui::devtools::protocol::Response AshDevToolsDOMAgent::disable() { |
115 Reset(); | 110 Reset(); |
116 return ui::devtools::protocol::Response::OK(); | 111 return ui::devtools::protocol::Response::OK(); |
117 } | 112 } |
118 | 113 |
119 ui::devtools::protocol::Response AshDevToolsDOMAgent::getDocument( | 114 ui::devtools::protocol::Response AshDevToolsDOMAgent::getDocument( |
120 std::unique_ptr<ui::devtools::protocol::DOM::Node>* out_root) { | 115 std::unique_ptr<ui::devtools::protocol::DOM::Node>* out_root) { |
121 *out_root = BuildInitialTree(); | 116 *out_root = BuildInitialTree(); |
122 return ui::devtools::protocol::Response::OK(); | 117 return ui::devtools::protocol::Response::OK(); |
123 } | 118 } |
124 | 119 |
125 ui::devtools::protocol::Response AshDevToolsDOMAgent::highlightNode( | 120 ui::devtools::protocol::Response AshDevToolsDOMAgent::highlightNode( |
126 std::unique_ptr<ui::devtools::protocol::DOM::HighlightConfig> | 121 std::unique_ptr<ui::devtools::protocol::DOM::HighlightConfig> |
127 highlight_config, | 122 highlight_config, |
128 ui::devtools::protocol::Maybe<int> node_id) { | 123 ui::devtools::protocol::Maybe<int> node_id) { |
129 return HighlightNode(std::move(highlight_config), node_id.fromJust()); | 124 return HighlightNode(std::move(highlight_config), node_id.fromJust()); |
130 } | 125 } |
131 | 126 |
132 ui::devtools::protocol::Response AshDevToolsDOMAgent::hideHighlight() { | 127 ui::devtools::protocol::Response AshDevToolsDOMAgent::hideHighlight() { |
133 if (widget_for_highlighting_ && widget_for_highlighting_->IsVisible()) | 128 if (widget_for_highlighting_ && widget_for_highlighting_->IsVisible()) |
134 widget_for_highlighting_->Hide(); | 129 widget_for_highlighting_->Hide(); |
135 return ui::devtools::protocol::Response::OK(); | 130 return ui::devtools::protocol::Response::OK(); |
136 } | 131 } |
137 | 132 |
138 // Handles removing windows. | 133 void AshDevToolsDOMAgent::OnUIElementAdded(UIElement* parent, |
139 void AshDevToolsDOMAgent::OnWindowHierarchyChanging( | 134 UIElement* child) { |
140 const HierarchyChangeParams& params) { | 135 // If tree is being built, don't add child to dom tree again. |
141 // Only trigger this when params.receiver == params.old_parent. | 136 if (is_building_tree_) |
142 // Only removals are handled here. Removing a node can occur as a result of | 137 return; |
143 // reorganizing a window or just destroying it. OnWindowHierarchyChanged | 138 DCHECK(node_id_to_ui_element_.count(parent->node_id())); |
144 // is only called if there is a new_parent. The only case this method isn't | 139 |
145 // called is when adding a node because old_parent is then null. | 140 const auto& children = parent->children(); |
146 // Finally, We only trigger this 0 or 1 times as an old_parent will | 141 auto iter = std::find(children.begin(), children.end(), child); |
147 // either exist and only call this callback once, or not at all. | 142 int prev_node_id = |
148 if (params.receiver == params.old_parent) | 143 (iter == children.end() - 1) ? 0 : (*std::next(iter))->node_id(); |
149 RemoveWindowTree(params.target, true); | 144 frontend()->childNodeInserted(parent->node_id(), prev_node_id, |
| 145 BuildTreeForUIElement(child)); |
150 } | 146 } |
151 | 147 |
152 // Handles adding windows. | 148 void AshDevToolsDOMAgent::OnUIElementReordered(UIElement* parent, |
153 void AshDevToolsDOMAgent::OnWindowHierarchyChanged( | 149 UIElement* child) { |
154 const HierarchyChangeParams& params) { | 150 if (child->type() == UIElementType::WINDOW && |
155 // Only trigger this when params.receiver == params.new_parent. | 151 IsHighlightingWindow( |
156 // If there is an old_parent + new_parent, then this window's node was | 152 UIElement::GetBackingElement<aura::Window, WindowElement>(child))) |
157 // removed in OnWindowHierarchyChanging and will now be added to the | 153 return; |
158 // new_parent. If there is only a new_parent, OnWindowHierarchyChanging is | 154 DCHECK(node_id_to_ui_element_.count(parent->node_id())); |
159 // never called and the window is only added here. | 155 |
160 if (params.receiver == params.new_parent) | 156 const auto& children = parent->children(); |
161 AddWindowTree(params.target); | 157 auto iter = std::find(children.begin(), children.end(), child); |
| 158 int prev_node_id = |
| 159 (iter == children.begin()) ? 0 : (*std::prev(iter))->node_id(); |
| 160 RemoveDomNode(child); |
| 161 frontend()->childNodeInserted(parent->node_id(), prev_node_id, |
| 162 BuildDomNodeFromUIElement(child)); |
162 } | 163 } |
163 | 164 |
164 void AshDevToolsDOMAgent::OnWindowStackingChanged(aura::Window* window) { | 165 void AshDevToolsDOMAgent::AddNodeIdMap(UIElement* ui_element) { |
165 RemoveWindowTree(window, false); | 166 node_id_to_ui_element_[ui_element->node_id()] = ui_element; |
166 AddWindowTree(window); | |
167 } | 167 } |
168 | 168 |
169 void AshDevToolsDOMAgent::OnWindowBoundsChanged(aura::Window* window, | 169 void AshDevToolsDOMAgent::RemoveNodeIdMap(UIElement* ui_element) { |
170 const gfx::Rect& old_bounds, | 170 node_id_to_ui_element_.erase(ui_element->node_id()); |
171 const gfx::Rect& new_bounds) { | |
172 for (auto& observer : observers_) | |
173 observer.OnWindowBoundsChanged(window); | |
174 } | 171 } |
175 | 172 |
176 void AshDevToolsDOMAgent::OnWillRemoveView(views::Widget* widget, | 173 bool AshDevToolsDOMAgent::OnUIElementRemoved(UIElement* ui_element) { |
177 views::View* view) { | 174 DCHECK(node_id_to_ui_element_.count(ui_element->node_id())); |
178 if (view == widget->GetRootView()) | 175 |
179 RemoveViewTree(view, nullptr, true); | 176 if (ui_element->type() == UIElementType::WINDOW && |
| 177 IsHighlightingWindow(static_cast<WindowElement*>(ui_element)->window())) |
| 178 return false; |
| 179 RemoveDomNode(ui_element); |
| 180 RemoveNodeIdMap(ui_element); |
| 181 return true; |
180 } | 182 } |
181 | 183 |
182 void AshDevToolsDOMAgent::OnWidgetBoundsChanged(views::Widget* widget, | 184 void AshDevToolsDOMAgent::OnUIElementBoundsChanged(UIElement* ui_element) { |
183 const gfx::Rect& new_bounds) { | |
184 for (auto& observer : observers_) | 185 for (auto& observer : observers_) |
185 observer.OnWidgetBoundsChanged(widget); | 186 observer.OnNodeBoundsChanged(ui_element->node_id()); |
186 } | 187 } |
187 | 188 |
188 void AshDevToolsDOMAgent::OnChildViewRemoved(views::View* parent, | 189 bool AshDevToolsDOMAgent::IsHighlightingWindow(aura::Window* window) { |
189 views::View* view) { | 190 return widget_for_highlighting_ && |
190 RemoveViewTree(view, parent, true); | 191 GetWidgetFromWindow(window) == widget_for_highlighting_.get(); |
191 } | |
192 | |
193 void AshDevToolsDOMAgent::OnChildViewAdded(views::View* parent, | |
194 views::View* view) { | |
195 AddViewTree(view); | |
196 } | |
197 | |
198 void AshDevToolsDOMAgent::OnChildViewReordered(views::View* parent, | |
199 views::View* view) { | |
200 RemoveViewTree(view, parent, false); | |
201 AddViewTree(view); | |
202 } | |
203 | |
204 void AshDevToolsDOMAgent::OnViewBoundsChanged(views::View* view) { | |
205 for (auto& observer : observers_) | |
206 observer.OnViewBoundsChanged(view); | |
207 } | |
208 | |
209 aura::Window* AshDevToolsDOMAgent::GetWindowFromNodeId(int nodeId) { | |
210 return node_id_to_window_map_.count(nodeId) ? node_id_to_window_map_[nodeId] | |
211 : nullptr; | |
212 } | |
213 | |
214 views::Widget* AshDevToolsDOMAgent::GetWidgetFromNodeId(int nodeId) { | |
215 return node_id_to_widget_map_.count(nodeId) ? node_id_to_widget_map_[nodeId] | |
216 : nullptr; | |
217 } | |
218 | |
219 views::View* AshDevToolsDOMAgent::GetViewFromNodeId(int nodeId) { | |
220 return node_id_to_view_map_.count(nodeId) ? node_id_to_view_map_[nodeId] | |
221 : nullptr; | |
222 } | |
223 | |
224 int AshDevToolsDOMAgent::GetNodeIdFromWindow(aura::Window* window) { | |
225 DCHECK(window_to_node_id_map_.count(window)); | |
226 return window_to_node_id_map_[window]; | |
227 } | |
228 | |
229 int AshDevToolsDOMAgent::GetNodeIdFromWidget(views::Widget* widget) { | |
230 DCHECK(widget_to_node_id_map_.count(widget)); | |
231 return widget_to_node_id_map_[widget]; | |
232 } | |
233 | |
234 int AshDevToolsDOMAgent::GetNodeIdFromView(views::View* view) { | |
235 DCHECK(view_to_node_id_map_.count(view)); | |
236 return view_to_node_id_map_[view]; | |
237 } | 192 } |
238 | 193 |
239 void AshDevToolsDOMAgent::AddObserver(AshDevToolsDOMAgentObserver* observer) { | 194 void AshDevToolsDOMAgent::AddObserver(AshDevToolsDOMAgentObserver* observer) { |
240 observers_.AddObserver(observer); | 195 observers_.AddObserver(observer); |
241 } | 196 } |
242 | 197 |
243 void AshDevToolsDOMAgent::RemoveObserver( | 198 void AshDevToolsDOMAgent::RemoveObserver( |
244 AshDevToolsDOMAgentObserver* observer) { | 199 AshDevToolsDOMAgentObserver* observer) { |
245 observers_.RemoveObserver(observer); | 200 observers_.RemoveObserver(observer); |
246 } | 201 } |
247 | 202 |
| 203 UIElement* AshDevToolsDOMAgent::GetElementFromNodeId(int node_id) { |
| 204 return node_id_to_ui_element_[node_id]; |
| 205 } |
| 206 |
| 207 void AshDevToolsDOMAgent::OnNodeBoundsChanged(int node_id) { |
| 208 for (auto& observer : observers_) |
| 209 observer.OnNodeBoundsChanged(node_id); |
| 210 } |
| 211 |
248 std::unique_ptr<ui::devtools::protocol::DOM::Node> | 212 std::unique_ptr<ui::devtools::protocol::DOM::Node> |
249 AshDevToolsDOMAgent::BuildInitialTree() { | 213 AshDevToolsDOMAgent::BuildInitialTree() { |
| 214 is_building_tree_ = true; |
250 std::unique_ptr<Array<DOM::Node>> children = Array<DOM::Node>::create(); | 215 std::unique_ptr<Array<DOM::Node>> children = Array<DOM::Node>::create(); |
251 for (aura::Window* window : Shell::GetAllRootWindows()) | 216 window_element_root_ = new WindowElement(nullptr, this, nullptr); |
252 children->addItem(BuildTreeForWindow(window)); | 217 for (aura::Window* window : Shell::GetAllRootWindows()) { |
253 return BuildNode("root", nullptr, std::move(children)); | 218 UIElement* window_element = |
| 219 new WindowElement(window, this, window_element_root_); |
| 220 |
| 221 children->addItem(BuildTreeForUIElement(window_element)); |
| 222 window_element_root_->AddChild(window_element); |
| 223 } |
| 224 std::unique_ptr<ui::devtools::protocol::DOM::Node> root_node = BuildNode( |
| 225 "root", nullptr, std::move(children), window_element_root_->node_id()); |
| 226 is_building_tree_ = false; |
| 227 return root_node; |
| 228 } |
| 229 |
| 230 std::unique_ptr<ui::devtools::protocol::DOM::Node> |
| 231 AshDevToolsDOMAgent::BuildTreeForUIElement(UIElement* ui_element) { |
| 232 if (ui_element->type() == UIElementType::WINDOW) { |
| 233 return BuildTreeForWindow( |
| 234 ui_element, |
| 235 UIElement::GetBackingElement<aura::Window, WindowElement>(ui_element)); |
| 236 } else if (ui_element->type() == UIElementType::WIDGET) { |
| 237 return BuildTreeForRootWidget( |
| 238 ui_element, |
| 239 UIElement::GetBackingElement<views::Widget, WidgetElement>(ui_element)); |
| 240 } else if (ui_element->type() == UIElementType::VIEW) { |
| 241 return BuildTreeForView( |
| 242 ui_element, |
| 243 UIElement::GetBackingElement<views::View, ViewElement>(ui_element)); |
| 244 } |
| 245 return nullptr; |
254 } | 246 } |
255 | 247 |
256 std::unique_ptr<DOM::Node> AshDevToolsDOMAgent::BuildTreeForWindow( | 248 std::unique_ptr<DOM::Node> AshDevToolsDOMAgent::BuildTreeForWindow( |
| 249 UIElement* window_element_root, |
257 aura::Window* window) { | 250 aura::Window* window) { |
258 DCHECK(!window_to_node_id_map_.count(window)); | |
259 std::unique_ptr<Array<DOM::Node>> children = Array<DOM::Node>::create(); | 251 std::unique_ptr<Array<DOM::Node>> children = Array<DOM::Node>::create(); |
260 views::Widget* widget = GetWidgetFromWindow(window); | 252 views::Widget* widget = GetWidgetFromWindow(window); |
261 if (widget) | 253 if (widget) { |
262 children->addItem(BuildTreeForRootWidget(widget)); | 254 UIElement* widget_element = |
| 255 new WidgetElement(widget, this, window_element_root); |
| 256 |
| 257 children->addItem(BuildTreeForRootWidget(widget_element, widget)); |
| 258 window_element_root->AddChild(widget_element); |
| 259 } |
263 for (aura::Window* child : window->children()) { | 260 for (aura::Window* child : window->children()) { |
264 if (!IsHighlightingWindow(child)) | 261 if (!IsHighlightingWindow(child)) { |
265 children->addItem(BuildTreeForWindow(child)); | 262 UIElement* window_element = |
| 263 new WindowElement(child, this, window_element_root); |
| 264 |
| 265 children->addItem(BuildTreeForWindow(window_element, child)); |
| 266 window_element_root->AddChild(window_element); |
| 267 } |
266 } | 268 } |
267 | |
268 std::unique_ptr<ui::devtools::protocol::DOM::Node> node = | 269 std::unique_ptr<ui::devtools::protocol::DOM::Node> node = |
269 BuildNode("Window", GetAttributes(window), std::move(children)); | 270 BuildNode("Window", GetAttributes(window_element_root), |
270 if (!window->HasObserver(this)) | 271 std::move(children), window_element_root->node_id()); |
271 window->AddObserver(this); | |
272 window_to_node_id_map_[window] = node->getNodeId(); | |
273 node_id_to_window_map_[node->getNodeId()] = window; | |
274 return node; | 272 return node; |
275 } | 273 } |
276 | 274 |
277 std::unique_ptr<DOM::Node> AshDevToolsDOMAgent::BuildTreeForRootWidget( | 275 std::unique_ptr<DOM::Node> AshDevToolsDOMAgent::BuildTreeForRootWidget( |
| 276 UIElement* widget_element, |
278 views::Widget* widget) { | 277 views::Widget* widget) { |
279 DCHECK(!widget_to_node_id_map_.count(widget)); | |
280 std::unique_ptr<Array<DOM::Node>> children = Array<DOM::Node>::create(); | 278 std::unique_ptr<Array<DOM::Node>> children = Array<DOM::Node>::create(); |
281 children->addItem(BuildTreeForView(widget->GetRootView())); | 279 |
| 280 UIElement* view_element = |
| 281 new ViewElement(widget->GetRootView(), this, widget_element); |
| 282 |
| 283 children->addItem(BuildTreeForView(view_element, widget->GetRootView())); |
| 284 widget_element->AddChild(view_element); |
| 285 |
282 std::unique_ptr<ui::devtools::protocol::DOM::Node> node = | 286 std::unique_ptr<ui::devtools::protocol::DOM::Node> node = |
283 BuildNode("Widget", GetAttributes(widget), std::move(children)); | 287 BuildNode("Widget", GetAttributes(widget_element), std::move(children), |
284 if (!widget->HasRemovalsObserver(this)) | 288 widget_element->node_id()); |
285 widget->AddRemovalsObserver(this); | |
286 widget_to_node_id_map_[widget] = node->getNodeId(); | |
287 node_id_to_widget_map_[node->getNodeId()] = widget; | |
288 return node; | 289 return node; |
289 } | 290 } |
290 | 291 |
291 std::unique_ptr<DOM::Node> AshDevToolsDOMAgent::BuildTreeForView( | 292 std::unique_ptr<DOM::Node> AshDevToolsDOMAgent::BuildTreeForView( |
| 293 UIElement* view_element, |
292 views::View* view) { | 294 views::View* view) { |
293 DCHECK(!view_to_node_id_map_.count(view)); | |
294 std::unique_ptr<Array<DOM::Node>> children = Array<DOM::Node>::create(); | 295 std::unique_ptr<Array<DOM::Node>> children = Array<DOM::Node>::create(); |
295 for (int i = 0, count = view->child_count(); i < count; i++) | 296 |
296 children->addItem(BuildTreeForView(view->child_at(i))); | 297 for (int i = 0, count = view->child_count(); i < count; i++) { |
| 298 UIElement* view_element_child = |
| 299 new ViewElement(view->child_at(i), this, view_element); |
| 300 |
| 301 children->addItem(BuildTreeForView(view_element_child, view->child_at(i))); |
| 302 view_element->AddChild(view_element_child); |
| 303 } |
297 std::unique_ptr<ui::devtools::protocol::DOM::Node> node = | 304 std::unique_ptr<ui::devtools::protocol::DOM::Node> node = |
298 BuildNode("View", GetAttributes(view), std::move(children)); | 305 BuildNode("View", GetAttributes(view_element), std::move(children), |
299 if (!view->HasObserver(this)) | 306 view_element->node_id()); |
300 view->AddObserver(this); | |
301 view_to_node_id_map_[view] = node->getNodeId(); | |
302 node_id_to_view_map_[node->getNodeId()] = view; | |
303 return node; | 307 return node; |
304 } | 308 } |
305 | 309 |
306 void AshDevToolsDOMAgent::AddWindowTree(aura::Window* window) { | 310 std::unique_ptr<DOM::Node> AshDevToolsDOMAgent::BuildDomNodeFromUIElement( |
307 if (IsHighlightingWindow(window)) | 311 UIElement* root) { |
308 return; | 312 std::unique_ptr<Array<DOM::Node>> children = Array<DOM::Node>::create(); |
| 313 for (auto* it : root->children()) |
| 314 children->addItem(BuildDomNodeFromUIElement(it)); |
309 | 315 |
310 DCHECK(window_to_node_id_map_.count(window->parent())); | 316 constexpr int kDomElementNodeType = 1; |
311 aura::Window* prev_sibling = FindPreviousSibling(window); | 317 std::unique_ptr<DOM::Node> node = DOM::Node::create() |
312 frontend()->childNodeInserted( | 318 .setNodeId(root->node_id()) |
313 window_to_node_id_map_[window->parent()], | 319 .setNodeName(root->GetTypeName()) |
314 prev_sibling ? window_to_node_id_map_[prev_sibling] : 0, | 320 .setNodeType(kDomElementNodeType) |
315 BuildTreeForWindow(window)); | 321 .setAttributes(GetAttributes(root)) |
| 322 .build(); |
| 323 node->setChildNodeCount(children->length()); |
| 324 node->setChildren(std::move(children)); |
| 325 return node; |
316 } | 326 } |
317 | 327 |
318 void AshDevToolsDOMAgent::RemoveWindowTree(aura::Window* window, | 328 void AshDevToolsDOMAgent::RemoveDomNode(UIElement* ui_element) { |
319 bool remove_observer) { | 329 for (auto* child_element : ui_element->children()) { |
320 DCHECK(window); | 330 if (child_element) |
321 if (IsHighlightingWindow(window)) | 331 RemoveDomNode(child_element); |
322 return; | 332 } |
323 | 333 frontend()->childNodeRemoved(ui_element->parent()->node_id(), |
324 if (GetWidgetFromWindow(window)) | 334 ui_element->node_id()); |
325 RemoveWidgetTree(GetWidgetFromWindow(window), remove_observer); | |
326 | |
327 for (aura::Window* child : window->children()) | |
328 RemoveWindowTree(child, remove_observer); | |
329 | |
330 RemoveWindowNode(window, remove_observer); | |
331 } | |
332 | |
333 void AshDevToolsDOMAgent::RemoveWindowNode(aura::Window* window, | |
334 bool remove_observer) { | |
335 WindowToNodeIdMap::iterator window_to_node_id_it = | |
336 window_to_node_id_map_.find(window); | |
337 DCHECK(window_to_node_id_it != window_to_node_id_map_.end()); | |
338 | |
339 int node_id = window_to_node_id_it->second; | |
340 int parent_id = GetNodeIdFromWindow(window->parent()); | |
341 | |
342 NodeIdToWindowMap::iterator node_id_to_window_it = | |
343 node_id_to_window_map_.find(node_id); | |
344 DCHECK(node_id_to_window_it != node_id_to_window_map_.end()); | |
345 | |
346 if (remove_observer) | |
347 window->RemoveObserver(this); | |
348 | |
349 node_id_to_window_map_.erase(node_id_to_window_it); | |
350 window_to_node_id_map_.erase(window_to_node_id_it); | |
351 frontend()->childNodeRemoved(parent_id, node_id); | |
352 } | |
353 | |
354 void AshDevToolsDOMAgent::RemoveWidgetTree(views::Widget* widget, | |
355 bool remove_observer) { | |
356 DCHECK(widget); | |
357 if (widget->GetRootView()) | |
358 RemoveViewTree(widget->GetRootView(), nullptr, remove_observer); | |
359 RemoveWidgetNode(widget, remove_observer); | |
360 } | |
361 | |
362 void AshDevToolsDOMAgent::RemoveWidgetNode(views::Widget* widget, | |
363 bool remove_observer) { | |
364 WidgetToNodeIdMap::iterator widget_to_node_id_it = | |
365 widget_to_node_id_map_.find(widget); | |
366 DCHECK(widget_to_node_id_it != widget_to_node_id_map_.end()); | |
367 | |
368 int node_id = widget_to_node_id_it->second; | |
369 int parent_id = GetNodeIdFromWindow(widget->GetNativeWindow()); | |
370 | |
371 if (remove_observer) | |
372 widget->RemoveRemovalsObserver(this); | |
373 | |
374 NodeIdToWidgetMap::iterator node_id_to_widget_it = | |
375 node_id_to_widget_map_.find(node_id); | |
376 DCHECK(node_id_to_widget_it != node_id_to_widget_map_.end()); | |
377 | |
378 widget_to_node_id_map_.erase(widget_to_node_id_it); | |
379 node_id_to_widget_map_.erase(node_id_to_widget_it); | |
380 frontend()->childNodeRemoved(parent_id, node_id); | |
381 } | |
382 | |
383 void AshDevToolsDOMAgent::AddViewTree(views::View* view) { | |
384 DCHECK(view_to_node_id_map_.count(view->parent())); | |
385 views::View* prev_sibling = FindPreviousSibling(view); | |
386 frontend()->childNodeInserted( | |
387 view_to_node_id_map_[view->parent()], | |
388 prev_sibling ? view_to_node_id_map_[prev_sibling] : 0, | |
389 BuildTreeForView(view)); | |
390 } | |
391 | |
392 void AshDevToolsDOMAgent::RemoveViewTree(views::View* view, | |
393 views::View* parent, | |
394 bool remove_observer) { | |
395 DCHECK(view); | |
396 for (int i = 0, count = view->child_count(); i < count; i++) | |
397 RemoveViewTree(view->child_at(i), view, remove_observer); | |
398 RemoveViewNode(view, parent, remove_observer); | |
399 } | |
400 | |
401 void AshDevToolsDOMAgent::RemoveViewNode(views::View* view, | |
402 views::View* parent, | |
403 bool remove_observer) { | |
404 ViewToNodeIdMap::iterator view_to_node_id_it = | |
405 view_to_node_id_map_.find(view); | |
406 DCHECK(view_to_node_id_it != view_to_node_id_map_.end()); | |
407 | |
408 int node_id = view_to_node_id_it->second; | |
409 int parent_id = 0; | |
410 if (parent) | |
411 parent_id = GetNodeIdFromView(parent); | |
412 else // views::RootView | |
413 parent_id = GetNodeIdFromWidget(view->GetWidget()); | |
414 | |
415 if (remove_observer) | |
416 view->RemoveObserver(this); | |
417 | |
418 NodeIdToViewMap::iterator node_id_to_view_it = | |
419 node_id_to_view_map_.find(node_id); | |
420 DCHECK(node_id_to_view_it != node_id_to_view_map_.end()); | |
421 | |
422 view_to_node_id_map_.erase(view_to_node_id_it); | |
423 node_id_to_view_map_.erase(node_id_to_view_it); | |
424 frontend()->childNodeRemoved(parent_id, node_id); | |
425 } | |
426 | |
427 void AshDevToolsDOMAgent::RemoveObservers() { | |
428 for (auto& pair : window_to_node_id_map_) | |
429 pair.first->RemoveObserver(this); | |
430 for (auto& pair : widget_to_node_id_map_) | |
431 pair.first->RemoveRemovalsObserver(this); | |
432 for (auto& pair : view_to_node_id_map_) | |
433 pair.first->RemoveObserver(this); | |
434 } | 335 } |
435 | 336 |
436 void AshDevToolsDOMAgent::Reset() { | 337 void AshDevToolsDOMAgent::Reset() { |
437 RemoveObservers(); | 338 is_building_tree_ = false; |
438 widget_for_highlighting_.reset(); | 339 widget_for_highlighting_.reset(); |
439 window_to_node_id_map_.clear(); | 340 window_element_root_->Destroy(); |
440 widget_to_node_id_map_.clear(); | 341 node_id_to_ui_element_.clear(); |
441 view_to_node_id_map_.clear(); | 342 observers_.Clear(); |
442 node_id_to_window_map_.clear(); | |
443 node_id_to_widget_map_.clear(); | |
444 node_id_to_view_map_.clear(); | |
445 node_ids = 1; | |
446 } | |
447 | |
448 AshDevToolsDOMAgent::WindowAndBoundsPair | |
449 AshDevToolsDOMAgent::GetNodeWindowAndBounds(int node_id) { | |
450 aura::Window* window = GetWindowFromNodeId(node_id); | |
451 if (window) | |
452 return std::make_pair(window, window->GetBoundsInScreen()); | |
453 | |
454 views::Widget* widget = GetWidgetFromNodeId(node_id); | |
455 if (widget) { | |
456 return std::make_pair(widget->GetNativeWindow(), | |
457 widget->GetWindowBoundsInScreen()); | |
458 } | |
459 | |
460 views::View* view = GetViewFromNodeId(node_id); | |
461 if (view) { | |
462 gfx::Rect bounds = view->GetBoundsInScreen(); | |
463 return std::make_pair(view->GetWidget()->GetNativeWindow(), bounds); | |
464 } | |
465 | |
466 return std::make_pair(nullptr, gfx::Rect()); | |
467 } | 343 } |
468 | 344 |
469 void AshDevToolsDOMAgent::InitializeHighlightingWidget() { | 345 void AshDevToolsDOMAgent::InitializeHighlightingWidget() { |
470 DCHECK(!widget_for_highlighting_); | 346 DCHECK(!widget_for_highlighting_); |
471 widget_for_highlighting_.reset(new views::Widget); | 347 widget_for_highlighting_.reset(new views::Widget); |
472 views::Widget::InitParams params; | 348 views::Widget::InitParams params; |
473 params.type = views::Widget::InitParams::TYPE_WINDOW_FRAMELESS; | 349 params.type = views::Widget::InitParams::TYPE_WINDOW_FRAMELESS; |
474 params.activatable = views::Widget::InitParams::ACTIVATABLE_NO; | 350 params.activatable = views::Widget::InitParams::ACTIVATABLE_NO; |
475 params.ownership = views::Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; | 351 params.ownership = views::Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; |
476 params.opacity = views::Widget::InitParams::WindowOpacity::TRANSLUCENT_WINDOW; | 352 params.opacity = views::Widget::InitParams::WindowOpacity::TRANSLUCENT_WINDOW; |
477 params.name = "HighlightingWidget"; | 353 params.name = "HighlightingWidget"; |
478 Shell::GetPrimaryRootWindowController() | 354 Shell::GetPrimaryRootWindowController() |
479 ->ConfigureWidgetInitParamsForContainer(widget_for_highlighting_.get(), | 355 ->ConfigureWidgetInitParamsForContainer(widget_for_highlighting_.get(), |
480 kShellWindowId_OverlayContainer, | 356 kShellWindowId_OverlayContainer, |
481 ¶ms); | 357 ¶ms); |
482 params.keep_on_top = true; | 358 params.keep_on_top = true; |
483 params.accept_events = false; | 359 params.accept_events = false; |
484 widget_for_highlighting_->Init(params); | 360 widget_for_highlighting_->Init(params); |
485 } | 361 } |
486 | 362 |
487 void AshDevToolsDOMAgent::UpdateHighlight( | 363 void AshDevToolsDOMAgent::UpdateHighlight( |
488 const WindowAndBoundsPair& window_and_bounds, | 364 const std::pair<aura::Window*, gfx::Rect>& window_and_bounds, |
489 SkColor background, | 365 SkColor background, |
490 SkColor border) { | 366 SkColor border) { |
491 constexpr int kBorderThickness = 1; | 367 constexpr int kBorderThickness = 1; |
492 views::View* root_view = widget_for_highlighting_->GetRootView(); | 368 views::View* root_view = widget_for_highlighting_->GetRootView(); |
493 root_view->SetBorder(views::CreateSolidBorder(kBorderThickness, border)); | 369 root_view->SetBorder(views::CreateSolidBorder(kBorderThickness, border)); |
494 root_view->set_background( | 370 root_view->set_background( |
495 views::Background::CreateSolidBackground(background)); | 371 views::Background::CreateSolidBackground(background)); |
496 display::Display display = | 372 display::Display display = |
497 display::Screen::GetScreen()->GetDisplayNearestWindow( | 373 display::Screen::GetScreen()->GetDisplayNearestWindow( |
498 window_and_bounds.first); | 374 window_and_bounds.first); |
499 widget_for_highlighting_->GetNativeWindow()->SetBoundsInScreen( | 375 widget_for_highlighting_->GetNativeWindow()->SetBoundsInScreen( |
500 window_and_bounds.second, display); | 376 window_and_bounds.second, display); |
501 } | 377 } |
502 | 378 |
503 ui::devtools::protocol::Response AshDevToolsDOMAgent::HighlightNode( | 379 ui::devtools::protocol::Response AshDevToolsDOMAgent::HighlightNode( |
504 std::unique_ptr<ui::devtools::protocol::DOM::HighlightConfig> | 380 std::unique_ptr<ui::devtools::protocol::DOM::HighlightConfig> |
505 highlight_config, | 381 highlight_config, |
506 int node_id) { | 382 int node_id) { |
507 if (!widget_for_highlighting_) | 383 if (!widget_for_highlighting_) |
508 InitializeHighlightingWidget(); | 384 InitializeHighlightingWidget(); |
509 | 385 |
510 WindowAndBoundsPair window_and_bounds(GetNodeWindowAndBounds(node_id)); | 386 std::pair<aura::Window*, gfx::Rect> window_and_bounds; |
511 | 387 if (node_id_to_ui_element_.count(node_id)) { |
512 if (!window_and_bounds.first) | 388 window_and_bounds = |
| 389 node_id_to_ui_element_[node_id]->GetNodeWindowAndBounds(); |
| 390 } else { |
| 391 window_and_bounds = |
| 392 std::make_pair<aura::Window*, gfx::Rect>(nullptr, gfx::Rect()); |
| 393 } |
| 394 if (!window_and_bounds.first) { |
513 return ui::devtools::protocol::Response::Error( | 395 return ui::devtools::protocol::Response::Error( |
514 "No node found with that id"); | 396 "No node found with that id"); |
515 | 397 } |
516 SkColor border_color = | 398 SkColor border_color = |
517 RGBAToSkColor(highlight_config->getBorderColor(nullptr)); | 399 RGBAToSkColor(highlight_config->getBorderColor(nullptr)); |
518 SkColor content_color = | 400 SkColor content_color = |
519 RGBAToSkColor(highlight_config->getContentColor(nullptr)); | 401 RGBAToSkColor(highlight_config->getContentColor(nullptr)); |
520 UpdateHighlight(window_and_bounds, content_color, border_color); | 402 UpdateHighlight(window_and_bounds, content_color, border_color); |
521 | 403 |
522 if (!widget_for_highlighting_->IsVisible()) | 404 if (!widget_for_highlighting_->IsVisible()) |
523 widget_for_highlighting_->Show(); | 405 widget_for_highlighting_->Show(); |
524 | 406 |
525 return ui::devtools::protocol::Response::OK(); | 407 return ui::devtools::protocol::Response::OK(); |
526 } | 408 } |
527 | 409 |
528 bool AshDevToolsDOMAgent::IsHighlightingWindow(aura::Window* window) { | |
529 return widget_for_highlighting_ && | |
530 GetWidgetFromWindow(window) == widget_for_highlighting_.get(); | |
531 } | |
532 | |
533 } // namespace devtools | 410 } // namespace devtools |
534 } // namespace ash | 411 } // namespace ash |
OLD | NEW |