| /* |
| * Copyright (C) 1999 Lars Knoll (knoll@kde.org) |
| * (C) 2000 Simon Hausmann <hausmann@kde.org> |
| * (C) 2000 Stefan Schimanski (1Stein@gmx.de) |
| * Copyright (C) 2004, 2005, 2006, 2008, 2009, 2010 Apple Inc. All rights reserved. |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Library General Public |
| * License as published by the Free Software Foundation; either |
| * version 2 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public License |
| * along with this library; see the file COPYING.LIB. If not, write to |
| * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| * |
| */ |
| |
| #include "config.h" |
| #include "RenderEmbeddedObject.h" |
| |
| #include "Chrome.h" |
| #include "ChromeClient.h" |
| #include "CSSValueKeywords.h" |
| #include "Font.h" |
| #include "FontSelector.h" |
| #include "Frame.h" |
| #include "FrameLoaderClient.h" |
| #include "GraphicsContext.h" |
| #include "HTMLEmbedElement.h" |
| #include "HTMLIFrameElement.h" |
| #include "HTMLNames.h" |
| #include "HTMLObjectElement.h" |
| #include "HTMLParamElement.h" |
| #include "LocalizedStrings.h" |
| #include "MIMETypeRegistry.h" |
| #include "MouseEvent.h" |
| #include "Page.h" |
| #include "PaintInfo.h" |
| #include "Path.h" |
| #include "PluginViewBase.h" |
| #include "RenderTheme.h" |
| #include "RenderView.h" |
| #include "RenderWidgetProtector.h" |
| #include "Settings.h" |
| #include "Text.h" |
| #include "TextRun.h" |
| |
| #if ENABLE(PLUGIN_PROXY_FOR_VIDEO) |
| #include "HTMLVideoElement.h" |
| #endif |
| |
| namespace WebCore { |
| |
| using namespace HTMLNames; |
| |
| static const float replacementTextRoundedRectHeight = 18; |
| static const float replacementTextRoundedRectLeftRightTextMargin = 6; |
| static const float replacementTextRoundedRectOpacity = 0.20f; |
| static const float replacementTextPressedRoundedRectOpacity = 0.65f; |
| static const float replacementTextRoundedRectRadius = 5; |
| static const float replacementTextTextOpacity = 0.55f; |
| static const float replacementTextPressedTextOpacity = 0.65f; |
| |
| static const Color& replacementTextRoundedRectPressedColor() |
| { |
| static const Color lightGray(205, 205, 205); |
| return lightGray; |
| } |
| |
| RenderEmbeddedObject::RenderEmbeddedObject(Element* element) |
| : RenderPart(element) |
| , m_hasFallbackContent(false) |
| , m_showsMissingPluginIndicator(false) |
| , m_missingPluginIndicatorIsPressed(false) |
| , m_mouseDownWasInMissingPluginIndicator(false) |
| { |
| view()->frameView()->setIsVisuallyNonEmpty(); |
| #if ENABLE(PLUGIN_PROXY_FOR_VIDEO) |
| if (element->hasTagName(videoTag) || element->hasTagName(audioTag)) |
| setHasIntrinsicSize(); |
| #endif |
| } |
| |
| RenderEmbeddedObject::~RenderEmbeddedObject() |
| { |
| if (frameView()) |
| frameView()->removeWidgetToUpdate(this); |
| } |
| |
| #if USE(ACCELERATED_COMPOSITING) |
| bool RenderEmbeddedObject::requiresLayer() const |
| { |
| if (RenderPart::requiresLayer()) |
| return true; |
| |
| return allowsAcceleratedCompositing(); |
| } |
| |
| bool RenderEmbeddedObject::allowsAcceleratedCompositing() const |
| { |
| return widget() && widget()->isPluginViewBase() && static_cast<PluginViewBase*>(widget())->platformLayer(); |
| } |
| #endif |
| |
| void RenderEmbeddedObject::setShowsMissingPluginIndicator() |
| { |
| ASSERT(m_replacementText.isEmpty()); |
| m_replacementText = missingPluginText(); |
| m_showsMissingPluginIndicator = true; |
| } |
| |
| void RenderEmbeddedObject::setShowsCrashedPluginIndicator() |
| { |
| ASSERT(m_replacementText.isEmpty()); |
| m_replacementText = crashedPluginText(); |
| } |
| |
| bool RenderEmbeddedObject::pluginCrashedOrWasMissing() const |
| { |
| return !m_replacementText.isNull(); |
| } |
| |
| void RenderEmbeddedObject::setMissingPluginIndicatorIsPressed(bool pressed) |
| { |
| if (m_missingPluginIndicatorIsPressed == pressed) |
| return; |
| |
| m_missingPluginIndicatorIsPressed = pressed; |
| repaint(); |
| } |
| |
| void RenderEmbeddedObject::paint(PaintInfo& paintInfo, int tx, int ty) |
| { |
| if (pluginCrashedOrWasMissing()) { |
| RenderReplaced::paint(paintInfo, tx, ty); |
| return; |
| } |
| |
| RenderPart::paint(paintInfo, tx, ty); |
| } |
| |
| void RenderEmbeddedObject::paintReplaced(PaintInfo& paintInfo, int tx, int ty) |
| { |
| if (!pluginCrashedOrWasMissing()) |
| return; |
| |
| if (paintInfo.phase == PaintPhaseSelection) |
| return; |
| |
| GraphicsContext* context = paintInfo.context; |
| if (context->paintingDisabled()) |
| return; |
| |
| FloatRect contentRect; |
| Path path; |
| FloatRect replacementTextRect; |
| Font font; |
| TextRun run(""); |
| float textWidth; |
| if (!getReplacementTextGeometry(tx, ty, contentRect, path, replacementTextRect, font, run, textWidth)) |
| return; |
| |
| context->save(); |
| context->clip(contentRect); |
| context->setAlpha(m_missingPluginIndicatorIsPressed ? replacementTextPressedRoundedRectOpacity : replacementTextRoundedRectOpacity); |
| context->setFillColor(m_missingPluginIndicatorIsPressed ? replacementTextRoundedRectPressedColor() : Color::white, style()->colorSpace()); |
| context->fillPath(path); |
| |
| const FontMetrics& fontMetrics = font.fontMetrics(); |
| float labelX = roundf(replacementTextRect.location().x() + (replacementTextRect.size().width() - textWidth) / 2); |
| float labelY = roundf(replacementTextRect.location().y() + (replacementTextRect.size().height() - fontMetrics.height()) / 2 + fontMetrics.ascent()); |
| context->setAlpha(m_missingPluginIndicatorIsPressed ? replacementTextPressedTextOpacity : replacementTextTextOpacity); |
| context->setFillColor(Color::black, style()->colorSpace()); |
| context->drawBidiText(font, run, FloatPoint(labelX, labelY)); |
| context->restore(); |
| } |
| |
| bool RenderEmbeddedObject::getReplacementTextGeometry(int tx, int ty, FloatRect& contentRect, Path& path, FloatRect& replacementTextRect, Font& font, TextRun& run, float& textWidth) |
| { |
| contentRect = contentBoxRect(); |
| contentRect.move(tx, ty); |
| |
| FontDescription fontDescription; |
| RenderTheme::defaultTheme()->systemFont(CSSValueWebkitSmallControl, fontDescription); |
| fontDescription.setWeight(FontWeightBold); |
| Settings* settings = document()->settings(); |
| ASSERT(settings); |
| if (!settings) |
| return false; |
| fontDescription.setRenderingMode(settings->fontRenderingMode()); |
| fontDescription.setComputedSize(fontDescription.specifiedSize()); |
| font = Font(fontDescription, 0, 0); |
| font.update(0); |
| |
| run = TextRun(m_replacementText.characters(), m_replacementText.length()); |
| textWidth = font.width(run); |
| |
| replacementTextRect.setSize(FloatSize(textWidth + replacementTextRoundedRectLeftRightTextMargin * 2, replacementTextRoundedRectHeight)); |
| float x = (contentRect.size().width() / 2 - replacementTextRect.size().width() / 2) + contentRect.location().x(); |
| float y = (contentRect.size().height() / 2 - replacementTextRect.size().height() / 2) + contentRect.location().y(); |
| replacementTextRect.setLocation(FloatPoint(x, y)); |
| |
| path.addRoundedRect(replacementTextRect, FloatSize(replacementTextRoundedRectRadius, replacementTextRoundedRectRadius)); |
| |
| return true; |
| } |
| |
| void RenderEmbeddedObject::layout() |
| { |
| ASSERT(needsLayout()); |
| |
| computeLogicalWidth(); |
| computeLogicalHeight(); |
| |
| RenderPart::layout(); |
| |
| m_overflow.clear(); |
| addShadowOverflow(); |
| |
| updateLayerTransform(); |
| |
| if (!widget() && frameView()) |
| frameView()->addWidgetToUpdate(this); |
| |
| setNeedsLayout(false); |
| } |
| |
| void RenderEmbeddedObject::viewCleared() |
| { |
| // This is required for <object> elements whose contents are rendered by WebCore (e.g. src="foo.html"). |
| if (node() && widget() && widget()->isFrameView()) { |
| FrameView* view = static_cast<FrameView*>(widget()); |
| int marginWidth = -1; |
| int marginHeight = -1; |
| if (node()->hasTagName(iframeTag)) { |
| HTMLIFrameElement* frame = static_cast<HTMLIFrameElement*>(node()); |
| marginWidth = frame->marginWidth(); |
| marginHeight = frame->marginHeight(); |
| } |
| if (marginWidth != -1) |
| view->setMarginWidth(marginWidth); |
| if (marginHeight != -1) |
| view->setMarginHeight(marginHeight); |
| } |
| } |
| |
| bool RenderEmbeddedObject::isInMissingPluginIndicator(MouseEvent* event) |
| { |
| FloatRect contentRect; |
| Path path; |
| FloatRect replacementTextRect; |
| Font font; |
| TextRun run(""); |
| float textWidth; |
| if (!getReplacementTextGeometry(0, 0, contentRect, path, replacementTextRect, font, run, textWidth)) |
| return false; |
| |
| return path.contains(absoluteToLocal(event->absoluteLocation(), false, true)); |
| } |
| |
| void RenderEmbeddedObject::handleMissingPluginIndicatorEvent(Event* event) |
| { |
| if (Page* page = document()->page()) { |
| if (!page->chrome()->client()->shouldMissingPluginMessageBeButton()) |
| return; |
| } |
| |
| if (!event->isMouseEvent()) |
| return; |
| |
| MouseEvent* mouseEvent = static_cast<MouseEvent*>(event); |
| HTMLPlugInElement* element = static_cast<HTMLPlugInElement*>(node()); |
| if (event->type() == eventNames().mousedownEvent && static_cast<MouseEvent*>(event)->button() == LeftButton) { |
| m_mouseDownWasInMissingPluginIndicator = isInMissingPluginIndicator(mouseEvent); |
| if (m_mouseDownWasInMissingPluginIndicator) { |
| if (Frame* frame = document()->frame()) { |
| frame->eventHandler()->setCapturingMouseEventsNode(element); |
| element->setIsCapturingMouseEvents(true); |
| } |
| setMissingPluginIndicatorIsPressed(true); |
| } |
| event->setDefaultHandled(); |
| } |
| if (event->type() == eventNames().mouseupEvent && static_cast<MouseEvent*>(event)->button() == LeftButton) { |
| if (m_missingPluginIndicatorIsPressed) { |
| if (Frame* frame = document()->frame()) { |
| frame->eventHandler()->setCapturingMouseEventsNode(0); |
| element->setIsCapturingMouseEvents(false); |
| } |
| setMissingPluginIndicatorIsPressed(false); |
| } |
| if (m_mouseDownWasInMissingPluginIndicator && isInMissingPluginIndicator(mouseEvent)) { |
| if (Page* page = document()->page()) |
| page->chrome()->client()->missingPluginButtonClicked(element); |
| } |
| m_mouseDownWasInMissingPluginIndicator = false; |
| event->setDefaultHandled(); |
| } |
| if (event->type() == eventNames().mousemoveEvent) { |
| setMissingPluginIndicatorIsPressed(m_mouseDownWasInMissingPluginIndicator && isInMissingPluginIndicator(mouseEvent)); |
| event->setDefaultHandled(); |
| } |
| } |
| |
| } |