/* * 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, 2009 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 "RenderFrame.h" #include "FrameView.h" #include "HTMLFrameElement.h" #include "RenderView.h" #ifdef FLATTEN_FRAMESET #include "Frame.h" #include "Document.h" #include "RenderView.h" #endif namespace WebCore { RenderFrame::RenderFrame(HTMLFrameElement* frame) : RenderPart(frame) { setInline(false); } FrameEdgeInfo RenderFrame::edgeInfo() const { HTMLFrameElement* element = static_cast<HTMLFrameElement*>(node()); return FrameEdgeInfo(element->noResize(), element->hasFrameBorder()); } void RenderFrame::viewCleared() { HTMLFrameElement* element = static_cast<HTMLFrameElement*>(node()); if (!element || !widget() || !widget()->isFrameView()) return; FrameView* view = static_cast<FrameView*>(widget()); int marginw = element->getMarginWidth(); int marginh = element->getMarginHeight(); if (marginw != -1) view->setMarginWidth(marginw); if (marginh != -1) view->setMarginHeight(marginh); } #ifdef FLATTEN_FRAMESET void RenderFrame::layout() { if (widget() && widget()->isFrameView()) { FrameView* view = static_cast<FrameView*>(widget()); RenderView* root = NULL; if (view->frame() && view->frame()->document() && view->frame()->document()->renderer() && view->frame()->document()->renderer()->isRenderView()) root = static_cast<RenderView*>(view->frame()->document()->renderer()); if (root) { // Resize the widget so that the RenderView will layout according to those dimensions. view->resize(width(), height()); view->layout(); // We can only grow in width and height because if positionFrames gives us a width and we become smaller, // then the fixup process of forcing the frame to fill extra space will fail. if (width() > root->docWidth()) { view->resize(root->docWidth(), 0); view->layout(); } // Honor the height set by RenderFrameSet::positionFrames unless our document height is larger. setHeight(max(root->docHeight(), height())); setWidth(max(root->docWidth(), width())); } } setNeedsLayout(false); } #endif void RenderFrame::layoutWithFlattening(bool fixedWidth, bool fixedHeight) { // NOTE: The width and height have been set at this point by // RenderFrameSet::positionFramesWithFlattening() FrameView* childFrameView = static_cast<FrameView*>(widget()); RenderView* childRoot = childFrameView ? static_cast<RenderView*>(childFrameView->frame()->contentRenderer()) : 0; HTMLFrameElement* element = static_cast<HTMLFrameElement*>(node()); // Do not expand framesets which has zero width or height if (!width() || !height() || !childRoot) { updateWidgetPosition(); if (childFrameView) childFrameView->layout(); setNeedsLayout(false); return; } // need to update to calculate min/max correctly updateWidgetPosition(); if (childRoot->prefWidthsDirty()) childRoot->calcPrefWidths(); // if scrollbars are off, and the width or height are fixed // we obey them and do not expand. With frame flattening // no subframe much ever become scrollable. bool isScrollable = element->scrollingMode() != ScrollbarAlwaysOff; // make sure minimum preferred width is enforced if (isScrollable || !fixedWidth || childRoot->isFrameSet()) setWidth(max(width(), childRoot->minPrefWidth())); // update again to pass the width to the child frame updateWidgetPosition(); childFrameView->layout(); // expand the frame by setting frame height = content height if (isScrollable || !fixedHeight || childRoot->isFrameSet()) setHeight(max(height(), childFrameView->contentsHeight())); if (isScrollable || !fixedWidth || childRoot->isFrameSet()) setWidth(max(width(), childFrameView->contentsWidth())); updateWidgetPosition(); ASSERT(!childFrameView->layoutPending()); ASSERT(!childRoot->needsLayout()); ASSERT(!childRoot->firstChild() || !childRoot->firstChild()->firstChild() || !childRoot->firstChild()->firstChild()->needsLayout()); setNeedsLayout(false); } } // namespace WebCore