| 1 | /* |
| 2 | * Copyright (C) 1999 Lars Knoll (knoll@kde.org) |
| 3 | * Copyright (C) 2000 Dirk Mueller (mueller@kde.org) |
| 4 | * Copyright (C) 2004, 2006, 2009, 2010 Apple Inc. All rights reserved. |
| 5 | * |
| 6 | * This library is free software; you can redistribute it and/or |
| 7 | * modify it under the terms of the GNU Library General Public |
| 8 | * License as published by the Free Software Foundation; either |
| 9 | * version 2 of the License, or (at your option) any later version. |
| 10 | * |
| 11 | * This library is distributed in the hope that it will be useful, |
| 12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 14 | * Library General Public License for more details. |
| 15 | * |
| 16 | * You should have received a copy of the GNU Library General Public License |
| 17 | * along with this library; see the file COPYING.LIB. If not, write to |
| 18 | * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
| 19 | * Boston, MA 02110-1301, USA. |
| 20 | * |
| 21 | */ |
| 22 | |
| 23 | #include "config.h" |
| 24 | #include "RenderWidget.h" |
| 25 | |
| 26 | #include "AXObjectCache.h" |
| 27 | #include "FloatRoundedRect.h" |
| 28 | #include "Frame.h" |
| 29 | #include "HTMLFrameOwnerElement.h" |
| 30 | #include "HitTestResult.h" |
| 31 | #include "RenderLayer.h" |
| 32 | #include "RenderLayerBacking.h" |
| 33 | #include "RenderView.h" |
| 34 | #include "SecurityOrigin.h" |
| 35 | #include <wtf/IsoMallocInlines.h> |
| 36 | #include <wtf/StackStats.h> |
| 37 | #include <wtf/Ref.h> |
| 38 | |
| 39 | namespace WebCore { |
| 40 | |
| 41 | WTF_MAKE_ISO_ALLOCATED_IMPL(RenderWidget); |
| 42 | |
| 43 | static HashMap<const Widget*, RenderWidget*>& widgetRendererMap() |
| 44 | { |
| 45 | static HashMap<const Widget*, RenderWidget*>* staticWidgetRendererMap = new HashMap<const Widget*, RenderWidget*>; |
| 46 | return *staticWidgetRendererMap; |
| 47 | } |
| 48 | |
| 49 | unsigned WidgetHierarchyUpdatesSuspensionScope::s_widgetHierarchyUpdateSuspendCount = 0; |
| 50 | |
| 51 | WidgetHierarchyUpdatesSuspensionScope::WidgetToParentMap& WidgetHierarchyUpdatesSuspensionScope::widgetNewParentMap() |
| 52 | { |
| 53 | static NeverDestroyed<WidgetToParentMap> map; |
| 54 | return map; |
| 55 | } |
| 56 | |
| 57 | void WidgetHierarchyUpdatesSuspensionScope::moveWidgets() |
| 58 | { |
| 59 | auto map = WTFMove(widgetNewParentMap()); |
| 60 | for (auto& entry : map) { |
| 61 | auto& child = *entry.key; |
| 62 | auto* currentParent = child.parent(); |
| 63 | auto* newParent = entry.value; |
| 64 | if (newParent != currentParent) { |
| 65 | if (currentParent) |
| 66 | currentParent->removeChild(child); |
| 67 | if (newParent) |
| 68 | newParent->addChild(child); |
| 69 | } |
| 70 | } |
| 71 | } |
| 72 | |
| 73 | static void moveWidgetToParentSoon(Widget& child, FrameView* parent) |
| 74 | { |
| 75 | if (!WidgetHierarchyUpdatesSuspensionScope::isSuspended()) { |
| 76 | if (parent) |
| 77 | parent->addChild(child); |
| 78 | else |
| 79 | child.removeFromParent(); |
| 80 | return; |
| 81 | } |
| 82 | WidgetHierarchyUpdatesSuspensionScope::scheduleWidgetToMove(child, parent); |
| 83 | } |
| 84 | |
| 85 | RenderWidget::RenderWidget(HTMLFrameOwnerElement& element, RenderStyle&& style) |
| 86 | : RenderReplaced(element, WTFMove(style)) |
| 87 | { |
| 88 | setInline(false); |
| 89 | } |
| 90 | |
| 91 | void RenderWidget::willBeDestroyed() |
| 92 | { |
| 93 | #if PLATFORM(IOS_FAMILY) |
| 94 | if (hasLayer()) |
| 95 | layer()->willBeDestroyed(); |
| 96 | #endif |
| 97 | |
| 98 | if (AXObjectCache* cache = document().existingAXObjectCache()) { |
| 99 | cache->childrenChanged(this->parent()); |
| 100 | cache->remove(this); |
| 101 | } |
| 102 | |
| 103 | setWidget(nullptr); |
| 104 | |
| 105 | RenderReplaced::willBeDestroyed(); |
| 106 | } |
| 107 | |
| 108 | RenderWidget::~RenderWidget() |
| 109 | { |
| 110 | ASSERT(!m_refCount); |
| 111 | } |
| 112 | |
| 113 | // Widgets are always placed on integer boundaries, so rounding the size is actually |
| 114 | // the desired behavior. This function is here because it's otherwise seldom what we |
| 115 | // want to do with a LayoutRect. |
| 116 | static inline IntRect roundedIntRect(const LayoutRect& rect) |
| 117 | { |
| 118 | return IntRect(roundedIntPoint(rect.location()), roundedIntSize(rect.size())); |
| 119 | } |
| 120 | |
| 121 | bool RenderWidget::setWidgetGeometry(const LayoutRect& frame) |
| 122 | { |
| 123 | IntRect clipRect = roundedIntRect(enclosingLayer()->childrenClipRect()); |
| 124 | IntRect newFrameRect = roundedIntRect(frame); |
| 125 | IntRect oldFrameRect = m_widget->frameRect(); |
| 126 | bool clipChanged = m_clipRect != clipRect; |
| 127 | bool boundsChanged = oldFrameRect != newFrameRect; |
| 128 | |
| 129 | if (!boundsChanged && !clipChanged) |
| 130 | return false; |
| 131 | |
| 132 | m_clipRect = clipRect; |
| 133 | |
| 134 | auto weakThis = makeWeakPtr(*this); |
| 135 | // These calls *may* cause this renderer to disappear from underneath... |
| 136 | if (boundsChanged) |
| 137 | m_widget->setFrameRect(newFrameRect); |
| 138 | else if (clipChanged) |
| 139 | m_widget->clipRectChanged(); |
| 140 | // ...so we follow up with a sanity check. |
| 141 | if (!weakThis) |
| 142 | return true; |
| 143 | |
| 144 | if (boundsChanged && isComposited()) |
| 145 | layer()->backing()->updateAfterWidgetResize(); |
| 146 | |
| 147 | return oldFrameRect.size() != newFrameRect.size(); |
| 148 | } |
| 149 | |
| 150 | bool RenderWidget::updateWidgetGeometry() |
| 151 | { |
| 152 | if (!m_widget->transformsAffectFrameRect()) |
| 153 | return setWidgetGeometry(absoluteContentBox()); |
| 154 | |
| 155 | LayoutRect contentBox = contentBoxRect(); |
| 156 | LayoutRect absoluteContentBox(localToAbsoluteQuad(FloatQuad(contentBox)).boundingBox()); |
| 157 | if (m_widget->isFrameView()) { |
| 158 | contentBox.setLocation(absoluteContentBox.location()); |
| 159 | return setWidgetGeometry(contentBox); |
| 160 | } |
| 161 | |
| 162 | return setWidgetGeometry(absoluteContentBox); |
| 163 | } |
| 164 | |
| 165 | void RenderWidget::setWidget(RefPtr<Widget>&& widget) |
| 166 | { |
| 167 | if (widget == m_widget) |
| 168 | return; |
| 169 | |
| 170 | if (m_widget) { |
| 171 | moveWidgetToParentSoon(*m_widget, nullptr); |
| 172 | view().frameView().willRemoveWidgetFromRenderTree(*m_widget); |
| 173 | widgetRendererMap().remove(m_widget.get()); |
| 174 | m_widget = nullptr; |
| 175 | } |
| 176 | m_widget = widget; |
| 177 | if (m_widget) { |
| 178 | widgetRendererMap().add(m_widget.get(), this); |
| 179 | view().frameView().didAddWidgetToRenderTree(*m_widget); |
| 180 | // If we've already received a layout, apply the calculated space to the |
| 181 | // widget immediately, but we have to have really been fully constructed. |
| 182 | if (hasInitializedStyle()) { |
| 183 | if (!needsLayout()) { |
| 184 | auto weakThis = makeWeakPtr(*this); |
| 185 | updateWidgetGeometry(); |
| 186 | if (!weakThis) |
| 187 | return; |
| 188 | } |
| 189 | |
| 190 | if (style().visibility() != Visibility::Visible) |
| 191 | m_widget->hide(); |
| 192 | else { |
| 193 | m_widget->show(); |
| 194 | repaint(); |
| 195 | } |
| 196 | } |
| 197 | moveWidgetToParentSoon(*m_widget, &view().frameView()); |
| 198 | } |
| 199 | } |
| 200 | |
| 201 | void RenderWidget::layout() |
| 202 | { |
| 203 | StackStats::LayoutCheckPoint layoutCheckPoint; |
| 204 | ASSERT(needsLayout()); |
| 205 | |
| 206 | clearNeedsLayout(); |
| 207 | } |
| 208 | |
| 209 | void RenderWidget::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle) |
| 210 | { |
| 211 | RenderReplaced::styleDidChange(diff, oldStyle); |
| 212 | if (m_widget) { |
| 213 | if (style().visibility() != Visibility::Visible) |
| 214 | m_widget->hide(); |
| 215 | else |
| 216 | m_widget->show(); |
| 217 | } |
| 218 | } |
| 219 | |
| 220 | void RenderWidget::paintContents(PaintInfo& paintInfo, const LayoutPoint& paintOffset) |
| 221 | { |
| 222 | if (paintInfo.requireSecurityOriginAccessForWidgets) { |
| 223 | if (auto contentDocument = frameOwnerElement().contentDocument()) { |
| 224 | if (!document().securityOrigin().canAccess(contentDocument->securityOrigin())) |
| 225 | return; |
| 226 | } |
| 227 | } |
| 228 | |
| 229 | IntPoint contentPaintOffset = roundedIntPoint(paintOffset + location() + contentBoxRect().location()); |
| 230 | // Tell the widget to paint now. This is the only time the widget is allowed |
| 231 | // to paint itself. That way it will composite properly with z-indexed layers. |
| 232 | LayoutRect paintRect = paintInfo.rect; |
| 233 | |
| 234 | OptionSet<PaintBehavior> oldBehavior = PaintBehavior::Normal; |
| 235 | if (is<FrameView>(*m_widget) && (paintInfo.paintBehavior & PaintBehavior::TileFirstPaint)) { |
| 236 | FrameView& frameView = downcast<FrameView>(*m_widget); |
| 237 | oldBehavior = frameView.paintBehavior(); |
| 238 | frameView.setPaintBehavior(oldBehavior | PaintBehavior::TileFirstPaint); |
| 239 | } |
| 240 | |
| 241 | IntPoint widgetLocation = m_widget->frameRect().location(); |
| 242 | IntSize widgetPaintOffset = contentPaintOffset - widgetLocation; |
| 243 | // When painting widgets into compositing layers, tx and ty are relative to the enclosing compositing layer, |
| 244 | // not the root. In this case, shift the CTM and adjust the paintRect to be root-relative to fix plug-in drawing. |
| 245 | if (!widgetPaintOffset.isZero()) { |
| 246 | paintInfo.context().translate(widgetPaintOffset); |
| 247 | paintRect.move(-widgetPaintOffset); |
| 248 | } |
| 249 | // FIXME: Remove repaintrect enclosing/integral snapping when RenderWidget becomes device pixel snapped. |
| 250 | m_widget->paint(paintInfo.context(), snappedIntRect(paintRect), paintInfo.requireSecurityOriginAccessForWidgets ? Widget::SecurityOriginPaintPolicy::AccessibleOriginOnly : Widget::SecurityOriginPaintPolicy::AnyOrigin); |
| 251 | |
| 252 | if (!widgetPaintOffset.isZero()) |
| 253 | paintInfo.context().translate(-widgetPaintOffset); |
| 254 | |
| 255 | if (is<FrameView>(*m_widget)) { |
| 256 | FrameView& frameView = downcast<FrameView>(*m_widget); |
| 257 | bool runOverlapTests = !frameView.useSlowRepaintsIfNotOverlapped(); |
| 258 | if (paintInfo.overlapTestRequests && runOverlapTests) { |
| 259 | ASSERT(!paintInfo.overlapTestRequests->contains(this) || (paintInfo.overlapTestRequests->get(this) == m_widget->frameRect())); |
| 260 | paintInfo.overlapTestRequests->set(this, m_widget->frameRect()); |
| 261 | } |
| 262 | if (paintInfo.paintBehavior & PaintBehavior::TileFirstPaint) |
| 263 | frameView.setPaintBehavior(oldBehavior); |
| 264 | } |
| 265 | } |
| 266 | |
| 267 | void RenderWidget::paint(PaintInfo& paintInfo, const LayoutPoint& paintOffset) |
| 268 | { |
| 269 | if (!shouldPaint(paintInfo, paintOffset)) |
| 270 | return; |
| 271 | |
| 272 | LayoutPoint adjustedPaintOffset = paintOffset + location(); |
| 273 | |
| 274 | if (hasVisibleBoxDecorations() && (paintInfo.phase == PaintPhase::Foreground || paintInfo.phase == PaintPhase::Selection)) |
| 275 | paintBoxDecorations(paintInfo, adjustedPaintOffset); |
| 276 | |
| 277 | if (paintInfo.phase == PaintPhase::Mask) { |
| 278 | paintMask(paintInfo, adjustedPaintOffset); |
| 279 | return; |
| 280 | } |
| 281 | |
| 282 | if ((paintInfo.phase == PaintPhase::Outline || paintInfo.phase == PaintPhase::SelfOutline) && hasOutline()) |
| 283 | paintOutline(paintInfo, LayoutRect(adjustedPaintOffset, size())); |
| 284 | |
| 285 | if (paintInfo.phase != PaintPhase::Foreground) |
| 286 | return; |
| 287 | |
| 288 | if (style().hasBorderRadius()) { |
| 289 | LayoutRect borderRect = LayoutRect(adjustedPaintOffset, size()); |
| 290 | |
| 291 | if (borderRect.isEmpty()) |
| 292 | return; |
| 293 | |
| 294 | // Push a clip if we have a border radius, since we want to round the foreground content that gets painted. |
| 295 | paintInfo.context().save(); |
| 296 | FloatRoundedRect roundedInnerRect = FloatRoundedRect(style().getRoundedInnerBorderFor(borderRect, |
| 297 | paddingTop() + borderTop(), paddingBottom() + borderBottom(), paddingLeft() + borderLeft(), paddingRight() + borderRight(), true, true)); |
| 298 | clipRoundedInnerRect(paintInfo.context(), borderRect, roundedInnerRect); |
| 299 | } |
| 300 | |
| 301 | if (m_widget) |
| 302 | paintContents(paintInfo, paintOffset); |
| 303 | |
| 304 | if (style().hasBorderRadius()) |
| 305 | paintInfo.context().restore(); |
| 306 | |
| 307 | // Paint a partially transparent wash over selected widgets. |
| 308 | if (isSelected() && !document().printing()) { |
| 309 | // FIXME: selectionRect() is in absolute, not painting coordinates. |
| 310 | paintInfo.context().fillRect(snappedIntRect(selectionRect()), selectionBackgroundColor()); |
| 311 | } |
| 312 | |
| 313 | if (hasLayer() && layer()->canResize()) |
| 314 | layer()->paintResizer(paintInfo.context(), roundedIntPoint(adjustedPaintOffset), paintInfo.rect); |
| 315 | } |
| 316 | |
| 317 | void RenderWidget::setOverlapTestResult(bool isOverlapped) |
| 318 | { |
| 319 | ASSERT(m_widget); |
| 320 | downcast<FrameView>(*m_widget).setIsOverlapped(isOverlapped); |
| 321 | } |
| 322 | |
| 323 | RenderWidget::ChildWidgetState RenderWidget::updateWidgetPosition() |
| 324 | { |
| 325 | if (!m_widget) |
| 326 | return ChildWidgetState::Destroyed; |
| 327 | |
| 328 | auto weakThis = makeWeakPtr(*this); |
| 329 | bool widgetSizeChanged = updateWidgetGeometry(); |
| 330 | if (!weakThis || !m_widget) |
| 331 | return ChildWidgetState::Destroyed; |
| 332 | |
| 333 | // if the frame size got changed, or if view needs layout (possibly indicating |
| 334 | // content size is wrong) we have to do a layout to set the right widget size. |
| 335 | if (is<FrameView>(*m_widget)) { |
| 336 | FrameView& frameView = downcast<FrameView>(*m_widget); |
| 337 | // Check the frame's page to make sure that the frame isn't in the process of being destroyed. |
| 338 | if ((widgetSizeChanged || frameView.needsLayout()) && frameView.frame().page() && frameView.frame().document()) |
| 339 | frameView.layoutContext().layout(); |
| 340 | } |
| 341 | return ChildWidgetState::Valid; |
| 342 | } |
| 343 | |
| 344 | IntRect RenderWidget::windowClipRect() const |
| 345 | { |
| 346 | return intersection(view().frameView().contentsToWindow(m_clipRect), view().frameView().windowClipRect()); |
| 347 | } |
| 348 | |
| 349 | void RenderWidget::setSelectionState(SelectionState state) |
| 350 | { |
| 351 | // The selection state for our containing block hierarchy is updated by the base class call. |
| 352 | RenderReplaced::setSelectionState(state); |
| 353 | |
| 354 | if (m_widget) |
| 355 | m_widget->setIsSelected(isSelected()); |
| 356 | } |
| 357 | |
| 358 | RenderWidget* RenderWidget::find(const Widget& widget) |
| 359 | { |
| 360 | return widgetRendererMap().get(&widget); |
| 361 | } |
| 362 | |
| 363 | bool RenderWidget::nodeAtPoint(const HitTestRequest& request, HitTestResult& result, const HitTestLocation& locationInContainer, const LayoutPoint& accumulatedOffset, HitTestAction action) |
| 364 | { |
| 365 | if (request.allowsChildFrameContent() && is<FrameView>(widget()) && downcast<FrameView>(*widget()).renderView()) { |
| 366 | FrameView& childFrameView = downcast<FrameView>(*widget()); |
| 367 | RenderView& childRoot = *childFrameView.renderView(); |
| 368 | |
| 369 | LayoutPoint adjustedLocation = accumulatedOffset + location(); |
| 370 | LayoutPoint contentOffset = LayoutPoint(borderLeft() + paddingLeft(), borderTop() + paddingTop()) - toIntSize(childFrameView.scrollPosition()); |
| 371 | HitTestLocation newHitTestLocation(locationInContainer, -adjustedLocation - contentOffset); |
| 372 | HitTestRequest newHitTestRequest(request.type() | HitTestRequest::ChildFrameHitTest); |
| 373 | HitTestResult childFrameResult(newHitTestLocation); |
| 374 | |
| 375 | bool isInsideChildFrame = childRoot.hitTest(newHitTestRequest, newHitTestLocation, childFrameResult); |
| 376 | |
| 377 | if (request.resultIsElementList()) |
| 378 | result.append(childFrameResult, request); |
| 379 | else if (isInsideChildFrame) |
| 380 | result = childFrameResult; |
| 381 | |
| 382 | if (isInsideChildFrame) |
| 383 | return true; |
| 384 | } |
| 385 | |
| 386 | bool hadResult = result.innerNode(); |
| 387 | bool inside = RenderReplaced::nodeAtPoint(request, result, locationInContainer, accumulatedOffset, action); |
| 388 | |
| 389 | // Check to see if we are really over the widget itself (and not just in the border/padding area). |
| 390 | if ((inside || result.isRectBasedTest()) && !hadResult && result.innerNode() == &frameOwnerElement()) |
| 391 | result.setIsOverWidget(contentBoxRect().contains(result.localPoint())); |
| 392 | return inside; |
| 393 | } |
| 394 | |
| 395 | bool RenderWidget::requiresLayer() const |
| 396 | { |
| 397 | return RenderReplaced::requiresLayer() || requiresAcceleratedCompositing(); |
| 398 | } |
| 399 | |
| 400 | bool RenderWidget::requiresAcceleratedCompositing() const |
| 401 | { |
| 402 | // If this is a renderer with a contentDocument and that document needs a layer, then we need a layer. |
| 403 | if (Document* contentDocument = frameOwnerElement().contentDocument()) { |
| 404 | if (RenderView* view = contentDocument->renderView()) |
| 405 | return view->usesCompositing(); |
| 406 | } |
| 407 | |
| 408 | return false; |
| 409 | } |
| 410 | |
| 411 | bool RenderWidget::needsPreferredWidthsRecalculation() const |
| 412 | { |
| 413 | if (RenderReplaced::needsPreferredWidthsRecalculation()) |
| 414 | return true; |
| 415 | return embeddedContentBox(); |
| 416 | } |
| 417 | |
| 418 | RenderBox* RenderWidget::embeddedContentBox() const |
| 419 | { |
| 420 | if (!is<FrameView>(widget())) |
| 421 | return nullptr; |
| 422 | return downcast<FrameView>(*widget()).embeddedContentBox(); |
| 423 | } |
| 424 | |
| 425 | } // namespace WebCore |
| 426 | |