зеркало из https://github.com/mozilla/pjs.git
261 строка
8.9 KiB
C++
261 строка
8.9 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
*
|
|
* The contents of this file are subject to the Netscape Public License
|
|
* Version 1.0 (the "NPL"); you may not use this file except in
|
|
* compliance with the NPL. You may obtain a copy of the NPL at
|
|
* http://www.mozilla.org/NPL/
|
|
*
|
|
* Software distributed under the NPL is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
|
|
* for the specific language governing rights and limitations under the
|
|
* NPL.
|
|
*
|
|
* The Initial Developer of this code under the NPL is Netscape
|
|
* Communications Corporation. Portions created by Netscape are
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
|
|
* Reserved.
|
|
*/
|
|
#include "nsPlaceholderFrame.h"
|
|
#include "nsIContent.h"
|
|
#include "nsIContentDelegate.h"
|
|
#include "nsIFloaterContainer.h"
|
|
#include "nsBodyFrame.h"
|
|
#include "nsIStyleContext.h"
|
|
#include "nsCSSLayout.h"
|
|
#include "nsIView.h"
|
|
#include "nsHTMLIIDs.h"
|
|
#include "nsIPresContext.h"
|
|
#include "nsCSSLineLayout.h"
|
|
|
|
nsresult
|
|
nsPlaceholderFrame::NewFrame(nsIFrame** aInstancePtrResult,
|
|
nsIContent* aContent,
|
|
nsIFrame* aParent)
|
|
{
|
|
NS_PRECONDITION(nsnull != aInstancePtrResult, "null ptr");
|
|
if (nsnull == aInstancePtrResult) {
|
|
return NS_ERROR_NULL_POINTER;
|
|
}
|
|
nsIFrame* it = new nsPlaceholderFrame(aContent, aParent);
|
|
if (nsnull == it) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
*aInstancePtrResult = it;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsPlaceholderFrame::nsPlaceholderFrame(nsIContent* aContent, nsIFrame* aParent)
|
|
: nsFrame(aContent, aParent)
|
|
{
|
|
}
|
|
|
|
nsPlaceholderFrame::~nsPlaceholderFrame()
|
|
{
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPlaceholderFrame::QueryInterface(REFNSIID aIID, void** aInstancePtrResult)
|
|
{
|
|
NS_PRECONDITION(nsnull != aInstancePtrResult, "null pointer");
|
|
if (nsnull == aInstancePtrResult) {
|
|
return NS_ERROR_NULL_POINTER;
|
|
}
|
|
if (aIID.Equals(kIInlineReflowIID)) {
|
|
nsIInlineReflow* tmp = this;
|
|
*aInstancePtrResult = (void*) tmp;
|
|
return NS_OK;
|
|
}
|
|
return nsFrame::QueryInterface(aIID, aInstancePtrResult);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPlaceholderFrame::FindTextRuns(nsCSSLineLayout& aLineLayout,
|
|
nsIReflowCommand* aReflowCommand)
|
|
{
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPlaceholderFrame::InlineReflow(nsCSSLineLayout& aLineLayout,
|
|
nsReflowMetrics& aDesiredSize,
|
|
const nsReflowState& aReflowState)
|
|
{
|
|
nsIPresContext& presContext = *aLineLayout.mPresContext;
|
|
|
|
// Get the floater container in which we're inserted
|
|
nsIFrame* containingBlock;
|
|
nsIFloaterContainer* container = nsnull;
|
|
|
|
for (containingBlock = mGeometricParent;
|
|
nsnull != containingBlock;
|
|
containingBlock->GetGeometricParent(containingBlock))
|
|
{
|
|
if (NS_OK == containingBlock->QueryInterface(kIFloaterContainerIID, (void**)&container)) {
|
|
break;
|
|
}
|
|
}
|
|
NS_ASSERTION(nsnull != container, "no floater container");
|
|
|
|
// Have we created the anchored item yet?
|
|
if (nsnull == mAnchoredItem) {
|
|
// If the content object is a container then wrap it in a body pseudo-frame
|
|
|
|
// XXX begin hack
|
|
// XXX More hack. We don't want to blindly wrap a body pseudo-frame around
|
|
// a table content object, because the body pseudo-frame will map the table's
|
|
// content children rather than create a table frame and let it map the
|
|
// children...
|
|
PRBool select = PR_FALSE;
|
|
nsIAtom* atom = mContent->GetTag();
|
|
nsAutoString tmp;
|
|
if (nsnull != atom) {
|
|
atom->ToString(tmp);
|
|
if (tmp.EqualsIgnoreCase("select") ||
|
|
tmp.EqualsIgnoreCase("table") ||
|
|
tmp.EqualsIgnoreCase("embed") ||
|
|
tmp.EqualsIgnoreCase("object") ||
|
|
tmp.EqualsIgnoreCase("applet")) {
|
|
select = PR_TRUE;
|
|
}
|
|
NS_RELEASE(atom);
|
|
}
|
|
// XXX end hack
|
|
|
|
if (mContent->CanContainChildren() && !select) {
|
|
nsBodyFrame::NewFrame(&mAnchoredItem, mContent, this);
|
|
|
|
// Use our style context for the pseudo-frame
|
|
mAnchoredItem->SetStyleContext(&presContext, mStyleContext);
|
|
} else {
|
|
// Create the anchored item
|
|
nsIContentDelegate* delegate = mContent->GetDelegate(&presContext);
|
|
nsresult rv = delegate->CreateFrame(&presContext, mContent,
|
|
mGeometricParent, mStyleContext,
|
|
mAnchoredItem);
|
|
NS_RELEASE(delegate);
|
|
if (NS_OK != rv) {
|
|
return rv;
|
|
}
|
|
}
|
|
|
|
// Notify our containing block that there's a new floater
|
|
container->AddFloater(&presContext, aReflowState, mAnchoredItem, this);
|
|
}
|
|
|
|
aDesiredSize.width = 0;
|
|
aDesiredSize.height = 0;
|
|
aDesiredSize.ascent = 0;
|
|
aDesiredSize.descent = 0;
|
|
if (nsnull != aDesiredSize.maxElementSize) {
|
|
aDesiredSize.maxElementSize->width = aDesiredSize.width;
|
|
aDesiredSize.maxElementSize->height = aDesiredSize.height;
|
|
}
|
|
return NS_FRAME_COMPLETE;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPlaceholderFrame::Paint(nsIPresContext& aPresContext,
|
|
nsIRenderingContext& aRenderingContext,
|
|
const nsRect& aDirtyRect)
|
|
{
|
|
if (nsIFrame::GetShowFrameBorders()) {
|
|
float p2t = aPresContext.GetPixelsToTwips();
|
|
aRenderingContext.SetColor(NS_RGB(0, 255, 255));
|
|
nscoord x = nscoord(-5 * p2t);
|
|
aRenderingContext.FillRect(x, 0, nscoord(13 * p2t), nscoord(3 * p2t));
|
|
nscoord y = nscoord(-10 * p2t);
|
|
aRenderingContext.FillRect(0, y, nscoord(3 * p2t), nscoord(10 * p2t));
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP nsPlaceholderFrame::ContentAppended(nsIPresShell* aShell,
|
|
nsIPresContext* aPresContext,
|
|
nsIContent* aContainer)
|
|
{
|
|
NS_ASSERTION(mContent == aContainer, "bad content-appended target");
|
|
|
|
// Forward the notification to the floater
|
|
if (nsnull != mAnchoredItem) {
|
|
return mAnchoredItem->ContentAppended(aShell, aPresContext, aContainer);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP nsPlaceholderFrame::ContentInserted(nsIPresShell* aShell,
|
|
nsIPresContext* aPresContext,
|
|
nsIContent* aContainer,
|
|
nsIContent* aChild,
|
|
PRInt32 aIndexInParent)
|
|
{
|
|
NS_ASSERTION(mContent == aContainer, "bad content-inserted target");
|
|
|
|
// Forward the notification to the floater
|
|
if (nsnull != mAnchoredItem) {
|
|
return mAnchoredItem->ContentInserted(aShell, aPresContext, aContainer,
|
|
aChild, aIndexInParent);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP nsPlaceholderFrame::ContentReplaced(nsIPresShell* aShell,
|
|
nsIPresContext* aPresContext,
|
|
nsIContent* aContainer,
|
|
nsIContent* aOldChild,
|
|
nsIContent* aNewChild,
|
|
PRInt32 aIndexInParent)
|
|
{
|
|
NS_ASSERTION(mContent == aContainer, "bad content-replaced target");
|
|
|
|
// Forward the notification to the floater
|
|
if (nsnull != mAnchoredItem) {
|
|
return mAnchoredItem->ContentReplaced(aShell, aPresContext, aContainer,
|
|
aOldChild, aNewChild, aIndexInParent);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP nsPlaceholderFrame::ContentDeleted(nsIPresShell* aShell,
|
|
nsIPresContext* aPresContext,
|
|
nsIContent* aContainer,
|
|
nsIContent* aChild,
|
|
PRInt32 aIndexInParent)
|
|
{
|
|
NS_ASSERTION(mContent == aContainer, "bad content-deleted target");
|
|
|
|
// Forward the notification to the floater
|
|
if (nsnull != mAnchoredItem) {
|
|
return mAnchoredItem->ContentDeleted(aShell, aPresContext, aContainer,
|
|
aChild, aIndexInParent);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP nsPlaceholderFrame::ContentChanged(nsIPresShell* aShell,
|
|
nsIPresContext* aPresContext,
|
|
nsIContent* aChild,
|
|
nsISupports* aSubContent)
|
|
{
|
|
NS_ASSERTION(mContent == aChild, "bad content-changed target");
|
|
|
|
// Forward the notification to the floater
|
|
if (nsnull != mAnchoredItem) {
|
|
return mAnchoredItem->ContentChanged(aShell, aPresContext, aChild, aSubContent);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP nsPlaceholderFrame::ListTag(FILE* out) const
|
|
{
|
|
fputs("*placeholder", out);
|
|
PRInt32 contentIndex;
|
|
GetContentIndex(contentIndex);
|
|
fprintf(out, "(%d)@%p", contentIndex, this);
|
|
return NS_OK;
|
|
}
|