1999-02-05 03:55:18 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
|
|
*
|
1999-11-06 03:40:37 +00:00
|
|
|
* The contents of this file are subject to the Netscape Public
|
|
|
|
* License Version 1.1 (the "License"); you may not use this file
|
|
|
|
* except in compliance with the License. You may obtain a copy of
|
|
|
|
* the License at http://www.mozilla.org/NPL/
|
1999-02-05 03:55:18 +00:00
|
|
|
*
|
1999-11-06 03:40:37 +00:00
|
|
|
* Software distributed under the License is distributed on an "AS
|
|
|
|
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
|
|
|
|
* implied. See the License for the specific language governing
|
|
|
|
* rights and limitations under the License.
|
1999-02-05 03:55:18 +00:00
|
|
|
*
|
1999-11-06 03:40:37 +00:00
|
|
|
* The Original Code is mozilla.org code.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is Netscape
|
1999-02-05 03:55:18 +00:00
|
|
|
* Communications Corporation. Portions created by Netscape are
|
1999-11-06 03:40:37 +00:00
|
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All
|
|
|
|
* Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
2000-02-02 22:24:56 +00:00
|
|
|
* Pierre Phaneuf <pp@ludusdesign.com>
|
1999-02-05 03:55:18 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "nsCSSFrameConstructor.h"
|
|
|
|
#include "nsIArena.h"
|
|
|
|
#include "nsCRT.h"
|
|
|
|
#include "nsIAtom.h"
|
|
|
|
#include "nsIURL.h"
|
|
|
|
#include "nsISupportsArray.h"
|
|
|
|
#include "nsHashtable.h"
|
|
|
|
#include "nsIHTMLContent.h"
|
|
|
|
#include "nsIHTMLAttributes.h"
|
|
|
|
#include "nsIStyleRule.h"
|
|
|
|
#include "nsIFrame.h"
|
|
|
|
#include "nsIStyleContext.h"
|
|
|
|
#include "nsHTMLAtoms.h"
|
|
|
|
#include "nsIPresContext.h"
|
|
|
|
#include "nsILinkHandler.h"
|
|
|
|
#include "nsIDocument.h"
|
|
|
|
#include "nsIHTMLTableCellElement.h"
|
1999-12-13 22:56:31 +00:00
|
|
|
#include "nsTableColGroupFrame.h"
|
1999-02-05 03:55:18 +00:00
|
|
|
#include "nsTableColFrame.h"
|
|
|
|
#include "nsHTMLIIDs.h"
|
|
|
|
#include "nsIStyleFrameConstruction.h"
|
|
|
|
#include "nsHTMLParts.h"
|
|
|
|
#include "nsIPresShell.h"
|
1999-04-20 00:05:14 +00:00
|
|
|
#include "nsIStyleSet.h"
|
1999-02-05 03:55:18 +00:00
|
|
|
#include "nsIViewManager.h"
|
|
|
|
#include "nsStyleConsts.h"
|
|
|
|
#include "nsTableOuterFrame.h"
|
|
|
|
#include "nsIXMLDocument.h"
|
|
|
|
#include "nsIWebShell.h"
|
|
|
|
#include "nsHTMLContainerFrame.h"
|
|
|
|
#include "nsINameSpaceManager.h"
|
|
|
|
#include "nsLayoutAtoms.h"
|
|
|
|
#include "nsIDOMHTMLSelectElement.h"
|
|
|
|
#include "nsIComboboxControlFrame.h"
|
1999-08-25 13:42:59 +00:00
|
|
|
#include "nsIListControlFrame.h"
|
1999-06-21 20:41:56 +00:00
|
|
|
#include "nsIRadioControlFrame.h"
|
1999-02-05 03:55:18 +00:00
|
|
|
#include "nsIListControlFrame.h"
|
|
|
|
#include "nsIDOMCharacterData.h"
|
|
|
|
#include "nsIDOMHTMLImageElement.h"
|
|
|
|
#include "nsITextContent.h"
|
1999-02-26 17:11:54 +00:00
|
|
|
#include "nsPlaceholderFrame.h"
|
1999-03-03 16:33:57 +00:00
|
|
|
#include "nsTableRowGroupFrame.h"
|
1999-03-25 06:41:43 +00:00
|
|
|
#include "nsStyleChangeList.h"
|
1999-03-31 06:47:40 +00:00
|
|
|
#include "nsIFormControl.h"
|
1999-03-30 15:22:54 +00:00
|
|
|
#include "nsCSSAtoms.h"
|
1999-04-11 03:10:20 +00:00
|
|
|
#include "nsIDeviceContext.h"
|
1999-04-27 22:14:54 +00:00
|
|
|
#include "nsTextFragment.h"
|
1999-06-30 22:17:43 +00:00
|
|
|
#include "nsISupportsArray.h"
|
|
|
|
#include "nsIAnonymousContentCreator.h"
|
1999-08-05 03:09:22 +00:00
|
|
|
#include "nsIFrameManager.h"
|
1999-08-19 14:29:55 +00:00
|
|
|
#include "nsIAttributeContent.h"
|
1999-10-12 00:16:06 +00:00
|
|
|
#include "nsIPref.h"
|
2000-01-09 02:04:36 +00:00
|
|
|
#include "nsLegendFrame.h"
|
2000-02-14 01:42:09 +00:00
|
|
|
#include "nsTitleFrame.h"
|
1999-02-05 03:55:18 +00:00
|
|
|
|
2000-02-12 03:10:01 +00:00
|
|
|
#include "nsIDOMWindow.h"
|
|
|
|
#include "nsPIDOMWindow.h"
|
2000-02-14 01:42:09 +00:00
|
|
|
#ifdef INCLUDE_XUL
|
|
|
|
#include "nsIDOMXULCommandDispatcher.h"
|
|
|
|
#include "nsIDOMXULDocument.h"
|
|
|
|
#endif
|
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
// XXX - temporary, this is for GfxList View
|
|
|
|
#include "nsViewsCID.h"
|
|
|
|
#include "nsWidgetsCID.h"
|
|
|
|
// to here
|
2000-02-12 03:10:01 +00:00
|
|
|
|
1999-08-31 03:09:40 +00:00
|
|
|
#include "nsInlineFrame.h"
|
1999-10-29 14:39:48 +00:00
|
|
|
#include "nsBlockFrame.h"
|
1999-12-17 03:28:50 +00:00
|
|
|
|
|
|
|
#include "nsGfxTextControlFrame.h"
|
2000-02-11 01:23:36 +00:00
|
|
|
#include "nsIScrollableFrame.h"
|
1999-12-17 03:28:50 +00:00
|
|
|
|
2000-01-11 08:25:59 +00:00
|
|
|
#include "nsIServiceManager.h"
|
|
|
|
#include "nsIXBLService.h"
|
|
|
|
|
1999-08-31 03:09:40 +00:00
|
|
|
#undef NOISY_FIRST_LETTER
|
|
|
|
|
1999-10-02 10:41:40 +00:00
|
|
|
#ifdef MOZ_MATHML
|
1999-09-21 01:15:30 +00:00
|
|
|
#include "nsMathMLAtoms.h"
|
|
|
|
#include "nsMathMLParts.h"
|
|
|
|
#endif
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
#ifdef INCLUDE_XUL
|
|
|
|
#include "nsXULAtoms.h"
|
|
|
|
#include "nsTreeFrame.h"
|
1999-08-24 00:44:21 +00:00
|
|
|
#include "nsTreeOuterFrame.h"
|
1999-06-23 03:02:21 +00:00
|
|
|
#include "nsTreeRowGroupFrame.h"
|
1999-08-28 00:45:18 +00:00
|
|
|
#include "nsTreeRowFrame.h"
|
1999-02-05 03:55:18 +00:00
|
|
|
#include "nsToolboxFrame.h"
|
|
|
|
#include "nsToolbarFrame.h"
|
|
|
|
#include "nsTreeIndentationFrame.h"
|
|
|
|
#include "nsTreeCellFrame.h"
|
1999-06-15 04:02:43 +00:00
|
|
|
#include "nsIDOMElement.h"
|
|
|
|
#include "nsIDOMDocument.h"
|
|
|
|
#include "nsDocument.h"
|
1999-07-01 21:11:38 +00:00
|
|
|
#include "nsToolbarItemFrame.h"
|
2000-02-25 08:37:49 +00:00
|
|
|
#include "nsXULCheckboxFrame.h"
|
2000-02-29 06:51:04 +00:00
|
|
|
#include "nsIScrollable.h"
|
1999-07-01 21:11:38 +00:00
|
|
|
|
1999-11-01 15:24:57 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
static PRBool gNoisyContentUpdates = PR_FALSE;
|
|
|
|
static PRBool gReallyNoisyContentUpdates = PR_FALSE;
|
|
|
|
static PRBool gNoisyInlineConstruction = PR_FALSE;
|
|
|
|
#endif
|
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
//------------------------------------------------------------------
|
|
|
|
// This is temporary while GfxLists and GfxDropdowns are being put in
|
|
|
|
static PRBool gDoGfxDropdown = PR_FALSE;
|
|
|
|
static PRBool gDoGfxListbox = PR_FALSE;
|
|
|
|
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
#define NEWGFX_LIST_SCROLLFRAME
|
2000-03-15 15:45:29 +00:00
|
|
|
//------------------------------------------------------------------
|
1999-11-19 15:41:19 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewGfxListControlFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-11-19 15:41:19 +00:00
|
|
|
|
1999-08-19 22:16:23 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewThumbFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-08-19 22:16:23 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewScrollPortFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-08-19 22:16:23 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewGfxScrollFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame, nsIDocument* aDocument );
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-04-21 22:46:15 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewTabFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-04-21 22:46:15 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewDeckFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-04-21 22:46:15 +00:00
|
|
|
|
2000-03-02 03:01:30 +00:00
|
|
|
nsresult
|
|
|
|
NS_NewSpringFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
NS_NewStackFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
|
|
|
|
1999-02-19 18:23:02 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewProgressMeterFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-02-19 18:23:02 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewTitledButtonFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-02-19 18:23:02 +00:00
|
|
|
|
2000-02-16 23:00:52 +00:00
|
|
|
nsresult
|
|
|
|
NS_NewXULTextFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
|
|
|
|
2000-02-14 01:42:09 +00:00
|
|
|
nsresult
|
|
|
|
NS_NewTitledBoxFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
NS_NewTitledBoxInnerFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
NS_NewTitleFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
|
|
|
|
1999-03-27 00:58:29 +00:00
|
|
|
nsresult
|
2000-03-02 03:01:30 +00:00
|
|
|
NS_NewBoxFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame, PRBool aIsRoot);
|
1999-03-27 00:58:29 +00:00
|
|
|
|
2000-03-11 10:38:36 +00:00
|
|
|
nsresult
|
|
|
|
NS_NewXULButtonFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame);
|
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewSliderFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewScrollbarFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewSpinnerFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewColorPickerFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewFontPickerFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewScrollbarButtonFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewScrollbarFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-06-15 04:02:43 +00:00
|
|
|
|
1999-06-30 22:17:43 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewGrippyFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-06-30 22:17:43 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewSplitterFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-07-18 06:29:43 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewMenuPopupFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-07-18 06:29:43 +00:00
|
|
|
|
1999-08-06 19:20:56 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewPopupSetFrame(nsIPresShell* aPresShell, nsIFrame** aNewFrame);
|
1999-08-06 19:20:56 +00:00
|
|
|
|
1999-07-18 06:37:06 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewMenuFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame, PRUint32 aFlags );
|
1999-07-18 07:07:55 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewMenuBarFrame ( nsIPresShell* aPresShell, nsIFrame** aNewFrame );
|
1999-07-18 07:07:55 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
//static NS_DEFINE_IID(kIStyleRuleIID, NS_ISTYLE_RULE_IID);
|
|
|
|
static NS_DEFINE_IID(kIStyleFrameConstructionIID, NS_ISTYLE_FRAME_CONSTRUCTION_IID);
|
|
|
|
static NS_DEFINE_IID(kIHTMLTableCellElementIID, NS_IHTMLTABLECELLELEMENT_IID);
|
|
|
|
static NS_DEFINE_IID(kIXMLDocumentIID, NS_IXMLDOCUMENT_IID);
|
|
|
|
static NS_DEFINE_IID(kIWebShellIID, NS_IWEB_SHELL_IID);
|
|
|
|
|
|
|
|
static NS_DEFINE_IID(kIDOMHTMLSelectElementIID, NS_IDOMHTMLSELECTELEMENT_IID);
|
|
|
|
static NS_DEFINE_IID(kIComboboxControlFrameIID, NS_ICOMBOBOXCONTROLFRAME_IID);
|
1999-06-21 20:41:56 +00:00
|
|
|
static NS_DEFINE_IID(kIRadioControlFrameIID, NS_IRADIOCONTROLFRAME_IID);
|
1999-02-05 03:55:18 +00:00
|
|
|
static NS_DEFINE_IID(kIListControlFrameIID, NS_ILISTCONTROLFRAME_IID);
|
|
|
|
static NS_DEFINE_IID(kIDOMHTMLImageElementIID, NS_IDOMHTMLIMAGEELEMENT_IID);
|
|
|
|
static NS_DEFINE_IID(kIDOMCharacterDataIID, NS_IDOMCHARACTERDATA_IID);
|
1999-02-16 04:41:15 +00:00
|
|
|
static NS_DEFINE_IID(kScrollViewIID, NS_ISCROLLABLEVIEW_IID);
|
1999-03-31 06:47:40 +00:00
|
|
|
static NS_DEFINE_IID(kIFormControlIID, NS_IFORMCONTROL_IID);
|
1999-04-12 22:14:31 +00:00
|
|
|
static NS_DEFINE_IID(kIFrameIID, NS_IFRAME_IID);
|
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
// Structure used when constructing formatting object trees.
|
|
|
|
struct nsFrameItems {
|
|
|
|
nsIFrame* childList;
|
|
|
|
nsIFrame* lastChild;
|
|
|
|
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameItems(nsIFrame* aFrame = nsnull);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Appends the frame to the end of the list
|
|
|
|
void AddChild(nsIFrame* aChild);
|
|
|
|
};
|
|
|
|
|
|
|
|
nsFrameItems::nsFrameItems(nsIFrame* aFrame)
|
1999-02-26 17:11:54 +00:00
|
|
|
: childList(aFrame), lastChild(aFrame)
|
|
|
|
{
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
nsFrameItems::AddChild(nsIFrame* aChild)
|
|
|
|
{
|
|
|
|
if (childList == nsnull) {
|
|
|
|
childList = lastChild = aChild;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
lastChild->SetNextSibling(aChild);
|
|
|
|
lastChild = aChild;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
|
|
|
// Structure used when constructing formatting object trees. Contains
|
|
|
|
// state information needed for absolutely positioned elements
|
|
|
|
struct nsAbsoluteItems : nsFrameItems {
|
1999-02-26 17:11:54 +00:00
|
|
|
// containing block for absolutely positioned elements
|
1999-04-28 19:08:14 +00:00
|
|
|
nsIFrame* containingBlock;
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-04-28 19:08:14 +00:00
|
|
|
nsAbsoluteItems(nsIFrame* aContainingBlock = nsnull);
|
1999-04-25 17:20:53 +00:00
|
|
|
|
|
|
|
// Appends the frame to the end of the list
|
|
|
|
void AddChild(nsIFrame* aChild);
|
1999-02-05 03:55:18 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
nsAbsoluteItems::nsAbsoluteItems(nsIFrame* aContainingBlock)
|
|
|
|
: containingBlock(aContainingBlock)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
1999-04-25 17:20:53 +00:00
|
|
|
// Additional behavior is that it sets the frame's NS_FRAME_OUT_OF_FLOW flag
|
|
|
|
void
|
|
|
|
nsAbsoluteItems::AddChild(nsIFrame* aChild)
|
|
|
|
{
|
|
|
|
nsFrameState frameState;
|
|
|
|
|
|
|
|
// Mark the frame as being moved out of the flow
|
|
|
|
aChild->GetFrameState(&frameState);
|
|
|
|
aChild->SetFrameState(frameState | NS_FRAME_OUT_OF_FLOW);
|
|
|
|
nsFrameItems::AddChild(aChild);
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-10 19:50:50 +00:00
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
1999-04-28 19:08:14 +00:00
|
|
|
// Structure for saving the existing state when pushing/poping containing
|
|
|
|
// blocks. The destructor restores the state to its previous state
|
|
|
|
class nsFrameConstructorSaveState {
|
|
|
|
public:
|
|
|
|
nsFrameConstructorSaveState();
|
|
|
|
~nsFrameConstructorSaveState();
|
|
|
|
|
|
|
|
private:
|
|
|
|
nsAbsoluteItems* mItems; // pointer to struct whose data we save/restore
|
1999-08-27 21:46:10 +00:00
|
|
|
PRBool* mFirstLetterStyle;
|
1999-08-31 03:09:40 +00:00
|
|
|
PRBool* mFirstLineStyle;
|
1999-08-27 21:46:10 +00:00
|
|
|
|
1999-04-28 19:08:14 +00:00
|
|
|
nsAbsoluteItems mSavedItems; // copy of original data
|
1999-08-27 21:46:10 +00:00
|
|
|
PRBool mSavedFirstLetterStyle;
|
1999-08-31 03:09:40 +00:00
|
|
|
PRBool mSavedFirstLineStyle;
|
1999-08-27 21:46:10 +00:00
|
|
|
|
1999-04-28 19:08:14 +00:00
|
|
|
friend class nsFrameConstructorState;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Structure used for maintaining state information during the
|
|
|
|
// frame construction process
|
|
|
|
class nsFrameConstructorState {
|
|
|
|
public:
|
1999-08-05 03:09:22 +00:00
|
|
|
nsCOMPtr<nsIPresShell> mPresShell;
|
|
|
|
nsCOMPtr<nsIFrameManager> mFrameManager;
|
|
|
|
|
1999-04-28 19:08:14 +00:00
|
|
|
// Containing block information for out-of-flow frammes
|
1999-08-05 03:09:22 +00:00
|
|
|
nsAbsoluteItems mFixedItems;
|
|
|
|
nsAbsoluteItems mAbsoluteItems;
|
|
|
|
nsAbsoluteItems mFloatedItems;
|
1999-08-27 21:46:10 +00:00
|
|
|
PRBool mFirstLetterStyle;
|
1999-08-31 03:09:40 +00:00
|
|
|
PRBool mFirstLineStyle;
|
1999-12-06 07:44:18 +00:00
|
|
|
nsCOMPtr<nsILayoutHistoryState> mFrameState;
|
1999-04-28 19:08:14 +00:00
|
|
|
|
|
|
|
// Constructor
|
1999-12-06 07:44:18 +00:00
|
|
|
nsFrameConstructorState(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aFixedContainingBlock,
|
|
|
|
nsIFrame* aAbsoluteContainingBlock,
|
|
|
|
nsIFrame* aFloaterContainingBlock,
|
|
|
|
nsILayoutHistoryState* aFrameState);
|
1999-04-28 19:08:14 +00:00
|
|
|
|
|
|
|
// Function to push the existing absolute containing block state and
|
|
|
|
// create a new scope
|
|
|
|
void PushAbsoluteContainingBlock(nsIFrame* aNewAbsoluteContainingBlock,
|
|
|
|
nsFrameConstructorSaveState& aSaveState);
|
|
|
|
|
|
|
|
// Function to push the existing floater containing block state and
|
|
|
|
// create a new scope
|
|
|
|
void PushFloaterContainingBlock(nsIFrame* aNewFloaterContainingBlock,
|
1999-08-27 21:46:10 +00:00
|
|
|
nsFrameConstructorSaveState& aSaveState,
|
1999-08-31 03:09:40 +00:00
|
|
|
PRBool aFirstLetterStyle,
|
|
|
|
PRBool aFirstLineStyle);
|
1999-04-28 19:08:14 +00:00
|
|
|
};
|
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
nsFrameConstructorState::nsFrameConstructorState(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aFixedContainingBlock,
|
|
|
|
nsIFrame* aAbsoluteContainingBlock,
|
|
|
|
nsIFrame* aFloaterContainingBlock,
|
|
|
|
nsILayoutHistoryState* aFrameState)
|
1999-04-28 19:08:14 +00:00
|
|
|
: mFixedItems(aFixedContainingBlock),
|
|
|
|
mAbsoluteItems(aAbsoluteContainingBlock),
|
1999-08-27 21:46:10 +00:00
|
|
|
mFloatedItems(aFloaterContainingBlock),
|
1999-09-20 03:46:09 +00:00
|
|
|
mFirstLetterStyle(PR_FALSE),
|
1999-12-06 07:44:18 +00:00
|
|
|
mFirstLineStyle(PR_FALSE),
|
|
|
|
mFrameState(aFrameState)
|
1999-04-28 19:08:14 +00:00
|
|
|
{
|
1999-08-05 03:09:22 +00:00
|
|
|
aPresContext->GetShell(getter_AddRefs(mPresShell));
|
|
|
|
mPresShell->GetFrameManager(getter_AddRefs(mFrameManager));
|
1999-04-28 19:08:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsFrameConstructorState::PushAbsoluteContainingBlock(nsIFrame* aNewAbsoluteContainingBlock,
|
|
|
|
nsFrameConstructorSaveState& aSaveState)
|
|
|
|
{
|
|
|
|
aSaveState.mItems = &mAbsoluteItems;
|
|
|
|
aSaveState.mSavedItems = mAbsoluteItems;
|
|
|
|
mAbsoluteItems = nsAbsoluteItems(aNewAbsoluteContainingBlock);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsFrameConstructorState::PushFloaterContainingBlock(nsIFrame* aNewFloaterContainingBlock,
|
1999-08-27 21:46:10 +00:00
|
|
|
nsFrameConstructorSaveState& aSaveState,
|
1999-08-31 03:09:40 +00:00
|
|
|
PRBool aFirstLetterStyle,
|
|
|
|
PRBool aFirstLineStyle)
|
1999-04-28 19:08:14 +00:00
|
|
|
{
|
|
|
|
aSaveState.mItems = &mFloatedItems;
|
1999-08-27 21:46:10 +00:00
|
|
|
aSaveState.mFirstLetterStyle = &mFirstLetterStyle;
|
1999-09-20 03:46:09 +00:00
|
|
|
aSaveState.mFirstLineStyle = &mFirstLineStyle;
|
1999-04-28 19:08:14 +00:00
|
|
|
aSaveState.mSavedItems = mFloatedItems;
|
1999-08-27 21:46:10 +00:00
|
|
|
aSaveState.mSavedFirstLetterStyle = mFirstLetterStyle;
|
1999-08-31 03:09:40 +00:00
|
|
|
aSaveState.mSavedFirstLineStyle = mFirstLineStyle;
|
1999-04-28 19:08:14 +00:00
|
|
|
mFloatedItems = nsAbsoluteItems(aNewFloaterContainingBlock);
|
1999-08-27 21:46:10 +00:00
|
|
|
mFirstLetterStyle = aFirstLetterStyle;
|
1999-08-31 03:09:40 +00:00
|
|
|
mFirstLineStyle = aFirstLineStyle;
|
1999-04-28 19:08:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsFrameConstructorSaveState::nsFrameConstructorSaveState()
|
1999-09-26 10:07:16 +00:00
|
|
|
: mItems(nsnull), mFirstLetterStyle(nsnull), mFirstLineStyle(nsnull),
|
|
|
|
mSavedFirstLetterStyle(PR_FALSE), mSavedFirstLineStyle(PR_FALSE)
|
1999-04-28 19:08:14 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
nsFrameConstructorSaveState::~nsFrameConstructorSaveState()
|
|
|
|
{
|
|
|
|
// Restore the state
|
|
|
|
if (mItems) {
|
|
|
|
*mItems = mSavedItems;
|
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
if (mFirstLetterStyle) {
|
|
|
|
*mFirstLetterStyle = mSavedFirstLetterStyle;
|
|
|
|
}
|
1999-08-31 03:09:40 +00:00
|
|
|
if (mFirstLineStyle) {
|
|
|
|
*mFirstLineStyle = mSavedFirstLineStyle;
|
|
|
|
}
|
1999-04-28 19:08:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
1999-02-10 19:50:50 +00:00
|
|
|
// Structure used when creating table frames.
|
|
|
|
struct nsTableCreator {
|
1999-08-24 00:44:21 +00:00
|
|
|
virtual nsresult CreateTableOuterFrame(nsIFrame** aNewFrame);
|
1999-05-11 22:03:29 +00:00
|
|
|
virtual nsresult CreateTableFrame(nsIFrame** aNewFrame);
|
|
|
|
virtual nsresult CreateTableRowGroupFrame(nsIFrame** aNewFrame);
|
|
|
|
virtual nsresult CreateTableColFrame(nsIFrame** aNewFrame);
|
|
|
|
virtual nsresult CreateTableColGroupFrame(nsIFrame** aNewFrame);
|
|
|
|
virtual nsresult CreateTableRowFrame(nsIFrame** aNewFrame);
|
|
|
|
virtual nsresult CreateTableCellFrame(nsIFrame** aNewFrame);
|
1999-09-21 01:15:30 +00:00
|
|
|
virtual nsresult CreateTableCellInnerFrame(nsIFrame** aNewFrame);
|
1999-12-01 10:37:20 +00:00
|
|
|
|
1999-06-10 09:31:30 +00:00
|
|
|
virtual PRBool IsTreeCreator() { return PR_FALSE; };
|
1999-12-04 23:49:50 +00:00
|
|
|
|
|
|
|
nsTableCreator(nsIPresShell* aPresShell)
|
|
|
|
{
|
|
|
|
mPresShell = aPresShell;
|
|
|
|
}
|
|
|
|
|
1999-12-05 04:57:13 +00:00
|
|
|
virtual ~nsTableCreator() {};
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCOMPtr<nsIPresShell> mPresShell;
|
1999-02-10 19:50:50 +00:00
|
|
|
};
|
|
|
|
|
1999-08-24 00:44:21 +00:00
|
|
|
nsresult
|
|
|
|
nsTableCreator::CreateTableOuterFrame(nsIFrame** aNewFrame) {
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTableOuterFrame(mPresShell, aNewFrame);
|
1999-08-24 00:44:21 +00:00
|
|
|
}
|
|
|
|
|
1999-02-10 19:50:50 +00:00
|
|
|
nsresult
|
1999-05-11 22:03:29 +00:00
|
|
|
nsTableCreator::CreateTableFrame(nsIFrame** aNewFrame) {
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTableFrame(mPresShell, aNewFrame);
|
1999-02-10 19:50:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-05-11 22:03:29 +00:00
|
|
|
nsTableCreator::CreateTableRowGroupFrame(nsIFrame** aNewFrame) {
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTableRowGroupFrame(mPresShell, aNewFrame);
|
1999-02-10 19:50:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-05-11 22:03:29 +00:00
|
|
|
nsTableCreator::CreateTableColFrame(nsIFrame** aNewFrame) {
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTableColFrame(mPresShell, aNewFrame);
|
1999-02-10 19:50:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-05-11 22:03:29 +00:00
|
|
|
nsTableCreator::CreateTableColGroupFrame(nsIFrame** aNewFrame) {
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTableColGroupFrame(mPresShell, aNewFrame);
|
1999-02-10 19:50:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-05-11 22:03:29 +00:00
|
|
|
nsTableCreator::CreateTableRowFrame(nsIFrame** aNewFrame) {
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTableRowFrame(mPresShell, aNewFrame);
|
1999-02-10 19:50:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-05-11 22:03:29 +00:00
|
|
|
nsTableCreator::CreateTableCellFrame(nsIFrame** aNewFrame) {
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTableCellFrame(mPresShell, aNewFrame);
|
1999-02-10 19:50:50 +00:00
|
|
|
}
|
|
|
|
|
1999-12-01 10:37:20 +00:00
|
|
|
nsresult
|
|
|
|
nsTableCreator::CreateTableCellInnerFrame(nsIFrame** aNewFrame) {
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTableCellInnerFrame(mPresShell, aNewFrame);
|
1999-12-01 10:37:20 +00:00
|
|
|
}
|
|
|
|
|
1999-02-10 19:50:50 +00:00
|
|
|
#ifdef INCLUDE_XUL
|
|
|
|
|
|
|
|
// Structure used when creating tree frames
|
|
|
|
struct nsTreeCreator: public nsTableCreator {
|
1999-08-24 00:44:21 +00:00
|
|
|
nsresult CreateTableOuterFrame(nsIFrame** aNewFrame);
|
1999-05-11 22:03:29 +00:00
|
|
|
nsresult CreateTableFrame(nsIFrame** aNewFrame);
|
|
|
|
nsresult CreateTableCellFrame(nsIFrame** aNewFrame);
|
1999-06-23 03:02:21 +00:00
|
|
|
nsresult CreateTableRowGroupFrame(nsIFrame** aNewFrame);
|
1999-08-28 00:45:18 +00:00
|
|
|
nsresult CreateTableRowFrame(nsIFrame** aNewFrame);
|
1999-12-01 10:37:20 +00:00
|
|
|
nsresult CreateTableCellInnerFrame(nsIFrame** aNewFrame);
|
1999-06-10 09:31:30 +00:00
|
|
|
|
|
|
|
PRBool IsTreeCreator() { return PR_TRUE; };
|
1999-12-04 23:49:50 +00:00
|
|
|
|
|
|
|
nsTreeCreator(nsIPresShell* aPresShell)
|
|
|
|
:nsTableCreator(aPresShell) {};
|
1999-02-10 19:50:50 +00:00
|
|
|
};
|
|
|
|
|
1999-08-24 00:44:21 +00:00
|
|
|
nsresult
|
|
|
|
nsTreeCreator::CreateTableOuterFrame(nsIFrame** aNewFrame)
|
|
|
|
{
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTreeOuterFrame(mPresShell, aNewFrame);
|
1999-08-24 00:44:21 +00:00
|
|
|
}
|
|
|
|
|
1999-02-10 19:50:50 +00:00
|
|
|
nsresult
|
1999-05-11 22:03:29 +00:00
|
|
|
nsTreeCreator::CreateTableFrame(nsIFrame** aNewFrame)
|
1999-02-10 19:50:50 +00:00
|
|
|
{
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTreeFrame(mPresShell, aNewFrame);
|
1999-02-10 19:50:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-05-11 22:03:29 +00:00
|
|
|
nsTreeCreator::CreateTableCellFrame(nsIFrame** aNewFrame)
|
1999-02-10 19:50:50 +00:00
|
|
|
{
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTreeCellFrame(mPresShell, aNewFrame);
|
1999-02-10 19:50:50 +00:00
|
|
|
}
|
|
|
|
|
1999-06-23 03:02:21 +00:00
|
|
|
nsresult
|
|
|
|
nsTreeCreator::CreateTableRowGroupFrame(nsIFrame** aNewFrame)
|
|
|
|
{
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTreeRowGroupFrame(mPresShell, aNewFrame);
|
1999-06-23 03:02:21 +00:00
|
|
|
}
|
|
|
|
|
1999-08-28 00:45:18 +00:00
|
|
|
nsresult
|
|
|
|
nsTreeCreator::CreateTableRowFrame(nsIFrame** aNewFrame) {
|
1999-12-04 23:49:50 +00:00
|
|
|
return NS_NewTreeRowFrame(mPresShell, aNewFrame);
|
1999-08-28 00:45:18 +00:00
|
|
|
}
|
|
|
|
|
1999-12-01 10:37:20 +00:00
|
|
|
nsresult
|
|
|
|
nsTreeCreator::CreateTableCellInnerFrame(nsIFrame** aNewFrame) {
|
2000-02-13 09:28:51 +00:00
|
|
|
return NS_NewBoxFrame(mPresShell, aNewFrame);
|
|
|
|
//return NS_NewTableCellInnerFrame(mPresShell, aNewFrame);
|
1999-12-01 10:37:20 +00:00
|
|
|
}
|
|
|
|
|
1999-02-10 19:50:50 +00:00
|
|
|
#endif // INCLUDE_XUL
|
|
|
|
|
1999-09-21 01:15:30 +00:00
|
|
|
//MathML Mod - RBS
|
1999-10-02 10:41:40 +00:00
|
|
|
#ifdef MOZ_MATHML
|
1999-09-21 01:15:30 +00:00
|
|
|
|
|
|
|
// Structure used when creating MathML mtable frames
|
|
|
|
struct nsMathMLmtableCreator: public nsTableCreator {
|
|
|
|
nsresult CreateTableCellInnerFrame(nsIFrame** aNewFrame);
|
1999-12-10 13:07:59 +00:00
|
|
|
|
|
|
|
nsMathMLmtableCreator(nsIPresShell* aPresShell)
|
|
|
|
:nsTableCreator(aPresShell) {};
|
1999-09-21 01:15:30 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsMathMLmtableCreator::CreateTableCellInnerFrame(nsIFrame** aNewFrame)
|
|
|
|
{
|
|
|
|
// only works if aNewFrame is an AreaFrame (to take care of the lineLayout logic)
|
1999-12-10 13:07:59 +00:00
|
|
|
return NS_NewMathMLmtdFrame(mPresShell, aNewFrame);
|
1999-09-21 01:15:30 +00:00
|
|
|
}
|
1999-10-02 10:41:40 +00:00
|
|
|
#endif // MOZ_MATHML
|
1999-09-21 01:15:30 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
2000-01-28 02:19:45 +00:00
|
|
|
// return the child list that aFrame belongs on. does not ADDREF
|
|
|
|
nsIAtom* GetChildListFor(const nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
nsIAtom* childList = nsnull;
|
2000-02-04 03:16:47 +00:00
|
|
|
nsCOMPtr<nsIAtom> frameType;
|
|
|
|
aFrame->GetFrameType(getter_AddRefs(frameType));
|
|
|
|
if (nsLayoutAtoms::tableCaptionFrame == frameType.get()) {
|
2000-01-28 02:19:45 +00:00
|
|
|
childList = nsLayoutAtoms::captionList;
|
|
|
|
}
|
|
|
|
return childList;
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
nsCSSFrameConstructor::nsCSSFrameConstructor(void)
|
|
|
|
: nsIStyleFrameConstruction(),
|
|
|
|
mDocument(nsnull),
|
|
|
|
mInitialContainingBlock(nsnull),
|
|
|
|
mFixedContainingBlock(nsnull),
|
1999-08-27 06:06:39 +00:00
|
|
|
mDocElementContainingBlock(nsnull),
|
|
|
|
mGfxScrollFrame(nsnull)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
NS_INIT_REFCNT();
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCSSFrameConstructor::~nsCSSFrameConstructor(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMPL_ISUPPORTS(nsCSSFrameConstructor, kIStyleFrameConstructionIID);
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::Init(nsIDocument* aDocument)
|
|
|
|
{
|
|
|
|
NS_PRECONDITION(aDocument, "null ptr");
|
|
|
|
if (! aDocument)
|
|
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
|
|
|
|
if (mDocument)
|
|
|
|
return NS_ERROR_ALREADY_INITIALIZED;
|
|
|
|
|
|
|
|
mDocument = aDocument; // not refcounted!
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-04-27 03:58:46 +00:00
|
|
|
// Helper function that determines the child list name that aChildFrame
|
|
|
|
// is contained in
|
|
|
|
static void
|
2000-01-22 01:16:50 +00:00
|
|
|
GetChildListNameFor(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIFrame* aChildFrame,
|
|
|
|
nsIAtom** aListName)
|
1999-04-27 03:58:46 +00:00
|
|
|
{
|
|
|
|
nsFrameState frameState;
|
|
|
|
nsIAtom* listName;
|
|
|
|
|
|
|
|
// See if the frame is moved out of the flow
|
|
|
|
aChildFrame->GetFrameState(&frameState);
|
|
|
|
if (frameState & NS_FRAME_OUT_OF_FLOW) {
|
|
|
|
// Look at the style information to tell
|
|
|
|
const nsStylePosition* position;
|
|
|
|
aChildFrame->GetStyleData(eStyleStruct_Position, (const nsStyleStruct*&)position);
|
|
|
|
|
|
|
|
if (NS_STYLE_POSITION_ABSOLUTE == position->mPosition) {
|
|
|
|
listName = nsLayoutAtoms::absoluteList;
|
|
|
|
} else if (NS_STYLE_POSITION_FIXED == position->mPosition) {
|
|
|
|
listName = nsLayoutAtoms::fixedList;
|
|
|
|
} else {
|
|
|
|
#ifdef NS_DEBUG
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
aChildFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct*&)display);
|
|
|
|
NS_ASSERTION(display->IsFloating(), "not a floated frame");
|
|
|
|
#endif
|
|
|
|
listName = nsLayoutAtoms::floaterList;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
listName = nsnull;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify that the frame is actually in that child list
|
|
|
|
#ifdef NS_DEBUG
|
|
|
|
nsIFrame* firstChild;
|
2000-01-22 01:16:50 +00:00
|
|
|
aParentFrame->FirstChild(aPresContext, listName, &firstChild);
|
1999-04-27 03:58:46 +00:00
|
|
|
|
|
|
|
nsFrameList frameList(firstChild);
|
|
|
|
NS_ASSERTION(frameList.ContainsFrame(aChildFrame), "not in child list");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
NS_IF_ADDREF(listName);
|
|
|
|
*aListName = listName;
|
|
|
|
}
|
|
|
|
|
1999-04-05 02:53:07 +00:00
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::CreateGeneratedFrameFor(nsIPresContext* aPresContext,
|
1999-09-07 03:09:33 +00:00
|
|
|
nsIDocument* aDocument,
|
1999-04-05 02:53:07 +00:00
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
const nsStyleContent* aStyleContent,
|
|
|
|
PRUint32 aContentIndex,
|
|
|
|
nsIFrame** aFrame)
|
|
|
|
{
|
|
|
|
*aFrame = nsnull; // initialize OUT parameter
|
|
|
|
|
|
|
|
// Get the content value
|
|
|
|
nsStyleContentType type;
|
|
|
|
nsAutoString contentString;
|
|
|
|
aStyleContent->GetContentAt(aContentIndex, type, contentString);
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
|
|
|
|
1999-04-05 02:53:07 +00:00
|
|
|
if (eStyleContentType_URL == type) {
|
|
|
|
nsIHTMLContent* imageContent;
|
|
|
|
|
|
|
|
// Create an HTML image content object, and set the SRC.
|
|
|
|
// XXX Check if it's an image type we can handle...
|
|
|
|
NS_NewHTMLImageElement(&imageContent, nsHTMLAtoms::img);
|
|
|
|
imageContent->SetHTMLAttribute(nsHTMLAtoms::src, contentString, PR_FALSE);
|
1999-09-07 03:09:33 +00:00
|
|
|
|
|
|
|
// Set aContent as the parent content and set the document object. This
|
|
|
|
// way event handling works
|
|
|
|
imageContent->SetParent(aContent);
|
|
|
|
imageContent->SetDocument(aDocument, PR_TRUE);
|
1999-04-05 02:53:07 +00:00
|
|
|
|
|
|
|
// Create an image frame and initialize it
|
|
|
|
nsIFrame* imageFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewImageFrame(shell, &imageFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
imageFrame->Init(aPresContext, imageContent, aParentFrame, aStyleContext, nsnull);
|
1999-04-05 02:53:07 +00:00
|
|
|
NS_RELEASE(imageContent);
|
|
|
|
|
|
|
|
// Return the image frame
|
|
|
|
*aFrame = imageFrame;
|
|
|
|
|
|
|
|
} else {
|
1999-08-19 14:29:55 +00:00
|
|
|
|
1999-04-05 02:53:07 +00:00
|
|
|
switch (type) {
|
|
|
|
case eStyleContentType_String:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eStyleContentType_Attr:
|
1999-08-19 14:29:55 +00:00
|
|
|
{
|
1999-06-10 05:24:00 +00:00
|
|
|
nsIAtom* attrName = nsnull;
|
|
|
|
PRInt32 attrNameSpace = kNameSpaceID_None;
|
1999-07-26 00:37:24 +00:00
|
|
|
PRInt32 barIndex = contentString.FindChar('|'); // CSS namespace delimiter
|
1999-06-10 05:24:00 +00:00
|
|
|
if (-1 != barIndex) {
|
|
|
|
nsAutoString nameSpaceVal;
|
|
|
|
contentString.Left(nameSpaceVal, barIndex);
|
|
|
|
PRInt32 error;
|
|
|
|
attrNameSpace = nameSpaceVal.ToInteger(&error, 10);
|
|
|
|
contentString.Cut(0, barIndex + 1);
|
|
|
|
if (contentString.Length()) {
|
|
|
|
attrName = NS_NewAtom(contentString);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
attrName = NS_NewAtom(contentString);
|
|
|
|
}
|
1999-08-19 14:29:55 +00:00
|
|
|
|
|
|
|
// Creates the content and frame and return if successful
|
|
|
|
nsresult rv = NS_ERROR_FAILURE;
|
|
|
|
if (nsnull != attrName) {
|
2000-02-08 13:17:50 +00:00
|
|
|
nsCOMPtr<nsIContent> content;
|
1999-08-19 14:29:55 +00:00
|
|
|
nsIFrame* textFrame = nsnull;
|
2000-02-08 13:17:50 +00:00
|
|
|
NS_NewAttributeContent(getter_AddRefs(content));
|
1999-08-19 14:29:55 +00:00
|
|
|
if (nsnull != content) {
|
|
|
|
nsCOMPtr<nsIAttributeContent> attrContent(do_QueryInterface(content));
|
|
|
|
if (attrContent) {
|
|
|
|
attrContent->Init(aContent, attrNameSpace, attrName);
|
|
|
|
}
|
|
|
|
|
1999-09-07 03:09:33 +00:00
|
|
|
// Set aContent as the parent content and set the document object. This
|
|
|
|
// way event handling works
|
|
|
|
content->SetParent(aContent);
|
|
|
|
content->SetDocument(aDocument, PR_TRUE);
|
|
|
|
|
1999-08-19 14:29:55 +00:00
|
|
|
// Create a text frame and initialize it
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewTextFrame(shell, &textFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
textFrame->Init(aPresContext, content, aParentFrame, aStyleContext, nsnull);
|
1999-08-19 14:29:55 +00:00
|
|
|
|
|
|
|
// Return the text frame
|
|
|
|
*aFrame = textFrame;
|
|
|
|
rv = NS_OK;
|
|
|
|
}
|
1999-06-10 05:24:00 +00:00
|
|
|
NS_RELEASE(attrName);
|
|
|
|
}
|
1999-08-19 14:29:55 +00:00
|
|
|
return rv;
|
1999-04-05 02:53:07 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eStyleContentType_Counter:
|
|
|
|
case eStyleContentType_Counters:
|
1999-04-27 22:14:54 +00:00
|
|
|
case eStyleContentType_URL:
|
1999-04-05 02:53:07 +00:00
|
|
|
return NS_ERROR_NOT_IMPLEMENTED; // XXX not supported yet...
|
|
|
|
|
|
|
|
case eStyleContentType_OpenQuote:
|
|
|
|
case eStyleContentType_CloseQuote:
|
|
|
|
{
|
|
|
|
PRUint32 quotesCount = aStyleContent->QuotesCount();
|
|
|
|
if (quotesCount > 0) {
|
|
|
|
nsAutoString openQuote, closeQuote;
|
|
|
|
|
|
|
|
// If the depth is greater than the number of pairs, the last pair
|
|
|
|
// is repeated
|
|
|
|
PRUint32 quoteDepth = 0; // XXX really track the nested quotes...
|
|
|
|
if (quoteDepth > quotesCount) {
|
|
|
|
quoteDepth = quotesCount - 1;
|
|
|
|
}
|
|
|
|
aStyleContent->GetQuotesAt(quoteDepth, openQuote, closeQuote);
|
|
|
|
if (eStyleContentType_OpenQuote == type) {
|
|
|
|
contentString = openQuote;
|
|
|
|
} else {
|
|
|
|
contentString = closeQuote;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
1999-04-20 00:28:58 +00:00
|
|
|
contentString = '\"';
|
1999-04-05 02:53:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eStyleContentType_NoOpenQuote:
|
|
|
|
case eStyleContentType_NoCloseQuote:
|
|
|
|
// XXX Adjust quote depth...
|
|
|
|
return NS_OK;
|
1999-08-19 14:29:55 +00:00
|
|
|
} // switch
|
1999-04-05 02:53:07 +00:00
|
|
|
|
1999-08-19 14:29:55 +00:00
|
|
|
|
1999-04-05 02:53:07 +00:00
|
|
|
// Create a text content node
|
1999-08-19 14:29:55 +00:00
|
|
|
nsIContent* textContent = nsnull;
|
1999-04-05 02:53:07 +00:00
|
|
|
nsIDOMCharacterData* domData;
|
1999-08-19 14:29:55 +00:00
|
|
|
nsIFrame* textFrame = nsnull;
|
1999-04-05 02:53:07 +00:00
|
|
|
|
|
|
|
NS_NewTextNode(&textContent);
|
1999-09-07 03:09:33 +00:00
|
|
|
if (textContent) {
|
|
|
|
// Set the text
|
1999-08-19 14:29:55 +00:00
|
|
|
textContent->QueryInterface(kIDOMCharacterDataIID, (void**)&domData);
|
|
|
|
domData->SetData(contentString);
|
|
|
|
NS_RELEASE(domData);
|
1999-04-05 02:53:07 +00:00
|
|
|
|
1999-11-11 21:43:15 +00:00
|
|
|
// Set aContent as the parent content and set the document object. This
|
|
|
|
// way event handling works
|
|
|
|
textContent->SetParent(aContent);
|
|
|
|
textContent->SetDocument(aDocument, PR_TRUE);
|
|
|
|
|
1999-08-19 14:29:55 +00:00
|
|
|
// Create a text frame and initialize it
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewTextFrame(shell, &textFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
textFrame->Init(aPresContext, textContent, aParentFrame, aStyleContext, nsnull);
|
1999-04-05 02:53:07 +00:00
|
|
|
|
1999-08-19 14:29:55 +00:00
|
|
|
NS_RELEASE(textContent);
|
|
|
|
}
|
1999-04-05 02:53:07 +00:00
|
|
|
|
|
|
|
// Return the text frame
|
|
|
|
*aFrame = textFrame;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-03-30 15:22:54 +00:00
|
|
|
PRBool
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::CreateGeneratedContentFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-07-05 18:46:09 +00:00
|
|
|
nsFrameConstructorState& aState,
|
1999-03-30 15:22:54 +00:00
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIAtom* aPseudoElement,
|
1999-07-05 18:46:09 +00:00
|
|
|
PRBool aForBlock,
|
1999-03-30 15:22:54 +00:00
|
|
|
nsIFrame** aResult)
|
|
|
|
{
|
|
|
|
*aResult = nsnull; // initialize OUT parameter
|
|
|
|
|
|
|
|
// Probe for the existence of the pseudo-element
|
2000-02-23 21:24:07 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> pseudoStyleContext;
|
1999-03-30 15:22:54 +00:00
|
|
|
aPresContext->ProbePseudoStyleContextFor(aContent, aPseudoElement, aStyleContext,
|
2000-02-23 21:24:07 +00:00
|
|
|
PR_FALSE, getter_AddRefs(pseudoStyleContext));
|
1999-03-30 15:22:54 +00:00
|
|
|
|
|
|
|
if (pseudoStyleContext) {
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
|
|
|
|
// See whether the generated content should be displayed.
|
|
|
|
display = (const nsStyleDisplay*)pseudoStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
1999-10-02 04:27:40 +00:00
|
|
|
if (NS_STYLE_DISPLAY_NONE == display->mDisplay) {
|
|
|
|
aState.mFrameManager->SetUndisplayedPseudoIn(pseudoStyleContext, aContent);
|
|
|
|
}
|
|
|
|
else { // has valid display type
|
1999-04-05 02:53:07 +00:00
|
|
|
// See if there was any content specified
|
1999-04-10 04:21:33 +00:00
|
|
|
const nsStyleContent* styleContent =
|
|
|
|
(const nsStyleContent*)pseudoStyleContext->GetStyleData(eStyleStruct_Content);
|
1999-04-02 18:50:45 +00:00
|
|
|
PRUint32 contentCount = styleContent->ContentCount();
|
|
|
|
|
1999-04-05 02:53:07 +00:00
|
|
|
if (contentCount > 0) {
|
|
|
|
PRUint8 displayValue = display->mDisplay;
|
|
|
|
|
|
|
|
// Make sure the 'display' property value is allowable
|
|
|
|
const nsStyleDisplay* subjectDisplay = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
if (subjectDisplay->IsBlockLevel()) {
|
|
|
|
// For block-level elements the only allowed 'display' values are:
|
|
|
|
// 'none', 'inline', 'block', and 'marker'
|
|
|
|
if ((NS_STYLE_DISPLAY_INLINE != displayValue) &&
|
|
|
|
(NS_STYLE_DISPLAY_BLOCK != displayValue) &&
|
|
|
|
(NS_STYLE_DISPLAY_MARKER != displayValue)) {
|
|
|
|
// Pseudo-element behaves as if the value were 'block'
|
|
|
|
displayValue = NS_STYLE_DISPLAY_BLOCK;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// For inline-level elements the only allowed 'display' values are
|
|
|
|
// 'none' and 'inline'
|
|
|
|
displayValue = NS_STYLE_DISPLAY_INLINE;
|
1999-04-02 18:50:45 +00:00
|
|
|
}
|
1999-03-30 15:22:54 +00:00
|
|
|
|
1999-04-05 02:53:07 +00:00
|
|
|
if (display->mDisplay != displayValue) {
|
|
|
|
// Reset the value
|
|
|
|
nsStyleDisplay* mutableDisplay = (nsStyleDisplay*)
|
|
|
|
pseudoStyleContext->GetMutableStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
mutableDisplay->mDisplay = displayValue;
|
|
|
|
}
|
|
|
|
|
1999-04-10 04:21:33 +00:00
|
|
|
// Also make sure the 'position' property is 'static'. :before and :after
|
|
|
|
// pseudo-elements can not be floated or positioned
|
|
|
|
const nsStylePosition * stylePosition =
|
|
|
|
(const nsStylePosition*)pseudoStyleContext->GetStyleData(eStyleStruct_Position);
|
|
|
|
if (NS_STYLE_POSITION_NORMAL != stylePosition->mPosition) {
|
|
|
|
// Reset the value
|
|
|
|
nsStylePosition* mutablePosition = (nsStylePosition*)
|
|
|
|
pseudoStyleContext->GetMutableStyleData(eStyleStruct_Position);
|
|
|
|
|
|
|
|
mutablePosition->mPosition = NS_STYLE_POSITION_NORMAL;
|
|
|
|
}
|
|
|
|
|
1999-04-05 02:53:07 +00:00
|
|
|
// Create a block box or an inline box depending on the value of
|
|
|
|
// the 'display' property
|
|
|
|
nsIFrame* containerFrame;
|
|
|
|
nsFrameItems childFrames;
|
2000-02-10 04:35:51 +00:00
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMElement>containerElement;
|
|
|
|
nsCOMPtr<nsIContent> containerContent;
|
|
|
|
nsCOMPtr<nsIDocument> document;
|
|
|
|
|
|
|
|
aContent->GetDocument(*getter_AddRefs(document));
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMDocument> domdoc(do_QueryInterface(document));
|
|
|
|
nsresult result;
|
|
|
|
result = domdoc->CreateElement("SPAN",getter_AddRefs(containerElement));//is the literal the correct way?
|
|
|
|
if (NS_SUCCEEDED(result) && containerElement)
|
|
|
|
{
|
|
|
|
containerContent = do_QueryInterface(containerElement);
|
2000-02-10 05:20:55 +00:00
|
|
|
}//do NOT use YET set this up for later checkin. this will be the content of the new frame.
|
2000-02-10 04:35:51 +00:00
|
|
|
|
1999-04-05 02:53:07 +00:00
|
|
|
if (NS_STYLE_DISPLAY_BLOCK == displayValue) {
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewBlockFrame(aPresShell, &containerFrame);
|
1999-04-05 02:53:07 +00:00
|
|
|
} else {
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewInlineFrame(aPresShell, &containerFrame);
|
1999-12-06 07:44:18 +00:00
|
|
|
}
|
2000-02-10 05:20:55 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
1999-12-06 07:44:18 +00:00
|
|
|
aFrame, pseudoStyleContext, nsnull, containerFrame);
|
1999-04-05 02:53:07 +00:00
|
|
|
|
1999-04-06 04:58:05 +00:00
|
|
|
// Mark the frame as being associated with generated content
|
|
|
|
nsFrameState frameState;
|
|
|
|
containerFrame->GetFrameState(&frameState);
|
|
|
|
frameState |= NS_FRAME_GENERATED_CONTENT;
|
|
|
|
containerFrame->SetFrameState(frameState);
|
|
|
|
|
1999-04-05 02:53:07 +00:00
|
|
|
// Create another pseudo style context to use for all the generated child
|
|
|
|
// frames
|
|
|
|
nsIStyleContext* textStyleContext;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::textPseudo,
|
|
|
|
pseudoStyleContext, PR_FALSE,
|
|
|
|
&textStyleContext);
|
|
|
|
|
|
|
|
// Now create content objects (and child frames) for each value of the
|
|
|
|
// 'content' property
|
1999-09-07 03:09:33 +00:00
|
|
|
|
1999-04-05 02:53:07 +00:00
|
|
|
for (PRUint32 contentIndex = 0; contentIndex < contentCount; contentIndex++) {
|
|
|
|
nsIFrame* frame;
|
|
|
|
nsresult result;
|
|
|
|
|
|
|
|
// Create a frame
|
1999-09-07 03:09:33 +00:00
|
|
|
result = CreateGeneratedFrameFor(aPresContext, document, containerFrame,
|
1999-07-05 18:46:09 +00:00
|
|
|
aContent, textStyleContext,
|
|
|
|
styleContent, contentIndex, &frame);
|
1999-04-05 02:53:07 +00:00
|
|
|
if (NS_SUCCEEDED(result) && frame) {
|
|
|
|
// Add it to the list of child frames
|
|
|
|
childFrames.AddChild(frame);
|
|
|
|
}
|
|
|
|
}
|
1999-03-30 15:22:54 +00:00
|
|
|
|
1999-04-05 02:53:07 +00:00
|
|
|
NS_RELEASE(textStyleContext);
|
|
|
|
if (childFrames.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
containerFrame->SetInitialChildList(aPresContext, nsnull, childFrames.childList);
|
1999-04-05 02:53:07 +00:00
|
|
|
}
|
|
|
|
*aResult = containerFrame;
|
1999-04-02 18:50:45 +00:00
|
|
|
return PR_TRUE;
|
|
|
|
}
|
1999-03-30 15:22:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-09-16 19:55:08 +00:00
|
|
|
#if 0
|
1999-08-27 21:46:10 +00:00
|
|
|
static PRBool
|
|
|
|
IsEmptyTextContent(nsIContent* aContent)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
1999-08-27 21:46:10 +00:00
|
|
|
PRBool result = PR_FALSE;
|
|
|
|
nsCOMPtr<nsITextContent> tc(do_QueryInterface(aContent));
|
|
|
|
if (tc) {
|
|
|
|
tc->IsOnlyWhitespace(&result);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
1999-03-30 15:22:54 +00:00
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
#include "nsIDOMHTMLParagraphElement.h"
|
1999-03-30 15:22:54 +00:00
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
static PRBool
|
|
|
|
IsHTMLParagraph(nsIContent* aContent)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIDOMHTMLParagraphElement> p(do_QueryInterface(aContent));
|
|
|
|
PRBool status = PR_FALSE;
|
|
|
|
if (p) {
|
|
|
|
status = PR_TRUE;
|
1999-03-30 15:22:54 +00:00
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
return status;
|
|
|
|
}
|
1999-03-30 15:22:54 +00:00
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
static PRBool
|
1999-08-31 03:09:40 +00:00
|
|
|
IsEmptyContainer(nsIContent* aContainer)
|
1999-08-27 21:46:10 +00:00
|
|
|
{
|
|
|
|
// Check each kid and if each kid is empty text content (or there
|
|
|
|
// are no kids) then return true.
|
|
|
|
PRInt32 i, numKids;
|
|
|
|
aContainer->ChildCount(numKids);
|
|
|
|
for (i = 0; i < numKids; i++) {
|
|
|
|
nsCOMPtr<nsIContent> kid;
|
|
|
|
aContainer->ChildAt(i, *getter_AddRefs(kid));
|
|
|
|
if (kid.get()) {
|
|
|
|
if (!IsEmptyTextContent(kid)) {
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
return PR_TRUE;
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
static PRBool
|
|
|
|
IsEmptyHTMLParagraph(nsIContent* aContent)
|
|
|
|
{
|
|
|
|
if (!IsHTMLParagraph(aContent)) {
|
|
|
|
// It's not an HTML paragraph so return false
|
|
|
|
return PR_FALSE;
|
1999-03-30 15:22:54 +00:00
|
|
|
}
|
1999-08-31 03:09:40 +00:00
|
|
|
return IsEmptyContainer(aContent);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-08-31 03:09:40 +00:00
|
|
|
#endif
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::CreateInputFrame(nsIPresShell *aPresShell,
|
|
|
|
nsIPresContext *aPresContext,
|
1999-08-03 14:41:48 +00:00
|
|
|
nsIContent *aContent,
|
|
|
|
nsIFrame *&aFrame,
|
2000-02-23 21:36:29 +00:00
|
|
|
nsIStyleContext *aStyleContext)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv;
|
|
|
|
|
|
|
|
// Figure out which type of input frame to create
|
|
|
|
nsAutoString val;
|
|
|
|
if (NS_OK == aContent->GetAttribute(kNameSpaceID_HTML, nsHTMLAtoms::type, val)) {
|
|
|
|
if (val.EqualsIgnoreCase("submit")) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructButtonControlFrame(aPresShell, aPresContext, aFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (val.EqualsIgnoreCase("reset")) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructButtonControlFrame(aPresShell, aPresContext, aFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (val.EqualsIgnoreCase("button")) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructButtonControlFrame(aPresShell, aPresContext, aFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (val.EqualsIgnoreCase("checkbox")) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructCheckboxControlFrame(aPresShell, aPresContext, aFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (val.EqualsIgnoreCase("file")) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewFileControlFrame(aPresShell, &aFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (val.EqualsIgnoreCase("hidden")) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructButtonControlFrame(aPresShell, aPresContext, aFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (val.EqualsIgnoreCase("image")) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewImageControlFrame(aPresShell, &aFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (val.EqualsIgnoreCase("password")) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTextControlFrame(aPresShell, aPresContext, aFrame, aContent);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (val.EqualsIgnoreCase("radio")) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructRadioControlFrame(aPresShell, aPresContext, aFrame, aContent, aStyleContext);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (val.EqualsIgnoreCase("text")) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTextControlFrame(aPresShell, aPresContext, aFrame, aContent);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTextControlFrame(aPresShell, aPresContext, aFrame, aContent);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
} else {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTextControlFrame(aPresShell, aPresContext, aFrame, aContent);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************
|
|
|
|
** BEGIN TABLE SECTION
|
|
|
|
****************************************************/
|
|
|
|
|
1999-02-26 09:15:03 +00:00
|
|
|
struct nsTableList {
|
|
|
|
nsTableList() {
|
|
|
|
mInner = mChild = nsnull;
|
|
|
|
}
|
|
|
|
void Set(nsIFrame* aInner, nsIFrame* aChild) {
|
|
|
|
mInner = aInner;
|
|
|
|
mChild = aChild;
|
|
|
|
}
|
|
|
|
nsIFrame* mInner;
|
|
|
|
nsIFrame* mChild;
|
1999-02-05 03:55:18 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Construct the outer, inner table frames and the children frames for the table.
|
|
|
|
// Tables can have any table-related child.
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructTableFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aNewFrame,
|
|
|
|
nsTableCreator& aTableCreator)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
nsIFrame* childList;
|
|
|
|
nsIFrame* innerFrame;
|
|
|
|
nsIFrame* innerChildList = nsnull;
|
|
|
|
nsIFrame* captionFrame = nsnull;
|
|
|
|
|
|
|
|
// Create an anonymous table outer frame which holds the caption and table frame
|
1999-08-24 00:44:21 +00:00
|
|
|
aTableCreator.CreateTableOuterFrame(&aNewFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Init the table outer frame and see if we need to create a view, e.g.
|
1999-12-06 07:44:18 +00:00
|
|
|
// the frame is absolutely positioned
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, aNewFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, aNewFrame,
|
1999-02-05 03:55:18 +00:00
|
|
|
aStyleContext, PR_FALSE);
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> parentStyleContext;
|
|
|
|
aParentFrame->GetStyleContext(getter_AddRefs(parentStyleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
#if 0
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> outerStyleContext;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableOuterPseudo,
|
|
|
|
parentStyleContext,
|
|
|
|
getter_AddRefs(outerStyleContext));
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewFrame->Init(aPresContext, aContent, aParentFrame, outerStyleContext);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, aNewFrame,
|
1999-02-05 03:55:18 +00:00
|
|
|
outerStyleContext, PR_FALSE);
|
|
|
|
#endif
|
|
|
|
// Create the inner table frame
|
1999-05-11 22:03:29 +00:00
|
|
|
aTableCreator.CreateTableFrame(&innerFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
childList = innerFrame;
|
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aNewFrame, aStyleContext, nsnull, innerFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
nsIFrame* lastChildFrame = nsnull;
|
|
|
|
PRInt32 count;
|
|
|
|
aContent->ChildCount(count);
|
|
|
|
|
|
|
|
for (PRInt32 i = 0; i < count; i++) { // iterate the child content
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIContent> childContent;
|
|
|
|
|
|
|
|
if (NS_SUCCEEDED(aContent->ChildAt(i, *getter_AddRefs(childContent)))) {
|
1999-02-26 09:15:03 +00:00
|
|
|
nsIFrame* childFrame = nsnull;
|
|
|
|
nsIFrame* ignore1;
|
|
|
|
nsIFrame* ignore2;
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> childStyleContext;
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Resolve the style context and get its display
|
1999-02-11 15:54:13 +00:00
|
|
|
aPresContext->ResolveStyleContextFor(childContent, aStyleContext,
|
1999-02-12 17:45:58 +00:00
|
|
|
PR_FALSE,
|
1999-02-11 15:54:13 +00:00
|
|
|
getter_AddRefs(childStyleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
const nsStyleDisplay* styleDisplay = (const nsStyleDisplay*)
|
|
|
|
childStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
switch (styleDisplay->mDisplay) {
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_CAPTION:
|
|
|
|
if (nsnull == captionFrame) { // only allow one caption
|
|
|
|
// XXX should absolute items be passed along?
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableCaptionFrame(aPresShell, aPresContext, aState, childContent, aNewFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
childStyleContext, ignore1, captionFrame, aTableCreator);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_HEADER_GROUP:
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_FOOTER_GROUP:
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_ROW_GROUP:
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_COLUMN_GROUP:
|
|
|
|
{
|
|
|
|
PRBool isRowGroup = (NS_STYLE_DISPLAY_TABLE_COLUMN_GROUP != styleDisplay->mDisplay);
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableGroupFrame(aPresShell, aPresContext, aState, childContent, innerFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
childStyleContext, isRowGroup, childFrame, ignore1,
|
|
|
|
aTableCreator);
|
1999-02-05 03:55:18 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_ROW:
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableRowFrame(aPresShell, aPresContext, aState, childContent, innerFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
childStyleContext, childFrame, ignore1, aTableCreator);
|
1999-02-05 03:55:18 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_COLUMN:
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableColFrame(aPresShell, aPresContext, aState, childContent, innerFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
childStyleContext, childFrame, ignore1, aTableCreator);
|
1999-02-05 03:55:18 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_CELL:
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableCellFrame(aPresShell, aPresContext, aState, childContent, innerFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
childStyleContext, childFrame, ignore1, ignore2, aTableCreator);
|
1999-02-05 03:55:18 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
//nsIFrame* nonTableRelatedFrame;
|
|
|
|
//nsIAtom* tag;
|
|
|
|
//childContent->GetTag(tag);
|
|
|
|
//ConstructFrameByTag(aPresContext, childContent, aNewFrame, tag, childStyleContext,
|
|
|
|
// aAbsoluteItems, nonTableRelatedFrame);
|
|
|
|
//childList->SetNextSibling(nonTableRelatedFrame);
|
|
|
|
//NS_IF_RELEASE(tag);
|
1999-02-12 01:35:37 +00:00
|
|
|
nsFrameItems childItems;
|
1999-12-04 23:49:50 +00:00
|
|
|
TableProcessChild(aPresShell, aPresContext, aState, childContent, innerFrame, parentStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
childItems, aTableCreator);
|
1999-02-12 01:35:37 +00:00
|
|
|
childFrame = childItems.childList; // XXX: Need to change this whole function to return a list of frames
|
|
|
|
// rather than a single frame. - DWH
|
1999-02-05 03:55:18 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// for every table related frame except captions, link into the child list
|
|
|
|
if (nsnull != childFrame) {
|
|
|
|
if (nsnull == lastChildFrame) {
|
|
|
|
innerChildList = childFrame;
|
|
|
|
} else {
|
|
|
|
lastChildFrame->SetNextSibling(childFrame);
|
|
|
|
}
|
|
|
|
lastChildFrame = childFrame;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-01-28 02:19:45 +00:00
|
|
|
// Set the inner table frame's initial primary list
|
1999-11-24 06:03:41 +00:00
|
|
|
innerFrame->SetInitialChildList(aPresContext, nsnull, innerChildList);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
2000-01-28 02:19:45 +00:00
|
|
|
// Set the outer table frame's primary and option lists
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewFrame->SetInitialChildList(aPresContext, nsnull, childList);
|
2000-01-28 02:19:45 +00:00
|
|
|
if (captionFrame) {
|
|
|
|
aNewFrame->SetInitialChildList(aPresContext, nsLayoutAtoms::captionList, captionFrame);
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructAnonymousTableFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIFrame*& aNewTopFrame,
|
|
|
|
nsIFrame*& aOuterFrame,
|
|
|
|
nsIFrame*& aInnerFrame,
|
|
|
|
nsTableCreator& aTableCreator)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
1999-02-26 09:15:03 +00:00
|
|
|
nsresult rv = NS_OK;
|
1999-12-13 22:56:31 +00:00
|
|
|
//NS_WARNING("an anonymous table frame was created. \n");
|
1999-02-26 09:15:03 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> parentStyleContext;
|
|
|
|
aParentFrame->GetStyleContext(getter_AddRefs(parentStyleContext));
|
|
|
|
const nsStyleDisplay* parentDisplay =
|
|
|
|
(const nsStyleDisplay*) parentStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
PRBool cellIsParent = PR_TRUE;
|
|
|
|
nsIFrame* cellFrame;
|
|
|
|
nsIFrame* cellBodyFrame;
|
|
|
|
nsCOMPtr<nsIStyleContext> cellStyleContext;
|
|
|
|
nsIFrame* ignore;
|
|
|
|
|
|
|
|
// construct the ancestors of anonymous table frames
|
|
|
|
switch(parentDisplay->mDisplay) {
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_CELL:
|
|
|
|
break;
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_ROW: // construct a cell
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableCellPseudo, parentStyleContext,
|
|
|
|
PR_FALSE, getter_AddRefs(cellStyleContext));
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableCellFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, cellStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
aNewTopFrame, cellFrame, cellBodyFrame, aTableCreator, PR_FALSE);
|
1999-02-26 09:15:03 +00:00
|
|
|
break;
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_ROW_GROUP: // construct a row & a cell
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_HEADER_GROUP:
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_FOOTER_GROUP:
|
|
|
|
{
|
|
|
|
nsIFrame* rowFrame;
|
|
|
|
nsCOMPtr<nsIStyleContext> rowStyleContext;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableRowPseudo,
|
|
|
|
parentStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(rowStyleContext));
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableRowFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, rowStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
aNewTopFrame, rowFrame, aTableCreator);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableCellPseudo, parentStyleContext,
|
|
|
|
PR_FALSE, getter_AddRefs(cellStyleContext));
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableCellFrame(aPresShell, aPresContext, aState, aContent, rowFrame, cellStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
ignore, cellFrame, cellBodyFrame, aTableCreator, PR_FALSE);
|
1999-11-24 06:03:41 +00:00
|
|
|
rowFrame->SetInitialChildList(aPresContext, nsnull, cellFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
break;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-26 09:15:03 +00:00
|
|
|
case NS_STYLE_DISPLAY_TABLE: // construct a row group, a row, & a cell
|
|
|
|
{
|
|
|
|
nsIFrame* groupFrame;
|
|
|
|
nsCOMPtr<nsIStyleContext> groupStyleContext;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableRowGroupPseudo,
|
|
|
|
parentStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(groupStyleContext));
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableGroupFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, groupStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
PR_TRUE, aNewTopFrame, groupFrame, aTableCreator);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
nsIFrame* rowFrame;
|
|
|
|
nsCOMPtr<nsIStyleContext> rowStyleContext;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableRowPseudo,
|
|
|
|
parentStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(rowStyleContext));
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableRowFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, rowStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
ignore, rowFrame, aTableCreator);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableCellPseudo, parentStyleContext,
|
|
|
|
PR_FALSE, getter_AddRefs(cellStyleContext));
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableCellFrame(aPresShell, aPresContext, aState, aContent, rowFrame, cellStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
ignore, cellFrame, cellBodyFrame, aTableCreator, PR_FALSE);
|
1999-11-24 06:03:41 +00:00
|
|
|
rowFrame->SetInitialChildList(aPresContext, nsnull, cellFrame);
|
|
|
|
groupFrame->SetInitialChildList(aPresContext, nsnull, rowFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
cellIsParent = PR_FALSE;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
1999-02-26 09:15:03 +00:00
|
|
|
// create the outer table, inner table frames and make the cell frame the parent of the
|
|
|
|
// outer frame
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
// create the outer table frame
|
|
|
|
nsIFrame* tableParent = nsnull;
|
|
|
|
nsIStyleContext* tableParentSC;
|
|
|
|
if (cellIsParent) {
|
|
|
|
tableParent = cellFrame;
|
|
|
|
tableParentSC = cellStyleContext;
|
|
|
|
} else {
|
|
|
|
tableParent = aParentFrame;
|
|
|
|
tableParentSC = parentStyleContext;
|
|
|
|
}
|
|
|
|
nsCOMPtr<nsIStyleContext> outerStyleContext;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableOuterPseudo,
|
|
|
|
tableParentSC, PR_FALSE,
|
|
|
|
getter_AddRefs(outerStyleContext));
|
1999-08-24 00:44:21 +00:00
|
|
|
rv = aTableCreator.CreateTableOuterFrame(&aOuterFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
tableParent, outerStyleContext, nsnull, aOuterFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
|
|
|
|
// create the inner table frame
|
|
|
|
nsCOMPtr<nsIStyleContext> innerStyleContext;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tablePseudo,
|
|
|
|
outerStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(innerStyleContext));
|
1999-05-11 22:03:29 +00:00
|
|
|
rv = aTableCreator.CreateTableFrame(&aInnerFrame);
|
1999-12-06 07:44:18 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aOuterFrame, innerStyleContext, nsnull, aInnerFrame);
|
|
|
|
|
2000-01-28 02:19:45 +00:00
|
|
|
//XXX when bug 2479 is fixed this may be a duplicate call
|
1999-11-24 06:03:41 +00:00
|
|
|
aOuterFrame->SetInitialChildList(aPresContext, nsnull, aInnerFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
|
|
|
|
if (cellIsParent) {
|
1999-11-24 06:03:41 +00:00
|
|
|
cellBodyFrame->SetInitialChildList(aPresContext, nsnull, aOuterFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
} else {
|
|
|
|
aNewTopFrame = aOuterFrame;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructTableCaptionFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aNewTopFrame,
|
|
|
|
nsIFrame*& aNewCaptionFrame,
|
|
|
|
nsTableCreator& aTableCreator)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
1999-12-04 23:49:50 +00:00
|
|
|
nsresult rv = NS_NewTableCaptionFrame(aPresShell, &aNewCaptionFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-26 09:15:03 +00:00
|
|
|
const nsStyleDisplay* parentDisplay = GetDisplay(aParentFrame);
|
|
|
|
|
2000-01-28 02:19:45 +00:00
|
|
|
nsIFrame* outerFrame = aParentFrame;
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_STYLE_DISPLAY_TABLE == parentDisplay->mDisplay) { // parent is an outer table
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
2000-01-28 02:19:45 +00:00
|
|
|
aParentFrame, aStyleContext, nsnull, aNewCaptionFrame);
|
1999-12-06 07:44:18 +00:00
|
|
|
|
2000-01-28 02:19:45 +00:00
|
|
|
// the caller is responsible for calling SetInitialChildList
|
1999-02-26 09:15:03 +00:00
|
|
|
aNewTopFrame = aNewCaptionFrame;
|
|
|
|
} else { // parent is not a table, need to create a new table
|
1999-12-13 22:56:31 +00:00
|
|
|
//NS_WARNING("a non table contains a table caption child. \n");
|
2000-01-28 02:19:45 +00:00
|
|
|
nsIFrame* innerFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
ConstructAnonymousTableFrame(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
aNewTopFrame, outerFrame, innerFrame, aTableCreator);
|
1999-02-26 09:15:03 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> outerStyleContext;
|
|
|
|
outerFrame->GetStyleContext(getter_AddRefs(outerStyleContext));
|
|
|
|
nsCOMPtr<nsIStyleContext> adjStyleContext;
|
|
|
|
aPresContext->ResolveStyleContextFor(aContent, outerStyleContext,
|
|
|
|
PR_FALSE, getter_AddRefs(adjStyleContext));
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
outerFrame, adjStyleContext, nsnull, aNewCaptionFrame);
|
2000-02-04 03:16:47 +00:00
|
|
|
aState.mFrameManager->SetPrimaryFrameFor(aContent, aNewCaptionFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-26 17:11:54 +00:00
|
|
|
|
1999-07-16 23:04:27 +00:00
|
|
|
// The caption frame is a floater container
|
1999-08-31 03:09:40 +00:00
|
|
|
PRBool haveFirstLetterStyle, haveFirstLineStyle;
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aContent, aStyleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
1999-07-16 23:04:27 +00:00
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
1999-08-27 21:46:10 +00:00
|
|
|
aState.PushFloaterContainingBlock(aNewCaptionFrame, floaterSaveState,
|
1999-08-31 03:09:40 +00:00
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
1999-07-16 23:04:27 +00:00
|
|
|
|
|
|
|
// Process the child content
|
1999-02-26 09:15:03 +00:00
|
|
|
nsFrameItems childItems;
|
1999-12-04 23:49:50 +00:00
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aContent, aNewCaptionFrame,
|
1999-08-27 21:46:10 +00:00
|
|
|
PR_TRUE, childItems, PR_TRUE);
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewCaptionFrame->SetInitialChildList(aPresContext, nsnull,
|
1999-08-27 21:46:10 +00:00
|
|
|
childItems.childList);
|
1999-07-16 23:04:27 +00:00
|
|
|
if (aState.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewCaptionFrame->SetInitialChildList(aPresContext,
|
1999-02-26 17:11:54 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
1999-07-16 23:04:27 +00:00
|
|
|
aState.mFloatedItems.childList);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if aParentFrame is a table, it is assummed that it is an inner table
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructTableGroupFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
PRBool aIsRowGroup,
|
|
|
|
nsIFrame*& aNewTopFrame,
|
|
|
|
nsIFrame*& aNewGroupFrame,
|
|
|
|
nsTableCreator& aTableCreator,
|
|
|
|
nsTableList* aToDo)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
const nsStyleDisplay* styleDisplay = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
1999-02-26 09:15:03 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> styleContext( dont_QueryInterface(aStyleContext) );
|
|
|
|
|
|
|
|
// TRUE if we are being called from above, FALSE if from below (e.g. row)
|
1999-02-05 03:55:18 +00:00
|
|
|
PRBool contentDisplayIsGroup =
|
|
|
|
(aIsRowGroup) ? (NS_STYLE_DISPLAY_TABLE_ROW_GROUP == styleDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_TABLE_HEADER_GROUP == styleDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_TABLE_FOOTER_GROUP == styleDisplay->mDisplay)
|
|
|
|
: (NS_STYLE_DISPLAY_TABLE_COLUMN_GROUP == styleDisplay->mDisplay);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (!contentDisplayIsGroup) {
|
|
|
|
NS_ASSERTION(aToDo, "null nsTableList when constructing from below");
|
1999-12-14 22:30:41 +00:00
|
|
|
if (!aToDo)
|
|
|
|
return NS_ERROR_NULL_POINTER;
|
1999-02-26 09:15:03 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> parentStyleContext;
|
|
|
|
aParentFrame->GetStyleContext(getter_AddRefs(parentStyleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
const nsStyleDisplay* parentDisplay =
|
|
|
|
(const nsStyleDisplay*) parentStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
if (NS_STYLE_DISPLAY_TABLE == parentDisplay->mDisplay) { // parent is an inner table
|
1999-02-26 09:15:03 +00:00
|
|
|
if (!contentDisplayIsGroup) { // called from row below
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIAtom* pseudoGroup = (aIsRowGroup) ? nsHTMLAtoms::tableRowGroupPseudo : nsHTMLAtoms::tableColGroupPseudo;
|
1999-02-11 15:54:13 +00:00
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, pseudoGroup, parentStyleContext,
|
1999-02-26 09:15:03 +00:00
|
|
|
PR_FALSE, getter_AddRefs(styleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-26 09:15:03 +00:00
|
|
|
// only process the group's children if we're called from above
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableGroupFrameOnly(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
styleContext, aIsRowGroup, aNewTopFrame, aNewGroupFrame,
|
|
|
|
aTableCreator, contentDisplayIsGroup);
|
1999-06-10 09:31:30 +00:00
|
|
|
} else if (aTableCreator.IsTreeCreator() &&
|
|
|
|
parentDisplay->mDisplay == NS_STYLE_DISPLAY_TABLE_ROW_GROUP) {
|
|
|
|
// We're a tree view. We want to allow nested row groups.
|
|
|
|
// only process the group's children if we're called from above
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableGroupFrameOnly(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-06-10 09:31:30 +00:00
|
|
|
styleContext, aIsRowGroup, aNewTopFrame, aNewGroupFrame,
|
|
|
|
aTableCreator, contentDisplayIsGroup);
|
1999-02-26 09:15:03 +00:00
|
|
|
} else { // construct anonymous frames
|
1999-12-13 22:56:31 +00:00
|
|
|
//NS_WARNING("a non table contains a table row or col group child. \n");
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* innerFrame;
|
|
|
|
nsIFrame* outerFrame;
|
1999-02-26 09:15:03 +00:00
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
ConstructAnonymousTableFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, aNewTopFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
outerFrame, innerFrame, aTableCreator);
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> innerStyleContext;
|
|
|
|
innerFrame->GetStyleContext(getter_AddRefs(innerStyleContext));
|
1999-02-26 09:15:03 +00:00
|
|
|
if (contentDisplayIsGroup) { // called from above
|
1999-02-12 17:45:58 +00:00
|
|
|
aPresContext->ResolveStyleContextFor(aContent, innerStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(styleContext));
|
1999-02-26 09:15:03 +00:00
|
|
|
} else { // called from row below
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIAtom* pseudoGroup = (aIsRowGroup) ? nsHTMLAtoms::tableRowGroupPseudo : nsHTMLAtoms::tableColGroupPseudo;
|
1999-02-11 15:54:13 +00:00
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, pseudoGroup, innerStyleContext,
|
1999-02-26 09:15:03 +00:00
|
|
|
PR_FALSE, getter_AddRefs(styleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-26 09:15:03 +00:00
|
|
|
nsIFrame* topFrame;
|
|
|
|
// only process the group's children if we're called from above
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableGroupFrameOnly(aPresShell, aPresContext, aState, aContent, innerFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
styleContext, aIsRowGroup, topFrame, aNewGroupFrame,
|
|
|
|
aTableCreator, contentDisplayIsGroup);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
if (contentDisplayIsGroup) { // called from above
|
1999-11-24 06:03:41 +00:00
|
|
|
innerFrame->SetInitialChildList(aPresContext, nsnull, topFrame);
|
1999-10-17 03:15:13 +00:00
|
|
|
// set the primary frame to avoid getting the anonymous table frame
|
|
|
|
aState.mFrameManager->SetPrimaryFrameFor(aContent, aNewGroupFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
} else { // called from row below
|
|
|
|
aToDo->Set(innerFrame, topFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructTableGroupFrameOnly(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
PRBool aIsRowGroup,
|
|
|
|
nsIFrame*& aNewTopFrame,
|
|
|
|
nsIFrame*& aNewGroupFrame,
|
|
|
|
nsTableCreator& aTableCreator,
|
|
|
|
PRBool aProcessChildren)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
const nsStyleDisplay* styleDisplay =
|
|
|
|
(const nsStyleDisplay*) aStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
rv = (aIsRowGroup) ? aTableCreator.CreateTableRowGroupFrame(&aNewGroupFrame)
|
|
|
|
: aTableCreator.CreateTableColGroupFrame(&aNewGroupFrame);
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
if (IsScrollable(aPresContext, styleDisplay)) {
|
|
|
|
|
1999-02-26 09:15:03 +00:00
|
|
|
// Create an area container for the frame
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
BuildScrollFrame(aPresShell, aPresContext, aState, aContent, aStyleContext, aNewGroupFrame, aParentFrame,
|
1999-10-12 00:16:06 +00:00
|
|
|
aNewTopFrame, aStyleContext);
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
} else {
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, aNewGroupFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
aNewTopFrame = aNewGroupFrame;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
1999-02-26 09:15:03 +00:00
|
|
|
if (aProcessChildren) {
|
|
|
|
nsFrameItems childItems;
|
1999-08-27 06:06:39 +00:00
|
|
|
|
1999-02-26 09:15:03 +00:00
|
|
|
if (aIsRowGroup) {
|
1999-06-23 03:02:21 +00:00
|
|
|
|
|
|
|
// Create some anonymous extras within the tree body.
|
|
|
|
if (aTableCreator.IsTreeCreator()) {
|
1999-07-01 05:16:56 +00:00
|
|
|
|
|
|
|
const nsStyleDisplay *parentDisplay;
|
1999-06-23 21:04:14 +00:00
|
|
|
aParentFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct *&)parentDisplay);
|
|
|
|
if (parentDisplay->mDisplay == NS_STYLE_DISPLAY_TABLE_ROW_GROUP) {
|
1999-06-23 03:02:21 +00:00
|
|
|
// We're the child of another row group. If it's lazy, we're lazy.
|
2000-01-13 19:20:25 +00:00
|
|
|
((nsTreeRowGroupFrame*)aNewGroupFrame)->SetFrameConstructor(this);
|
1999-06-23 03:02:21 +00:00
|
|
|
}
|
1999-07-01 05:16:56 +00:00
|
|
|
else if (parentDisplay->mDisplay == NS_STYLE_DISPLAY_TABLE) {
|
|
|
|
// We're the child of a table.
|
|
|
|
// See if our parent is a tree.
|
|
|
|
// We will want to have a scrollbar.
|
|
|
|
((nsTreeRowGroupFrame*)aNewGroupFrame)->SetFrameConstructor(this);
|
|
|
|
}
|
1999-06-23 03:02:21 +00:00
|
|
|
}
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
TableProcessChildren(aPresShell, aPresContext, aState, aContent, aNewGroupFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
childItems, aTableCreator);
|
1999-02-26 09:15:03 +00:00
|
|
|
} else {
|
1999-12-04 23:49:50 +00:00
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aContent, aNewGroupFrame,
|
1999-08-27 21:46:10 +00:00
|
|
|
PR_FALSE, childItems, PR_FALSE);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewGroupFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-08-27 06:06:39 +00:00
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructTableRowFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aNewTopFrame,
|
|
|
|
nsIFrame*& aNewRowFrame,
|
|
|
|
nsTableCreator& aTableCreator,
|
|
|
|
nsTableList* aToDo)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
1999-02-26 09:15:03 +00:00
|
|
|
|
|
|
|
// TRUE if we are being called from above, FALSE if from below (e.g. cell)
|
1999-02-05 03:55:18 +00:00
|
|
|
PRBool contentDisplayIsRow = (NS_STYLE_DISPLAY_TABLE_ROW == display->mDisplay);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (!contentDisplayIsRow) {
|
|
|
|
NS_ASSERTION(aToDo, "null nsTableList when constructing from below");
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> groupStyleContext;
|
1999-02-18 23:55:10 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> styleContext( dont_QueryInterface(aStyleContext) );
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
const nsStyleDisplay* parentDisplay = GetDisplay(aParentFrame);
|
|
|
|
if ((NS_STYLE_DISPLAY_TABLE_ROW_GROUP == parentDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_TABLE_HEADER_GROUP == parentDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_TABLE_FOOTER_GROUP == parentDisplay->mDisplay)) {
|
|
|
|
if (!contentDisplayIsRow) { // content is from some (soon to be) child of ours
|
1999-10-12 00:16:06 +00:00
|
|
|
// EDV this will not work.
|
|
|
|
// 1) This crashes because the aParent is already what you think it is, event if it
|
|
|
|
// is scrollable.
|
|
|
|
// 2) If the we had gfx scrollbars turned on the first child would have been
|
|
|
|
// a scrollport!!
|
|
|
|
//aParentFrame = TableGetAsNonScrollFrame(aPresContext, aParentFrame, parentDisplay);
|
1999-02-11 15:54:13 +00:00
|
|
|
aParentFrame->GetStyleContext(getter_AddRefs(groupStyleContext));
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableRowPseudo,
|
1999-02-26 09:15:03 +00:00
|
|
|
groupStyleContext, PR_FALSE,
|
1999-02-11 15:54:13 +00:00
|
|
|
getter_AddRefs(styleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-26 09:15:03 +00:00
|
|
|
// only process the row's children if we're called from above
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableRowFrameOnly(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
styleContext, contentDisplayIsRow, aNewRowFrame,
|
|
|
|
aTableCreator);
|
1999-02-26 09:15:03 +00:00
|
|
|
aNewTopFrame = aNewRowFrame;
|
1999-02-05 03:55:18 +00:00
|
|
|
} else { // construct an anonymous row group frame
|
|
|
|
nsIFrame* groupFrame;
|
1999-02-26 09:15:03 +00:00
|
|
|
nsTableList localToDo;
|
|
|
|
nsTableList* toDo = (aToDo) ? aToDo : &localToDo;
|
|
|
|
|
|
|
|
// it may also need to create a table frame
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableGroupFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, styleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
PR_TRUE, aNewTopFrame, groupFrame, aTableCreator, toDo);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
groupFrame->GetStyleContext(getter_AddRefs(groupStyleContext));
|
|
|
|
if (contentDisplayIsRow) { // called from above
|
|
|
|
aPresContext->ResolveStyleContextFor(aContent, groupStyleContext,
|
|
|
|
PR_FALSE, getter_AddRefs(styleContext));
|
|
|
|
} else { // called from cell below
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableRowPseudo,
|
|
|
|
groupStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(styleContext));
|
|
|
|
}
|
|
|
|
// only process the row's children if we're called from above
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableRowFrameOnly(aPresShell, aPresContext, aState, aContent, groupFrame, styleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
contentDisplayIsRow, aNewRowFrame, aTableCreator);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
1999-10-17 03:15:13 +00:00
|
|
|
|
2000-01-28 02:19:45 +00:00
|
|
|
//XXX when bug 2479 is fixed this may be a duplicate call
|
1999-11-24 06:03:41 +00:00
|
|
|
groupFrame->SetInitialChildList(aPresContext, nsnull, aNewRowFrame);
|
|
|
|
if (contentDisplayIsRow) { // called from above
|
|
|
|
// set the primary frame to avoid getting the anonymous row group frame
|
|
|
|
aState.mFrameManager->SetPrimaryFrameFor(aContent, aNewRowFrame);
|
|
|
|
}
|
|
|
|
|
1999-02-26 09:15:03 +00:00
|
|
|
if (contentDisplayIsRow) { // called from above
|
|
|
|
TableProcessTableList(aPresContext, *toDo);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
1999-11-23 02:06:39 +00:00
|
|
|
|
|
|
|
if (aState.mFrameManager)
|
|
|
|
aState.mFrameManager->SetPrimaryFrameFor(aContent, aNewRowFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructTableRowFrameOnly(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
PRBool aProcessChildren,
|
|
|
|
nsIFrame*& aNewRowFrame,
|
|
|
|
nsTableCreator& aTableCreator)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
1999-05-11 22:03:29 +00:00
|
|
|
nsresult rv = aTableCreator.CreateTableRowFrame(&aNewRowFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, aNewRowFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (aProcessChildren) {
|
|
|
|
nsFrameItems childItems;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = TableProcessChildren(aPresShell, aPresContext, aState, aContent, aNewRowFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
childItems, aTableCreator);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewRowFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-26 09:15:03 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-12-13 22:56:31 +00:00
|
|
|
nsCSSFrameConstructor::ConstructTableColFrame(nsIPresShell* aPresShell,
|
1999-12-04 23:49:50 +00:00
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aNewTopFrame,
|
|
|
|
nsIFrame*& aNewColFrame,
|
|
|
|
nsTableCreator& aTableCreator)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
1999-02-27 19:02:48 +00:00
|
|
|
// the content display here is always table-col and were always called from above
|
1999-02-05 03:55:18 +00:00
|
|
|
const nsStyleDisplay* parentDisplay = GetDisplay(aParentFrame);
|
|
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN_GROUP == parentDisplay->mDisplay) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableColFrameOnly(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
aStyleContext, aNewColFrame, aTableCreator);
|
1999-02-26 09:15:03 +00:00
|
|
|
aNewTopFrame = aNewColFrame;
|
1999-02-05 03:55:18 +00:00
|
|
|
} else { // construct anonymous col group frame
|
1999-02-27 19:02:48 +00:00
|
|
|
nsTableList toDo;
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* groupFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableGroupFrame(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
aStyleContext, PR_FALSE, aNewTopFrame, groupFrame,
|
|
|
|
aTableCreator, &toDo);
|
1999-02-27 19:02:48 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
nsCOMPtr<nsIStyleContext> groupStyleContext;
|
|
|
|
groupFrame->GetStyleContext(getter_AddRefs(groupStyleContext));
|
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
aPresContext->ResolveStyleContextFor(aContent, groupStyleContext,
|
|
|
|
PR_FALSE, getter_AddRefs(styleContext));
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableColFrameOnly(aPresShell, aPresContext, aState, aContent, groupFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
styleContext, aNewColFrame, aTableCreator);
|
1999-02-27 19:02:48 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
1999-12-13 22:56:31 +00:00
|
|
|
aState.mFrameManager->SetPrimaryFrameFor(aContent, aNewColFrame);
|
2000-01-28 02:19:45 +00:00
|
|
|
//XXX when bug 2479 is fixed this may be a duplicate call
|
1999-11-24 06:03:41 +00:00
|
|
|
groupFrame->SetInitialChildList(aPresContext, nsnull, aNewColFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-27 19:02:48 +00:00
|
|
|
// if an anoymous table got created, then set its initial child list
|
|
|
|
TableProcessTableList(aPresContext, toDo);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-02-27 19:02:48 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructTableColFrameOnly(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aNewColFrame,
|
|
|
|
nsTableCreator& aTableCreator)
|
1999-02-27 19:02:48 +00:00
|
|
|
{
|
1999-05-11 22:03:29 +00:00
|
|
|
nsresult rv = aTableCreator.CreateTableColFrame(&aNewColFrame);
|
1999-02-27 19:02:48 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, aNewColFrame);
|
1999-02-27 19:02:48 +00:00
|
|
|
nsFrameItems colChildItems;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ProcessChildren(aPresShell, aPresContext, aState, aContent, aNewColFrame,
|
1999-08-27 21:46:10 +00:00
|
|
|
PR_FALSE, colChildItems, PR_FALSE);
|
1999-02-27 19:02:48 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewColFrame->SetInitialChildList(aPresContext, nsnull,
|
1999-02-27 19:02:48 +00:00
|
|
|
colChildItems.childList);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructTableCellFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aNewTopFrame,
|
|
|
|
nsIFrame*& aNewCellFrame,
|
|
|
|
nsIFrame*& aNewCellBodyFrame,
|
|
|
|
nsTableCreator& aTableCreator,
|
|
|
|
PRBool aProcessChildren)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
1999-02-26 09:15:03 +00:00
|
|
|
// FALSE if we are being called to wrap a cell around the content
|
1999-02-05 03:55:18 +00:00
|
|
|
PRBool contentDisplayIsCell = (NS_STYLE_DISPLAY_TABLE_CELL == display->mDisplay);
|
|
|
|
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> parentStyleContext;
|
|
|
|
aParentFrame->GetStyleContext(getter_AddRefs(parentStyleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
const nsStyleDisplay* parentDisplay = (const nsStyleDisplay*)
|
|
|
|
parentStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
1999-02-18 23:55:10 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> styleContext( dont_QueryInterface(aStyleContext) );
|
1999-02-05 03:55:18 +00:00
|
|
|
PRBool wrapContent = PR_FALSE;
|
|
|
|
|
|
|
|
if (NS_STYLE_DISPLAY_TABLE_ROW == parentDisplay->mDisplay) {
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> styleContextRelease;
|
1999-02-26 09:15:03 +00:00
|
|
|
if (!contentDisplayIsCell) { // need to wrap
|
1999-02-11 15:54:13 +00:00
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableCellPseudo,
|
|
|
|
parentStyleContext,
|
1999-02-12 17:45:58 +00:00
|
|
|
PR_FALSE,
|
1999-02-11 15:54:13 +00:00
|
|
|
getter_AddRefs(styleContext));
|
1999-02-26 09:15:03 +00:00
|
|
|
wrapContent = PR_TRUE;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableCellFrameOnly(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
styleContext, aNewCellFrame, aNewCellBodyFrame,
|
|
|
|
aTableCreator, aProcessChildren);
|
1999-02-26 09:15:03 +00:00
|
|
|
aNewTopFrame = aNewCellFrame;
|
|
|
|
} else { // the cell needs some ancestors to be fabricated
|
1999-12-13 22:56:31 +00:00
|
|
|
//NS_WARNING("WARNING - a non table row contains a table cell child. \n");
|
1999-02-26 09:15:03 +00:00
|
|
|
nsTableList toDo;
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* rowFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableRowFrame(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
aStyleContext, aNewTopFrame, rowFrame, aTableCreator,
|
|
|
|
&toDo);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
nsCOMPtr<nsIStyleContext> rowStyleContext;
|
|
|
|
rowFrame->GetStyleContext(getter_AddRefs(rowStyleContext));
|
|
|
|
if (contentDisplayIsCell) {
|
|
|
|
aPresContext->ResolveStyleContextFor(aContent, rowStyleContext,
|
|
|
|
PR_FALSE, getter_AddRefs(styleContext));
|
|
|
|
} else {
|
|
|
|
wrapContent = PR_TRUE; // XXX
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::tableCellPseudo,
|
|
|
|
rowStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(styleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableCellFrameOnly(aPresShell, aPresContext, aState, aContent, rowFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
styleContext, aNewCellFrame, aNewCellBodyFrame,
|
|
|
|
aTableCreator, aProcessChildren);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
2000-01-28 02:19:45 +00:00
|
|
|
//XXX when bug 2479 is fixed this may be a duplicate call
|
1999-11-24 06:03:41 +00:00
|
|
|
rowFrame->SetInitialChildList(aPresContext, nsnull, aNewCellFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
TableProcessTableList(aPresContext, toDo);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructTableCellFrameOnly(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aNewCellFrame,
|
|
|
|
nsIFrame*& aNewCellBodyFrame,
|
|
|
|
nsTableCreator& aTableCreator,
|
|
|
|
PRBool aProcessChildren)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv;
|
|
|
|
|
|
|
|
// Create a table cell frame
|
1999-05-11 22:03:29 +00:00
|
|
|
rv = aTableCreator.CreateTableCellFrame(&aNewCellFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
|
|
|
|
// Initialize the table cell frame
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, aNewCellFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
// Create an area frame that will format the cell's content
|
1999-09-21 01:15:30 +00:00
|
|
|
rv = aTableCreator.CreateTableCellInnerFrame(&aNewCellBodyFrame);
|
1999-12-01 10:37:20 +00:00
|
|
|
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewCellFrame->Destroy(aPresContext);
|
1999-02-26 09:15:03 +00:00
|
|
|
aNewCellFrame = nsnull;
|
|
|
|
return rv;
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-26 09:15:03 +00:00
|
|
|
// Resolve pseudo style and initialize the body cell frame
|
|
|
|
nsCOMPtr<nsIStyleContext> bodyPseudoStyle;
|
1999-08-27 21:46:10 +00:00
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent,
|
|
|
|
nsHTMLAtoms::cellContentPseudo,
|
1999-02-26 09:15:03 +00:00
|
|
|
aStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(bodyPseudoStyle));
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aNewCellFrame, bodyPseudoStyle, nsnull, aNewCellBodyFrame);
|
1999-02-26 09:15:03 +00:00
|
|
|
|
|
|
|
if (aProcessChildren) {
|
1999-08-31 03:09:40 +00:00
|
|
|
PRBool haveFirstLetterStyle, haveFirstLineStyle;
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aContent, aStyleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
1999-08-27 21:46:10 +00:00
|
|
|
|
1999-07-16 23:04:27 +00:00
|
|
|
// The area frame is a floater container
|
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
1999-08-27 21:46:10 +00:00
|
|
|
aState.PushFloaterContainingBlock(aNewCellBodyFrame, floaterSaveState,
|
1999-08-31 03:09:40 +00:00
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
1999-07-16 23:04:27 +00:00
|
|
|
|
|
|
|
// Process the child content
|
1999-02-05 03:55:18 +00:00
|
|
|
nsFrameItems childItems;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ProcessChildren(aPresShell, aPresContext, aState, aContent,
|
1999-08-27 21:46:10 +00:00
|
|
|
aNewCellBodyFrame, PR_TRUE, childItems, PR_TRUE);
|
1999-02-26 09:15:03 +00:00
|
|
|
if (NS_FAILED(rv)) return rv;
|
1999-08-25 05:52:10 +00:00
|
|
|
|
|
|
|
// if there are any anonymous children create frames for them
|
|
|
|
nsCOMPtr<nsIAtom> tagName;
|
|
|
|
aContent->GetTag(*getter_AddRefs(tagName));
|
1999-08-27 06:06:39 +00:00
|
|
|
if (tagName && tagName.get() == nsXULAtoms::treecell) {
|
1999-12-04 23:49:50 +00:00
|
|
|
CreateAnonymousTreeCellFrames(aPresShell, aPresContext, tagName, aState, aContent, aNewCellBodyFrame,
|
1999-09-28 00:57:31 +00:00
|
|
|
aNewCellFrame, childItems);
|
1999-08-27 06:06:39 +00:00
|
|
|
}
|
1999-08-25 05:52:10 +00:00
|
|
|
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewCellBodyFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-07-16 23:04:27 +00:00
|
|
|
if (aState.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewCellBodyFrame->SetInitialChildList(aPresContext,
|
1999-02-26 17:11:54 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
1999-07-16 23:04:27 +00:00
|
|
|
aState.mFloatedItems.childList);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewCellFrame->SetInitialChildList(aPresContext, nsnull, aNewCellBodyFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is only called by table row groups and rows. It allows children that are not
|
|
|
|
// table related to have a cell wrapped around them.
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::TableProcessChildren(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsFrameItems& aChildItems,
|
|
|
|
nsTableCreator& aTableCreator)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
1999-02-12 01:35:37 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
// Iterate the child content objects and construct a frame
|
|
|
|
PRInt32 count;
|
|
|
|
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> parentStyleContext;
|
|
|
|
aParentFrame->GetStyleContext(getter_AddRefs(parentStyleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-06-23 03:02:21 +00:00
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
parentStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
if (aTableCreator.IsTreeCreator() &&
|
|
|
|
(display->mDisplay == NS_STYLE_DISPLAY_TABLE_ROW_GROUP)) {
|
|
|
|
// Stop the processing if we're lazy. The tree row group frame builds its children
|
|
|
|
// as needed.
|
2000-01-13 19:20:25 +00:00
|
|
|
return NS_OK;
|
1999-06-23 03:02:21 +00:00
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
aContent->ChildCount(count);
|
|
|
|
for (PRInt32 i = 0; i < count; i++) {
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIContent> childContent;
|
|
|
|
aContent->ChildAt(i, *getter_AddRefs(childContent));
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = TableProcessChild(aPresShell, aPresContext, aState, childContent, aParentFrame, parentStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
aChildItems, aTableCreator);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::TableProcessChild(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aChildContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aParentStyleContext,
|
|
|
|
nsFrameItems& aChildItems,
|
|
|
|
nsTableCreator& aTableCreator)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
1999-02-12 01:35:37 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
if (nsnull != aChildContent) {
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> childStyleContext;
|
|
|
|
aPresContext->ResolveStyleContextFor(aChildContent, aParentStyleContext,
|
1999-02-12 17:45:58 +00:00
|
|
|
PR_FALSE,
|
1999-02-11 15:54:13 +00:00
|
|
|
getter_AddRefs(childStyleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
const nsStyleDisplay* childDisplay = (const nsStyleDisplay*)
|
|
|
|
childStyleContext->GetStyleData(eStyleStruct_Display);
|
1999-04-30 19:51:59 +00:00
|
|
|
if (IsTableRelated(childDisplay->mDisplay)) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructFrame(aPresShell, aPresContext, aState, aChildContent, aParentFrame, aChildItems);
|
1999-02-05 03:55:18 +00:00
|
|
|
} else {
|
1999-02-11 23:08:28 +00:00
|
|
|
nsCOMPtr<nsIAtom> tag;
|
1999-02-11 15:54:13 +00:00
|
|
|
aChildContent->GetTag(*getter_AddRefs(tag));
|
1999-02-26 17:11:54 +00:00
|
|
|
|
|
|
|
// XXX this needs to be fixed so that the form can work without
|
|
|
|
// a frame. This is *disgusting*
|
|
|
|
|
1999-03-31 06:47:40 +00:00
|
|
|
// forms, form controls need a frame but it can't be a child of an inner table
|
|
|
|
nsIFormControl* formControl = nsnull;
|
|
|
|
nsresult fcResult = aChildContent->QueryInterface(kIFormControlIID, (void**)&formControl);
|
|
|
|
NS_IF_RELEASE(formControl);
|
|
|
|
if ((nsHTMLAtoms::form == tag.get()) || NS_SUCCEEDED(fcResult)) {
|
1999-02-05 03:55:18 +00:00
|
|
|
// if the parent is a table, put the form in the outer table frame
|
|
|
|
const nsStyleDisplay* parentDisplay = (const nsStyleDisplay*)
|
|
|
|
aParentStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
if (parentDisplay->mDisplay == NS_STYLE_DISPLAY_TABLE) {
|
1999-02-12 01:35:37 +00:00
|
|
|
nsIFrame* outerFrame;
|
|
|
|
aParentFrame->GetParent(&outerFrame);
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructFrame(aPresShell, aPresContext, aState, aChildContent, outerFrame, aChildItems);
|
1999-02-12 01:35:37 +00:00
|
|
|
// XXX: Seems like this is going into the inner frame's child list instead of the outer frame. - DWH
|
1999-02-05 03:55:18 +00:00
|
|
|
} else {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructFrame(aPresShell, aPresContext, aState, aChildContent, aParentFrame, aChildItems);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-11 23:08:28 +00:00
|
|
|
// wrap it in a table cell, row, row group, table if it is a valid tag or display
|
|
|
|
// and not whitespace. For example we don't allow map, head, body, etc.
|
|
|
|
} else {
|
|
|
|
if (TableIsValidCellContent(aPresContext, aParentFrame, aChildContent)) {
|
|
|
|
PRBool needCell = PR_TRUE;
|
|
|
|
nsIDOMCharacterData* domData = nsnull;
|
|
|
|
nsresult rv2 = aChildContent->QueryInterface(kIDOMCharacterDataIID, (void**)&domData);
|
|
|
|
if ((NS_OK == rv2) && (nsnull != domData)) {
|
|
|
|
nsString charData;
|
|
|
|
domData->GetData(charData);
|
|
|
|
charData = charData.StripWhitespace();
|
|
|
|
if ((charData.Length() <= 0) && (charData != " ")) { // XXX check this
|
|
|
|
needCell = PR_FALSE; // only contains whitespace, don't create cell
|
|
|
|
}
|
|
|
|
NS_RELEASE(domData);
|
|
|
|
}
|
|
|
|
if (needCell) {
|
1999-02-26 09:15:03 +00:00
|
|
|
nsIFrame* cellBodyFrame;
|
1999-02-11 23:08:28 +00:00
|
|
|
nsIFrame* cellFrame;
|
1999-02-12 01:35:37 +00:00
|
|
|
nsIFrame* childFrame; // XXX: Need to change the ConstructTableCell function to return lists of frames. - DWH
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableCellFrame(aPresShell, aPresContext, aState, aChildContent, aParentFrame, childStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
childFrame, cellFrame, cellBodyFrame, aTableCreator);
|
1999-02-12 01:35:37 +00:00
|
|
|
aChildItems.AddChild(childFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-02-11 23:08:28 +00:00
|
|
|
PRBool
|
|
|
|
nsCSSFrameConstructor::TableIsValidCellContent(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIContent* aContent)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIAtom> tag;
|
|
|
|
aContent->GetTag(*getter_AddRefs(tag));
|
|
|
|
|
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
|
|
|
|
nsresult rv = ResolveStyleContext(aPresContext, aParentFrame, aContent, tag, getter_AddRefs(styleContext));
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
styleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
if (NS_STYLE_DISPLAY_NONE != display->mDisplay) {
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check tags first
|
1999-02-17 04:39:22 +00:00
|
|
|
if ( (nsHTMLAtoms::img == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::hr == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::br == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::wbr == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::input == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::textarea == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::select == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::applet == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::embed == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::fieldset == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::legend == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::object == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::form == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::iframe == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::spacer == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::button == tag.get()) ||
|
|
|
|
(nsHTMLAtoms::label == tag.get() )) {
|
1999-02-11 23:08:28 +00:00
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef INCLUDE_XUL
|
1999-02-17 04:39:22 +00:00
|
|
|
if ( (nsXULAtoms::button == tag.get()) ||
|
2000-02-25 08:37:49 +00:00
|
|
|
(nsXULAtoms::titledbutton == tag.get()) ||
|
2000-02-14 01:42:09 +00:00
|
|
|
(nsXULAtoms::image == tag.get()) ||
|
1999-06-30 22:17:43 +00:00
|
|
|
(nsXULAtoms::grippy == tag.get()) ||
|
|
|
|
(nsXULAtoms::splitter == tag.get()) ||
|
1999-04-20 21:57:41 +00:00
|
|
|
(nsXULAtoms::slider == tag.get()) ||
|
|
|
|
(nsXULAtoms::spinner == tag.get()) ||
|
|
|
|
(nsXULAtoms::scrollbar == tag.get()) ||
|
1999-06-15 04:02:43 +00:00
|
|
|
(nsXULAtoms::scrollbarbutton == tag.get()) ||
|
|
|
|
(nsXULAtoms::thumb == tag.get()) ||
|
1999-04-20 21:57:41 +00:00
|
|
|
(nsXULAtoms::colorpicker == tag.get()) ||
|
|
|
|
(nsXULAtoms::fontpicker == tag.get()) ||
|
1999-02-17 04:39:22 +00:00
|
|
|
(nsXULAtoms::text == tag.get()) ||
|
|
|
|
(nsXULAtoms::widget == tag.get()) ||
|
|
|
|
(nsXULAtoms::tree == tag.get()) ||
|
|
|
|
(nsXULAtoms::treechildren == tag.get()) ||
|
|
|
|
(nsXULAtoms::treeitem == tag.get()) ||
|
1999-06-23 03:02:21 +00:00
|
|
|
(nsXULAtoms::treerow == tag.get()) ||
|
1999-02-17 04:39:22 +00:00
|
|
|
(nsXULAtoms::treecell == tag.get()) ||
|
|
|
|
(nsXULAtoms::treeindentation == tag.get()) ||
|
1999-06-29 20:20:40 +00:00
|
|
|
(nsXULAtoms::treecol == tag.get()) ||
|
|
|
|
(nsXULAtoms::treecolgroup == tag.get()) ||
|
|
|
|
(nsXULAtoms::treefoot == tag.get()) ||
|
1999-07-31 11:29:03 +00:00
|
|
|
(nsXULAtoms::menu == tag.get()) ||
|
|
|
|
(nsXULAtoms::menuitem == tag.get()) ||
|
|
|
|
(nsXULAtoms::menubar == tag.get()) ||
|
|
|
|
(nsXULAtoms::menupopup == tag.get()) ||
|
1999-09-10 08:47:12 +00:00
|
|
|
(nsXULAtoms::popupset == tag.get()) ||
|
|
|
|
(nsXULAtoms::popup == tag.get()) ||
|
1999-02-17 04:39:22 +00:00
|
|
|
(nsXULAtoms::toolbox == tag.get()) ||
|
|
|
|
(nsXULAtoms::toolbar == tag.get()) ||
|
1999-07-01 21:11:38 +00:00
|
|
|
(nsXULAtoms::toolbaritem == tag.get()) ||
|
2000-03-02 03:01:30 +00:00
|
|
|
(nsXULAtoms::stack == tag.get()) ||
|
1999-04-21 22:46:15 +00:00
|
|
|
(nsXULAtoms::deck == tag.get()) ||
|
2000-03-02 03:01:30 +00:00
|
|
|
(nsXULAtoms::spring == tag.get()) ||
|
|
|
|
(nsXULAtoms::title == tag.get()) ||
|
|
|
|
(nsXULAtoms::titledbox == tag.get()) ||
|
1999-04-21 22:46:15 +00:00
|
|
|
(nsXULAtoms::tabcontrol == tag.get()) ||
|
|
|
|
(nsXULAtoms::tabbox == tag.get()) ||
|
|
|
|
(nsXULAtoms::tabpanel == tag.get()) ||
|
|
|
|
(nsXULAtoms::tabpage == tag.get()) ||
|
2000-03-15 03:16:43 +00:00
|
|
|
(nsXULAtoms::progressbar == tag.get()) ||
|
2000-02-25 08:37:49 +00:00
|
|
|
(nsXULAtoms::checkbox == tag.get()) ||
|
|
|
|
(nsXULAtoms::radio == tag.get()) ||
|
|
|
|
(nsXULAtoms::menulist == tag.get()) ||
|
|
|
|
(nsXULAtoms::menubutton == tag.get()) ||
|
|
|
|
(nsXULAtoms::textfield == tag.get()) ||
|
2000-03-02 10:00:09 +00:00
|
|
|
(nsXULAtoms::textarea == tag.get())
|
2000-02-25 08:37:49 +00:00
|
|
|
) {
|
1999-02-11 23:08:28 +00:00
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
#endif
|
1999-09-21 01:15:30 +00:00
|
|
|
|
|
|
|
//MathML Mod - DJF
|
1999-10-02 10:41:40 +00:00
|
|
|
#ifdef MOZ_MATHML
|
1999-09-21 01:15:30 +00:00
|
|
|
if ( (nsMathMLAtoms::math == tag.get()) ) {
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
#endif
|
1999-02-11 23:08:28 +00:00
|
|
|
|
|
|
|
// we should check for display type as well - later
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult
|
1999-02-26 09:15:03 +00:00
|
|
|
nsCSSFrameConstructor::TableProcessTableList(nsIPresContext* aPresContext,
|
|
|
|
nsTableList& aTableList)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
1999-02-26 09:15:03 +00:00
|
|
|
nsresult rv = NS_OK;
|
|
|
|
nsIFrame* inner = (nsIFrame*)aTableList.mInner;
|
|
|
|
if (inner) {
|
|
|
|
nsIFrame* child = (nsIFrame*)aTableList.mChild;
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = inner->SetInitialChildList(aPresContext, nsnull, child);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-26 09:15:03 +00:00
|
|
|
return rv;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsIFrame*
|
|
|
|
nsCSSFrameConstructor::TableGetAsNonScrollFrame(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
const nsStyleDisplay* aDisplay)
|
|
|
|
{
|
|
|
|
if (nsnull == aFrame) {
|
|
|
|
return nsnull;
|
|
|
|
}
|
|
|
|
nsIFrame* result = aFrame;
|
|
|
|
if (IsScrollable(aPresContext, aDisplay)) {
|
2000-01-22 01:16:50 +00:00
|
|
|
aFrame->FirstChild(aPresContext, nsnull, &result);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
// nsIAtom* pseudoTag;
|
|
|
|
// styleContext->GetPseudoType(pseudoTag);
|
1999-08-08 01:16:50 +00:00
|
|
|
// if (pseudoTag != nsLayoutAtoms::scrolledContentPseudo) {
|
1999-02-05 03:55:18 +00:00
|
|
|
// NS_IF_RELEASE(pseudoTag);
|
|
|
|
|
|
|
|
const nsStyleDisplay*
|
|
|
|
nsCSSFrameConstructor:: GetDisplay(nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
if (nsnull == aFrame) {
|
|
|
|
return nsnull;
|
|
|
|
}
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
aFrame->GetStyleContext(getter_AddRefs(styleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
const nsStyleDisplay* display =
|
|
|
|
(const nsStyleDisplay*)styleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
return display;
|
|
|
|
}
|
|
|
|
|
1999-05-03 16:07:47 +00:00
|
|
|
PRBool
|
1999-04-30 19:51:59 +00:00
|
|
|
nsCSSFrameConstructor::IsTableRelated(PRUint8 aDisplay)
|
|
|
|
{
|
|
|
|
return (aDisplay == NS_STYLE_DISPLAY_TABLE) ||
|
|
|
|
(aDisplay == NS_STYLE_DISPLAY_TABLE_HEADER_GROUP) ||
|
|
|
|
(aDisplay == NS_STYLE_DISPLAY_TABLE_ROW_GROUP) ||
|
|
|
|
(aDisplay == NS_STYLE_DISPLAY_TABLE_FOOTER_GROUP) ||
|
|
|
|
(aDisplay == NS_STYLE_DISPLAY_TABLE_ROW) ||
|
|
|
|
(aDisplay == NS_STYLE_DISPLAY_TABLE_COLUMN) ||
|
|
|
|
(aDisplay == NS_STYLE_DISPLAY_TABLE_CELL);
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
/***********************************************
|
|
|
|
* END TABLE SECTION
|
|
|
|
***********************************************/
|
|
|
|
|
1999-04-30 19:51:59 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructDocElementTableFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-30 19:51:59 +00:00
|
|
|
nsIContent* aDocElement,
|
|
|
|
nsIFrame* aParentFrame,
|
1999-12-06 07:44:18 +00:00
|
|
|
nsIFrame*& aNewTableFrame,
|
|
|
|
nsILayoutHistoryState* aFrameState)
|
1999-04-30 19:51:59 +00:00
|
|
|
{
|
1999-12-06 07:44:18 +00:00
|
|
|
nsFrameConstructorState state(aPresContext, nsnull, nsnull, nsnull, aFrameState);
|
1999-04-30 19:51:59 +00:00
|
|
|
nsFrameItems frameItems;
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
ConstructFrame(aPresShell, aPresContext, state, aDocElement, aParentFrame, frameItems);
|
1999-04-30 19:51:59 +00:00
|
|
|
aNewTableFrame = frameItems.childList;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
/**
|
|
|
|
* New one
|
|
|
|
*/
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructDocElementFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aDocElement,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aParentStyleContext,
|
|
|
|
nsIFrame*& aNewFrame)
|
|
|
|
{
|
1999-11-10 23:51:44 +00:00
|
|
|
// how the root frame hierarchy should look
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
---------------No Scrollbars------
|
|
|
|
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
AreaFrame or BoxFrame (InitialContainingBlock)
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
---------------Native Scrollbars------
|
|
|
|
|
|
|
|
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
ScrollFrame
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
^
|
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
AreaFrame or BoxFrame (InitialContainingBlock)
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
---------------Gfx Scrollbars ------
|
|
|
|
|
|
|
|
|
|
|
|
GfxScrollFrame
|
|
|
|
|
|
|
|
^
|
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
ScrollPort
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
^
|
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
AreaFrame or BoxFrame (InitialContainingBlock)
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
*/
|
|
|
|
|
2000-01-14 09:28:54 +00:00
|
|
|
if (!mTempFrameTreeState)
|
|
|
|
aPresShell->GetHistoryState(getter_AddRefs(mTempFrameTreeState));
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
// ----- reattach gfx scrollbars ------
|
|
|
|
// Gfx scrollframes were created in the root frame but the primary frame map may have been destroyed if a
|
|
|
|
// new style sheet was loaded so lets reattach the frames to their content.
|
|
|
|
if (mGfxScrollFrame)
|
|
|
|
{
|
|
|
|
nsIFrame* scrollPort = nsnull;
|
2000-01-22 01:16:50 +00:00
|
|
|
mGfxScrollFrame->FirstChild(aPresContext, nsnull, &scrollPort);
|
1999-11-10 23:51:44 +00:00
|
|
|
|
|
|
|
nsIFrame* gfxScrollbarFrame1 = nsnull;
|
|
|
|
nsIFrame* gfxScrollbarFrame2 = nsnull;
|
|
|
|
nsresult rv = scrollPort->GetNextSibling(&gfxScrollbarFrame1);
|
|
|
|
rv = gfxScrollbarFrame1->GetNextSibling(&gfxScrollbarFrame2);
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
|
|
gfxScrollbarFrame1->GetContent(getter_AddRefs(content));
|
|
|
|
aState.mFrameManager->SetPrimaryFrameFor(content, gfxScrollbarFrame1);
|
|
|
|
gfxScrollbarFrame2->GetContent(getter_AddRefs(content));
|
|
|
|
aState.mFrameManager->SetPrimaryFrameFor(content, gfxScrollbarFrame2);
|
|
|
|
}
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// --------- CREATE AREA OR BOX FRAME -------
|
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
aPresContext->ResolveStyleContextFor(aDocElement, aParentStyleContext,
|
|
|
|
PR_FALSE,
|
|
|
|
getter_AddRefs(styleContext));
|
|
|
|
|
|
|
|
const nsStyleDisplay* display =
|
|
|
|
(const nsStyleDisplay*)styleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
PRBool docElemIsTable = IsTableRelated(display->mDisplay);
|
|
|
|
|
|
|
|
|
|
|
|
// --------- IF SCROLLABLE WRAP IN SCROLLFRAME --------
|
|
|
|
|
|
|
|
PRBool isScrollable = IsScrollable(aPresContext, display);
|
|
|
|
PRBool isPaginated = PR_FALSE;
|
|
|
|
aPresContext->IsPaginated(&isPaginated);
|
1999-10-12 20:50:01 +00:00
|
|
|
nsIFrame* scrollFrame = nsnull;
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
// build a scrollframe
|
|
|
|
if (!isPaginated && isScrollable) {
|
|
|
|
nsIFrame* newScrollFrame = nsnull;
|
1999-11-02 05:36:08 +00:00
|
|
|
nsCOMPtr<nsIDocument> document;
|
|
|
|
aDocElement->GetDocument(*getter_AddRefs(document));
|
1999-12-08 01:56:28 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> newContext;
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
BeginBuildingScrollFrame( aPresShell, aPresContext,
|
1999-10-12 00:16:06 +00:00
|
|
|
aState,
|
|
|
|
aDocElement,
|
|
|
|
styleContext,
|
|
|
|
aParentFrame,
|
|
|
|
nsLayoutAtoms::scrolledContentPseudo,
|
|
|
|
document,
|
1999-10-12 20:50:01 +00:00
|
|
|
scrollFrame,
|
1999-12-08 01:56:28 +00:00
|
|
|
newContext,
|
1999-10-12 00:16:06 +00:00
|
|
|
newScrollFrame);
|
|
|
|
|
1999-12-08 01:56:28 +00:00
|
|
|
styleContext = newContext;
|
1999-10-12 00:16:06 +00:00
|
|
|
aParentFrame = newScrollFrame;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIFrame* contentFrame = nsnull;
|
|
|
|
PRBool isBlockFrame = PR_FALSE;
|
|
|
|
|
|
|
|
if (docElemIsTable) {
|
|
|
|
// if the document is a table then just populate it.
|
1999-12-06 07:44:18 +00:00
|
|
|
ConstructDocElementTableFrame(aPresShell, aPresContext, aDocElement,
|
|
|
|
aParentFrame, contentFrame,
|
|
|
|
aState.mFrameState);
|
1999-10-12 00:16:06 +00:00
|
|
|
contentFrame->GetStyleContext(getter_AddRefs(styleContext));
|
|
|
|
} else {
|
|
|
|
// otherwise build a box or a block
|
|
|
|
PRInt32 nameSpaceID;
|
|
|
|
if (NS_SUCCEEDED(aDocElement->GetNameSpaceID(nameSpaceID)) &&
|
|
|
|
nameSpaceID == nsXULAtoms::nameSpaceID) {
|
2000-03-02 03:01:30 +00:00
|
|
|
NS_NewBoxFrame(aPresShell, &contentFrame, PR_TRUE);
|
1999-10-12 00:16:06 +00:00
|
|
|
}
|
|
|
|
else {
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewDocumentElementFrame(aPresShell, &contentFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
isBlockFrame = PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// initialize the child
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aDocElement,
|
|
|
|
aParentFrame, styleContext, nsnull, contentFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
}
|
|
|
|
|
2000-01-27 20:21:28 +00:00
|
|
|
// set the primary frame
|
|
|
|
aState.mFrameManager->SetPrimaryFrameFor(aDocElement, contentFrame);
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// Finish building the scrollframe
|
|
|
|
if (isScrollable) {
|
|
|
|
FinishBuildingScrollFrame(aPresContext,
|
|
|
|
aState,
|
|
|
|
aDocElement,
|
|
|
|
aParentFrame,
|
|
|
|
contentFrame,
|
|
|
|
styleContext);
|
2000-01-31 14:04:41 +00:00
|
|
|
// primary is set above (to the contentFrame)
|
|
|
|
|
|
|
|
aNewFrame = scrollFrame;
|
1999-10-12 20:50:01 +00:00
|
|
|
} else {
|
|
|
|
// if not scrollable the new frame is the content frame.
|
|
|
|
aNewFrame = contentFrame;
|
1999-10-12 00:16:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
mInitialContainingBlock = contentFrame;
|
|
|
|
|
|
|
|
// if it was a table then we don't need to process our children.
|
|
|
|
if (!docElemIsTable) {
|
|
|
|
// Process the child content
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
|
|
|
nsFrameItems childItems;
|
|
|
|
|
|
|
|
// XXX these next lines are wrong for BoxFrame
|
|
|
|
PRBool haveFirstLetterStyle, haveFirstLineStyle;
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aDocElement, styleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
|
|
|
aState.PushAbsoluteContainingBlock(contentFrame, absoluteSaveState);
|
|
|
|
aState.PushFloaterContainingBlock(contentFrame, floaterSaveState,
|
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
1999-12-04 23:49:50 +00:00
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aDocElement, contentFrame,
|
1999-10-12 00:16:06 +00:00
|
|
|
PR_TRUE, childItems, isBlockFrame);
|
|
|
|
|
|
|
|
// See if the document element has a fixed background attachment.
|
|
|
|
// Note: the reason we wait until after processing the document element's
|
|
|
|
// children is because of special treatment of the background for the HTML
|
|
|
|
// element. See BodyFixupRule::MapStyleInto() for details
|
|
|
|
const nsStyleColor* color;
|
|
|
|
color = (const nsStyleColor*)styleContext->GetStyleData(eStyleStruct_Color);
|
|
|
|
if (NS_STYLE_BG_ATTACHMENT_FIXED == color->mBackgroundAttachment) {
|
|
|
|
// Fixed background attachments are handled by setting the
|
|
|
|
// NS_VIEW_PUBLIC_FLAG_DONT_BITBLT flag bit on the view.
|
|
|
|
//
|
|
|
|
// If the document element's frame is scrollable, then set the bit on its
|
|
|
|
// view; otherwise, set it on the root frame's view. This avoids
|
|
|
|
// unnecessarily creating another view and should be faster
|
|
|
|
nsIView* view;
|
|
|
|
|
|
|
|
if (IsScrollable(aPresContext, display)) {
|
1999-10-26 04:44:41 +00:00
|
|
|
contentFrame->GetView(aPresContext, &view);
|
1999-10-12 00:16:06 +00:00
|
|
|
} else {
|
|
|
|
nsIFrame* parentFrame;
|
|
|
|
|
|
|
|
contentFrame->GetParent(&parentFrame);
|
1999-10-26 04:44:41 +00:00
|
|
|
parentFrame->GetView(aPresContext, &view);
|
1999-10-12 00:16:06 +00:00
|
|
|
}
|
|
|
|
|
2000-03-09 05:54:58 +00:00
|
|
|
// Not all shells have scroll frames, even in scrollable presContext (bug 30317)
|
|
|
|
if (view) {
|
|
|
|
PRUint32 viewFlags;
|
|
|
|
view->GetViewFlags(&viewFlags);
|
|
|
|
view->SetViewFlags(viewFlags | NS_VIEW_PUBLIC_FLAG_DONT_BITBLT);
|
|
|
|
}
|
1999-10-12 00:16:06 +00:00
|
|
|
}
|
2000-02-10 21:36:28 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// Set the initial child lists
|
1999-11-24 06:03:41 +00:00
|
|
|
contentFrame->SetInitialChildList(aPresContext, nsnull,
|
2000-02-10 21:36:28 +00:00
|
|
|
childItems.childList);
|
|
|
|
|
|
|
|
// only support absolute positioning if we are a block.
|
|
|
|
// if we are a box don't do it.
|
|
|
|
if (isBlockFrame) {
|
|
|
|
if (aState.mAbsoluteItems.childList) {
|
|
|
|
contentFrame->SetInitialChildList(aPresContext,
|
|
|
|
nsLayoutAtoms::absoluteList,
|
|
|
|
aState.mAbsoluteItems.childList);
|
|
|
|
}
|
|
|
|
if (aState.mFloatedItems.childList) {
|
|
|
|
contentFrame->SetInitialChildList(aPresContext,
|
|
|
|
nsLayoutAtoms::floaterList,
|
|
|
|
aState.mFloatedItems.childList);
|
|
|
|
}
|
1999-10-12 00:16:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
1999-04-30 19:51:59 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
|
1999-08-19 22:16:23 +00:00
|
|
|
NS_IMETHODIMP
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructRootFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-08-19 22:16:23 +00:00
|
|
|
nsIContent* aDocElement,
|
|
|
|
nsIFrame*& aNewFrame)
|
|
|
|
{
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// how the root frame hierarchy should look
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
/*
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
---------------No Scrollbars------
|
|
|
|
|
|
|
|
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
ViewPortFrame (FixedContainingBlock) <---- RootView
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
^
|
|
|
|
|
|
|
|
|
RootFrame(DocElementContainingBlock)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
---------------Native Scrollbars------
|
|
|
|
|
|
|
|
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
ViewPortFrame (FixedContainingBlock) <---- RootView
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
^
|
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
ScrollFrame <--- RootScrollableView
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
^
|
|
|
|
|
|
|
|
|
RootFrame(DocElementContainingBlock)
|
|
|
|
|
|
|
|
|
|
|
|
---------------Gfx Scrollbars ------
|
|
|
|
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
ViewPortFrame (FixedContainingBlock) <---- RootView
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
^
|
|
|
|
|
|
|
|
|
GfxScrollFrame
|
|
|
|
|
|
|
|
^
|
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
ScrollPort <--- RootScrollableView
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
^
|
|
|
|
|
|
|
|
|
RootFrame(DocElementContainingBlock)
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
// --------- BUILD VIEWPORT -----------
|
|
|
|
nsIFrame* viewportFrame = nsnull;
|
|
|
|
nsCOMPtr<nsIStyleContext> viewportPseudoStyle;
|
1999-08-19 22:16:23 +00:00
|
|
|
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(nsnull, nsLayoutAtoms::viewportPseudo,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsnull, PR_FALSE,
|
|
|
|
getter_AddRefs(viewportPseudoStyle));
|
1999-08-19 22:16:23 +00:00
|
|
|
|
|
|
|
{ // ensure that the viewport thinks it is a block frame, layout goes pootsy if it doesn't
|
|
|
|
nsStyleDisplay* display = (nsStyleDisplay*)viewportPseudoStyle->GetMutableStyleData(eStyleStruct_Display);
|
|
|
|
display->mDisplay = NS_STYLE_DISPLAY_BLOCK;
|
|
|
|
}
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewViewportFrame(aPresShell, &viewportFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
|
1999-11-24 06:03:41 +00:00
|
|
|
viewportFrame->Init(aPresContext, nsnull, nsnull, viewportPseudoStyle, nsnull);
|
1999-08-19 22:16:23 +00:00
|
|
|
|
|
|
|
// Bind the viewport frame to the root view
|
|
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(presShell));
|
|
|
|
nsCOMPtr<nsIViewManager> viewManager;
|
|
|
|
presShell->GetViewManager(getter_AddRefs(viewManager));
|
|
|
|
nsIView* rootView;
|
|
|
|
|
|
|
|
viewManager->GetRootView(rootView);
|
1999-10-26 04:44:41 +00:00
|
|
|
viewportFrame->SetView(aPresContext, rootView);
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// The viewport is the containing block for 'fixed' elements
|
|
|
|
mFixedContainingBlock = viewportFrame;
|
|
|
|
|
|
|
|
// --------- CREATE ROOT FRAME -------
|
|
|
|
|
|
|
|
|
|
|
|
// Create the root frame. The document element's frame is a child of the
|
|
|
|
// root frame.
|
|
|
|
//
|
|
|
|
// The root frame serves two purposes:
|
|
|
|
// - reserves space for any margins needed for the document element's frame
|
|
|
|
// - makes sure that the document element's frame covers the entire canvas
|
|
|
|
|
|
|
|
PRBool isPaginated = PR_FALSE;
|
|
|
|
aPresContext->IsPaginated(&isPaginated);
|
|
|
|
nsIFrame* rootFrame = nsnull;
|
|
|
|
nsIAtom* rootPseudo;
|
|
|
|
|
|
|
|
if (!isPaginated) {
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewRootFrame(aPresShell, &rootFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
rootPseudo = nsLayoutAtoms::canvasPseudo;
|
|
|
|
mDocElementContainingBlock = rootFrame;
|
|
|
|
} else {
|
|
|
|
// Create a page sequence frame
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewSimplePageSequenceFrame(aPresShell, &rootFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
rootPseudo = nsLayoutAtoms::pageSequencePseudo;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// --------- IF SCROLLABLE WRAP IN SCROLLFRAME --------
|
|
|
|
|
|
|
|
// If the device supports scrolling (e.g., in galley mode on the screen and
|
1999-08-19 22:16:23 +00:00
|
|
|
// for print-preview, but not when printing), then create a scroll frame that
|
|
|
|
// will act as the scrolling mechanism for the viewport.
|
|
|
|
// XXX Do we even need a viewport when printing to a printer?
|
|
|
|
// XXX It would be nice to have a better way to query for whether the device
|
|
|
|
// is scrollable
|
|
|
|
PRBool isScrollable = PR_TRUE;
|
|
|
|
if (aPresContext) {
|
|
|
|
nsIDeviceContext* dc;
|
|
|
|
aPresContext->GetDeviceContext(&dc);
|
|
|
|
if (dc) {
|
|
|
|
PRBool supportsWidgets;
|
|
|
|
if (NS_SUCCEEDED(dc->SupportsNativeWidgets(supportsWidgets))) {
|
|
|
|
isScrollable = supportsWidgets;
|
|
|
|
}
|
|
|
|
NS_RELEASE(dc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// As long as the webshell doesn't prohibit it, and the device supports
|
|
|
|
// it, create a scroll frame that will act as the scolling mechanism for
|
|
|
|
// the viewport.
|
2000-02-29 06:51:04 +00:00
|
|
|
//
|
|
|
|
// Threre are three possible values stored in the docshell:
|
|
|
|
// 1) NS_STYLE_OVERFLOW_HIDDEN = no scrollbars
|
|
|
|
// 2) NS_STYLE_OVERFLOW_AUTO = scrollbars appear if needed
|
|
|
|
// 3) NS_STYLE_OVERFLOW_SCROLL = scrollbars always
|
|
|
|
// Only need to create a scroll frame/view for cases 2 and 3.
|
|
|
|
// Currently OVERFLOW_SCROLL isn't honored, as
|
|
|
|
// scrollportview::SetScrollPref is not implemented.
|
|
|
|
PRInt32 nameSpaceID; // Never create scrollbars for XUL documents
|
|
|
|
if (NS_SUCCEEDED(aDocElement->GetNameSpaceID(nameSpaceID)) &&
|
|
|
|
nameSpaceID == nsXULAtoms::nameSpaceID) {
|
|
|
|
isScrollable = PR_FALSE;
|
|
|
|
} else {
|
|
|
|
nsresult rv;
|
|
|
|
nsCOMPtr<nsISupports> container;
|
|
|
|
if (nsnull != aPresContext) {
|
|
|
|
aPresContext->GetContainer(getter_AddRefs(container));
|
|
|
|
if (nsnull != container) {
|
|
|
|
nsCOMPtr<nsIScrollable> scrollableContainer = do_QueryInterface(container, &rv);
|
|
|
|
if (NS_SUCCEEDED(rv) && scrollableContainer) {
|
|
|
|
PRInt32 scrolling = -1;
|
|
|
|
// XXX We should get prefs for X and Y and deal with these independently!
|
|
|
|
scrollableContainer->GetCurrentScrollbarPreferences(nsIScrollable::ScrollOrientation_Y,&scrolling);
|
|
|
|
if (NS_STYLE_OVERFLOW_HIDDEN == scrolling) {
|
|
|
|
isScrollable = PR_FALSE;
|
|
|
|
}
|
|
|
|
// XXX NS_STYLE_OVERFLOW_SCROLL should create 'always on' scrollbars
|
1999-08-19 22:16:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIFrame* newFrame = rootFrame;
|
|
|
|
nsCOMPtr<nsIStyleContext> rootPseudoStyle;
|
|
|
|
// we must create a state because if the scrollbars are GFX it needs the
|
|
|
|
// state to build the scrollbar frames.
|
|
|
|
nsFrameConstructorState state(aPresContext,
|
1999-12-06 07:44:18 +00:00
|
|
|
nsnull,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsnull,
|
|
|
|
nsnull,
|
|
|
|
nsnull);
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIFrame* parentFrame = viewportFrame;
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
if (isScrollable) {
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// built the frame. We give it the content we are wrapping which is the document,
|
|
|
|
// the root frame, the parent view port frame, and we should get back the new
|
|
|
|
// frame and the scrollable view if one was created.
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// resolve a context for the scrollframe
|
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(nsnull,
|
1999-12-08 01:56:28 +00:00
|
|
|
nsLayoutAtoms::viewportScrollPseudo,
|
|
|
|
viewportPseudoStyle, PR_FALSE,
|
|
|
|
getter_AddRefs(styleContext));
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
nsIFrame* newScrollableFrame = nsnull;
|
1999-11-02 06:40:38 +00:00
|
|
|
nsCOMPtr<nsIDocument> document;
|
|
|
|
aDocElement->GetDocument(*getter_AddRefs(document));
|
1999-10-12 00:16:06 +00:00
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
BeginBuildingScrollFrame( aPresShell,
|
|
|
|
aPresContext,
|
1999-10-12 00:16:06 +00:00
|
|
|
state,
|
|
|
|
nsnull,
|
|
|
|
styleContext,
|
|
|
|
viewportFrame,
|
|
|
|
rootPseudo,
|
|
|
|
document,
|
|
|
|
newFrame,
|
|
|
|
rootPseudoStyle,
|
|
|
|
newScrollableFrame);
|
|
|
|
|
|
|
|
// Inform the view manager about the root scrollable view
|
|
|
|
// get the scrolling view
|
|
|
|
nsIView* view = nsnull;
|
1999-10-26 04:44:41 +00:00
|
|
|
newScrollableFrame->GetView(aPresContext, &view);
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIScrollableView* scrollableView;
|
|
|
|
view->QueryInterface(kScrollViewIID, (void**)&scrollableView);
|
|
|
|
viewManager->SetRootScrollableView(scrollableView);
|
|
|
|
parentFrame = newScrollableFrame;
|
1999-11-10 23:51:44 +00:00
|
|
|
|
|
|
|
// if gfx scrollbars store them
|
|
|
|
if (HasGfxScrollbars(aPresContext)) {
|
|
|
|
mGfxScrollFrame = newFrame;
|
|
|
|
} else {
|
|
|
|
mGfxScrollFrame = nsnull;
|
|
|
|
}
|
1999-10-12 00:16:06 +00:00
|
|
|
} else {
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(nsnull, rootPseudo,
|
|
|
|
viewportPseudoStyle,
|
|
|
|
PR_FALSE,
|
|
|
|
getter_AddRefs(rootPseudoStyle));
|
1999-08-19 22:16:23 +00:00
|
|
|
}
|
|
|
|
|
1999-11-24 06:03:41 +00:00
|
|
|
rootFrame->Init(aPresContext, nsnull, parentFrame, rootPseudoStyle, nsnull);
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
if (isScrollable) {
|
|
|
|
FinishBuildingScrollFrame(aPresContext,
|
|
|
|
state,
|
|
|
|
aDocElement,
|
|
|
|
parentFrame,
|
|
|
|
rootFrame,
|
|
|
|
rootPseudoStyle);
|
2000-01-31 14:04:41 +00:00
|
|
|
|
|
|
|
// set the primary frame to the root frame
|
|
|
|
state.mFrameManager->SetPrimaryFrameFor(aDocElement, rootFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
}
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
if (isPaginated) {
|
1999-08-19 22:16:23 +00:00
|
|
|
// Create the first page
|
|
|
|
nsIFrame* pageFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewPageFrame(aPresShell, &pageFrame);
|
1999-08-19 22:16:23 +00:00
|
|
|
|
|
|
|
// The page is the containing block for 'fixed' elements. which are repeated
|
|
|
|
// on every page
|
|
|
|
mFixedContainingBlock = pageFrame;
|
|
|
|
|
|
|
|
// Initialize the page and force it to have a view. This makes printing of
|
|
|
|
// the pages easier and faster.
|
|
|
|
nsCOMPtr<nsIStyleContext> pagePseudoStyle;
|
|
|
|
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(nsnull, nsLayoutAtoms::pagePseudo,
|
1999-10-12 00:16:06 +00:00
|
|
|
rootPseudoStyle, PR_FALSE,
|
1999-08-19 22:16:23 +00:00
|
|
|
getter_AddRefs(pagePseudoStyle));
|
|
|
|
|
1999-11-24 06:03:41 +00:00
|
|
|
pageFrame->Init(aPresContext, nsnull, rootFrame, pagePseudoStyle,
|
1999-08-19 22:16:23 +00:00
|
|
|
nsnull);
|
1999-11-24 06:03:41 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, pageFrame,
|
1999-08-19 22:16:23 +00:00
|
|
|
pagePseudoStyle, PR_TRUE);
|
|
|
|
|
|
|
|
// The eventual parent of the document element frame
|
|
|
|
mDocElementContainingBlock = pageFrame;
|
|
|
|
|
|
|
|
|
|
|
|
// Set the initial child lists
|
1999-11-24 06:03:41 +00:00
|
|
|
rootFrame->SetInitialChildList(aPresContext, nsnull, pageFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
1999-11-24 06:03:41 +00:00
|
|
|
viewportFrame->SetInitialChildList(aPresContext, nsnull, newFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
aNewFrame = viewportFrame;
|
1999-11-10 23:51:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::CreatePlaceholderFrameFor(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-08-05 03:09:22 +00:00
|
|
|
nsIFrameManager* aFrameManager,
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame* aParentFrame,
|
1999-04-25 16:58:42 +00:00
|
|
|
nsIFrame** aPlaceholderFrame)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
1999-04-25 16:58:42 +00:00
|
|
|
nsPlaceholderFrame* placeholderFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
nsresult rv = NS_NewPlaceholderFrame(aPresShell, (nsIFrame**)&placeholderFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
// The placeholder frame gets a pseudo style context
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> placeholderPseudoStyle;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::placeholderPseudo,
|
|
|
|
aStyleContext,
|
1999-02-12 17:45:58 +00:00
|
|
|
PR_FALSE,
|
1999-02-11 15:54:13 +00:00
|
|
|
getter_AddRefs(placeholderPseudoStyle));
|
1999-11-24 06:03:41 +00:00
|
|
|
placeholderFrame->Init(aPresContext, aContent, aParentFrame,
|
1999-02-25 03:27:57 +00:00
|
|
|
placeholderPseudoStyle, nsnull);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Add mapping from absolutely positioned frame to its placeholder frame
|
1999-08-05 03:09:22 +00:00
|
|
|
aFrameManager->SetPlaceholderFrameFor(aFrame, placeholderFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-04-25 16:58:42 +00:00
|
|
|
// The placeholder frame has a pointer back to the out-of-flow frame
|
|
|
|
placeholderFrame->SetOutOfFlowFrame(aFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
|
1999-04-25 16:58:42 +00:00
|
|
|
*aPlaceholderFrame = NS_STATIC_CAST(nsIFrame*, placeholderFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-08-06 05:11:39 +00:00
|
|
|
|
|
|
|
nsWidgetRendering
|
|
|
|
nsCSSFrameConstructor::GetFormElementRenderingMode(nsIPresContext* aPresContext,
|
|
|
|
nsWidgetType aWidgetType)
|
|
|
|
{
|
|
|
|
if (!aPresContext) { return eWidgetRendering_Gfx;}
|
|
|
|
|
|
|
|
nsWidgetRendering mode;
|
|
|
|
aPresContext->GetWidgetRenderingMode(&mode);
|
|
|
|
|
|
|
|
switch (mode)
|
|
|
|
{
|
|
|
|
case eWidgetRendering_Gfx:
|
|
|
|
return eWidgetRendering_Gfx;
|
|
|
|
|
|
|
|
case eWidgetRendering_PartialGfx:
|
|
|
|
switch (aWidgetType)
|
|
|
|
{
|
|
|
|
case eWidgetType_Button:
|
|
|
|
case eWidgetType_Checkbox:
|
|
|
|
case eWidgetType_Radio:
|
|
|
|
case eWidgetType_Text:
|
|
|
|
return eWidgetRendering_Gfx;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return eWidgetRendering_Native;
|
|
|
|
}
|
|
|
|
|
|
|
|
case eWidgetRendering_Native:
|
|
|
|
PRBool useNativeWidgets = PR_FALSE;
|
|
|
|
nsIDeviceContext* dc;
|
|
|
|
aPresContext->GetDeviceContext(&dc);
|
|
|
|
if (dc) {
|
|
|
|
PRBool supportsWidgets;
|
|
|
|
if (NS_SUCCEEDED(dc->SupportsNativeWidgets(supportsWidgets))) {
|
|
|
|
useNativeWidgets = supportsWidgets;
|
|
|
|
}
|
|
|
|
NS_RELEASE(dc);
|
|
|
|
}
|
|
|
|
if (useNativeWidgets)
|
|
|
|
return eWidgetRendering_Native;
|
|
|
|
else
|
|
|
|
return eWidgetRendering_Gfx;
|
|
|
|
}
|
|
|
|
return eWidgetRendering_Gfx;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-06-21 20:41:56 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructRadioControlFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-06-21 20:41:56 +00:00
|
|
|
nsIFrame*& aNewFrame,
|
1999-08-03 14:41:48 +00:00
|
|
|
nsIContent* aContent,
|
|
|
|
nsIStyleContext* aStyleContext)
|
1999-06-21 20:41:56 +00:00
|
|
|
{
|
1999-08-06 05:11:39 +00:00
|
|
|
nsresult rv = NS_OK;
|
|
|
|
if (GetFormElementRenderingMode(aPresContext, eWidgetType_Radio) == eWidgetRendering_Gfx)
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewGfxRadioControlFrame(aPresShell, &aNewFrame);
|
1999-08-06 05:11:39 +00:00
|
|
|
else
|
2000-01-31 22:46:55 +00:00
|
|
|
NS_ASSERTION(0, "We longer support native widgets");
|
1999-08-06 05:11:39 +00:00
|
|
|
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
aNewFrame = nsnull;
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-06-21 20:41:56 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> radioStyle;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::radioPseudo,
|
1999-08-03 14:41:48 +00:00
|
|
|
aStyleContext, PR_FALSE, getter_AddRefs(radioStyle));
|
1999-06-21 20:41:56 +00:00
|
|
|
nsIRadioControlFrame* radio = nsnull;
|
2000-01-31 22:46:55 +00:00
|
|
|
if (aNewFrame != nsnull && NS_SUCCEEDED(aNewFrame->QueryInterface(kIRadioControlFrameIID, (void**)&radio))) {
|
1999-06-21 20:41:56 +00:00
|
|
|
radio->SetRadioButtonFaceStyleContext(radioStyle);
|
|
|
|
NS_RELEASE(radio);
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-08-06 05:11:39 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructCheckboxControlFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-08-06 05:11:39 +00:00
|
|
|
nsIFrame*& aNewFrame)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
if (GetFormElementRenderingMode(aPresContext, eWidgetType_Checkbox) == eWidgetRendering_Gfx)
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewGfxCheckboxControlFrame(aPresShell, &aNewFrame);
|
1999-08-06 05:11:39 +00:00
|
|
|
else
|
2000-01-31 22:46:55 +00:00
|
|
|
NS_ASSERTION(0, "We longer support native widgets");
|
|
|
|
|
1999-08-06 05:11:39 +00:00
|
|
|
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
aNewFrame = nsnull;
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructButtonControlFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-08-06 05:11:39 +00:00
|
|
|
nsIFrame*& aNewFrame)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
if (GetFormElementRenderingMode(aPresContext, eWidgetType_Button) == eWidgetRendering_Gfx)
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewGfxButtonControlFrame(aPresShell, &aNewFrame);
|
1999-08-06 05:11:39 +00:00
|
|
|
else
|
2000-01-31 22:46:55 +00:00
|
|
|
NS_ASSERTION(0, "We longer support native widgets");
|
1999-08-06 05:11:39 +00:00
|
|
|
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
aNewFrame = nsnull;
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-06-12 22:32:41 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructTextControlFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-08-19 22:11:03 +00:00
|
|
|
nsIFrame*& aNewFrame,
|
|
|
|
nsIContent* aContent)
|
1999-06-12 22:32:41 +00:00
|
|
|
{
|
|
|
|
if (!aPresContext) { return NS_ERROR_NULL_POINTER;}
|
|
|
|
nsresult rv = NS_OK;
|
1999-08-19 22:11:03 +00:00
|
|
|
|
|
|
|
//Do we want an Autocomplete input text widget?
|
|
|
|
nsString val1;
|
|
|
|
nsString val2;
|
|
|
|
if ((NS_OK == aContent->GetAttribute(kNameSpaceID_HTML, nsHTMLAtoms::autocompletetimeout, val1)) ||
|
|
|
|
(NS_OK == aContent->GetAttribute(kNameSpaceID_HTML, nsHTMLAtoms::autocompletetype, val2))) {
|
|
|
|
if (! val1.IsEmpty() || ! val2.IsEmpty()) {
|
|
|
|
//ducarroz: How can I check if I am in a xul document?
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewGfxAutoTextControlFrame(aPresShell, &aNewFrame);
|
1999-08-19 22:11:03 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
aNewFrame = nsnull;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-06-12 22:32:41 +00:00
|
|
|
nsWidgetRendering mode;
|
|
|
|
aPresContext->GetWidgetRenderingMode(&mode);
|
|
|
|
if (eWidgetRendering_Gfx == mode)
|
|
|
|
{
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewGfxTextControlFrame(aPresShell, &aNewFrame);
|
1999-06-12 22:32:41 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
aNewFrame = nsnull;
|
|
|
|
}
|
1999-12-17 03:28:50 +00:00
|
|
|
if (aNewFrame)
|
|
|
|
((nsGfxTextControlFrame*)aNewFrame)->SetFrameConstructor(this);
|
1999-06-12 22:32:41 +00:00
|
|
|
}
|
|
|
|
if (!aNewFrame)
|
|
|
|
{
|
2000-01-31 22:46:55 +00:00
|
|
|
NS_ASSERTION(0, "We longer support native widgets");
|
1999-06-12 22:32:41 +00:00
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-08-19 22:16:23 +00:00
|
|
|
PRBool
|
1999-08-27 06:06:39 +00:00
|
|
|
nsCSSFrameConstructor::HasGfxScrollbars(nsIPresContext* aPresContext)
|
1999-08-19 22:16:23 +00:00
|
|
|
{
|
2000-03-02 07:13:02 +00:00
|
|
|
nsCOMPtr<nsIPref> pref(do_GetService(NS_PREF_PROGID));
|
1999-10-12 00:16:06 +00:00
|
|
|
PRBool gfx = PR_FALSE;
|
1999-10-19 03:00:47 +00:00
|
|
|
if (pref) {
|
|
|
|
pref->GetBoolPref("nglayout.widget.gfxscrollbars", &gfx);
|
|
|
|
}
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
return gfx;
|
1999-08-27 06:06:39 +00:00
|
|
|
|
1999-08-19 22:16:23 +00:00
|
|
|
/*
|
|
|
|
nsWidgetRendering mode;
|
|
|
|
aPresContext->GetWidgetRenderingMode(&mode);
|
1999-08-27 06:06:39 +00:00
|
|
|
return (eWidgetRendering_Native != mode);
|
|
|
|
*/
|
1999-08-19 22:16:23 +00:00
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructSelectFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIAtom* aTag,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aNewFrame,
|
|
|
|
PRBool& aProcessChildren,
|
|
|
|
PRBool aIsAbsolutelyPositioned,
|
|
|
|
PRBool& aFrameHasBeenInitialized,
|
|
|
|
PRBool aIsFixedPositioned,
|
|
|
|
nsFrameItems& aFrameItems)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
1999-11-19 15:41:19 +00:00
|
|
|
nsresult rv = NS_OK;
|
1999-03-02 22:43:26 +00:00
|
|
|
nsWidgetRendering mode;
|
|
|
|
aPresContext->GetWidgetRenderingMode(&mode);
|
1999-04-12 22:14:31 +00:00
|
|
|
const PRInt32 kNoSizeSpecified = -1;
|
1999-03-22 21:32:12 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
PRBool hasGfxScrollbars = HasGfxScrollbars(aPresContext);
|
|
|
|
|
1999-03-22 21:32:12 +00:00
|
|
|
if (eWidgetRendering_Gfx == mode) {
|
1999-08-31 03:09:40 +00:00
|
|
|
// Construct a frame-based listbox or combobox
|
|
|
|
nsCOMPtr<nsIDOMHTMLSelectElement> sel(do_QueryInterface(aContent));
|
1999-04-12 22:14:31 +00:00
|
|
|
PRInt32 size = 1;
|
1999-08-31 03:09:40 +00:00
|
|
|
if (sel) {
|
|
|
|
sel->GetSize(&size);
|
1999-07-20 22:32:41 +00:00
|
|
|
PRBool multipleSelect = PR_FALSE;
|
1999-08-31 03:09:40 +00:00
|
|
|
sel->GetMultiple(&multipleSelect);
|
1999-07-20 22:32:41 +00:00
|
|
|
// Construct a combobox if size=1 or no size is specified and its multiple select
|
2000-03-14 12:03:21 +00:00
|
|
|
if (((1 == size || 0 == size) || (kNoSizeSpecified == size)) && (PR_FALSE == multipleSelect)) {
|
fixed bugs #6303, #6753, #6756, #6759
Re-wrote nsComboboxFrame.
removed the obsolete nsHTMLAtoms: comboText,comoTextSelected,comTextSelectedFocus,dropDownVisible,
dropdownHidden, dropDownBtnOut, dropDownBtnPressed,
Added nsHTMLAtoms::combobox, nsLayoutAtoms::popupList
Renamed dropDownList to dropDownListPseudo
Added "arrow.gif" as to be used the background-image for the combobox button
ua.css - added rules for select to differentiate between comboboxes and listboxes.
Added style rules to more closely match the XPTOOLKIT XPWidgets look.
removed the following :-moz-combobox-text, -moz-combobox-textselected
nsIFormControlFrame.h - Added SetSuggestedSize method.
nsButtonControlFrame - Implemented SetSuggestedSize.
nsCSSFrameConstructor.cpp - Rewrote ConstructSelectFrame.
nsIWidget.h -Added GetAbsoluteBounds method.
nsWindow.cpp - Implemented GetAbsoluteBounds.
1999-07-14 22:00:24 +00:00
|
|
|
// Construct a frame-based combo box.
|
|
|
|
// The frame-based combo box is built out of tree parts. A display area, a button and
|
|
|
|
// a dropdown list. The display area and button are created through anonymous content.
|
|
|
|
// The drop-down list's frame is created explicitly. The combobox frame shares it's content
|
|
|
|
// with the drop-down list.
|
1999-12-05 20:43:26 +00:00
|
|
|
PRUint32 flags = NS_BLOCK_SHRINK_WRAP | (aIsAbsolutelyPositioned?NS_BLOCK_SPACE_MGR:0);
|
fixed bugs #6303, #6753, #6756, #6759
Re-wrote nsComboboxFrame.
removed the obsolete nsHTMLAtoms: comboText,comoTextSelected,comTextSelectedFocus,dropDownVisible,
dropdownHidden, dropDownBtnOut, dropDownBtnPressed,
Added nsHTMLAtoms::combobox, nsLayoutAtoms::popupList
Renamed dropDownList to dropDownListPseudo
Added "arrow.gif" as to be used the background-image for the combobox button
ua.css - added rules for select to differentiate between comboboxes and listboxes.
Added style rules to more closely match the XPTOOLKIT XPWidgets look.
removed the following :-moz-combobox-text, -moz-combobox-textselected
nsIFormControlFrame.h - Added SetSuggestedSize method.
nsButtonControlFrame - Implemented SetSuggestedSize.
nsCSSFrameConstructor.cpp - Rewrote ConstructSelectFrame.
nsIWidget.h -Added GetAbsoluteBounds method.
nsWindow.cpp - Implemented GetAbsoluteBounds.
1999-07-14 22:00:24 +00:00
|
|
|
nsIFrame * comboboxFrame;
|
1999-12-05 20:43:26 +00:00
|
|
|
rv = NS_NewComboboxControlFrame(aPresShell, &comboboxFrame, flags);
|
fixed bugs #6303, #6753, #6756, #6759
Re-wrote nsComboboxFrame.
removed the obsolete nsHTMLAtoms: comboText,comoTextSelected,comTextSelectedFocus,dropDownVisible,
dropdownHidden, dropDownBtnOut, dropDownBtnPressed,
Added nsHTMLAtoms::combobox, nsLayoutAtoms::popupList
Renamed dropDownList to dropDownListPseudo
Added "arrow.gif" as to be used the background-image for the combobox button
ua.css - added rules for select to differentiate between comboboxes and listboxes.
Added style rules to more closely match the XPTOOLKIT XPWidgets look.
removed the following :-moz-combobox-text, -moz-combobox-textselected
nsIFormControlFrame.h - Added SetSuggestedSize method.
nsButtonControlFrame - Implemented SetSuggestedSize.
nsCSSFrameConstructor.cpp - Rewrote ConstructSelectFrame.
nsIWidget.h -Added GetAbsoluteBounds method.
nsWindow.cpp - Implemented GetAbsoluteBounds.
1999-07-14 22:00:24 +00:00
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
// Determine geometric parent for the combobox frame
|
fixed bugs #6303, #6753, #6756, #6759
Re-wrote nsComboboxFrame.
removed the obsolete nsHTMLAtoms: comboText,comoTextSelected,comTextSelectedFocus,dropDownVisible,
dropdownHidden, dropDownBtnOut, dropDownBtnPressed,
Added nsHTMLAtoms::combobox, nsLayoutAtoms::popupList
Renamed dropDownList to dropDownListPseudo
Added "arrow.gif" as to be used the background-image for the combobox button
ua.css - added rules for select to differentiate between comboboxes and listboxes.
Added style rules to more closely match the XPTOOLKIT XPWidgets look.
removed the following :-moz-combobox-text, -moz-combobox-textselected
nsIFormControlFrame.h - Added SetSuggestedSize method.
nsButtonControlFrame - Implemented SetSuggestedSize.
nsCSSFrameConstructor.cpp - Rewrote ConstructSelectFrame.
nsIWidget.h -Added GetAbsoluteBounds method.
nsWindow.cpp - Implemented GetAbsoluteBounds.
1999-07-14 22:00:24 +00:00
|
|
|
nsIFrame* geometricParent = aParentFrame;
|
|
|
|
if (aIsAbsolutelyPositioned) {
|
|
|
|
geometricParent = aState.mAbsoluteItems.containingBlock;
|
|
|
|
} else if (aIsFixedPositioned) {
|
|
|
|
geometricParent = aState.mFixedItems.containingBlock;
|
|
|
|
}
|
1999-12-06 07:44:18 +00:00
|
|
|
// Initialize the combobox frame
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
geometricParent, aStyleContext, nsnull, comboboxFrame);
|
|
|
|
|
1999-12-07 00:05:23 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, comboboxFrame,
|
|
|
|
aStyleContext, PR_FALSE);
|
2000-03-15 15:45:29 +00:00
|
|
|
if (hasGfxScrollbars && gDoGfxDropdown) {
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
|
|
// Combobox - New GFX Implementation
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
|
|
// Construct a frame-based list box
|
|
|
|
nsIComboboxControlFrame* comboBox = nsnull;
|
|
|
|
if (NS_SUCCEEDED(comboboxFrame->QueryInterface(kIComboboxControlFrameIID, (void**)&comboBox))) {
|
|
|
|
comboBox->SetFrameConstructor(this);
|
|
|
|
|
|
|
|
nsIFrame * listFrame;
|
|
|
|
rv = NS_NewGfxListControlFrame(aPresShell, &listFrame);
|
|
|
|
|
|
|
|
// Notify the listbox that it is being used as a dropdown list.
|
|
|
|
nsIListControlFrame * listControlFrame;
|
|
|
|
if (NS_SUCCEEDED(listFrame->QueryInterface(kIListControlFrameIID, (void**)&listControlFrame))) {
|
|
|
|
listControlFrame->SetComboboxFrame(comboboxFrame);
|
|
|
|
}
|
|
|
|
// Notify combobox that it should use the listbox as it's popup
|
|
|
|
comboBox->SetDropDown(listFrame);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIStyleContext> gfxListStyle;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent,
|
|
|
|
nsHTMLAtoms::dropDownListPseudo,
|
|
|
|
aStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(gfxListStyle));
|
|
|
|
|
|
|
|
// initialize the list control
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
comboboxFrame, gfxListStyle, nsnull, listFrame);
|
2000-03-18 14:25:02 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, listFrame,
|
|
|
|
gfxListStyle, PR_TRUE);
|
2000-03-18 14:25:02 +00:00
|
|
|
nsIView * view;
|
|
|
|
listFrame->GetView(aPresContext, &view);
|
|
|
|
if (view != nsnull) {
|
|
|
|
nsWidgetInitData widgetData;
|
|
|
|
view->SetFloating(PR_TRUE);
|
|
|
|
widgetData.mWindowType = eWindowType_popup;
|
|
|
|
widgetData.mBorderStyle = eBorderStyle_default;
|
2000-03-15 15:45:29 +00:00
|
|
|
|
|
|
|
#ifdef XP_MAC
|
2000-03-18 14:25:02 +00:00
|
|
|
static NS_DEFINE_IID(kCPopUpCID, NS_POPUP_CID);
|
|
|
|
view->CreateWidget(kCPopUpCID, &widgetData, nsnull);
|
2000-03-15 15:45:29 +00:00
|
|
|
#else
|
2000-03-18 14:25:02 +00:00
|
|
|
static NS_DEFINE_IID(kCChildCID, NS_CHILD_CID);
|
|
|
|
view->CreateWidget(kCChildCID, &widgetData, nsnull);
|
2000-03-15 15:45:29 +00:00
|
|
|
#endif
|
2000-03-18 14:25:02 +00:00
|
|
|
}
|
2000-03-15 15:45:29 +00:00
|
|
|
|
|
|
|
// create the area frame we are scrolling
|
|
|
|
PRUint32 flags = NS_BLOCK_SHRINK_WRAP | (aIsAbsolutelyPositioned?NS_BLOCK_SPACE_MGR:0);
|
|
|
|
nsIFrame* scrolledFrame = nsnull;
|
|
|
|
NS_NewSelectsAreaFrame(aPresShell, &scrolledFrame, flags);
|
|
|
|
|
|
|
|
// resolve a style for our gfx scrollframe based on the list frames style
|
|
|
|
// resolve a style for our gfx scrollframe based on the list frames style
|
|
|
|
nsCOMPtr<nsIStyleContext> scrollFrameStyle;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent,
|
|
|
|
nsLayoutAtoms::selectScrolledContentPseudo,
|
|
|
|
gfxListStyle, PR_FALSE,
|
|
|
|
getter_AddRefs(scrollFrameStyle));
|
2000-03-14 12:03:21 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
listFrame, scrollFrameStyle, nsnull, scrolledFrame);
|
|
|
|
|
|
|
|
// XXX Temporary for Bug 19416
|
|
|
|
{
|
|
|
|
nsIView * lstView;
|
|
|
|
scrolledFrame->GetView(aPresContext, &lstView);
|
|
|
|
if (lstView) {
|
|
|
|
lstView->IgnoreSetPosition(PR_TRUE);
|
|
|
|
}
|
2000-03-14 12:03:21 +00:00
|
|
|
}
|
2000-03-15 15:45:29 +00:00
|
|
|
// this is what is returned when the scrollframe is created.
|
|
|
|
// newScrollFrame - Either a gfx scrollframe or a native scrollframe that was created
|
|
|
|
// scrolledFrameStyleContext - The resolved style context of the scrolledframe you passed in.
|
|
|
|
// this is not the style of the scrollFrame.
|
1999-12-07 00:05:23 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
nsIFrame* newScrollFrame = nsnull;
|
2000-03-14 12:03:21 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
nsFrameItems anonChildItems;
|
|
|
|
// Create display and button frames from the combobox'es anonymous content
|
|
|
|
CreateAnonymousFrames(aPresShell, aPresContext, nsHTMLAtoms::combobox, aState, aContent, comboboxFrame,
|
|
|
|
anonChildItems);
|
2000-03-14 12:03:21 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
comboboxFrame->SetInitialChildList(aPresContext, nsnull,
|
|
|
|
anonChildItems.childList);
|
2000-03-14 12:03:21 +00:00
|
|
|
|
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
/* scroll frame */
|
2000-03-14 12:03:21 +00:00
|
|
|
|
|
|
|
#ifdef NEWGFX_LIST_SCROLLFRAME
|
2000-03-15 15:45:29 +00:00
|
|
|
nsIStyleContext * scrolledFrameStyleContext = nsnull;
|
2000-03-14 12:03:21 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
// ok take the style context, the Select area frame to scroll,the listFrame, and its parent
|
|
|
|
// and build the scrollframe.
|
|
|
|
BuildScrollFrame(aPresShell, aPresContext, aState, aContent, scrollFrameStyle, scrolledFrame,
|
|
|
|
listFrame, newScrollFrame, scrolledFrameStyleContext);
|
2000-03-14 12:03:21 +00:00
|
|
|
#else
|
2000-03-15 15:45:29 +00:00
|
|
|
newScrollFrame = scrolledFrame;
|
2000-03-14 12:03:21 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
// The area frame is a floater container
|
|
|
|
PRBool haveFirstLetterStyle, haveFirstLineStyle;
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aContent, aStyleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
|
|
|
aState.PushFloaterContainingBlock(scrolledFrame, floaterSaveState,
|
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
|
|
|
|
|
|
|
// Process children
|
|
|
|
nsFrameItems childItems;
|
|
|
|
|
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aContent, scrolledFrame, PR_FALSE,
|
|
|
|
childItems, PR_TRUE);
|
|
|
|
|
|
|
|
// if a select is being created with zero options we need to create
|
|
|
|
// a special pseudo frame so it can be sized as best it can
|
|
|
|
nsCOMPtr<nsIDOMHTMLSelectElement> selectElement;
|
|
|
|
nsresult result = aContent->QueryInterface(NS_GET_IID(nsIDOMHTMLSelectElement),
|
|
|
|
(void**)getter_AddRefs(selectElement));
|
|
|
|
if (NS_SUCCEEDED(result) && selectElement) {
|
|
|
|
PRUint32 numOptions = 0;
|
|
|
|
result = selectElement->GetLength(&numOptions);
|
|
|
|
if (NS_SUCCEEDED(result) && 0 == numOptions) {
|
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
nsIFrame* generatedFrame = nsnull;
|
|
|
|
scrolledFrame->GetStyleContext(getter_AddRefs(styleContext));
|
|
|
|
if (CreateGeneratedContentFrame(aPresShell, aPresContext, aState, scrolledFrame, aContent,
|
|
|
|
styleContext, nsLayoutAtoms::dummyOptionPseudo,
|
|
|
|
PR_FALSE, &generatedFrame)) {
|
|
|
|
// Add the generated frame to the child list
|
|
|
|
childItems.AddChild(generatedFrame);
|
|
|
|
}
|
2000-03-14 12:03:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
// Set the scrolled frame's initial child lists
|
|
|
|
scrolledFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
2000-03-14 12:03:21 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
// set the floated list
|
|
|
|
if (aState.mFloatedItems.childList) {
|
|
|
|
scrolledFrame->SetInitialChildList(aPresContext,
|
|
|
|
nsLayoutAtoms::floaterList,
|
|
|
|
aState.mFloatedItems.childList);
|
|
|
|
}
|
2000-03-14 12:03:21 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
// now we need to set the initial child list of the listFrame and this of course will be the gfx scrollframe
|
|
|
|
listFrame->SetInitialChildList(aPresContext, nsnull, newScrollFrame);
|
2000-03-14 12:03:21 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
// Initialize the additional popup child list which contains the dropdown list frame.
|
|
|
|
nsFrameItems popupItems;
|
|
|
|
popupItems.AddChild(listFrame);
|
|
|
|
comboboxFrame->SetInitialChildList(aPresContext, nsLayoutAtoms::popupList,
|
|
|
|
popupItems.childList);
|
|
|
|
|
|
|
|
// our new frame retured is the top frame which is the list frame.
|
|
|
|
//aNewFrame = listFrame;
|
|
|
|
|
|
|
|
// yes we have already initialized our frame
|
|
|
|
// Don't process, the children, They are already processed by the InitializeScrollFrame
|
|
|
|
// call above.
|
|
|
|
aProcessChildren = PR_FALSE;
|
|
|
|
aNewFrame = comboboxFrame;
|
|
|
|
aFrameHasBeenInitialized = PR_TRUE;
|
1999-04-12 22:14:31 +00:00
|
|
|
}
|
2000-03-15 15:45:29 +00:00
|
|
|
} else {
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
|
|
// Combobox - Old Native Implementation
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
|
|
nsIComboboxControlFrame* comboBox = nsnull;
|
|
|
|
if (NS_SUCCEEDED(comboboxFrame->QueryInterface(kIComboboxControlFrameIID, (void**)&comboBox))) {
|
|
|
|
comboBox->SetFrameConstructor(this);
|
|
|
|
|
|
|
|
// Create a listbox
|
|
|
|
nsIFrame * listFrame;
|
|
|
|
rv = NS_NewListControlFrame(aPresShell, &listFrame);
|
|
|
|
|
|
|
|
// Notify the listbox that it is being used as a dropdown list.
|
|
|
|
nsIListControlFrame * listControlFrame;
|
|
|
|
if (NS_SUCCEEDED(listFrame->QueryInterface(kIListControlFrameIID, (void**)&listControlFrame))) {
|
|
|
|
listControlFrame->SetComboboxFrame(comboboxFrame);
|
|
|
|
}
|
|
|
|
// Notify combobox that it should use the listbox as it's popup
|
|
|
|
comboBox->SetDropDown(listFrame);
|
1999-04-12 22:14:31 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
// Resolve psuedo element style for the dropdown list
|
|
|
|
nsCOMPtr<nsIStyleContext> listStyle;
|
|
|
|
rv = aPresContext->ResolvePseudoStyleContextFor(aContent,
|
|
|
|
nsHTMLAtoms::dropDownListPseudo,
|
|
|
|
aStyleContext,
|
|
|
|
PR_FALSE,
|
|
|
|
getter_AddRefs(listStyle));
|
|
|
|
|
|
|
|
// Initialize the scroll frame positioned. Note that it is NOT initialized as
|
|
|
|
// absolutely positioned.
|
|
|
|
nsIFrame* newFrame = nsnull;
|
|
|
|
nsIFrame* scrolledFrame = nsnull;
|
|
|
|
NS_NewSelectsAreaFrame(aPresShell, &scrolledFrame, flags);
|
|
|
|
|
|
|
|
InitializeSelectFrame(aPresShell, aPresContext, aState, listFrame, scrolledFrame, aContent, comboboxFrame,
|
|
|
|
listStyle, PR_FALSE, PR_FALSE, PR_TRUE);
|
|
|
|
|
|
|
|
newFrame = listFrame;
|
|
|
|
// XXX Temporary for Bug 19416
|
|
|
|
{
|
|
|
|
nsIView * lstView;
|
|
|
|
scrolledFrame->GetView(aPresContext, &lstView);
|
|
|
|
if (lstView) {
|
|
|
|
lstView->IgnoreSetPosition(PR_TRUE);
|
|
|
|
}
|
1999-12-14 22:21:58 +00:00
|
|
|
}
|
fixed bugs #6303, #6753, #6756, #6759
Re-wrote nsComboboxFrame.
removed the obsolete nsHTMLAtoms: comboText,comoTextSelected,comTextSelectedFocus,dropDownVisible,
dropdownHidden, dropDownBtnOut, dropDownBtnPressed,
Added nsHTMLAtoms::combobox, nsLayoutAtoms::popupList
Renamed dropDownList to dropDownListPseudo
Added "arrow.gif" as to be used the background-image for the combobox button
ua.css - added rules for select to differentiate between comboboxes and listboxes.
Added style rules to more closely match the XPTOOLKIT XPWidgets look.
removed the following :-moz-combobox-text, -moz-combobox-textselected
nsIFormControlFrame.h - Added SetSuggestedSize method.
nsButtonControlFrame - Implemented SetSuggestedSize.
nsCSSFrameConstructor.cpp - Rewrote ConstructSelectFrame.
nsIWidget.h -Added GetAbsoluteBounds method.
nsWindow.cpp - Implemented GetAbsoluteBounds.
1999-07-14 22:00:24 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
// Set flag so the events go to the listFrame not child frames.
|
|
|
|
// XXX: We should replace this with a real widget manager similar
|
|
|
|
// to how the nsFormControlFrame works. Re-directing events is a temporary Kludge.
|
|
|
|
nsIView *listView;
|
|
|
|
listFrame->GetView(aPresContext, &listView);
|
|
|
|
NS_ASSERTION(nsnull != listView,"ListFrame's view is nsnull");
|
|
|
|
nsIWidget * viewWidget;
|
|
|
|
listView->GetWidget(viewWidget);
|
|
|
|
//viewWidget->SetOverrideShow(PR_TRUE);
|
|
|
|
NS_RELEASE(viewWidget);
|
|
|
|
//listView->SetViewFlags(NS_VIEW_PUBLIC_FLAG_DONT_CHECK_CHILDREN);
|
|
|
|
|
|
|
|
nsIFrame* frame = nsnull;
|
|
|
|
if (NS_SUCCEEDED(comboboxFrame->QueryInterface(kIFrameIID, (void**)&frame))) {
|
|
|
|
nsFrameItems childItems;
|
1999-04-12 22:14:31 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
// Create display and button frames from the combobox'es anonymous content
|
|
|
|
CreateAnonymousFrames(aPresShell, aPresContext, nsHTMLAtoms::combobox, aState, aContent, frame,
|
|
|
|
childItems);
|
fixed bugs #6303, #6753, #6756, #6759
Re-wrote nsComboboxFrame.
removed the obsolete nsHTMLAtoms: comboText,comoTextSelected,comTextSelectedFocus,dropDownVisible,
dropdownHidden, dropDownBtnOut, dropDownBtnPressed,
Added nsHTMLAtoms::combobox, nsLayoutAtoms::popupList
Renamed dropDownList to dropDownListPseudo
Added "arrow.gif" as to be used the background-image for the combobox button
ua.css - added rules for select to differentiate between comboboxes and listboxes.
Added style rules to more closely match the XPTOOLKIT XPWidgets look.
removed the following :-moz-combobox-text, -moz-combobox-textselected
nsIFormControlFrame.h - Added SetSuggestedSize method.
nsButtonControlFrame - Implemented SetSuggestedSize.
nsCSSFrameConstructor.cpp - Rewrote ConstructSelectFrame.
nsIWidget.h -Added GetAbsoluteBounds method.
nsWindow.cpp - Implemented GetAbsoluteBounds.
1999-07-14 22:00:24 +00:00
|
|
|
|
2000-03-15 15:45:29 +00:00
|
|
|
frame->SetInitialChildList(aPresContext, nsnull,
|
|
|
|
childItems.childList);
|
|
|
|
|
|
|
|
// Initialize the additional popup child list which contains the dropdown list frame.
|
|
|
|
nsFrameItems popupItems;
|
|
|
|
popupItems.AddChild(listFrame);
|
|
|
|
frame->SetInitialChildList(aPresContext, nsLayoutAtoms::popupList,
|
|
|
|
popupItems.childList);
|
|
|
|
}
|
|
|
|
// Don't process, the children, They are already processed by the InitializeScrollFrame
|
|
|
|
// call above.
|
|
|
|
aProcessChildren = PR_FALSE;
|
|
|
|
aNewFrame = comboboxFrame;
|
|
|
|
aFrameHasBeenInitialized = PR_TRUE;
|
fixed bugs #6303, #6753, #6756, #6759
Re-wrote nsComboboxFrame.
removed the obsolete nsHTMLAtoms: comboText,comoTextSelected,comTextSelectedFocus,dropDownVisible,
dropdownHidden, dropDownBtnOut, dropDownBtnPressed,
Added nsHTMLAtoms::combobox, nsLayoutAtoms::popupList
Renamed dropDownList to dropDownListPseudo
Added "arrow.gif" as to be used the background-image for the combobox button
ua.css - added rules for select to differentiate between comboboxes and listboxes.
Added style rules to more closely match the XPTOOLKIT XPWidgets look.
removed the following :-moz-combobox-text, -moz-combobox-textselected
nsIFormControlFrame.h - Added SetSuggestedSize method.
nsButtonControlFrame - Implemented SetSuggestedSize.
nsCSSFrameConstructor.cpp - Rewrote ConstructSelectFrame.
nsIWidget.h -Added GetAbsoluteBounds method.
nsWindow.cpp - Implemented GetAbsoluteBounds.
1999-07-14 22:00:24 +00:00
|
|
|
}
|
|
|
|
}
|
2000-03-15 15:45:29 +00:00
|
|
|
} else if (hasGfxScrollbars && gDoGfxListbox) {
|
|
|
|
|
2000-03-14 12:03:21 +00:00
|
|
|
///////////////////////////////////////////////////////////////////
|
|
|
|
// ListBox - New GFX Implementation
|
|
|
|
///////////////////////////////////////////////////////////////////
|
2000-03-15 15:45:29 +00:00
|
|
|
// Construct a frame-based list box
|
1999-11-19 15:41:19 +00:00
|
|
|
nsIFrame * listFrame;
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
rv = NS_NewGfxListControlFrame(aPresShell, &listFrame);
|
1999-11-19 15:41:19 +00:00
|
|
|
|
|
|
|
// initialize the list control
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, listFrame);
|
1999-11-19 15:41:19 +00:00
|
|
|
|
|
|
|
// create the area frame we are scrolling
|
1999-12-05 20:43:26 +00:00
|
|
|
PRUint32 flags = NS_BLOCK_SHRINK_WRAP | (aIsAbsolutelyPositioned?NS_BLOCK_SPACE_MGR:0);
|
1999-11-19 15:41:19 +00:00
|
|
|
nsIFrame* scrolledFrame = nsnull;
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
NS_NewSelectsAreaFrame(aPresShell, &scrolledFrame, flags);
|
1999-11-19 15:41:19 +00:00
|
|
|
|
|
|
|
// resolve a style for our gfx scrollframe based on the list frames style
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> scrollFrameStyle;
|
1999-11-19 15:41:19 +00:00
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent,
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
nsLayoutAtoms::selectScrolledContentPseudo,
|
1999-11-19 15:41:19 +00:00
|
|
|
aStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(scrollFrameStyle));
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
listFrame, scrollFrameStyle, nsnull, scrolledFrame);
|
|
|
|
|
1999-11-19 15:41:19 +00:00
|
|
|
// this is what is returned when the scrollframe is created.
|
|
|
|
// newScrollFrame - Either a gfx scrollframe or a native scrollframe that was created
|
|
|
|
// scrolledFrameStyleContext - The resolved style context of the scrolledframe you passed in.
|
|
|
|
// this is not the style of the scrollFrame.
|
|
|
|
|
|
|
|
nsIFrame* newScrollFrame = nsnull;
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
|
|
|
|
/* scroll frame */
|
|
|
|
|
|
|
|
#ifdef NEWGFX_LIST_SCROLLFRAME
|
1999-11-19 15:41:19 +00:00
|
|
|
nsIStyleContext * scrolledFrameStyleContext = nsnull;
|
|
|
|
|
|
|
|
// ok take the style context, the Select area frame to scroll,the listFrame, and its parent
|
|
|
|
// and build the scrollframe.
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
BuildScrollFrame(aPresShell, aPresContext, aState, aContent, scrollFrameStyle, scrolledFrame,
|
1999-11-19 15:41:19 +00:00
|
|
|
listFrame, newScrollFrame, scrolledFrameStyleContext);
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
#else
|
|
|
|
newScrollFrame = scrolledFrame;
|
|
|
|
#endif
|
1999-11-19 15:41:19 +00:00
|
|
|
|
|
|
|
// The area frame is a floater container
|
|
|
|
PRBool haveFirstLetterStyle, haveFirstLineStyle;
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aContent, aStyleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
|
|
|
aState.PushFloaterContainingBlock(scrolledFrame, floaterSaveState,
|
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
|
|
|
|
|
|
|
// Process children
|
|
|
|
nsFrameItems childItems;
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aContent, scrolledFrame, PR_FALSE,
|
1999-11-19 15:41:19 +00:00
|
|
|
childItems, PR_TRUE);
|
|
|
|
|
|
|
|
// if a select is being created with zero options we need to create
|
|
|
|
// a special pseudo frame so it can be sized as best it can
|
|
|
|
nsCOMPtr<nsIDOMHTMLSelectElement> selectElement;
|
2000-02-02 22:24:56 +00:00
|
|
|
nsresult result = aContent->QueryInterface(NS_GET_IID(nsIDOMHTMLSelectElement),
|
1999-11-19 15:41:19 +00:00
|
|
|
(void**)getter_AddRefs(selectElement));
|
|
|
|
if (NS_SUCCEEDED(result) && selectElement) {
|
|
|
|
PRUint32 numOptions = 0;
|
|
|
|
result = selectElement->GetLength(&numOptions);
|
|
|
|
if (NS_SUCCEEDED(result) && 0 == numOptions) {
|
2000-02-23 21:32:23 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
1999-11-19 15:41:19 +00:00
|
|
|
nsIFrame* generatedFrame = nsnull;
|
2000-03-14 12:03:21 +00:00
|
|
|
scrolledFrame->GetStyleContext(getter_AddRefs(styleContext));
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
if (CreateGeneratedContentFrame(aPresShell, aPresContext, aState, scrolledFrame, aContent,
|
1999-11-19 15:41:19 +00:00
|
|
|
styleContext, nsLayoutAtoms::dummyOptionPseudo,
|
2000-03-14 12:03:21 +00:00
|
|
|
PR_FALSE, &generatedFrame)) {
|
1999-11-19 15:41:19 +00:00
|
|
|
// Add the generated frame to the child list
|
|
|
|
childItems.AddChild(generatedFrame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the scrolled frame's initial child lists
|
1999-11-24 06:03:41 +00:00
|
|
|
scrolledFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-11-19 15:41:19 +00:00
|
|
|
|
|
|
|
// set the floated list
|
|
|
|
if (aState.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
scrolledFrame->SetInitialChildList(aPresContext,
|
1999-11-19 15:41:19 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
|
|
|
aState.mFloatedItems.childList);
|
|
|
|
}
|
|
|
|
|
|
|
|
// now we need to set the initial child list of the listFrame and this of course will be the gfx scrollframe
|
1999-11-24 06:03:41 +00:00
|
|
|
listFrame->SetInitialChildList(aPresContext, nsnull, newScrollFrame);
|
1999-11-19 15:41:19 +00:00
|
|
|
|
|
|
|
// our new frame retured is the top frame which is the list frame.
|
|
|
|
aNewFrame = listFrame;
|
|
|
|
|
|
|
|
// yes we have already initialized our frame
|
|
|
|
aFrameHasBeenInitialized = PR_TRUE;
|
2000-03-15 15:45:29 +00:00
|
|
|
} else {
|
2000-03-14 12:03:21 +00:00
|
|
|
///////////////////////////////////////////////////////////////////
|
|
|
|
// ListBox - Old Native Implementation
|
|
|
|
///////////////////////////////////////////////////////////////////
|
1999-03-02 22:43:26 +00:00
|
|
|
nsIFrame * listFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewListControlFrame(aPresShell, &listFrame);
|
1999-03-02 22:43:26 +00:00
|
|
|
aNewFrame = listFrame;
|
1999-12-05 20:43:26 +00:00
|
|
|
|
|
|
|
PRUint32 flags = NS_BLOCK_SHRINK_WRAP | (aIsAbsolutelyPositioned?NS_BLOCK_SPACE_MGR:0);
|
1999-08-19 22:16:23 +00:00
|
|
|
nsIFrame* scrolledFrame = nsnull;
|
1999-12-05 20:43:26 +00:00
|
|
|
NS_NewSelectsAreaFrame(aPresShell, &scrolledFrame, flags);
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// ******* this code stolen from Initialze ScrollFrame ********
|
|
|
|
// please adjust this code to use BuildScrollFrame.
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
InitializeSelectFrame(aPresShell, aPresContext, aState, listFrame, scrolledFrame, aContent, aParentFrame,
|
1999-10-12 00:16:06 +00:00
|
|
|
aStyleContext, aIsAbsolutelyPositioned, aIsFixedPositioned, PR_FALSE);
|
1999-03-02 22:43:26 +00:00
|
|
|
|
1999-08-19 22:16:23 +00:00
|
|
|
aNewFrame = listFrame;
|
|
|
|
|
1999-04-12 22:14:31 +00:00
|
|
|
// Set flag so the events go to the listFrame not child frames.
|
|
|
|
// XXX: We should replace this with a real widget manager similar
|
|
|
|
// to how the nsFormControlFrame works.
|
|
|
|
// Re-directing events is a temporary Kludge.
|
1999-03-02 22:43:26 +00:00
|
|
|
nsIView *listView;
|
1999-10-26 04:44:41 +00:00
|
|
|
listFrame->GetView(aPresContext, &listView);
|
1999-03-02 22:43:26 +00:00
|
|
|
NS_ASSERTION(nsnull != listView,"ListFrame's view is nsnull");
|
1999-09-03 14:57:47 +00:00
|
|
|
//listView->SetViewFlags(NS_VIEW_PUBLIC_FLAG_DONT_CHECK_CHILDREN);
|
1999-03-02 22:43:26 +00:00
|
|
|
aFrameHasBeenInitialized = PR_TRUE;
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
} else {
|
2000-01-31 22:46:55 +00:00
|
|
|
NS_ASSERTION(0, "We longer support native widgets");
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-03-02 22:43:26 +00:00
|
|
|
}
|
|
|
|
else {
|
1999-04-12 22:14:31 +00:00
|
|
|
// Not frame based. Use a SelectFrame which creates a native widget.
|
2000-01-31 22:46:55 +00:00
|
|
|
NS_ASSERTION(0, "We longer support native widgets");
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
}
|
1999-03-02 22:43:26 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
/**
|
|
|
|
* Used to be InitializeScrollFrame but now its only used for the select tag
|
|
|
|
* But the select tag should really be fixed to use GFX scrollbars that can
|
|
|
|
* be create with BuildScrollFrame.
|
|
|
|
*/
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::InitializeSelectFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIFrame* scrollFrame,
|
|
|
|
nsIFrame* scrolledFrame,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
PRBool aIsAbsolutelyPositioned,
|
|
|
|
PRBool aIsFixedPositioned,
|
|
|
|
PRBool aCreateBlock)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
1999-10-12 00:16:06 +00:00
|
|
|
// Initialize it
|
|
|
|
nsIFrame* geometricParent = aParentFrame;
|
|
|
|
|
|
|
|
if (aIsAbsolutelyPositioned) {
|
|
|
|
geometricParent = aState.mAbsoluteItems.containingBlock;
|
|
|
|
} else if (aIsFixedPositioned) {
|
|
|
|
geometricParent = aState.mFixedItems.containingBlock;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
nsCOMPtr<nsIStyleContext> scrollPseudoStyle;
|
|
|
|
nsCOMPtr<nsIStyleContext> scrolledPseudoStyle;
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent,
|
|
|
|
nsLayoutAtoms::scrolledContentPseudo,
|
1999-10-14 14:55:08 +00:00
|
|
|
aStyleContext, PR_FALSE,
|
1999-10-12 00:16:06 +00:00
|
|
|
getter_AddRefs(scrolledPseudoStyle));
|
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
geometricParent, aStyleContext, nsnull, scrollFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
// Initialize the frame and force it to have a view
|
1999-10-12 00:16:06 +00:00
|
|
|
// the scrolled frame is anonymous and does not have a content node
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
scrollFrame, scrolledPseudoStyle, nsnull, scrolledFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, scrolledFrame,
|
1999-10-12 00:16:06 +00:00
|
|
|
scrolledPseudoStyle, PR_TRUE);
|
|
|
|
|
|
|
|
|
|
|
|
// The area frame is a floater container
|
|
|
|
PRBool haveFirstLetterStyle, haveFirstLineStyle;
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aContent, aStyleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
|
|
|
aState.PushFloaterContainingBlock(scrolledFrame, floaterSaveState,
|
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
|
|
|
|
|
|
|
// Process children
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
nsFrameItems childItems;
|
|
|
|
PRBool isPositionedContainingBlock = aIsAbsolutelyPositioned ||
|
|
|
|
aIsFixedPositioned;
|
|
|
|
|
|
|
|
if (isPositionedContainingBlock) {
|
|
|
|
// The area frame becomes a container for child frames that are
|
|
|
|
// absolutely positioned
|
|
|
|
aState.PushAbsoluteContainingBlock(scrolledFrame, absoluteSaveState);
|
|
|
|
}
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aContent, scrolledFrame, PR_FALSE,
|
1999-10-12 00:16:06 +00:00
|
|
|
childItems, PR_TRUE);
|
|
|
|
|
|
|
|
// if a select is being created with zero options we need to create
|
|
|
|
// a special pseudo frame so it can be sized as best it can
|
|
|
|
nsCOMPtr<nsIDOMHTMLSelectElement> selectElement;
|
2000-02-02 22:24:56 +00:00
|
|
|
nsresult result = aContent->QueryInterface(NS_GET_IID(nsIDOMHTMLSelectElement),
|
1999-10-12 00:16:06 +00:00
|
|
|
(void**)getter_AddRefs(selectElement));
|
|
|
|
if (NS_SUCCEEDED(result) && selectElement) {
|
|
|
|
PRUint32 numOptions = 0;
|
|
|
|
result = selectElement->GetLength(&numOptions);
|
|
|
|
if (NS_SUCCEEDED(result) && 0 == numOptions) {
|
2000-02-23 21:32:23 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIFrame* generatedFrame = nsnull;
|
2000-02-23 21:32:23 +00:00
|
|
|
scrolledFrame->GetStyleContext(getter_AddRefs(styleContext));
|
1999-12-04 23:49:50 +00:00
|
|
|
if (CreateGeneratedContentFrame(aPresShell, aPresContext, aState, scrolledFrame, aContent,
|
1999-10-12 00:16:06 +00:00
|
|
|
styleContext, nsLayoutAtoms::dummyOptionPseudo,
|
1999-10-29 14:39:48 +00:00
|
|
|
PR_FALSE, &generatedFrame)) {
|
1999-10-12 00:16:06 +00:00
|
|
|
// Add the generated frame to the child list
|
|
|
|
childItems.AddChild(generatedFrame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// Set the scrolled frame's initial child lists
|
1999-11-24 06:03:41 +00:00
|
|
|
scrolledFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-10-12 00:16:06 +00:00
|
|
|
if (isPositionedContainingBlock && aState.mAbsoluteItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
scrolledFrame->SetInitialChildList(aPresContext,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsLayoutAtoms::absoluteList,
|
|
|
|
aState.mAbsoluteItems.childList);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aState.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
scrolledFrame->SetInitialChildList(aPresContext,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
|
|
|
aState.mFloatedItems.childList);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the scroll frame's initial child list
|
1999-11-24 06:03:41 +00:00
|
|
|
scrollFrame->SetInitialChildList(aPresContext, nsnull, scrolledFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2000-02-14 01:42:09 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Used to be InitializeScrollFrame but now its only used for the select tag
|
|
|
|
* But the select tag should really be fixed to use GFX scrollbars that can
|
|
|
|
* be create with BuildScrollFrame.
|
|
|
|
*/
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::ConstructTitledBoxFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIAtom* aTag,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aNewFrame)
|
|
|
|
{
|
|
|
|
nsIFrame * newFrame;
|
|
|
|
nsresult rv = NS_NewTitledBoxFrame(aPresShell, &newFrame);
|
|
|
|
if (!NS_SUCCEEDED(rv)) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
|
|
|
|
|
|
|
// Initialize it
|
|
|
|
nsIFrame* geometricParent = aParentFrame;
|
|
|
|
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
geometricParent, aStyleContext, nsnull, newFrame);
|
|
|
|
|
|
|
|
// cache our display type
|
|
|
|
const nsStyleDisplay* styleDisplay;
|
|
|
|
newFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct*&) styleDisplay);
|
|
|
|
|
|
|
|
nsIFrame * boxFrame;
|
|
|
|
NS_NewTitledBoxInnerFrame(shell, &boxFrame);
|
|
|
|
|
|
|
|
|
|
|
|
// Resolve style and initialize the frame
|
|
|
|
nsIStyleContext* styleContext;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsXULAtoms::titledboxContentPseudo,
|
|
|
|
aStyleContext, PR_FALSE, &styleContext);
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, nsnull,
|
|
|
|
newFrame, styleContext, nsnull, boxFrame);
|
|
|
|
|
|
|
|
NS_RELEASE(styleContext);
|
|
|
|
|
|
|
|
nsFrameItems childItems;
|
|
|
|
|
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aContent, boxFrame, PR_FALSE,
|
|
|
|
childItems, PR_TRUE);
|
|
|
|
|
|
|
|
static NS_DEFINE_IID(kTitleFrameCID, NS_TITLE_FRAME_CID);
|
|
|
|
nsIFrame * child = childItems.childList;
|
|
|
|
nsIFrame * previous = nsnull;
|
|
|
|
nsIFrame* titleFrame = nsnull;
|
|
|
|
while (nsnull != child) {
|
|
|
|
nsresult result = child->QueryInterface(kTitleFrameCID, (void**)&titleFrame);
|
|
|
|
if (NS_SUCCEEDED(result) && titleFrame) {
|
|
|
|
if (nsnull != previous) {
|
|
|
|
nsIFrame * nxt;
|
|
|
|
titleFrame->GetNextSibling(&nxt);
|
|
|
|
previous->SetNextSibling(nxt);
|
|
|
|
titleFrame->SetNextSibling(boxFrame);
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
nsIFrame * nxt;
|
|
|
|
titleFrame->GetNextSibling(&nxt);
|
|
|
|
childItems.childList = nxt;
|
|
|
|
titleFrame->SetNextSibling(boxFrame);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
previous = child;
|
|
|
|
child->GetNextSibling(&child);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the scrolled frame's initial child lists
|
|
|
|
boxFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
|
|
|
|
|
|
|
if (titleFrame)
|
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, titleFrame);
|
|
|
|
else
|
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, boxFrame);
|
|
|
|
|
|
|
|
// our new frame retured is the top frame which is the list frame.
|
|
|
|
aNewFrame = newFrame;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-01-09 02:04:36 +00:00
|
|
|
/**
|
|
|
|
* Used to be InitializeScrollFrame but now its only used for the select tag
|
|
|
|
* But the select tag should really be fixed to use GFX scrollbars that can
|
|
|
|
* be create with BuildScrollFrame.
|
|
|
|
*/
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::ConstructFieldSetFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIAtom* aTag,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aNewFrame,
|
|
|
|
PRBool& aProcessChildren,
|
|
|
|
PRBool aIsAbsolutelyPositioned,
|
|
|
|
PRBool& aFrameHasBeenInitialized,
|
|
|
|
PRBool aIsFixedPositioned)
|
|
|
|
{
|
|
|
|
nsIFrame * newFrame;
|
2000-01-18 15:40:25 +00:00
|
|
|
PRUint32 flags = aIsAbsolutelyPositioned ? NS_BLOCK_SPACE_MGR : 0;
|
|
|
|
nsresult rv = NS_NewFieldSetFrame(aPresShell, &newFrame, flags);
|
2000-01-31 14:26:25 +00:00
|
|
|
if (!NS_SUCCEEDED(rv)) {
|
2000-01-31 14:04:41 +00:00
|
|
|
return rv;
|
|
|
|
}
|
2000-01-09 02:04:36 +00:00
|
|
|
|
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
|
|
|
|
|
|
|
// Initialize it
|
|
|
|
nsIFrame* geometricParent = aParentFrame;
|
|
|
|
|
|
|
|
if (aIsAbsolutelyPositioned) {
|
|
|
|
geometricParent = aState.mAbsoluteItems.containingBlock;
|
|
|
|
} else if (aIsFixedPositioned) {
|
|
|
|
geometricParent = aState.mFixedItems.containingBlock;
|
|
|
|
}
|
|
|
|
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
geometricParent, aStyleContext, nsnull, newFrame);
|
|
|
|
|
|
|
|
// See if we need to create a view, e.g. the frame is absolutely
|
|
|
|
// positioned
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
|
|
|
aStyleContext, PR_FALSE);
|
|
|
|
|
|
|
|
// cache our display type
|
|
|
|
const nsStyleDisplay* styleDisplay;
|
|
|
|
newFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct*&) styleDisplay);
|
2000-02-10 21:36:28 +00:00
|
|
|
|
|
|
|
/*
|
2000-01-18 15:40:25 +00:00
|
|
|
PRUint32 childFlags = (NS_STYLE_DISPLAY_BLOCK != styleDisplay->mDisplay) ? NS_BLOCK_SHRINK_WRAP : 0;
|
|
|
|
// inherit the FieldSet state
|
|
|
|
PRUint32 parentState;
|
|
|
|
newFrame->GetFrameState( &parentState );
|
|
|
|
childFlags |= parentState;
|
2000-02-10 21:36:28 +00:00
|
|
|
childFlags |= NS_BLOCK_SHRINK_WRAP;
|
|
|
|
*/
|
2000-01-09 02:04:36 +00:00
|
|
|
|
|
|
|
nsIFrame * areaFrame;
|
2000-02-10 21:36:28 +00:00
|
|
|
// NS_NewAreaFrame(shell, &areaFrame, childFlags);
|
|
|
|
|
|
|
|
|
|
|
|
NS_NewBlockFrame(shell, &areaFrame);
|
2000-01-09 02:04:36 +00:00
|
|
|
|
|
|
|
// Resolve style and initialize the frame
|
|
|
|
nsIStyleContext* styleContext;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::fieldsetContentPseudo,
|
|
|
|
aStyleContext, PR_FALSE, &styleContext);
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
newFrame, styleContext, nsnull, areaFrame);
|
|
|
|
|
|
|
|
NS_RELEASE(styleContext);
|
|
|
|
|
|
|
|
|
|
|
|
// The area frame is a floater container
|
|
|
|
PRBool haveFirstLetterStyle, haveFirstLineStyle;
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aContent, aStyleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
|
|
|
aState.PushFloaterContainingBlock(areaFrame, floaterSaveState,
|
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
|
|
|
|
|
|
|
// Process children
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
nsFrameItems childItems;
|
|
|
|
PRBool isPositionedContainingBlock = aIsAbsolutelyPositioned ||
|
|
|
|
aIsFixedPositioned;
|
|
|
|
|
|
|
|
if (isPositionedContainingBlock) {
|
|
|
|
// The area frame becomes a container for child frames that are
|
|
|
|
// absolutely positioned
|
|
|
|
aState.PushAbsoluteContainingBlock(areaFrame, absoluteSaveState);
|
|
|
|
}
|
|
|
|
|
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aContent, areaFrame, PR_FALSE,
|
|
|
|
childItems, PR_TRUE);
|
|
|
|
|
|
|
|
static NS_DEFINE_IID(kLegendFrameCID, NS_LEGEND_FRAME_CID);
|
|
|
|
nsIFrame * child = childItems.childList;
|
|
|
|
nsIFrame * previous = nsnull;
|
|
|
|
nsIFrame* legendFrame = nsnull;
|
|
|
|
while (nsnull != child) {
|
|
|
|
nsresult result = child->QueryInterface(kLegendFrameCID, (void**)&legendFrame);
|
|
|
|
if (NS_SUCCEEDED(result) && legendFrame) {
|
|
|
|
if (nsnull != previous) {
|
|
|
|
nsIFrame * nxt;
|
|
|
|
legendFrame->GetNextSibling(&nxt);
|
|
|
|
previous->SetNextSibling(nxt);
|
|
|
|
areaFrame->SetNextSibling(legendFrame);
|
|
|
|
legendFrame->SetParent(newFrame);
|
|
|
|
legendFrame->SetNextSibling(nsnull);
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
nsIFrame * nxt;
|
|
|
|
legendFrame->GetNextSibling(&nxt);
|
|
|
|
childItems.childList = nxt;
|
|
|
|
areaFrame->SetNextSibling(legendFrame);
|
|
|
|
legendFrame->SetParent(newFrame);
|
|
|
|
legendFrame->SetNextSibling(nsnull);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2000-03-21 21:40:23 +00:00
|
|
|
previous = child;
|
2000-01-09 02:04:36 +00:00
|
|
|
child->GetNextSibling(&child);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the scrolled frame's initial child lists
|
|
|
|
areaFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
|
|
|
if (isPositionedContainingBlock && aState.mAbsoluteItems.childList) {
|
|
|
|
areaFrame->SetInitialChildList(aPresContext,
|
|
|
|
nsLayoutAtoms::absoluteList,
|
|
|
|
aState.mAbsoluteItems.childList);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aState.mFloatedItems.childList) {
|
|
|
|
areaFrame->SetInitialChildList(aPresContext,
|
|
|
|
nsLayoutAtoms::floaterList,
|
|
|
|
aState.mFloatedItems.childList);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the scroll frame's initial child list
|
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, areaFrame);
|
|
|
|
|
|
|
|
// our new frame retured is the top frame which is the list frame.
|
|
|
|
aNewFrame = newFrame;
|
|
|
|
|
|
|
|
// yes we have already initialized our frame
|
|
|
|
aFrameHasBeenInitialized = PR_TRUE;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
|
|
|
nsIFrame* newChildList = aChildList;
|
|
|
|
|
|
|
|
// Set the geometric and content parent for each of the child frames
|
|
|
|
// that will go into the area frame's child list.
|
|
|
|
// The legend frame does not go into the list
|
|
|
|
nsIFrame* lastNewFrame = nsnull;
|
|
|
|
for (nsIFrame* frame = aChildList; nsnull != frame;) {
|
|
|
|
nsIFrame* legendFrame = nsnull;
|
|
|
|
nsresult result = frame->QueryInterface(kLegendFrameCID, (void**)&legendFrame);
|
|
|
|
if (NS_SUCCEEDED(result) && legendFrame) {
|
|
|
|
if (mLegendFrame) { // we already have a legend, destroy it
|
|
|
|
frame->GetNextSibling(&frame);
|
|
|
|
if (lastNewFrame) {
|
|
|
|
lastNewFrame->SetNextSibling(frame);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
aChildList = frame;
|
|
|
|
}
|
|
|
|
legendFrame->Destroy(aPresContext);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
nsIFrame* nextFrame;
|
|
|
|
frame->GetNextSibling(&nextFrame);
|
|
|
|
if (lastNewFrame) {
|
|
|
|
lastNewFrame->SetNextSibling(nextFrame);
|
|
|
|
} else {
|
|
|
|
newChildList = nextFrame;
|
|
|
|
}
|
|
|
|
frame->SetParent(this);
|
|
|
|
mFrames.FirstChild()->SetNextSibling(frame);
|
|
|
|
mLegendFrame = frame;
|
|
|
|
mLegendFrame->SetNextSibling(nsnull);
|
|
|
|
frame = nextFrame;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
frame->SetParent(mFrames.FirstChild());
|
|
|
|
lastNewFrame = frame;
|
|
|
|
frame->GetNextSibling(&frame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Queue up the frames for the content frame
|
|
|
|
return mFrames.FirstChild()->SetInitialChildList(aPresContext, nsnull, newChildList);
|
|
|
|
#endif
|
|
|
|
}
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructFrameByTag(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIAtom* aTag,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsFrameItems& aFrameItems)
|
|
|
|
{
|
|
|
|
PRBool processChildren = PR_FALSE; // whether we should process child content
|
|
|
|
PRBool isAbsolutelyPositioned = PR_FALSE;
|
|
|
|
PRBool isFixedPositioned = PR_FALSE;
|
|
|
|
PRBool isFloating = PR_FALSE;
|
|
|
|
PRBool canBePositioned = PR_TRUE;
|
|
|
|
PRBool frameHasBeenInitialized = PR_FALSE;
|
|
|
|
nsIFrame* newFrame = nsnull; // the frame we construct
|
|
|
|
PRBool isReplaced = PR_FALSE;
|
|
|
|
PRBool addToHashTable = PR_TRUE;
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
if (nsLayoutAtoms::textTagName == aTag) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewTextFrame(aPresShell, &newFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
// Text frames don't go in the content->frame hash table, because
|
|
|
|
// they're anonymous. This keeps the hash table smaller
|
|
|
|
addToHashTable = PR_FALSE;
|
|
|
|
isReplaced = PR_TRUE; // XXX kipp: temporary
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
nsIHTMLContent *htmlContent;
|
|
|
|
|
|
|
|
// Ignore the tag if it's not HTML content
|
|
|
|
if (NS_SUCCEEDED(aContent->QueryInterface(kIHTMLContentIID, (void **)&htmlContent))) {
|
|
|
|
NS_RELEASE(htmlContent);
|
|
|
|
|
|
|
|
// See if the element is absolute or fixed positioned
|
|
|
|
const nsStylePosition* position = (const nsStylePosition*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Position);
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
if (NS_STYLE_POSITION_ABSOLUTE == position->mPosition) {
|
|
|
|
isAbsolutelyPositioned = PR_TRUE;
|
|
|
|
}
|
|
|
|
else if (NS_STYLE_POSITION_FIXED == position->mPosition) {
|
|
|
|
isFixedPositioned = PR_TRUE;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-26 17:11:54 +00:00
|
|
|
else if (NS_STYLE_FLOAT_NONE != display->mFloats) {
|
|
|
|
isFloating = PR_TRUE;
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Create a frame based on the tag
|
|
|
|
if (nsHTMLAtoms::img == aTag) {
|
1999-02-13 05:59:19 +00:00
|
|
|
isReplaced = PR_TRUE;
|
1999-02-06 03:45:11 +00:00
|
|
|
// XXX If image display is turned off, then use ConstructAlternateImageFrame()
|
|
|
|
// instead...
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewImageFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::hr == aTag) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewHRFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::br == aTag) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewBRFrame(aPresShell, &newFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
isReplaced = PR_TRUE;
|
1999-07-06 03:52:33 +00:00
|
|
|
// BR frames don't go in the content->frame hash table: typically
|
|
|
|
// there are many BR content objects and this would increase the size
|
|
|
|
// of the hash table, and it's doubtful we need the mapping anyway
|
|
|
|
addToHashTable = PR_FALSE;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::wbr == aTag) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewWBRFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::input == aTag) {
|
1999-02-13 05:59:19 +00:00
|
|
|
isReplaced = PR_TRUE;
|
2000-02-23 21:36:29 +00:00
|
|
|
rv = CreateInputFrame(aPresShell, aPresContext, aContent, newFrame, aStyleContext);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::textarea == aTag) {
|
1999-02-13 05:59:19 +00:00
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTextControlFrame(aPresShell, aPresContext, newFrame, aContent);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::select == aTag) {
|
1999-02-13 05:59:19 +00:00
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructSelectFrame(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
aTag, aStyleContext, newFrame, processChildren,
|
|
|
|
isAbsolutelyPositioned, frameHasBeenInitialized,
|
|
|
|
isFixedPositioned, aFrameItems);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::applet == aTag) {
|
1999-02-13 05:59:19 +00:00
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewObjectFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::embed == aTag) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewObjectFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::fieldset == aTag) {
|
2000-01-31 22:46:55 +00:00
|
|
|
#define DO_NEWFIELDSET
|
2000-01-09 02:04:36 +00:00
|
|
|
#ifdef DO_NEWFIELDSET
|
|
|
|
rv = ConstructFieldSetFrame(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
|
|
|
aTag, aStyleContext, newFrame, processChildren,
|
|
|
|
isAbsolutelyPositioned, frameHasBeenInitialized,
|
|
|
|
isFixedPositioned);
|
|
|
|
processChildren = PR_FALSE;
|
|
|
|
#else
|
2000-01-18 15:40:25 +00:00
|
|
|
rv = NS_NewFieldSetFrame(aPresShell, &newFrame, isAbsolutelyPositioned ? NS_BLOCK_SPACE_MGR : 0);
|
1999-02-05 03:55:18 +00:00
|
|
|
processChildren = PR_TRUE;
|
2000-01-09 02:04:36 +00:00
|
|
|
#endif
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::legend == aTag) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewLegendFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
processChildren = PR_TRUE;
|
|
|
|
canBePositioned = PR_FALSE;
|
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::object == aTag) {
|
1999-02-13 05:59:19 +00:00
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewObjectFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::form == aTag) {
|
2000-03-15 04:38:08 +00:00
|
|
|
PRBool isOutOfFlow = isFloating || isAbsolutelyPositioned || isFixedPositioned;
|
|
|
|
|
|
|
|
rv = NS_NewFormFrame(aPresShell, &newFrame,
|
|
|
|
isOutOfFlow ? NS_BLOCK_SPACE_MGR|NS_BLOCK_MARGIN_ROOT : 0);
|
1999-07-22 23:28:16 +00:00
|
|
|
processChildren = PR_TRUE;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::frameset == aTag) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewHTMLFramesetFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
canBePositioned = PR_FALSE;
|
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::iframe == aTag) {
|
1999-02-13 05:59:19 +00:00
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewHTMLFrameOuterFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::spacer == aTag) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewSpacerFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
canBePositioned = PR_FALSE;
|
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::button == aTag) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewHTMLButtonControlFrame(aPresShell, &newFrame);
|
1999-03-06 19:43:13 +00:00
|
|
|
// the html4 button needs to act just like a
|
|
|
|
// regular button except contain html content
|
|
|
|
// so it must be replaced or html outside it will
|
|
|
|
// draw into its borders. -EDV
|
|
|
|
isReplaced = PR_TRUE;
|
1999-02-05 03:55:18 +00:00
|
|
|
processChildren = PR_TRUE;
|
|
|
|
}
|
|
|
|
else if (nsHTMLAtoms::label == aTag) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewLabelFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
processChildren = PR_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we succeeded in creating a frame then initialize it, process its
|
|
|
|
// children (if requested), and set the initial child list
|
|
|
|
if (NS_SUCCEEDED(rv) && (nsnull != newFrame)) {
|
1999-02-13 05:59:19 +00:00
|
|
|
// If the frame is a replaced element, then set the frame state bit
|
|
|
|
if (isReplaced) {
|
|
|
|
nsFrameState state;
|
|
|
|
newFrame->GetFrameState(&state);
|
|
|
|
newFrame->SetFrameState(state | NS_FRAME_REPLACED_ELEMENT);
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
if (!frameHasBeenInitialized) {
|
|
|
|
nsIFrame* geometricParent = aParentFrame;
|
|
|
|
|
1999-07-16 18:17:14 +00:00
|
|
|
// Makes sure we use the correct parent frame pointer when initializing
|
|
|
|
// the frame
|
|
|
|
if (isFloating) {
|
|
|
|
geometricParent = aState.mFloatedItems.containingBlock;
|
|
|
|
|
|
|
|
} else if (canBePositioned) {
|
|
|
|
if (isAbsolutelyPositioned) {
|
1999-04-28 19:08:14 +00:00
|
|
|
geometricParent = aState.mAbsoluteItems.containingBlock;
|
1999-02-05 03:55:18 +00:00
|
|
|
} else if (isFixedPositioned) {
|
1999-04-28 19:08:14 +00:00
|
|
|
geometricParent = aState.mFixedItems.containingBlock;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
geometricParent, aStyleContext, nsnull, newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
// See if we need to create a view, e.g. the frame is absolutely
|
|
|
|
// positioned
|
1999-11-24 06:03:41 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-05 03:55:18 +00:00
|
|
|
aStyleContext, PR_FALSE);
|
|
|
|
|
|
|
|
// Process the child content if requested
|
|
|
|
nsFrameItems childItems;
|
|
|
|
if (processChildren) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ProcessChildren(aPresShell, aPresContext, aState, aContent, newFrame,
|
1999-08-27 21:46:10 +00:00
|
|
|
PR_TRUE, childItems, PR_FALSE);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
1999-06-30 22:17:43 +00:00
|
|
|
// if there are any anonymous children create frames for them
|
1999-12-04 23:49:50 +00:00
|
|
|
CreateAnonymousFrames(aPresShell, aPresContext, aTag, aState, aContent, newFrame,
|
1999-08-27 21:46:10 +00:00
|
|
|
childItems);
|
1999-06-30 22:17:43 +00:00
|
|
|
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
// Set the frame's initial child list
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
1999-03-30 15:22:54 +00:00
|
|
|
// If the frame is positioned, then create a placeholder frame
|
1999-07-16 18:17:14 +00:00
|
|
|
if (canBePositioned && (isAbsolutelyPositioned || isFixedPositioned)) {
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* placeholderFrame;
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
CreatePlaceholderFrameFor(aPresShell, aPresContext, aState.mFrameManager, aContent,
|
1999-08-05 03:09:22 +00:00
|
|
|
newFrame, aStyleContext, aParentFrame, &placeholderFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
// Add the positioned frame to its containing block's list of
|
|
|
|
// child frames
|
1999-02-05 03:55:18 +00:00
|
|
|
if (isAbsolutelyPositioned) {
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mAbsoluteItems.AddChild(newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
} else {
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mFixedItems.AddChild(newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add the placeholder frame to the flow
|
|
|
|
aFrameItems.AddChild(placeholderFrame);
|
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
} else if (isFloating) {
|
|
|
|
nsIFrame* placeholderFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
CreatePlaceholderFrameFor(aPresShell, aPresContext, aState.mFrameManager, aContent, newFrame,
|
1999-04-25 16:58:42 +00:00
|
|
|
aStyleContext, aParentFrame, &placeholderFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
|
|
|
|
// Add the floating frame to its containing block's list of child frames
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mFloatedItems.AddChild(newFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
|
|
|
|
// Add the placeholder frame to the flow
|
|
|
|
aFrameItems.AddChild(placeholderFrame);
|
1999-07-01 14:34:35 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
} else {
|
|
|
|
// Add the newly constructed frame to the flow
|
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
}
|
1999-07-01 14:34:35 +00:00
|
|
|
|
|
|
|
if (addToHashTable) {
|
|
|
|
// Add a mapping from content object to primary frame. Note that for
|
|
|
|
// floated and positioned frames this is the out-of-flow frame and not
|
|
|
|
// the placeholder frame
|
1999-08-05 03:09:22 +00:00
|
|
|
aState.mFrameManager->SetPrimaryFrameFor(aContent, newFrame);
|
1999-07-01 14:34:35 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-06-30 22:17:43 +00:00
|
|
|
// after the node has been constructed and initialized create any
|
|
|
|
// anonymous content a node needs.
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::CreateAnonymousFrames(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-06-30 22:17:43 +00:00
|
|
|
nsIAtom* aTag,
|
|
|
|
nsFrameConstructorState& aState,
|
1999-08-27 06:06:39 +00:00
|
|
|
nsIContent* aParent,
|
1999-06-30 22:17:43 +00:00
|
|
|
nsIFrame* aNewFrame,
|
|
|
|
nsFrameItems& aChildItems)
|
|
|
|
{
|
2000-02-15 09:28:28 +00:00
|
|
|
if (aTag == nsXULAtoms::treecell)
|
|
|
|
return NS_OK; // Don't even allow the XBL check. The inner cell frame throws it off.
|
|
|
|
// There's a separate special method for XBL treecells.
|
|
|
|
|
2000-01-11 03:40:41 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
aNewFrame->GetStyleContext(getter_AddRefs(styleContext));
|
|
|
|
|
|
|
|
const nsStyleUserInterface* ui= (const nsStyleUserInterface*)
|
|
|
|
styleContext->GetStyleData(eStyleStruct_UserInterface);
|
|
|
|
|
2000-01-11 08:25:59 +00:00
|
|
|
if (ui->mBehavior != "") {
|
|
|
|
// Get the XBL loader.
|
|
|
|
nsresult rv;
|
2000-01-12 10:27:57 +00:00
|
|
|
NS_WITH_SERVICE(nsIXBLService, xblService, "component://netscape/xbl", &rv);
|
2000-01-11 08:25:59 +00:00
|
|
|
if (!xblService)
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
// Load the bindings.
|
|
|
|
xblService->LoadBindings(aParent, ui->mBehavior);
|
|
|
|
|
|
|
|
// Retrieve the anonymous content that we should build.
|
|
|
|
nsCOMPtr<nsISupportsArray> anonymousItems;
|
2000-03-14 11:09:46 +00:00
|
|
|
nsCOMPtr<nsIContent> childElement;
|
|
|
|
xblService->GetContentList(aParent, getter_AddRefs(anonymousItems), getter_AddRefs(childElement));
|
2000-01-11 08:25:59 +00:00
|
|
|
|
|
|
|
if (!anonymousItems)
|
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
// Build the frames for the anonymous content.
|
|
|
|
PRUint32 count = 0;
|
|
|
|
anonymousItems->Count(&count);
|
|
|
|
|
|
|
|
for (PRUint32 i=0; i < count; i++)
|
|
|
|
{
|
|
|
|
// get our child's content and set its parent to our content
|
|
|
|
nsCOMPtr<nsISupports> node;
|
|
|
|
anonymousItems->GetElementAt(i,getter_AddRefs(node));
|
|
|
|
|
|
|
|
nsCOMPtr<nsIContent> content(do_QueryInterface(node));
|
|
|
|
|
|
|
|
// create the frame and attach it to our frame
|
|
|
|
ConstructFrame(aPresShell, aPresContext, aState, content, aNewFrame, aChildItems);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2000-01-11 03:40:41 +00:00
|
|
|
|
1999-08-27 06:06:39 +00:00
|
|
|
// only these tags types can have anonymous content. We do this check for performance
|
1999-06-30 22:17:43 +00:00
|
|
|
// reasons. If we did a query interface on every tag it would be very inefficient.
|
|
|
|
if (aTag != nsHTMLAtoms::input &&
|
fixed bugs #6303, #6753, #6756, #6759
Re-wrote nsComboboxFrame.
removed the obsolete nsHTMLAtoms: comboText,comoTextSelected,comTextSelectedFocus,dropDownVisible,
dropdownHidden, dropDownBtnOut, dropDownBtnPressed,
Added nsHTMLAtoms::combobox, nsLayoutAtoms::popupList
Renamed dropDownList to dropDownListPseudo
Added "arrow.gif" as to be used the background-image for the combobox button
ua.css - added rules for select to differentiate between comboboxes and listboxes.
Added style rules to more closely match the XPTOOLKIT XPWidgets look.
removed the following :-moz-combobox-text, -moz-combobox-textselected
nsIFormControlFrame.h - Added SetSuggestedSize method.
nsButtonControlFrame - Implemented SetSuggestedSize.
nsCSSFrameConstructor.cpp - Rewrote ConstructSelectFrame.
nsIWidget.h -Added GetAbsoluteBounds method.
nsWindow.cpp - Implemented GetAbsoluteBounds.
1999-07-14 22:00:24 +00:00
|
|
|
aTag != nsHTMLAtoms::combobox &&
|
1999-06-30 22:17:43 +00:00
|
|
|
aTag != nsXULAtoms::slider &&
|
|
|
|
aTag != nsXULAtoms::splitter &&
|
1999-07-18 03:16:58 +00:00
|
|
|
aTag != nsXULAtoms::scrollbar &&
|
1999-07-31 11:29:03 +00:00
|
|
|
aTag != nsXULAtoms::menu &&
|
1999-08-27 06:06:39 +00:00
|
|
|
aTag != nsXULAtoms::menuitem
|
fixed bugs #6303, #6753, #6756, #6759
Re-wrote nsComboboxFrame.
removed the obsolete nsHTMLAtoms: comboText,comoTextSelected,comTextSelectedFocus,dropDownVisible,
dropdownHidden, dropDownBtnOut, dropDownBtnPressed,
Added nsHTMLAtoms::combobox, nsLayoutAtoms::popupList
Renamed dropDownList to dropDownListPseudo
Added "arrow.gif" as to be used the background-image for the combobox button
ua.css - added rules for select to differentiate between comboboxes and listboxes.
Added style rules to more closely match the XPTOOLKIT XPWidgets look.
removed the following :-moz-combobox-text, -moz-combobox-textselected
nsIFormControlFrame.h - Added SetSuggestedSize method.
nsButtonControlFrame - Implemented SetSuggestedSize.
nsCSSFrameConstructor.cpp - Rewrote ConstructSelectFrame.
nsIWidget.h -Added GetAbsoluteBounds method.
nsWindow.cpp - Implemented GetAbsoluteBounds.
1999-07-14 22:00:24 +00:00
|
|
|
) {
|
1999-06-30 22:17:43 +00:00
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
}
|
1999-08-27 06:06:39 +00:00
|
|
|
|
fixed bugs #6303, #6753, #6756, #6759
Re-wrote nsComboboxFrame.
removed the obsolete nsHTMLAtoms: comboText,comoTextSelected,comTextSelectedFocus,dropDownVisible,
dropdownHidden, dropDownBtnOut, dropDownBtnPressed,
Added nsHTMLAtoms::combobox, nsLayoutAtoms::popupList
Renamed dropDownList to dropDownListPseudo
Added "arrow.gif" as to be used the background-image for the combobox button
ua.css - added rules for select to differentiate between comboboxes and listboxes.
Added style rules to more closely match the XPTOOLKIT XPWidgets look.
removed the following :-moz-combobox-text, -moz-combobox-textselected
nsIFormControlFrame.h - Added SetSuggestedSize method.
nsButtonControlFrame - Implemented SetSuggestedSize.
nsCSSFrameConstructor.cpp - Rewrote ConstructSelectFrame.
nsIWidget.h -Added GetAbsoluteBounds method.
nsWindow.cpp - Implemented GetAbsoluteBounds.
1999-07-14 22:00:24 +00:00
|
|
|
// get the document
|
1999-09-20 00:02:59 +00:00
|
|
|
nsCOMPtr<nsIDocument> doc;
|
|
|
|
nsresult rv = aParent->GetDocument(*getter_AddRefs(doc));
|
fixed bugs #6303, #6753, #6756, #6759
Re-wrote nsComboboxFrame.
removed the obsolete nsHTMLAtoms: comboText,comoTextSelected,comTextSelectedFocus,dropDownVisible,
dropdownHidden, dropDownBtnOut, dropDownBtnPressed,
Added nsHTMLAtoms::combobox, nsLayoutAtoms::popupList
Renamed dropDownList to dropDownListPseudo
Added "arrow.gif" as to be used the background-image for the combobox button
ua.css - added rules for select to differentiate between comboboxes and listboxes.
Added style rules to more closely match the XPTOOLKIT XPWidgets look.
removed the following :-moz-combobox-text, -moz-combobox-textselected
nsIFormControlFrame.h - Added SetSuggestedSize method.
nsButtonControlFrame - Implemented SetSuggestedSize.
nsCSSFrameConstructor.cpp - Rewrote ConstructSelectFrame.
nsIWidget.h -Added GetAbsoluteBounds method.
nsWindow.cpp - Implemented GetAbsoluteBounds.
1999-07-14 22:00:24 +00:00
|
|
|
if (NS_FAILED(rv) || !doc)
|
|
|
|
return rv;
|
1999-06-30 22:17:43 +00:00
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
return CreateAnonymousFrames(aPresShell, aPresContext, aState, aParent, doc, aNewFrame, aChildItems);
|
1999-08-27 06:06:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// after the node has been constructed and initialized create any
|
|
|
|
// anonymous content a node needs.
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::CreateAnonymousFrames(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-08-27 06:06:39 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aParent,
|
|
|
|
nsIDocument* aDocument,
|
|
|
|
nsIFrame* aNewFrame,
|
|
|
|
nsFrameItems& aChildItems)
|
|
|
|
{
|
|
|
|
|
|
|
|
nsCOMPtr<nsIAnonymousContentCreator> creator(do_QueryInterface(aNewFrame));
|
|
|
|
|
|
|
|
if (!creator)
|
1999-06-30 22:17:43 +00:00
|
|
|
return NS_OK;
|
|
|
|
|
1999-08-27 06:06:39 +00:00
|
|
|
nsCOMPtr<nsISupportsArray> anonymousItems;
|
|
|
|
NS_NewISupportsArray(getter_AddRefs(anonymousItems));
|
|
|
|
|
|
|
|
|
2000-01-22 01:16:50 +00:00
|
|
|
creator->CreateAnonymousContent(aPresContext, *anonymousItems);
|
1999-08-27 06:06:39 +00:00
|
|
|
|
|
|
|
PRUint32 count = 0;
|
|
|
|
anonymousItems->Count(&count);
|
|
|
|
|
|
|
|
for (PRUint32 i=0; i < count; i++)
|
|
|
|
{
|
|
|
|
// get our child's content and set its parent to our content
|
|
|
|
nsCOMPtr<nsISupports> node;
|
|
|
|
anonymousItems->GetElementAt(i,getter_AddRefs(node));
|
|
|
|
|
|
|
|
nsCOMPtr<nsIContent> content(do_QueryInterface(node));
|
|
|
|
content->SetParent(aParent);
|
|
|
|
content->SetDocument(aDocument, PR_TRUE);
|
|
|
|
|
|
|
|
// create the frame and attach it to our frame
|
1999-12-04 23:49:50 +00:00
|
|
|
ConstructFrame(aPresShell, aPresContext, aState, content, aNewFrame, aChildItems);
|
1999-08-27 06:06:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// after the node has been constructed and initialized create any
|
|
|
|
// anonymous content a node needs.
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::CreateAnonymousTreeCellFrames(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-08-27 06:06:39 +00:00
|
|
|
nsIAtom* aTag,
|
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aParent,
|
|
|
|
nsIFrame* aNewFrame,
|
1999-09-28 00:57:31 +00:00
|
|
|
nsIFrame* aNewCellFrame,
|
1999-08-27 06:06:39 +00:00
|
|
|
nsFrameItems& aChildItems)
|
|
|
|
{
|
2000-02-15 09:28:28 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
aNewCellFrame->GetStyleContext(getter_AddRefs(styleContext));
|
1999-09-28 00:57:31 +00:00
|
|
|
|
2000-02-15 09:28:28 +00:00
|
|
|
const nsStyleUserInterface* ui= (const nsStyleUserInterface*)
|
|
|
|
styleContext->GetStyleData(eStyleStruct_UserInterface);
|
1999-08-27 06:06:39 +00:00
|
|
|
|
2000-02-15 09:28:28 +00:00
|
|
|
if (ui->mBehavior != "") {
|
|
|
|
// Get the XBL loader.
|
|
|
|
nsresult rv;
|
|
|
|
NS_WITH_SERVICE(nsIXBLService, xblService, "component://netscape/xbl", &rv);
|
|
|
|
if (!xblService)
|
|
|
|
return rv;
|
1999-08-25 05:52:10 +00:00
|
|
|
|
2000-02-15 09:28:28 +00:00
|
|
|
// Load the bindings.
|
|
|
|
xblService->LoadBindings(aParent, ui->mBehavior);
|
|
|
|
|
|
|
|
// Retrieve the anonymous content that we should build.
|
2000-03-14 11:09:46 +00:00
|
|
|
nsCOMPtr<nsIContent> childElement;
|
2000-02-15 09:28:28 +00:00
|
|
|
nsCOMPtr<nsISupportsArray> anonymousItems;
|
2000-03-14 11:09:46 +00:00
|
|
|
xblService->GetContentList(aParent, getter_AddRefs(anonymousItems), getter_AddRefs(childElement));
|
1999-08-25 05:52:10 +00:00
|
|
|
|
2000-02-15 09:28:28 +00:00
|
|
|
if (!anonymousItems)
|
|
|
|
return NS_OK;
|
1999-08-25 05:52:10 +00:00
|
|
|
|
2000-02-15 09:28:28 +00:00
|
|
|
// Build the frames for the anonymous content.
|
|
|
|
PRUint32 count = 0;
|
|
|
|
anonymousItems->Count(&count);
|
1999-11-30 00:53:41 +00:00
|
|
|
|
2000-02-15 09:28:28 +00:00
|
|
|
for (PRUint32 i=0; i < count; i++)
|
|
|
|
{
|
|
|
|
// get our child's content and set its parent to our content
|
|
|
|
nsCOMPtr<nsISupports> node;
|
|
|
|
anonymousItems->GetElementAt(i,getter_AddRefs(node));
|
1999-08-25 05:52:10 +00:00
|
|
|
|
2000-02-15 09:28:28 +00:00
|
|
|
nsCOMPtr<nsIContent> content(do_QueryInterface(node));
|
|
|
|
|
|
|
|
// create the frame and attach it to our frame
|
|
|
|
ConstructFrame(aPresShell, aPresContext, aState, content, aNewFrame, aChildItems);
|
1999-08-25 05:52:10 +00:00
|
|
|
}
|
1999-06-30 22:17:43 +00:00
|
|
|
|
2000-02-15 09:28:28 +00:00
|
|
|
return NS_OK;
|
1999-09-28 00:57:31 +00:00
|
|
|
}
|
1999-06-30 22:17:43 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
#ifdef INCLUDE_XUL
|
|
|
|
nsresult
|
2000-03-11 10:38:36 +00:00
|
|
|
nsCSSFrameConstructor::ConstructXULFrame(nsIPresShell* aPresShell,
|
1999-12-04 23:49:50 +00:00
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIAtom* aTag,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsFrameItems& aFrameItems,
|
2000-03-11 10:38:36 +00:00
|
|
|
PRBool aXBLBaseTag,
|
2000-01-10 11:18:12 +00:00
|
|
|
PRBool& aHaltProcessing)
|
1999-10-12 00:16:06 +00:00
|
|
|
{
|
|
|
|
PRBool primaryFrameSet = PR_FALSE;
|
1999-02-05 03:55:18 +00:00
|
|
|
PRBool processChildren = PR_FALSE; // whether we should process child content
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
PRBool isAbsolutelyPositioned = PR_FALSE;
|
1999-02-10 19:50:50 +00:00
|
|
|
PRBool isFixedPositioned = PR_FALSE;
|
1999-03-04 23:39:06 +00:00
|
|
|
PRBool isReplaced = PR_FALSE;
|
1999-07-20 07:03:16 +00:00
|
|
|
PRBool frameHasBeenInitialized = PR_FALSE;
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// this is the new frame that will be created
|
1999-02-10 19:50:50 +00:00
|
|
|
nsIFrame* newFrame = nsnull;
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
// this is the also the new frame that is created. But if a scroll frame is needed
|
|
|
|
// the content will be mapped to the scrollframe and topFrame will point to it.
|
|
|
|
// newFrame will still point to the child that we created like a "div" for example.
|
|
|
|
nsIFrame* topFrame = nsnull;
|
|
|
|
|
1999-02-10 19:50:50 +00:00
|
|
|
nsIFrame* ignore = nsnull; // I have no idea what this is used for.
|
1999-12-04 23:49:50 +00:00
|
|
|
nsTreeCreator treeCreator(aPresShell); // Used to make tree views.
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
NS_ASSERTION(aTag != nsnull, "null XUL tag");
|
|
|
|
if (aTag == nsnull)
|
|
|
|
return NS_OK;
|
|
|
|
|
2000-03-11 10:38:36 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
PRInt32 nameSpaceID;
|
|
|
|
if (NS_SUCCEEDED(aContent->GetNameSpaceID(nameSpaceID)) &&
|
|
|
|
nameSpaceID == nsXULAtoms::nameSpaceID) {
|
2000-03-11 10:38:36 +00:00
|
|
|
|
|
|
|
// The following code allows the user to specify the base tag
|
|
|
|
// of a XUL object using XBL. XUL objects (like boxes, menus, etc.)
|
|
|
|
// can then be extended arbitrarily.
|
|
|
|
if (!aXBLBaseTag) {
|
|
|
|
const nsStyleUserInterface* ui= (const nsStyleUserInterface*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_UserInterface);
|
|
|
|
|
|
|
|
// Ensure that our XBL bindings are installed.
|
|
|
|
if (ui->mBehavior != "") {
|
|
|
|
// Get the XBL loader.
|
|
|
|
nsresult rv;
|
|
|
|
NS_WITH_SERVICE(nsIXBLService, xblService, "component://netscape/xbl", &rv);
|
|
|
|
if (!xblService)
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
// Load the bindings.
|
|
|
|
xblService->LoadBindings(aContent, ui->mBehavior);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIAtom> baseTag;
|
|
|
|
xblService->GetBaseTag(aContent, getter_AddRefs(baseTag));
|
|
|
|
|
|
|
|
if (baseTag) {
|
|
|
|
// Construct the frame using the XBL base tag.
|
|
|
|
return ConstructXULFrame( aPresShell,
|
|
|
|
aPresContext,
|
|
|
|
aState,
|
|
|
|
aContent,
|
|
|
|
aParentFrame,
|
|
|
|
baseTag,
|
|
|
|
aStyleContext,
|
|
|
|
aFrameItems,
|
|
|
|
PR_TRUE,
|
|
|
|
aHaltProcessing );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
// See if the element is absolutely positioned
|
|
|
|
const nsStylePosition* position = (const nsStylePosition*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Position);
|
|
|
|
if (NS_STYLE_POSITION_ABSOLUTE == position->mPosition)
|
|
|
|
isAbsolutelyPositioned = PR_TRUE;
|
|
|
|
|
|
|
|
// Create a frame based on the tag
|
2000-03-02 03:01:30 +00:00
|
|
|
// box is first because it is created the most.
|
|
|
|
// BOX CONSTRUCTION
|
|
|
|
if (aTag == nsXULAtoms::box || aTag == nsXULAtoms::tabbox ||
|
2000-03-02 10:47:58 +00:00
|
|
|
aTag == nsXULAtoms::tabpage || aTag == nsXULAtoms::tabcontrol ||
|
|
|
|
aTag == nsXULAtoms::radiogroup) {
|
2000-03-02 03:01:30 +00:00
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
|
|
|
rv = NS_NewBoxFrame(aPresShell, &newFrame);
|
|
|
|
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
// Boxes can scroll.
|
|
|
|
if (IsScrollable(aPresContext, display)) {
|
|
|
|
|
|
|
|
// set the top to be the newly created scrollframe
|
|
|
|
BuildScrollFrame(aPresShell, aPresContext, aState, aContent, aStyleContext, newFrame, aParentFrame,
|
|
|
|
topFrame, aStyleContext);
|
|
|
|
|
|
|
|
// we have a scrollframe so the parent becomes the scroll frame.
|
|
|
|
newFrame->GetParent(&aParentFrame);
|
|
|
|
|
|
|
|
primaryFrameSet = PR_TRUE;
|
|
|
|
|
|
|
|
frameHasBeenInitialized = PR_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
} // End of BOX CONSTRUCTION logic
|
2000-03-11 10:38:36 +00:00
|
|
|
|
|
|
|
// BUTTON CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::button) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
|
|
|
rv = NS_NewXULButtonFrame(aPresShell, &newFrame);
|
|
|
|
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
// Boxes can scroll.
|
|
|
|
if (IsScrollable(aPresContext, display)) {
|
|
|
|
|
|
|
|
// set the top to be the newly created scrollframe
|
|
|
|
BuildScrollFrame(aPresShell, aPresContext, aState, aContent, aStyleContext, newFrame, aParentFrame,
|
|
|
|
topFrame, aStyleContext);
|
|
|
|
|
|
|
|
// we have a scrollframe so the parent becomes the scroll frame.
|
|
|
|
newFrame->GetParent(&aParentFrame);
|
|
|
|
|
|
|
|
primaryFrameSet = PR_TRUE;
|
|
|
|
|
|
|
|
frameHasBeenInitialized = PR_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
} // End of BUTTON CONSTRUCTION logic
|
2000-03-02 03:01:30 +00:00
|
|
|
|
|
|
|
// TITLED BUTTON CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::titledbutton ||
|
|
|
|
aTag == nsXULAtoms::image) {
|
|
|
|
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
|
|
|
rv = NS_NewTitledButtonFrame(aPresShell, &newFrame);
|
|
|
|
}
|
|
|
|
// End of TITLED BUTTON CONSTRUCTION logic
|
|
|
|
|
|
|
|
else if (aTag == nsXULAtoms::spring) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
|
|
|
rv = NS_NewSpringFrame(aPresShell, &newFrame);
|
|
|
|
}
|
|
|
|
// End of TITLED BUTTON CONSTRUCTION logic
|
|
|
|
|
|
|
|
// TEXT CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::text) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
|
|
|
rv = NS_NewXULTextFrame(aPresShell, &newFrame);
|
|
|
|
}
|
|
|
|
// End of TEXT CONSTRUCTION logic
|
|
|
|
|
|
|
|
// Menu Construction
|
|
|
|
else if (aTag == nsXULAtoms::menu ||
|
|
|
|
aTag == nsXULAtoms::menuitem ||
|
|
|
|
aTag == nsXULAtoms::menulist ||
|
|
|
|
aTag == nsXULAtoms::menubutton) {
|
|
|
|
// A derived class box frame
|
|
|
|
// that has custom reflow to prevent menu children
|
|
|
|
// from becoming part of the flow.
|
|
|
|
processChildren = PR_TRUE; // Will need this to be custom.
|
|
|
|
isReplaced = PR_TRUE;
|
|
|
|
rv = NS_NewMenuFrame(aPresShell, &newFrame, (aTag != nsXULAtoms::menuitem));
|
|
|
|
}
|
|
|
|
else if (aTag == nsXULAtoms::menubar) {
|
2000-03-11 03:08:04 +00:00
|
|
|
#if defined(XP_MAC) || defined(RHAPSODY) // The Mac uses its native menu bar.
|
2000-03-02 03:01:30 +00:00
|
|
|
aHaltProcessing = PR_TRUE;
|
|
|
|
return NS_OK;
|
|
|
|
#else
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
rv = NS_NewMenuBarFrame(aPresShell, &newFrame);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else if (aTag == nsXULAtoms::popupset) {
|
|
|
|
// This frame contains child popups
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
|
|
|
rv = NS_NewPopupSetFrame(aPresShell, &newFrame);
|
|
|
|
}
|
|
|
|
else if (aTag == nsXULAtoms::menupopup || aTag == nsXULAtoms::popup) {
|
|
|
|
// This is its own frame that derives from
|
|
|
|
// box.
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
|
|
|
rv = NS_NewMenuPopupFrame(aPresShell, &newFrame);
|
|
|
|
}
|
|
|
|
else if (aTag == nsXULAtoms::title) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
|
|
|
rv = NS_NewTitleFrame(aPresShell, &newFrame);
|
|
|
|
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
// Boxes can scroll.
|
|
|
|
if (IsScrollable(aPresContext, display)) {
|
|
|
|
|
|
|
|
// set the top to be the newly created scrollframe
|
|
|
|
BuildScrollFrame(aPresShell, aPresContext, aState, aContent, aStyleContext, newFrame, aParentFrame,
|
|
|
|
topFrame, aStyleContext);
|
|
|
|
|
|
|
|
// we have a scrollframe so the parent becomes the scroll frame.
|
|
|
|
newFrame->GetParent(&aParentFrame);
|
|
|
|
|
|
|
|
primaryFrameSet = PR_TRUE;
|
|
|
|
|
|
|
|
frameHasBeenInitialized = PR_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
} // End of BOX CONSTRUCTION logic
|
|
|
|
|
|
|
|
else if (aTag == nsXULAtoms::titledbox) {
|
|
|
|
|
|
|
|
ConstructTitledBoxFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, aTag, aStyleContext, newFrame);
|
|
|
|
processChildren = PR_FALSE;
|
|
|
|
isReplaced = PR_TRUE;
|
|
|
|
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
// Boxes can scroll.
|
|
|
|
if (IsScrollable(aPresContext, display)) {
|
|
|
|
|
|
|
|
// set the top to be the newly created scrollframe
|
|
|
|
BuildScrollFrame(aPresShell, aPresContext, aState, aContent, aStyleContext, newFrame, aParentFrame,
|
|
|
|
topFrame, aStyleContext);
|
|
|
|
|
|
|
|
// we have a scrollframe so the parent becomes the scroll frame.
|
|
|
|
newFrame->GetParent(&aParentFrame);
|
|
|
|
|
|
|
|
primaryFrameSet = PR_TRUE;
|
|
|
|
|
|
|
|
frameHasBeenInitialized = PR_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (aTag == nsXULAtoms::spinner)
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewSpinnerFrame(aPresShell, &newFrame);
|
1999-04-20 21:57:41 +00:00
|
|
|
else if (aTag == nsXULAtoms::colorpicker)
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewColorPickerFrame(aPresShell, &newFrame);
|
1999-04-20 21:57:41 +00:00
|
|
|
else if (aTag == nsXULAtoms::fontpicker)
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewFontPickerFrame(aPresShell, &newFrame);
|
1999-10-14 10:55:24 +00:00
|
|
|
else if (aTag == nsXULAtoms::iframe) {
|
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewHTMLFrameOuterFrame(aPresShell, &newFrame);
|
1999-10-14 10:55:24 +00:00
|
|
|
}
|
|
|
|
else if (aTag == nsXULAtoms::editor) {
|
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewHTMLFrameOuterFrame(aPresShell, &newFrame);
|
1999-10-14 10:55:24 +00:00
|
|
|
}
|
|
|
|
else if (aTag == nsXULAtoms::browser) {
|
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewHTMLFrameOuterFrame(aPresShell, &newFrame);
|
1999-10-14 10:55:24 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
// TREE CONSTRUCTION
|
|
|
|
// The following code is used to construct a tree view from the XUL content
|
1999-06-23 03:02:21 +00:00
|
|
|
// model.
|
|
|
|
else if (aTag == nsXULAtoms::treeitem ||
|
|
|
|
aTag == nsXULAtoms::treechildren) {
|
1999-06-10 09:31:30 +00:00
|
|
|
nsIFrame* newTopFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableGroupFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, aStyleContext,
|
1999-06-10 09:31:30 +00:00
|
|
|
PR_TRUE, newTopFrame, newFrame, treeCreator, nsnull);
|
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
return rv;
|
|
|
|
}
|
1999-02-26 17:11:54 +00:00
|
|
|
else if (aTag == nsXULAtoms::tree)
|
|
|
|
{
|
1999-02-10 19:50:50 +00:00
|
|
|
nsIFrame* geometricParent = aParentFrame;
|
|
|
|
if (NS_STYLE_POSITION_ABSOLUTE == position->mPosition) {
|
|
|
|
isAbsolutelyPositioned = PR_TRUE;
|
1999-04-28 19:08:14 +00:00
|
|
|
aParentFrame = aState.mAbsoluteItems.containingBlock;
|
1999-02-10 19:50:50 +00:00
|
|
|
}
|
|
|
|
if (NS_STYLE_POSITION_FIXED == position->mPosition) {
|
|
|
|
isFixedPositioned = PR_TRUE;
|
1999-04-28 19:08:14 +00:00
|
|
|
aParentFrame = aState.mFixedItems.containingBlock;
|
1999-02-10 19:50:50 +00:00
|
|
|
}
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableFrame(aPresShell, aPresContext, aState, aContent, geometricParent, aStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
newFrame, treeCreator);
|
1999-02-10 19:50:50 +00:00
|
|
|
// Note: table construction function takes care of initializing the frame,
|
1999-02-05 03:55:18 +00:00
|
|
|
// processing children, and setting the initial child list
|
1999-02-10 19:50:50 +00:00
|
|
|
if (isAbsolutelyPositioned || isFixedPositioned) {
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* placeholderFrame;
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
CreatePlaceholderFrameFor(aPresShell, aPresContext, aState.mFrameManager, aContent,
|
1999-08-05 03:09:22 +00:00
|
|
|
newFrame, aStyleContext, aParentFrame, &placeholderFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-10 19:50:50 +00:00
|
|
|
// Add the positioned frame to its containing block's list of child frames
|
|
|
|
if (isAbsolutelyPositioned) {
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mAbsoluteItems.AddChild(newFrame);
|
1999-02-10 19:50:50 +00:00
|
|
|
} else {
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mFixedItems.AddChild(newFrame);
|
1999-02-10 19:50:50 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Add the placeholder frame to the flow
|
1999-02-10 19:50:50 +00:00
|
|
|
aFrameItems.AddChild(placeholderFrame);
|
|
|
|
|
|
|
|
} else {
|
1999-02-05 03:55:18 +00:00
|
|
|
// Add the table frame to the flow
|
1999-02-10 19:50:50 +00:00
|
|
|
aFrameItems.AddChild(newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-07-06 03:52:33 +00:00
|
|
|
// Make sure we add a mapping in the content->frame hash table
|
|
|
|
goto addToHashTable;
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-06-23 03:02:21 +00:00
|
|
|
else if (aTag == nsXULAtoms::treerow)
|
1999-02-26 17:11:54 +00:00
|
|
|
{
|
|
|
|
// A tree item causes a table row to be constructed that is always
|
|
|
|
// slaved to the nearest enclosing table row group (regardless of how
|
|
|
|
// deeply nested it is within other tree items).
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableRowFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, aStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
newFrame, ignore, treeCreator);
|
1999-02-26 17:11:54 +00:00
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
return rv;
|
1999-06-23 03:02:21 +00:00
|
|
|
}
|
|
|
|
else if (aTag == nsXULAtoms::treecell)
|
|
|
|
{
|
2000-01-10 11:18:12 +00:00
|
|
|
// We could be in a hidden column.
|
|
|
|
// XXX This is so disgusting.
|
|
|
|
if (nsTreeCellFrame::ShouldBuildCell(aParentFrame, aContent)) {
|
|
|
|
// We make a tree cell frame and process the children.
|
|
|
|
nsIFrame* ignore2;
|
|
|
|
rv = ConstructTableCellFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, aStyleContext,
|
1999-06-23 03:02:21 +00:00
|
|
|
newFrame, ignore, ignore2, treeCreator);
|
2000-01-10 11:18:12 +00:00
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
}
|
|
|
|
else aHaltProcessing = PR_TRUE;
|
1999-06-23 03:02:21 +00:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
else if (aTag == nsXULAtoms::treeindentation)
|
|
|
|
{
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewTreeIndentationFrame(aPresShell, &newFrame);
|
1999-06-23 03:02:21 +00:00
|
|
|
}
|
|
|
|
// End of TREE CONSTRUCTION code here (there's more later on in the function)
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
// TOOLBAR CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::toolbox) {
|
|
|
|
processChildren = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewToolboxFrame(aPresShell, &newFrame);
|
1999-08-27 06:06:39 +00:00
|
|
|
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
if (IsScrollable(aPresContext, display)) {
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// build the scrollframe
|
|
|
|
// set the top to be the newly created scrollframe
|
1999-12-04 23:49:50 +00:00
|
|
|
BuildScrollFrame(aPresShell, aPresContext, aState, aContent, aStyleContext, newFrame, aParentFrame,
|
1999-10-12 00:16:06 +00:00
|
|
|
topFrame, aStyleContext);
|
|
|
|
|
|
|
|
// we have a scrollframe so the parent becomes the scroll frame.
|
|
|
|
newFrame->GetParent(&aParentFrame);
|
|
|
|
primaryFrameSet = PR_TRUE;
|
|
|
|
|
1999-08-27 06:06:39 +00:00
|
|
|
frameHasBeenInitialized = PR_TRUE;
|
|
|
|
|
|
|
|
}
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
|
|
|
else if (aTag == nsXULAtoms::toolbar) {
|
1999-08-27 06:06:39 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
processChildren = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewToolbarFrame(aPresShell, &newFrame);
|
1999-08-27 06:06:39 +00:00
|
|
|
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
if (IsScrollable(aPresContext, display)) {
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// set the top to be the newly created scrollframe
|
1999-12-04 23:49:50 +00:00
|
|
|
BuildScrollFrame(aPresShell, aPresContext, aState, aContent, aStyleContext, newFrame, aParentFrame,
|
1999-10-12 00:16:06 +00:00
|
|
|
topFrame, aStyleContext);
|
|
|
|
|
|
|
|
// we have a scrollframe so the parent becomes the scroll frame.
|
|
|
|
newFrame->GetParent(&aParentFrame);
|
|
|
|
primaryFrameSet = PR_TRUE;
|
|
|
|
|
1999-08-27 06:06:39 +00:00
|
|
|
frameHasBeenInitialized = PR_TRUE;
|
|
|
|
|
|
|
|
}
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-07-01 21:11:38 +00:00
|
|
|
else if (aTag == nsXULAtoms::toolbaritem) {
|
|
|
|
processChildren = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewToolbarItemFrame(aPresShell, &newFrame);
|
1999-07-01 21:11:38 +00:00
|
|
|
}
|
1999-02-26 17:11:54 +00:00
|
|
|
// End of TOOLBAR CONSTRUCTION logic
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
// PROGRESS METER CONSTRUCTION
|
2000-03-15 03:16:43 +00:00
|
|
|
else if (aTag == nsXULAtoms::progressbar) {
|
1999-02-26 17:11:54 +00:00
|
|
|
processChildren = PR_TRUE;
|
1999-03-04 23:39:06 +00:00
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewProgressMeterFrame(aPresShell, &newFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
|
|
|
// End of PROGRESS METER CONSTRUCTION logic
|
2000-03-02 10:00:09 +00:00
|
|
|
|
2000-02-25 08:37:49 +00:00
|
|
|
// XULCHECKBOX CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::checkbox ||
|
|
|
|
aTag == nsXULAtoms::radio) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
|
|
|
rv = NS_NewXULCheckboxFrame(aPresShell, &newFrame);
|
|
|
|
}
|
|
|
|
// End of XULCHECKBOX CONSTRUCTION logic
|
|
|
|
|
2000-03-02 03:01:30 +00:00
|
|
|
// STACK CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::stack) {
|
|
|
|
processChildren = PR_TRUE;
|
2000-02-16 23:00:52 +00:00
|
|
|
isReplaced = PR_TRUE;
|
2000-03-02 03:01:30 +00:00
|
|
|
rv = NS_NewStackFrame(aPresShell, &newFrame);
|
2000-02-16 23:00:52 +00:00
|
|
|
}
|
2000-03-02 03:01:30 +00:00
|
|
|
// End of STACK CONSTRUCTION logic
|
2000-02-16 23:00:52 +00:00
|
|
|
|
1999-04-21 22:46:15 +00:00
|
|
|
// DECK CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::deck || aTag == nsXULAtoms::tabpanel) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewDeckFrame(aPresShell, &newFrame);
|
1999-04-21 22:46:15 +00:00
|
|
|
}
|
|
|
|
// End of DECK CONSTRUCTION logic
|
|
|
|
|
|
|
|
// TAB CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::tab) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewTabFrame(aPresShell, &newFrame);
|
1999-04-21 22:46:15 +00:00
|
|
|
}
|
|
|
|
// End of TAB CONSTRUCTION logic
|
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
// SLIDER CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::slider) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewSliderFrame(aPresShell, &newFrame);
|
1999-06-15 04:02:43 +00:00
|
|
|
}
|
|
|
|
// End of SLIDER CONSTRUCTION logic
|
|
|
|
|
|
|
|
// SCROLLBAR CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::scrollbar) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewScrollbarFrame(aPresShell, &newFrame);
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
// End of SCROLLBAR CONSTRUCTION logic
|
|
|
|
|
|
|
|
// SCROLLBUTTON CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::scrollbarbutton) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewScrollbarButtonFrame(aPresShell, &newFrame);
|
1999-06-15 04:02:43 +00:00
|
|
|
}
|
|
|
|
// End of SCROLLBUTTON CONSTRUCTION logic
|
|
|
|
|
1999-06-30 22:17:43 +00:00
|
|
|
// SPLITTER CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::splitter) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewSplitterFrame(aPresShell, &newFrame);
|
1999-06-30 22:17:43 +00:00
|
|
|
}
|
|
|
|
// End of SPLITTER CONSTRUCTION logic
|
|
|
|
|
|
|
|
// GRIPPY CONSTRUCTION
|
|
|
|
else if (aTag == nsXULAtoms::grippy) {
|
|
|
|
processChildren = PR_TRUE;
|
|
|
|
isReplaced = PR_TRUE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewGrippyFrame(aPresShell, &newFrame);
|
1999-06-30 22:17:43 +00:00
|
|
|
}
|
|
|
|
// End of GRIPPY CONSTRUCTION logic
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If we succeeded in creating a frame then initialize it, process its
|
|
|
|
// children (if requested), and set the initial child list
|
1999-02-10 19:50:50 +00:00
|
|
|
if (NS_SUCCEEDED(rv) && newFrame != nsnull) {
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
// if no top frame was created then the top is the new frame
|
|
|
|
if (topFrame == nsnull)
|
|
|
|
topFrame = newFrame;
|
|
|
|
|
1999-03-04 23:39:06 +00:00
|
|
|
// If the frame is a replaced element, then set the frame state bit
|
|
|
|
if (isReplaced) {
|
|
|
|
nsFrameState state;
|
|
|
|
newFrame->GetFrameState(&state);
|
|
|
|
newFrame->SetFrameState(state | NS_FRAME_REPLACED_ELEMENT);
|
|
|
|
}
|
|
|
|
|
2000-02-10 21:36:28 +00:00
|
|
|
// xul does not support absolute positioning
|
|
|
|
nsIFrame* geometricParent = aParentFrame;
|
|
|
|
|
|
|
|
/*
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIFrame* geometricParent = isAbsolutelyPositioned
|
1999-07-20 07:03:16 +00:00
|
|
|
? aState.mAbsoluteItems.containingBlock
|
|
|
|
: aParentFrame;
|
2000-02-10 21:36:28 +00:00
|
|
|
*/
|
1999-10-12 00:16:06 +00:00
|
|
|
// if the new frame was already initialized to initialize it again.
|
|
|
|
if (!frameHasBeenInitialized) {
|
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
geometricParent, aStyleContext, nsnull, newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-07-20 07:03:16 +00:00
|
|
|
// See if we need to create a view, e.g. the frame is absolutely positioned
|
1999-11-24 06:03:41 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-07-20 07:03:16 +00:00
|
|
|
aStyleContext, PR_FALSE);
|
1999-10-12 00:16:06 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-07-20 07:03:16 +00:00
|
|
|
// Process the child content if requested
|
|
|
|
nsFrameItems childItems;
|
|
|
|
if (processChildren) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ProcessChildren(aPresShell, aPresContext, aState, aContent, newFrame,
|
1999-08-27 21:46:10 +00:00
|
|
|
PR_FALSE, childItems, PR_FALSE);
|
1999-10-12 00:16:06 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-07-20 07:03:16 +00:00
|
|
|
// if there are any anonymous children create frames for them
|
2000-03-11 10:38:36 +00:00
|
|
|
nsCOMPtr<nsIAtom> tag(aTag);
|
|
|
|
if (aXBLBaseTag) {
|
|
|
|
aContent->GetTag(*getter_AddRefs(tag));
|
|
|
|
}
|
|
|
|
|
|
|
|
CreateAnonymousFrames(aPresShell, aPresContext, tag, aState, aContent, newFrame,
|
1999-07-20 07:03:16 +00:00
|
|
|
childItems);
|
|
|
|
|
|
|
|
// Set the frame's initial child list
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-07-20 07:03:16 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
|
|
|
|
1999-07-20 07:03:16 +00:00
|
|
|
// Add the new frame to our list of frame items.
|
1999-10-12 00:16:06 +00:00
|
|
|
aFrameItems.AddChild(topFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-07-01 14:34:35 +00:00
|
|
|
// If the frame is absolutely positioned, then create a placeholder frame
|
1999-02-10 19:50:50 +00:00
|
|
|
if (isAbsolutelyPositioned || isFixedPositioned) {
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* placeholderFrame;
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
CreatePlaceholderFrameFor(aPresShell, aPresContext, aState.mFrameManager, aContent,
|
1999-08-05 03:09:22 +00:00
|
|
|
newFrame, aStyleContext, aParentFrame, &placeholderFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-10 19:50:50 +00:00
|
|
|
// Add the positioned frame to its containing block's list of child frames
|
|
|
|
if (isAbsolutelyPositioned) {
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mAbsoluteItems.AddChild(newFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
} else {
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mFixedItems.AddChild(newFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Add the placeholder frame to the flow
|
|
|
|
aFrameItems.AddChild(placeholderFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-07-06 03:52:33 +00:00
|
|
|
}
|
1999-07-01 14:34:35 +00:00
|
|
|
|
1999-07-06 03:52:33 +00:00
|
|
|
addToHashTable:
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
if (topFrame) {
|
|
|
|
// the top frame is always what we map the content to. This is the frame that contains a pointer
|
|
|
|
// to the content node.
|
|
|
|
|
1999-07-01 14:34:35 +00:00
|
|
|
// Add a mapping from content object to primary frame. Note that for
|
|
|
|
// floated and positioned frames this is the out-of-flow frame and not
|
|
|
|
// the placeholder frame
|
1999-10-12 00:16:06 +00:00
|
|
|
if (!primaryFrameSet)
|
|
|
|
aState.mFrameManager->SetPrimaryFrameFor(aContent, topFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-08-27 06:06:39 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::BeginBuildingScrollFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-08-27 06:06:39 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIStyleContext* aContentStyle,
|
1999-08-27 06:06:39 +00:00
|
|
|
nsIFrame* aParentFrame,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIAtom* aScrolledPseudo,
|
|
|
|
nsIDocument* aDocument,
|
1999-08-27 06:06:39 +00:00
|
|
|
nsIFrame*& aNewFrame,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsCOMPtr<nsIStyleContext>& aScrolledChildStyle,
|
|
|
|
nsIFrame*& aScrollableFrame)
|
1999-08-27 06:06:39 +00:00
|
|
|
{
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIFrame* scrollFrame = nsnull;
|
|
|
|
nsIFrame* parentFrame = nsnull;
|
|
|
|
nsIFrame* gfxScrollFrame = nsnull;
|
1999-08-27 06:06:39 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
nsFrameItems anonymousItems;
|
1999-08-27 06:06:39 +00:00
|
|
|
|
1999-11-11 21:48:17 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> contentStyle = dont_QueryInterface(aContentStyle);
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
PRBool isGfx = HasGfxScrollbars(aPresContext);
|
1999-08-27 06:06:39 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
if (isGfx) {
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
BuildGfxScrollFrame(aPresShell, aPresContext, aState, aContent, aDocument, aParentFrame,
|
1999-11-11 21:48:17 +00:00
|
|
|
contentStyle, gfxScrollFrame, anonymousItems);
|
1999-08-27 06:06:39 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
scrollFrame = anonymousItems.childList;
|
|
|
|
parentFrame = gfxScrollFrame;
|
1999-08-27 06:06:39 +00:00
|
|
|
aNewFrame = gfxScrollFrame;
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
// we used the style that was passed in. So resolve another one.
|
|
|
|
nsCOMPtr<nsIStyleContext> scrollPseudoStyle;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent,
|
|
|
|
nsLayoutAtoms::scrolledContentPseudo,
|
1999-11-11 21:48:17 +00:00
|
|
|
contentStyle, PR_FALSE,
|
1999-12-08 01:56:28 +00:00
|
|
|
getter_AddRefs(scrollPseudoStyle));
|
1999-10-12 00:16:06 +00:00
|
|
|
|
1999-12-08 01:56:28 +00:00
|
|
|
contentStyle = scrollPseudoStyle;
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
parentFrame, contentStyle, nsnull, scrollFrame);
|
1999-08-27 06:06:39 +00:00
|
|
|
} else {
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewScrollFrame(aPresShell, &scrollFrame);
|
1999-08-27 06:06:39 +00:00
|
|
|
aNewFrame = scrollFrame;
|
1999-10-12 00:16:06 +00:00
|
|
|
parentFrame = aParentFrame;
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
parentFrame, contentStyle, nsnull, scrollFrame);
|
1999-08-27 06:06:39 +00:00
|
|
|
}
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// initialize the scrolled frame
|
|
|
|
nsCOMPtr<nsIStyleContext> scrolledPseudoStyle;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent,
|
|
|
|
aScrolledPseudo,
|
1999-11-11 21:48:17 +00:00
|
|
|
contentStyle, PR_FALSE,
|
1999-10-12 00:16:06 +00:00
|
|
|
getter_AddRefs(scrolledPseudoStyle));
|
|
|
|
|
|
|
|
|
|
|
|
aScrollableFrame = scrollFrame;
|
|
|
|
|
|
|
|
// set the child frame for the gfxscrollbar if the is one. This frames will be the
|
|
|
|
// 2 scrollbars and the scrolled frame.
|
|
|
|
if (gfxScrollFrame) {
|
1999-11-24 06:03:41 +00:00
|
|
|
gfxScrollFrame->SetInitialChildList(aPresContext, nsnull, anonymousItems.childList);
|
1999-10-12 00:16:06 +00:00
|
|
|
}
|
|
|
|
|
1999-11-10 23:51:44 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
aScrolledChildStyle = scrolledPseudoStyle;
|
|
|
|
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::FinishBuildingScrollFrame(nsIPresContext* aPresContext,
|
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aScrollFrame,
|
|
|
|
nsIFrame* aScrolledFrame,
|
|
|
|
nsIStyleContext* aScrolledContentStyle)
|
|
|
|
|
|
|
|
{
|
|
|
|
// create a view
|
1999-11-24 06:03:41 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, aScrolledFrame,
|
1999-10-12 00:16:06 +00:00
|
|
|
aScrolledContentStyle, PR_TRUE);
|
|
|
|
|
|
|
|
// the the scroll frames child list
|
1999-11-24 06:03:41 +00:00
|
|
|
aScrollFrame->SetInitialChildList(aPresContext, nsnull, aScrolledFrame);
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-08-27 06:06:39 +00:00
|
|
|
|
1999-08-19 22:16:23 +00:00
|
|
|
/**
|
1999-11-10 23:51:44 +00:00
|
|
|
* Called to wrap a scrollframe or gfx scrollframe around a frame. The hierarchy will look like this
|
|
|
|
*
|
|
|
|
* ------ for native scrollbars -----
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* ScrollFrame
|
|
|
|
* ^
|
|
|
|
* |
|
|
|
|
* Frame (scrolled frame you passed in)
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* ------- for gfx scrollbars ------
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* GfxScrollFrame
|
|
|
|
* ^
|
|
|
|
* |
|
|
|
|
* ScrollPort
|
|
|
|
* ^
|
|
|
|
* |
|
|
|
|
* Frame (scrolled frame you passed in)
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*-----------------------------------
|
|
|
|
* LEGEND:
|
|
|
|
*
|
|
|
|
* ScrollFrame: This is a frame that has a view that manages native scrollbars. It implements
|
|
|
|
* nsIScrollableView. It also manages clipping and scrolling of native widgets by
|
|
|
|
* having a native scrolling window.
|
|
|
|
*
|
|
|
|
* GfxScrollFrame: This is a frame that manages gfx cross platform frame based scrollbars.
|
|
|
|
*
|
|
|
|
* ScrollPort: This is similar to the ScrollFrame above in that is clips and scrolls its children
|
|
|
|
* with a native scrolling window. But because it is contained in a GfxScrollFrame
|
|
|
|
* it does not have any code to do scrollbars so it is much simpler. Infact it only has
|
|
|
|
* 1 view attached to it. Where the ScrollFrame above has 5!
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* @param aContent the content node of the child to wrap.
|
|
|
|
* @param aScrolledFrame The frame of the content to wrap. This should not be
|
|
|
|
* Initialized. This method will initialize it with a scrolled pseudo
|
|
|
|
* and no nsIContent. The content will be attached to the scrollframe
|
|
|
|
* returned.
|
|
|
|
* @param aContentStyle the style context that has already been resolved for the content being passed in.
|
|
|
|
*
|
|
|
|
* @param aParentFrame The parent to attach the scroll frame to
|
|
|
|
*
|
|
|
|
* @param aNewFrame The new scrollframe or gfx scrollframe that we create. It will contain the
|
|
|
|
* scrolled frame you passed in. (returned)
|
|
|
|
* @param aScrolledContentStyle the style that was resolved for the scrolled frame. (returned)
|
1999-08-19 22:16:23 +00:00
|
|
|
*/
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::BuildScrollFrame (nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-08-19 22:16:23 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIStyleContext* aContentStyle,
|
1999-08-19 22:16:23 +00:00
|
|
|
nsIFrame* aScrolledFrame,
|
|
|
|
nsIFrame* aParentFrame,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIFrame*& aNewFrame,
|
|
|
|
nsIStyleContext*& aScrolledContentStyle)
|
1999-08-19 22:16:23 +00:00
|
|
|
{
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIFrame *scrollFrame;
|
1999-11-02 06:40:38 +00:00
|
|
|
nsCOMPtr<nsIDocument> document;
|
|
|
|
aContent->GetDocument(*getter_AddRefs(document));
|
1999-10-12 00:16:06 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> scrolledContentStyle;
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
BeginBuildingScrollFrame(aPresShell, aPresContext,
|
1999-10-12 00:16:06 +00:00
|
|
|
aState,
|
|
|
|
aContent,
|
|
|
|
aContentStyle,
|
|
|
|
aParentFrame,
|
|
|
|
nsLayoutAtoms::scrolledContentPseudo,
|
|
|
|
document,
|
|
|
|
aNewFrame,
|
|
|
|
scrolledContentStyle,
|
|
|
|
scrollFrame);
|
1999-12-06 07:44:18 +00:00
|
|
|
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
scrollFrame, scrolledContentStyle, nsnull, aScrolledFrame);
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
FinishBuildingScrollFrame(aPresContext,
|
|
|
|
aState,
|
|
|
|
aContent,
|
|
|
|
scrollFrame,
|
|
|
|
aScrolledFrame,
|
|
|
|
scrolledContentStyle);
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
aScrolledContentStyle = scrolledContentStyle;
|
1999-08-19 22:16:23 +00:00
|
|
|
|
2000-01-31 14:04:41 +00:00
|
|
|
// now set the primary frame to the ScrollFrame
|
|
|
|
aState.mFrameManager->SetPrimaryFrameFor( aContent, aNewFrame );
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
return NS_OK;
|
1999-08-19 22:16:23 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
/**
|
|
|
|
* If we are building GFX scrollframes this will create one
|
|
|
|
*/
|
1999-08-19 22:16:23 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::BuildGfxScrollFrame (nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-08-19 22:16:23 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsIDocument* aDocument,
|
1999-08-19 22:16:23 +00:00
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aNewFrame,
|
|
|
|
nsFrameItems& aAnonymousFrames)
|
|
|
|
{
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewGfxScrollFrame(aPresShell, &aNewFrame,aDocument);
|
1999-08-19 22:16:23 +00:00
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, aNewFrame);
|
1999-08-19 22:16:23 +00:00
|
|
|
|
|
|
|
nsIFrame* scrollbox = nsnull;
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewScrollPortFrame(aPresShell, &scrollbox);
|
1999-08-19 22:16:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
aAnonymousFrames.AddChild(scrollbox);
|
|
|
|
|
|
|
|
// if there are any anonymous children for the nsScrollFrame create frames for them.
|
1999-12-04 23:49:50 +00:00
|
|
|
CreateAnonymousFrames(aPresShell, aPresContext, aState, aContent, aDocument, aNewFrame,
|
1999-08-19 22:16:23 +00:00
|
|
|
aAnonymousFrames);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructFrameByDisplayType(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
const nsStyleDisplay* aDisplay,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsFrameItems& aFrameItems)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
1999-10-12 00:16:06 +00:00
|
|
|
PRBool primaryFrameSet = PR_FALSE;
|
1999-02-05 03:55:18 +00:00
|
|
|
PRBool isAbsolutelyPositioned = PR_FALSE;
|
|
|
|
PRBool isFixedPositioned = PR_FALSE;
|
1999-02-26 17:11:54 +00:00
|
|
|
PRBool isFloating = PR_FALSE;
|
1999-02-05 03:55:18 +00:00
|
|
|
PRBool isBlock = aDisplay->IsBlockLevel();
|
|
|
|
nsIFrame* newFrame = nsnull; // the frame we construct
|
1999-11-01 15:24:57 +00:00
|
|
|
nsIFrame* newBlock = nsnull;
|
|
|
|
nsIFrame* nextInline = nsnull;
|
1999-12-04 23:49:50 +00:00
|
|
|
nsTableCreator tableCreator(aPresShell); // Used to make table frames.
|
1999-07-06 03:52:33 +00:00
|
|
|
PRBool addToHashTable = PR_TRUE;
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
// Get the position syle info
|
|
|
|
const nsStylePosition* position = (const nsStylePosition*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Position);
|
|
|
|
|
|
|
|
// The frame is also a block if it's an inline frame that's floated or
|
|
|
|
// absolutely positioned
|
1999-02-26 17:11:54 +00:00
|
|
|
if (NS_STYLE_FLOAT_NONE != aDisplay->mFloats) {
|
|
|
|
isFloating = PR_TRUE;
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
if ((NS_STYLE_DISPLAY_INLINE == aDisplay->mDisplay) &&
|
1999-02-26 17:11:54 +00:00
|
|
|
(isFloating || position->IsAbsolutelyPositioned())) {
|
1999-02-05 03:55:18 +00:00
|
|
|
isBlock = PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the frame is a block-level frame and is scrollable, then wrap it
|
|
|
|
// in a scroll frame.
|
|
|
|
// XXX Ignore tables for the time being
|
|
|
|
if ((isBlock && (aDisplay->mDisplay != NS_STYLE_DISPLAY_TABLE)) &&
|
|
|
|
IsScrollable(aPresContext, aDisplay)) {
|
|
|
|
|
|
|
|
// See if it's absolute positioned or fixed positioned
|
|
|
|
if (NS_STYLE_POSITION_ABSOLUTE == position->mPosition) {
|
|
|
|
isAbsolutelyPositioned = PR_TRUE;
|
|
|
|
} else if (NS_STYLE_POSITION_FIXED == position->mPosition) {
|
|
|
|
isFixedPositioned = PR_TRUE;
|
|
|
|
}
|
|
|
|
|
1999-10-12 00:16:06 +00:00
|
|
|
// Initialize it
|
|
|
|
nsIFrame* geometricParent = aParentFrame;
|
|
|
|
|
|
|
|
if (isAbsolutelyPositioned) {
|
|
|
|
geometricParent = aState.mAbsoluteItems.containingBlock;
|
|
|
|
} else if (isFixedPositioned) {
|
|
|
|
geometricParent = aState.mFixedItems.containingBlock;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIFrame* scrolledFrame = nsnull;
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewAreaFrame(aPresShell, &scrolledFrame, NS_BLOCK_SPACE_MGR |
|
1999-10-14 04:47:18 +00:00
|
|
|
NS_BLOCK_SHRINK_WRAP | NS_BLOCK_MARGIN_ROOT);
|
1999-10-12 00:16:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
nsIStyleContext* newStyle = nsnull;
|
1999-08-19 22:16:23 +00:00
|
|
|
// Build the scrollframe it
|
1999-12-04 23:49:50 +00:00
|
|
|
BuildScrollFrame(aPresShell, aPresContext, aState, aContent, aStyleContext, scrolledFrame, geometricParent,
|
1999-10-12 00:16:06 +00:00
|
|
|
newFrame, newStyle);
|
|
|
|
|
|
|
|
// buildscrollframe sets the primary frame.
|
|
|
|
primaryFrameSet = PR_TRUE;
|
|
|
|
|
|
|
|
|
|
|
|
//-----
|
|
|
|
|
|
|
|
// The area frame is a floater container
|
|
|
|
PRBool haveFirstLetterStyle, haveFirstLineStyle;
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aContent, aStyleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
|
|
|
aState.PushFloaterContainingBlock(scrolledFrame, floaterSaveState,
|
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
|
|
|
|
|
|
|
// Process children
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
nsFrameItems childItems;
|
|
|
|
PRBool isPositionedContainingBlock = isAbsolutelyPositioned ||
|
|
|
|
isFixedPositioned;
|
|
|
|
|
|
|
|
if (isPositionedContainingBlock) {
|
|
|
|
// The area frame becomes a container for child frames that are
|
|
|
|
// absolutely positioned
|
|
|
|
aState.PushAbsoluteContainingBlock(scrolledFrame, absoluteSaveState);
|
|
|
|
}
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aContent, scrolledFrame, PR_FALSE,
|
1999-10-12 00:16:06 +00:00
|
|
|
childItems, PR_TRUE);
|
|
|
|
|
|
|
|
// Set the scrolled frame's initial child lists
|
1999-11-24 06:03:41 +00:00
|
|
|
scrolledFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-10-12 00:16:06 +00:00
|
|
|
if (isPositionedContainingBlock && aState.mAbsoluteItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
scrolledFrame->SetInitialChildList(aPresContext,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsLayoutAtoms::absoluteList,
|
|
|
|
aState.mAbsoluteItems.childList);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aState.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
scrolledFrame->SetInitialChildList(aPresContext,
|
1999-10-12 00:16:06 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
|
|
|
aState.mFloatedItems.childList);
|
|
|
|
}
|
|
|
|
///------
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
// See if the frame is absolute or fixed positioned
|
1999-02-26 17:11:54 +00:00
|
|
|
else if (position->IsAbsolutelyPositioned() &&
|
|
|
|
((NS_STYLE_DISPLAY_BLOCK == aDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_INLINE == aDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_LIST_ITEM == aDisplay->mDisplay))) {
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
if (NS_STYLE_POSITION_ABSOLUTE == position->mPosition) {
|
|
|
|
isAbsolutelyPositioned = PR_TRUE;
|
|
|
|
} else {
|
|
|
|
isFixedPositioned = PR_TRUE;
|
|
|
|
}
|
|
|
|
|
1999-05-11 22:03:29 +00:00
|
|
|
// Create a frame to wrap up the absolute positioned item
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewAbsoluteItemWrapperFrame(aPresShell, &newFrame);
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
(isAbsolutelyPositioned
|
|
|
|
? aState.mAbsoluteItems.containingBlock
|
|
|
|
: aState.mFixedItems.containingBlock),
|
|
|
|
aStyleContext, nsnull, newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Create a view
|
1999-11-24 06:03:41 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-05 03:55:18 +00:00
|
|
|
aStyleContext, PR_FALSE);
|
|
|
|
|
|
|
|
// Process the child content. The area frame becomes a container for child
|
|
|
|
// frames that are absolutely positioned
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
|
|
|
nsFrameItems childItems;
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-08-31 03:09:40 +00:00
|
|
|
PRBool haveFirstLetterStyle = PR_FALSE, haveFirstLineStyle = PR_FALSE;
|
|
|
|
if (aDisplay->IsBlockLevel()) {
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aContent, aStyleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
|
|
|
}
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.PushAbsoluteContainingBlock(newFrame, absoluteSaveState);
|
1999-08-31 03:09:40 +00:00
|
|
|
aState.PushFloaterContainingBlock(newFrame, floaterSaveState,
|
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
1999-12-04 23:49:50 +00:00
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aContent, newFrame, PR_TRUE,
|
1999-08-27 21:46:10 +00:00
|
|
|
childItems, PR_TRUE);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
// Set the frame's initial child list(s)
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-04-28 19:08:14 +00:00
|
|
|
if (aState.mAbsoluteItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsLayoutAtoms::absoluteList,
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mAbsoluteItems.childList);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-04-28 19:08:14 +00:00
|
|
|
if (aState.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext,
|
1999-02-26 17:11:54 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mFloatedItems.childList);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
// See if the frame is floated, and it's a block or inline frame
|
1999-02-26 17:11:54 +00:00
|
|
|
else if (isFloating &&
|
|
|
|
((NS_STYLE_DISPLAY_BLOCK == aDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_INLINE == aDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_LIST_ITEM == aDisplay->mDisplay))) {
|
1999-02-05 03:55:18 +00:00
|
|
|
// Create an area frame
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewFloatingItemWrapperFrame(aPresShell, &newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Initialize the frame
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aState.mFloatedItems.containingBlock,
|
|
|
|
aStyleContext, nsnull, newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// See if we need to create a view
|
1999-11-24 06:03:41 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-05 03:55:18 +00:00
|
|
|
aStyleContext, PR_FALSE);
|
|
|
|
|
|
|
|
// Process the child content
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
|
|
|
nsFrameItems childItems;
|
|
|
|
|
1999-08-31 03:09:40 +00:00
|
|
|
PRBool haveFirstLetterStyle = PR_FALSE, haveFirstLineStyle = PR_FALSE;
|
|
|
|
if (aDisplay->IsBlockLevel()) {
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aContent, aStyleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
|
|
|
}
|
|
|
|
aState.PushFloaterContainingBlock(newFrame, floaterSaveState,
|
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
1999-12-04 23:49:50 +00:00
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aContent, newFrame,
|
1999-08-27 21:46:10 +00:00
|
|
|
PR_TRUE, childItems, PR_TRUE);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
// Set the frame's initial child list(s)
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-04-28 19:08:14 +00:00
|
|
|
if (aState.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext,
|
1999-02-26 17:11:54 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mFloatedItems.childList);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
|
|
|
}
|
1999-04-19 19:10:15 +00:00
|
|
|
// See if it's relatively positioned
|
1999-07-20 04:16:56 +00:00
|
|
|
else if ((NS_STYLE_POSITION_RELATIVE == position->mPosition) &&
|
|
|
|
((NS_STYLE_DISPLAY_BLOCK == aDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_INLINE == aDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_LIST_ITEM == aDisplay->mDisplay))) {
|
1999-04-19 19:10:15 +00:00
|
|
|
// Is it block-level or inline-level?
|
1999-08-27 21:46:10 +00:00
|
|
|
PRBool isBlockFrame = PR_FALSE;
|
1999-07-24 23:08:34 +00:00
|
|
|
if ((NS_STYLE_DISPLAY_BLOCK == aDisplay->mDisplay) ||
|
1999-07-20 04:16:56 +00:00
|
|
|
(NS_STYLE_DISPLAY_LIST_ITEM == aDisplay->mDisplay)) {
|
1999-05-11 22:03:29 +00:00
|
|
|
// Create a wrapper frame. No space manager, though
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewRelativeItemWrapperFrame(aPresShell, &newFrame);
|
1999-08-27 21:46:10 +00:00
|
|
|
isBlockFrame = PR_TRUE;
|
1999-04-19 19:10:15 +00:00
|
|
|
} else {
|
|
|
|
// Create a positioned inline frame
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewPositionedInlineFrame(aPresShell, &newFrame);
|
1999-04-19 19:10:15 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
// Initialize the frame
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
// Create a view
|
1999-11-24 06:03:41 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-05 03:55:18 +00:00
|
|
|
aStyleContext, PR_FALSE);
|
|
|
|
|
|
|
|
// Process the child content. Relatively positioned frames becomes a
|
|
|
|
// container for child frames that are positioned
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
|
|
|
nsFrameItems childItems;
|
|
|
|
|
|
|
|
aState.PushAbsoluteContainingBlock(newFrame, absoluteSaveState);
|
1999-04-19 19:10:15 +00:00
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
if (isBlockFrame) {
|
1999-08-31 03:09:40 +00:00
|
|
|
PRBool haveFirstLetterStyle, haveFirstLineStyle;
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aContent, aStyleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
|
|
|
aState.PushFloaterContainingBlock(newFrame, floaterSaveState,
|
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
1999-04-19 19:10:15 +00:00
|
|
|
}
|
1999-12-04 23:49:50 +00:00
|
|
|
ProcessChildren(aPresShell, aPresContext, aState, aContent, newFrame, PR_TRUE,
|
1999-08-27 21:46:10 +00:00
|
|
|
childItems, isBlockFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Set the frame's initial child list
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-04-28 19:08:14 +00:00
|
|
|
if (aState.mAbsoluteItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsLayoutAtoms::absoluteList,
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mAbsoluteItems.childList);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
if (isBlockFrame && aState.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext,
|
1999-02-26 17:11:54 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mFloatedItems.childList);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
// See if it's a block frame of some sort
|
|
|
|
else if ((NS_STYLE_DISPLAY_BLOCK == aDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_LIST_ITEM == aDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_RUN_IN == aDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_COMPACT == aDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_INLINE_BLOCK == aDisplay->mDisplay)) {
|
|
|
|
// Create the block frame
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewBlockFrame(aPresShell, &newFrame);
|
1999-08-27 21:46:10 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
// That worked so construct the block and its children
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructBlock(aPresShell, aPresContext, aState, aDisplay, aContent,
|
1999-08-27 21:46:10 +00:00
|
|
|
aParentFrame, aStyleContext, newFrame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// See if it's an inline frame of some sort
|
|
|
|
else if ((NS_STYLE_DISPLAY_INLINE == aDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_MARKER == aDisplay->mDisplay)) {
|
|
|
|
// Create the inline frame
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewInlineFrame(aPresShell, &newFrame);
|
1999-08-27 21:46:10 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
// That worked so construct the inline and its children
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructInline(aPresShell, aPresContext, aState, aDisplay, aContent,
|
1999-11-01 15:24:57 +00:00
|
|
|
aParentFrame, aStyleContext, newFrame,
|
|
|
|
&newBlock, &nextInline);
|
1999-08-27 21:46:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// To keep the hash table small don't add inline frames (they're
|
|
|
|
// typically things like FONT and B), because we can quickly
|
|
|
|
// find them if we need to
|
|
|
|
addToHashTable = PR_FALSE;
|
|
|
|
}
|
1999-03-30 15:22:54 +00:00
|
|
|
// otherwise let the display property influence the frame type to create
|
1999-02-26 17:11:54 +00:00
|
|
|
else {
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* ignore;
|
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
// XXX This section now only handles table frames; should be
|
|
|
|
// factored out probably
|
|
|
|
|
1999-03-30 15:22:54 +00:00
|
|
|
// Use the 'display' property to choose a frame type
|
1999-02-05 03:55:18 +00:00
|
|
|
switch (aDisplay->mDisplay) {
|
|
|
|
case NS_STYLE_DISPLAY_TABLE:
|
|
|
|
{
|
|
|
|
nsIFrame* geometricParent = aParentFrame;
|
|
|
|
if (NS_STYLE_POSITION_ABSOLUTE == position->mPosition) {
|
|
|
|
isAbsolutelyPositioned = PR_TRUE;
|
1999-09-29 05:06:17 +00:00
|
|
|
geometricParent = aState.mAbsoluteItems.containingBlock;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
if (NS_STYLE_POSITION_FIXED == position->mPosition) {
|
|
|
|
isFixedPositioned = PR_TRUE;
|
1999-09-29 05:06:17 +00:00
|
|
|
geometricParent = aState.mFixedItems.containingBlock;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableFrame(aPresShell, aPresContext, aState, aContent, geometricParent,
|
1999-04-28 19:08:14 +00:00
|
|
|
aStyleContext, newFrame, tableCreator);
|
1999-03-17 18:17:24 +00:00
|
|
|
// Note: table construction function takes care of initializing
|
|
|
|
// the frame, processing children, and setting the initial child
|
|
|
|
// list
|
|
|
|
goto nearly_done;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// the next 5 cases are only relevant if the parent is not a table, ConstructTableFrame handles children
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_CAPTION:
|
|
|
|
{
|
2000-03-02 06:09:37 +00:00
|
|
|
// aParentFrame may be an inner table frame rather than an outer frame
|
|
|
|
// In this case we need to get the outer frame.
|
|
|
|
nsIFrame* parentFrame = aParentFrame;
|
|
|
|
nsIFrame* outerFrame = nsnull;
|
|
|
|
aParentFrame->GetParent(&outerFrame);
|
|
|
|
nsCOMPtr<nsIAtom> frameType;
|
|
|
|
if (outerFrame) {
|
|
|
|
outerFrame->GetFrameType(getter_AddRefs(frameType));
|
|
|
|
if (nsLayoutAtoms::tableOuterFrame == frameType.get()) {
|
|
|
|
parentFrame = outerFrame;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rv = ConstructTableCaptionFrame(aPresShell, aPresContext, aState, aContent, parentFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
aStyleContext, newFrame, ignore, tableCreator);
|
1999-02-05 03:55:18 +00:00
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_ROW_GROUP:
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_HEADER_GROUP:
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_FOOTER_GROUP:
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_COLUMN_GROUP:
|
|
|
|
{
|
|
|
|
PRBool isRowGroup = (NS_STYLE_DISPLAY_TABLE_COLUMN_GROUP != aDisplay->mDisplay);
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableGroupFrame(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
aStyleContext, isRowGroup, newFrame, ignore, tableCreator);
|
1999-02-05 03:55:18 +00:00
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_COLUMN:
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableColFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, aStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
newFrame, ignore, tableCreator);
|
1999-02-05 03:55:18 +00:00
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_ROW:
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableRowFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, aStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
newFrame, ignore, tableCreator);
|
1999-02-05 03:55:18 +00:00
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
case NS_STYLE_DISPLAY_TABLE_CELL:
|
1999-02-26 09:15:03 +00:00
|
|
|
{
|
|
|
|
nsIFrame* ignore2;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructTableCellFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, aStyleContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
newFrame, ignore, ignore2, tableCreator);
|
1999-02-26 09:15:03 +00:00
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
return rv;
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
// Don't create any frame for content that's not displayed...
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-02-17 17:02:27 +00:00
|
|
|
// If the frame is absolutely positioned, then create a placeholder frame
|
1999-03-17 18:17:24 +00:00
|
|
|
nearly_done:
|
1999-02-05 03:55:18 +00:00
|
|
|
if (isAbsolutelyPositioned || isFixedPositioned) {
|
|
|
|
nsIFrame* placeholderFrame;
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
CreatePlaceholderFrameFor(aPresShell, aPresContext, aState.mFrameManager, aContent,
|
1999-08-05 03:09:22 +00:00
|
|
|
newFrame, aStyleContext, aParentFrame, &placeholderFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Add the positioned frame to its containing block's list of child frames
|
|
|
|
if (isAbsolutelyPositioned) {
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mAbsoluteItems.AddChild(newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
} else {
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mFixedItems.AddChild(newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add the placeholder frame to the flow
|
|
|
|
aFrameItems.AddChild(placeholderFrame);
|
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
} else if (isFloating) {
|
|
|
|
nsIFrame* placeholderFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
CreatePlaceholderFrameFor(aPresShell, aPresContext, aState.mFrameManager, aContent, newFrame,
|
1999-04-25 16:58:42 +00:00
|
|
|
aStyleContext, aParentFrame, &placeholderFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
|
|
|
|
// Add the floating frame to its containing block's list of child frames
|
1999-04-28 19:08:14 +00:00
|
|
|
aState.mFloatedItems.AddChild(newFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
|
|
|
|
// Add the placeholder frame to the flow
|
|
|
|
aFrameItems.AddChild(placeholderFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
} else if (nsnull != newFrame) {
|
|
|
|
// Add the frame we just created to the flowed list
|
|
|
|
aFrameItems.AddChild(newFrame);
|
1999-11-01 15:24:57 +00:00
|
|
|
if (newBlock) {
|
|
|
|
aFrameItems.AddChild(newBlock);
|
|
|
|
if (nextInline) {
|
|
|
|
aFrameItems.AddChild(nextInline);
|
|
|
|
}
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
1999-07-06 03:52:33 +00:00
|
|
|
if (newFrame && addToHashTable) {
|
1999-07-01 14:34:35 +00:00
|
|
|
// Add a mapping from content object to primary frame. Note that for
|
|
|
|
// floated and positioned frames this is the out-of-flow frame and not
|
|
|
|
// the placeholder frame
|
1999-10-12 00:16:06 +00:00
|
|
|
if (!primaryFrameSet)
|
|
|
|
aState.mFrameManager->SetPrimaryFrameFor(aContent, newFrame);
|
1999-07-01 14:34:35 +00:00
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
2000-01-22 01:16:50 +00:00
|
|
|
nsCSSFrameConstructor::GetAdjustedParentFrame(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aCurrentParentFrame,
|
|
|
|
PRUint8 aChildDisplayType,
|
|
|
|
nsIFrame*& aNewParentFrame)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
NS_PRECONDITION(nsnull!=aCurrentParentFrame, "bad arg aCurrentParentFrame");
|
|
|
|
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
// by default, the new parent frame is the given current parent frame
|
|
|
|
aNewParentFrame = aCurrentParentFrame;
|
|
|
|
if (nsnull != aCurrentParentFrame) {
|
|
|
|
const nsStyleDisplay* currentParentDisplay;
|
|
|
|
aCurrentParentFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct *&)currentParentDisplay);
|
|
|
|
if (NS_STYLE_DISPLAY_TABLE == currentParentDisplay->mDisplay) {
|
|
|
|
if (NS_STYLE_DISPLAY_TABLE_CAPTION != aChildDisplayType) {
|
|
|
|
nsIFrame *innerTableFrame = nsnull;
|
2000-01-22 01:16:50 +00:00
|
|
|
aCurrentParentFrame->FirstChild(aPresContext, nsnull, &innerTableFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
if (nsnull != innerTableFrame) {
|
|
|
|
const nsStyleDisplay* innerTableDisplay;
|
|
|
|
innerTableFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct *&)innerTableDisplay);
|
|
|
|
if (NS_STYLE_DISPLAY_TABLE == innerTableDisplay->mDisplay) {
|
|
|
|
// we were given the outer table frame, use the inner table frame
|
|
|
|
aNewParentFrame=innerTableFrame;
|
|
|
|
} // else we were already given the inner table frame
|
|
|
|
} // else the current parent has no children and cannot be an outer table frame
|
1999-03-23 14:16:52 +00:00
|
|
|
} else { // else the child is a caption and really belongs to the outer table frame
|
|
|
|
nsIFrame* parFrame = nsnull;
|
|
|
|
aCurrentParentFrame->GetParent(&parFrame);
|
|
|
|
const nsStyleDisplay* parDisplay;
|
|
|
|
aCurrentParentFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct *&)parDisplay);
|
|
|
|
if (NS_STYLE_DISPLAY_TABLE == parDisplay->mDisplay) {
|
|
|
|
aNewParentFrame = parFrame; // aNewParentFrame was an inner frame
|
|
|
|
}
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
rv = NS_ERROR_NULL_POINTER;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_POSTCONDITION(nsnull!=aNewParentFrame, "bad result null aNewParentFrame");
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsCSSFrameConstructor::IsScrollable(nsIPresContext* aPresContext,
|
|
|
|
const nsStyleDisplay* aDisplay)
|
|
|
|
{
|
|
|
|
// For the time being it's scrollable if the overflow property is auto or
|
|
|
|
// scroll, regardless of whether the width or height is fixed in size
|
2000-01-10 03:23:44 +00:00
|
|
|
switch (aDisplay->mOverflow) {
|
|
|
|
case NS_STYLE_OVERFLOW_SCROLL:
|
|
|
|
case NS_STYLE_OVERFLOW_AUTO:
|
|
|
|
case NS_STYLE_OVERFLOW_SCROLLBARS_NONE:
|
|
|
|
case NS_STYLE_OVERFLOW_SCROLLBARS_HORIZONTAL:
|
|
|
|
case NS_STYLE_OVERFLOW_SCROLLBARS_VERTICAL:
|
|
|
|
return PR_TRUE;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
1999-02-11 23:08:28 +00:00
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::InitAndRestoreFrame(nsIPresContext* aPresContext,
|
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame* aPrevInFlow,
|
|
|
|
nsIFrame* aNewFrame)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
NS_ASSERTION(aNewFrame, "Null frame cannot be initialized");
|
|
|
|
|
|
|
|
// Initialize the frame
|
|
|
|
rv = aNewFrame->Init(aPresContext, aContent, aParentFrame,
|
|
|
|
aStyleContext, aPrevInFlow);
|
|
|
|
|
|
|
|
if (aState.mFrameState && aState.mFrameManager) {
|
1999-12-06 09:03:16 +00:00
|
|
|
aState.mFrameManager->RestoreFrameState(aPresContext, aNewFrame, aState.mFrameState);
|
1999-12-06 07:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-02-11 23:08:28 +00:00
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::ResolveStyleContext(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIAtom* aTag,
|
|
|
|
nsIStyleContext** aStyleContext)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
// Resolve the style context based on the content object and the parent
|
|
|
|
// style context
|
|
|
|
nsCOMPtr<nsIStyleContext> parentStyleContext;
|
|
|
|
|
|
|
|
aParentFrame->GetStyleContext(getter_AddRefs(parentStyleContext));
|
|
|
|
if (nsLayoutAtoms::textTagName == aTag) {
|
|
|
|
// Use a special pseudo element style context for text
|
|
|
|
nsCOMPtr<nsIContent> parentContent;
|
|
|
|
if (nsnull != aParentFrame) {
|
|
|
|
aParentFrame->GetContent(getter_AddRefs(parentContent));
|
|
|
|
}
|
|
|
|
rv = aPresContext->ResolvePseudoStyleContextFor(parentContent,
|
|
|
|
nsHTMLAtoms::textPseudo,
|
|
|
|
parentStyleContext,
|
1999-02-12 17:45:58 +00:00
|
|
|
PR_FALSE,
|
1999-02-11 23:08:28 +00:00
|
|
|
aStyleContext);
|
|
|
|
} else if (nsLayoutAtoms::commentTagName == aTag) {
|
|
|
|
// Use a special pseudo element style context for comments
|
|
|
|
nsCOMPtr<nsIContent> parentContent;
|
|
|
|
if (nsnull != aParentFrame) {
|
|
|
|
aParentFrame->GetContent(getter_AddRefs(parentContent));
|
|
|
|
}
|
|
|
|
rv = aPresContext->ResolvePseudoStyleContextFor(parentContent,
|
|
|
|
nsHTMLAtoms::commentPseudo,
|
|
|
|
parentStyleContext,
|
1999-02-12 17:45:58 +00:00
|
|
|
PR_FALSE,
|
1999-02-11 23:08:28 +00:00
|
|
|
aStyleContext);
|
1999-03-31 20:49:25 +00:00
|
|
|
} else if (nsLayoutAtoms::processingInstructionTagName == aTag) {
|
|
|
|
// Use a special pseudo element style context for comments
|
|
|
|
nsCOMPtr<nsIContent> parentContent;
|
|
|
|
if (nsnull != aParentFrame) {
|
|
|
|
aParentFrame->GetContent(getter_AddRefs(parentContent));
|
|
|
|
}
|
|
|
|
rv = aPresContext->ResolvePseudoStyleContextFor(parentContent,
|
|
|
|
nsHTMLAtoms::processingInstructionPseudo,
|
|
|
|
parentStyleContext,
|
|
|
|
PR_FALSE,
|
|
|
|
aStyleContext);
|
1999-02-11 23:08:28 +00:00
|
|
|
} else {
|
|
|
|
rv = aPresContext->ResolveStyleContextFor(aContent, parentStyleContext,
|
1999-02-12 17:45:58 +00:00
|
|
|
PR_FALSE,
|
1999-02-11 23:08:28 +00:00
|
|
|
aStyleContext);
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-09-21 01:15:30 +00:00
|
|
|
// MathML Mod - RBS
|
1999-10-02 10:41:40 +00:00
|
|
|
#ifdef MOZ_MATHML
|
1999-09-21 01:15:30 +00:00
|
|
|
nsresult
|
1999-12-10 13:07:59 +00:00
|
|
|
nsCSSFrameConstructor::ConstructMathMLFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-09-21 01:15:30 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIAtom* aTag,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsFrameItems& aFrameItems)
|
|
|
|
{
|
|
|
|
PRBool processChildren = PR_TRUE; // Whether we should process child content.
|
|
|
|
// MathML frames are inline frames.
|
|
|
|
// processChildren = PR_TRUE for inline frames.
|
|
|
|
// see case NS_STYLE_DISPLAY_INLINE in
|
|
|
|
// ConstructFrameByDisplayType()
|
|
|
|
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
PRBool isAbsolutelyPositioned = PR_FALSE;
|
|
|
|
PRBool isFixedPositioned = PR_FALSE;
|
|
|
|
PRBool isReplaced = PR_FALSE;
|
|
|
|
|
|
|
|
NS_ASSERTION(aTag != nsnull, "null MathML tag");
|
|
|
|
if (aTag == nsnull)
|
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
// Make sure that we remain confined in the MathML world
|
|
|
|
PRInt32 nameSpaceID;
|
|
|
|
rv = aContent->GetNameSpaceID(nameSpaceID);
|
2000-01-07 15:25:15 +00:00
|
|
|
if (NS_FAILED(rv) || nameSpaceID != nsMathMLAtoms::nameSpaceID)
|
1999-09-21 01:15:30 +00:00
|
|
|
return NS_OK;
|
1999-10-02 10:41:40 +00:00
|
|
|
|
1999-09-21 01:15:30 +00:00
|
|
|
// Initialize the new frame
|
|
|
|
nsIFrame* newFrame = nsnull;
|
|
|
|
nsIFrame* ignore = nsnull;
|
1999-12-10 13:07:59 +00:00
|
|
|
nsMathMLmtableCreator mathTableCreator(aPresShell); // Used to make table views.
|
1999-09-21 01:15:30 +00:00
|
|
|
|
|
|
|
// See if the element is absolute or fixed positioned
|
|
|
|
const nsStylePosition* position = (const nsStylePosition*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Position);
|
|
|
|
if (NS_STYLE_POSITION_ABSOLUTE == position->mPosition) {
|
|
|
|
isAbsolutelyPositioned = PR_TRUE;
|
|
|
|
}
|
|
|
|
else if (NS_STYLE_POSITION_FIXED == position->mPosition) {
|
|
|
|
isFixedPositioned = PR_TRUE;
|
|
|
|
}
|
|
|
|
|
2000-01-07 15:25:15 +00:00
|
|
|
if (aTag == nsMathMLAtoms::mi_)
|
1999-12-10 13:07:59 +00:00
|
|
|
rv = NS_NewMathMLmiFrame(aPresShell, &newFrame);
|
2000-01-07 15:25:15 +00:00
|
|
|
else if (aTag == nsMathMLAtoms::mn_)
|
|
|
|
rv = NS_NewMathMLmnFrame(aPresShell, &newFrame);
|
1999-12-10 13:07:59 +00:00
|
|
|
else if (aTag == nsMathMLAtoms::mo_)
|
|
|
|
rv = NS_NewMathMLmoFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::mfrac_)
|
|
|
|
rv = NS_NewMathMLmfracFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::msup_)
|
|
|
|
rv = NS_NewMathMLmsupFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::msub_)
|
|
|
|
rv = NS_NewMathMLmsubFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::msubsup_)
|
|
|
|
rv = NS_NewMathMLmsubsupFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::munder_)
|
|
|
|
rv = NS_NewMathMLmunderFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::mover_)
|
|
|
|
rv = NS_NewMathMLmoverFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::munderover_)
|
|
|
|
rv = NS_NewMathMLmunderoverFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::mphantom_)
|
|
|
|
rv = NS_NewMathMLmphantomFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::mpadded_)
|
|
|
|
rv = NS_NewMathMLmpaddedFrame(aPresShell, &newFrame);
|
2000-01-27 12:28:41 +00:00
|
|
|
else if (aTag == nsMathMLAtoms::mspace_)
|
|
|
|
rv = NS_NewMathMLmspaceFrame(aPresShell, &newFrame);
|
2000-01-30 23:33:38 +00:00
|
|
|
else if (aTag == nsMathMLAtoms::ms_)
|
|
|
|
rv = NS_NewMathMLmsFrame(aPresShell, &newFrame);
|
1999-12-10 13:07:59 +00:00
|
|
|
else if (aTag == nsMathMLAtoms::mfenced_)
|
|
|
|
rv = NS_NewMathMLmfencedFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::mmultiscripts_)
|
|
|
|
rv = NS_NewMathMLmmultiscriptsFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::mstyle_)
|
|
|
|
rv = NS_NewMathMLmstyleFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::msqrt_)
|
|
|
|
rv = NS_NewMathMLmsqrtFrame(aPresShell, &newFrame);
|
|
|
|
else if (aTag == nsMathMLAtoms::mroot_)
|
|
|
|
rv = NS_NewMathMLmrootFrame(aPresShell, &newFrame);
|
2000-02-07 08:55:51 +00:00
|
|
|
else if (aTag == nsMathMLAtoms::maction_)
|
|
|
|
rv = NS_NewMathMLmactionFrame(aPresShell, &newFrame);
|
1999-12-10 13:07:59 +00:00
|
|
|
else if (aTag == nsMathMLAtoms::mrow_ ||
|
|
|
|
aTag == nsMathMLAtoms::mtext_ ||
|
|
|
|
aTag == nsMathMLAtoms::merror_ ||
|
|
|
|
aTag == nsMathMLAtoms::none_ ||
|
|
|
|
aTag == nsMathMLAtoms::mprescripts_ )
|
|
|
|
rv = NS_NewMathMLmrowFrame(aPresShell, &newFrame);
|
1999-09-21 01:15:30 +00:00
|
|
|
// CONSTRUCTION of MTABLE elements
|
1999-12-10 13:07:59 +00:00
|
|
|
else if (aTag == nsMathMLAtoms::mtable_) {
|
1999-09-21 01:15:30 +00:00
|
|
|
// <mtable> is an inline-table, for the moment, we just do what
|
|
|
|
// <table> does, and wait until nsLineLayout::TreatFrameAsBlock
|
|
|
|
// can handle NS_STYLE_DISPLAY_INLINE_TABLE.
|
|
|
|
nsIFrame* geometricParent = aParentFrame;
|
|
|
|
if (isAbsolutelyPositioned) {
|
|
|
|
aParentFrame = aState.mAbsoluteItems.containingBlock;
|
|
|
|
}
|
|
|
|
else if (isFixedPositioned) {
|
|
|
|
aParentFrame = aState.mFixedItems.containingBlock;
|
|
|
|
}
|
1999-12-10 13:07:59 +00:00
|
|
|
rv = ConstructTableFrame(aPresShell, aPresContext, aState, aContent, geometricParent, aStyleContext,
|
1999-09-21 01:15:30 +00:00
|
|
|
newFrame, mathTableCreator);
|
|
|
|
// Note: table construction function takes care of initializing the frame,
|
|
|
|
// processing children, and setting the initial child list
|
|
|
|
if (isAbsolutelyPositioned || isFixedPositioned) {
|
|
|
|
nsIFrame* placeholderFrame;
|
1999-12-10 13:07:59 +00:00
|
|
|
CreatePlaceholderFrameFor(aPresShell, aPresContext, aState.mFrameManager, aContent, newFrame,
|
1999-09-21 01:15:30 +00:00
|
|
|
aStyleContext, aParentFrame, &placeholderFrame);
|
|
|
|
// Add the positioned frame to its containing block's list of child frames
|
|
|
|
if (isAbsolutelyPositioned) {
|
|
|
|
aState.mAbsoluteItems.AddChild(newFrame);
|
|
|
|
} else {
|
|
|
|
aState.mFixedItems.AddChild(newFrame);
|
|
|
|
}
|
|
|
|
// Add the placeholder frame to the flow
|
|
|
|
aFrameItems.AddChild(placeholderFrame);
|
|
|
|
} else {
|
|
|
|
// Add the table frame to the flow
|
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
1999-12-10 13:07:59 +00:00
|
|
|
else if (aTag == nsMathMLAtoms::mtd_) {
|
1999-09-21 01:15:30 +00:00
|
|
|
nsIFrame* ignore2;
|
1999-12-10 13:07:59 +00:00
|
|
|
rv = ConstructTableCellFrame(aPresShell, aPresContext, aState, aContent, aParentFrame, aStyleContext,
|
1999-09-21 01:15:30 +00:00
|
|
|
newFrame, ignore, ignore2, mathTableCreator);
|
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
// End CONSTRUCTION of MTABLE elements
|
|
|
|
|
|
|
|
else {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we succeeded in creating a frame then initialize it, process its
|
|
|
|
// children (if requested), and set the initial child list
|
|
|
|
if (NS_SUCCEEDED(rv) && newFrame != nsnull) {
|
|
|
|
// If the frame is a replaced element, then set the frame state bit
|
|
|
|
if (isReplaced) {
|
|
|
|
nsFrameState state;
|
|
|
|
newFrame->GetFrameState(&state);
|
|
|
|
newFrame->SetFrameState(state | NS_FRAME_REPLACED_ELEMENT);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIFrame* geometricParent = isAbsolutelyPositioned
|
|
|
|
? aState.mAbsoluteItems.containingBlock
|
|
|
|
: aParentFrame;
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
geometricParent, aStyleContext, nsnull, newFrame);
|
1999-09-21 01:15:30 +00:00
|
|
|
|
|
|
|
// See if we need to create a view, e.g. the frame is absolutely positioned
|
1999-11-24 06:03:41 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-09-21 01:15:30 +00:00
|
|
|
aStyleContext, PR_FALSE);
|
|
|
|
|
|
|
|
// Add the new frame to our list of frame items.
|
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
|
|
|
|
// Process the child content if requested
|
|
|
|
nsFrameItems childItems;
|
|
|
|
if (processChildren) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ProcessChildren(aPresShell, aPresContext, aState, aContent, newFrame, PR_TRUE,
|
1999-09-21 01:15:30 +00:00
|
|
|
childItems, PR_FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the frame's initial child list
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-09-21 01:15:30 +00:00
|
|
|
|
|
|
|
// If the frame is absolutely positioned then create a placeholder frame
|
|
|
|
if (isAbsolutelyPositioned || isFixedPositioned) {
|
|
|
|
nsIFrame* placeholderFrame;
|
|
|
|
|
1999-12-10 13:07:59 +00:00
|
|
|
CreatePlaceholderFrameFor(aPresShell, aPresContext, aState.mFrameManager, aContent, newFrame,
|
1999-09-21 01:15:30 +00:00
|
|
|
aStyleContext, aParentFrame, &placeholderFrame);
|
|
|
|
|
|
|
|
// Add the positioned frame to its containing block's list of child frames
|
|
|
|
if (isAbsolutelyPositioned) {
|
|
|
|
aState.mAbsoluteItems.AddChild(newFrame);
|
|
|
|
} else {
|
|
|
|
aState.mFixedItems.AddChild(newFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the placeholder frame to the flow
|
|
|
|
aFrameItems.AddChild(placeholderFrame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
1999-10-02 10:41:40 +00:00
|
|
|
#endif // MOZ_MATHML
|
1999-09-21 01:15:30 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsFrameItems& aFrameItems)
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
NS_PRECONDITION(nsnull != aParentFrame, "no parent frame");
|
|
|
|
|
|
|
|
nsresult rv;
|
|
|
|
|
|
|
|
// Get the element's tag
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIAtom> tag;
|
|
|
|
aContent->GetTag(*getter_AddRefs(tag));
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-11 23:08:28 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
rv = ResolveStyleContext(aPresContext, aParentFrame, aContent, tag, getter_AddRefs(styleContext));
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
// Pre-check for display "none" - if we find that, don't create
|
|
|
|
// any frame at all
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
styleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
1999-10-02 04:27:40 +00:00
|
|
|
if (NS_STYLE_DISPLAY_NONE == display->mDisplay) {
|
|
|
|
aState.mFrameManager->SetUndisplayedContent(aContent, styleContext);
|
|
|
|
}
|
|
|
|
else {
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* lastChild = aFrameItems.lastChild;
|
|
|
|
|
|
|
|
// Handle specific frame types
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructFrameByTag(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-04-28 19:08:14 +00:00
|
|
|
tag, styleContext, aFrameItems);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
#ifdef INCLUDE_XUL
|
|
|
|
// Failing to find a matching HTML frame, try creating a specialized
|
|
|
|
// XUL frame. This is temporary, pending planned factoring of this
|
|
|
|
// whole process into separate, pluggable steps.
|
|
|
|
if (NS_SUCCEEDED(rv) && ((nsnull == aFrameItems.childList) ||
|
|
|
|
(lastChild == aFrameItems.lastChild))) {
|
|
|
|
PRBool haltProcessing = PR_FALSE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructXULFrame(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
2000-03-11 10:38:36 +00:00
|
|
|
tag, styleContext, aFrameItems, PR_FALSE, haltProcessing);
|
1999-02-05 03:55:18 +00:00
|
|
|
if (haltProcessing) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1999-09-21 01:15:30 +00:00
|
|
|
// MathML Mod - RBS
|
1999-10-02 10:41:40 +00:00
|
|
|
#ifdef MOZ_MATHML
|
1999-09-21 01:15:30 +00:00
|
|
|
if (NS_SUCCEEDED(rv) && ((nsnull == aFrameItems.childList) ||
|
|
|
|
(lastChild == aFrameItems.lastChild))) {
|
1999-12-10 13:07:59 +00:00
|
|
|
rv = ConstructMathMLFrame(aPresShell, aPresContext, aState, aContent, aParentFrame,
|
1999-09-21 01:15:30 +00:00
|
|
|
tag, styleContext, aFrameItems);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
if (NS_SUCCEEDED(rv) && ((nsnull == aFrameItems.childList) ||
|
|
|
|
(lastChild == aFrameItems.lastChild))) {
|
|
|
|
// When there is no explicit frame to create, assume it's a
|
|
|
|
// container and let display style dictate the rest
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructFrameByDisplayType(aPresShell, aPresContext, aState, display, aContent,
|
1999-11-01 15:24:57 +00:00
|
|
|
aParentFrame, styleContext, aFrameItems);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-03-01 16:57:35 +00:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::ReconstructDocElementHierarchy(nsIPresContext* aPresContext)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
|
|
|
|
1999-03-01 16:57:35 +00:00
|
|
|
if (nsnull != mDocument) {
|
|
|
|
nsCOMPtr<nsIContent> rootContent(dont_AddRef(mDocument->GetRootContent()));
|
1999-12-06 07:44:18 +00:00
|
|
|
|
1999-03-01 16:57:35 +00:00
|
|
|
if (rootContent) {
|
1999-12-06 07:44:18 +00:00
|
|
|
// Before removing the frames associated with the content object, ask them to save their
|
|
|
|
// state onto a temporary state object.
|
|
|
|
CaptureStateForFramesOf(aPresContext, rootContent, mTempFrameTreeState);
|
|
|
|
|
1999-08-05 03:09:22 +00:00
|
|
|
nsFrameConstructorState state(aPresContext, mFixedContainingBlock,
|
1999-12-06 07:44:18 +00:00
|
|
|
nsnull, nsnull, mTempFrameTreeState);
|
1999-08-05 03:09:22 +00:00
|
|
|
nsIFrame* docElementFrame;
|
1999-03-01 16:57:35 +00:00
|
|
|
|
1999-08-05 03:09:22 +00:00
|
|
|
// Get the frame that corresponds to the document element
|
|
|
|
state.mFrameManager->GetPrimaryFrameFor(rootContent, &docElementFrame);
|
|
|
|
|
|
|
|
// Clear the hash tables that map from content to frame and out-of-flow
|
|
|
|
// frame to placeholder frame
|
|
|
|
state.mFrameManager->ClearPrimaryFrameMap();
|
|
|
|
state.mFrameManager->ClearPlaceholderFrameMap();
|
1999-10-02 04:27:40 +00:00
|
|
|
state.mFrameManager->ClearUndisplayedContentMap();
|
1999-08-05 03:09:22 +00:00
|
|
|
|
|
|
|
if (docElementFrame) {
|
|
|
|
nsIFrame* docParentFrame;
|
|
|
|
docElementFrame->GetParent(&docParentFrame);
|
|
|
|
|
|
|
|
if (docParentFrame) {
|
|
|
|
// Remove the old document element hieararchy
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mFrameManager->RemoveFrame(aPresContext, *shell,
|
1999-08-05 03:09:22 +00:00
|
|
|
docParentFrame, nsnull,
|
|
|
|
docElementFrame);
|
1999-12-06 07:44:18 +00:00
|
|
|
// XXX Remove any existing fixed items...
|
1999-08-05 03:09:22 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-12-06 07:44:18 +00:00
|
|
|
// Create the new document element hierarchy
|
1999-08-05 03:09:22 +00:00
|
|
|
nsIFrame* newChild;
|
|
|
|
nsCOMPtr<nsIStyleContext> rootPseudoStyle;
|
1999-03-01 16:57:35 +00:00
|
|
|
|
1999-08-05 03:09:22 +00:00
|
|
|
docParentFrame->GetStyleContext(getter_AddRefs(rootPseudoStyle));
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructDocElementFrame(shell, aPresContext, state, rootContent,
|
1999-08-05 03:09:22 +00:00
|
|
|
docParentFrame, rootPseudoStyle,
|
|
|
|
newChild);
|
|
|
|
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mFrameManager->InsertFrames(aPresContext, *shell,
|
1999-08-05 03:09:22 +00:00
|
|
|
docParentFrame, nsnull,
|
|
|
|
nsnull, newChild);
|
|
|
|
|
|
|
|
// Tell the fixed containing block about its 'fixed' frames
|
|
|
|
if (state.mFixedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
state.mFrameManager->InsertFrames(aPresContext, *shell,
|
1999-08-05 03:09:22 +00:00
|
|
|
mFixedContainingBlock, nsLayoutAtoms::fixedList,
|
|
|
|
nsnull, state.mFixedItems.childList);
|
1999-03-01 16:57:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
1999-03-01 16:57:35 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-03-01 16:57:35 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame*
|
1999-08-05 03:09:22 +00:00
|
|
|
nsCSSFrameConstructor::GetFrameFor(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
// Get the primary frame associated with the content
|
|
|
|
nsIFrame* frame;
|
1999-02-12 17:45:58 +00:00
|
|
|
aPresShell->GetPrimaryFrameFor(aContent, &frame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
if (nsnull != frame) {
|
1999-08-23 14:05:39 +00:00
|
|
|
// Check to see if the content is a select and
|
|
|
|
// then if it has a drop down (thus making it a combobox)
|
|
|
|
// The drop down is a ListControlFrame derived from a
|
1999-08-25 13:42:59 +00:00
|
|
|
// nsScrollFrame then get the area frame and that will be the parent
|
|
|
|
// What is unclear here, is if any of this fails, should it return
|
|
|
|
// the nsComboboxControlFrame or null?
|
1999-08-23 14:05:39 +00:00
|
|
|
nsCOMPtr<nsIDOMHTMLSelectElement> selectElement;
|
2000-02-02 22:24:56 +00:00
|
|
|
nsresult res = aContent->QueryInterface(NS_GET_IID(nsIDOMHTMLSelectElement),
|
1999-08-23 14:05:39 +00:00
|
|
|
(void**)getter_AddRefs(selectElement));
|
|
|
|
if (NS_SUCCEEDED(res) && selectElement) {
|
|
|
|
nsIComboboxControlFrame * comboboxFrame;
|
2000-02-02 22:24:56 +00:00
|
|
|
res = frame->QueryInterface(NS_GET_IID(nsIComboboxControlFrame),
|
1999-08-23 14:05:39 +00:00
|
|
|
(void**)&comboboxFrame);
|
1999-08-25 13:42:59 +00:00
|
|
|
nsIFrame * listFrame;
|
1999-08-23 20:59:08 +00:00
|
|
|
if (NS_SUCCEEDED(res) && comboboxFrame) {
|
1999-08-23 14:05:39 +00:00
|
|
|
comboboxFrame->GetDropDown(&listFrame);
|
1999-08-25 13:42:59 +00:00
|
|
|
} else {
|
2000-03-18 14:25:02 +00:00
|
|
|
listFrame = frame;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (listFrame != nsnull) {
|
|
|
|
nsIListControlFrame * list;
|
|
|
|
res = listFrame->QueryInterface(NS_GET_IID(nsIListControlFrame),
|
|
|
|
(void**)&list);
|
|
|
|
if (NS_SUCCEEDED(res) && list) {
|
|
|
|
list->GetOptionsContainer(aPresContext, &frame);
|
1999-08-25 13:42:59 +00:00
|
|
|
}
|
1999-08-23 14:05:39 +00:00
|
|
|
}
|
|
|
|
} else {
|
2000-01-27 02:19:12 +00:00
|
|
|
// If the primary frame is a scroll frame, then get the scrolled frame.
|
|
|
|
// That's the frame that gets the reflow command
|
2000-01-27 20:21:28 +00:00
|
|
|
const nsStyleDisplay* display;
|
|
|
|
frame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct*&)display);
|
|
|
|
|
2000-02-11 01:23:36 +00:00
|
|
|
// If the primary frame supports IScrollableFrame, then get the scrolled frame.
|
|
|
|
// That's the frame that gets the reflow command
|
|
|
|
nsIScrollableFrame *pScrollableFrame = nsnull;
|
|
|
|
if (NS_SUCCEEDED( frame->QueryInterface(nsIScrollableFrame::GetIID(),
|
|
|
|
(void **)&pScrollableFrame) ))
|
|
|
|
{
|
|
|
|
pScrollableFrame->GetScrolledFrame( aPresContext, frame );
|
2000-01-27 20:21:28 +00:00
|
|
|
}
|
2000-02-11 01:23:36 +00:00
|
|
|
|
1999-08-23 14:05:39 +00:00
|
|
|
// if we get an outer table frame use its 1st child which is a table inner frame
|
|
|
|
// if we get a table cell frame use its 1st child which is an area frame
|
|
|
|
else if ((NS_STYLE_DISPLAY_TABLE == display->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_TABLE_CELL == display->mDisplay)) {
|
2000-01-22 01:16:50 +00:00
|
|
|
frame->FirstChild(aPresContext, nsnull, &frame);
|
1999-08-23 14:05:39 +00:00
|
|
|
}
|
1999-05-06 22:29:28 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return frame;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIFrame*
|
|
|
|
nsCSSFrameConstructor::GetAbsoluteContainingBlock(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
NS_PRECONDITION(nsnull != mInitialContainingBlock, "no initial containing block");
|
|
|
|
|
1999-04-20 22:08:33 +00:00
|
|
|
// Starting with aFrame, look for a frame that is absolutely positioned or
|
|
|
|
// relatively positioned
|
1999-04-20 21:51:55 +00:00
|
|
|
nsIFrame* containingBlock = nsnull;
|
|
|
|
for (nsIFrame* frame = aFrame; frame; frame->GetParent(&frame)) {
|
1999-02-05 03:55:18 +00:00
|
|
|
const nsStylePosition* position;
|
|
|
|
|
1999-04-20 22:08:33 +00:00
|
|
|
// Is it positioned?
|
1999-04-20 21:51:55 +00:00
|
|
|
frame->GetStyleData(eStyleStruct_Position, (const nsStyleStruct*&)position);
|
1999-04-20 22:08:33 +00:00
|
|
|
if ((position->mPosition == NS_STYLE_POSITION_ABSOLUTE) ||
|
|
|
|
(position->mPosition == NS_STYLE_POSITION_RELATIVE)) {
|
1999-02-05 03:55:18 +00:00
|
|
|
const nsStyleDisplay* display;
|
|
|
|
|
|
|
|
// If it's a table then ignore it, because for the time being tables
|
|
|
|
// are not containers for absolutely positioned child frames
|
1999-04-20 21:51:55 +00:00
|
|
|
frame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct*&)display);
|
1999-02-05 03:55:18 +00:00
|
|
|
if (display->mDisplay != NS_STYLE_DISPLAY_TABLE) {
|
1999-04-20 21:51:55 +00:00
|
|
|
nsIAtom* frameType;
|
|
|
|
frame->GetFrameType(&frameType);
|
|
|
|
|
|
|
|
if (nsLayoutAtoms::scrollFrame == frameType) {
|
|
|
|
// We want the scrolled frame, not the scroll frame
|
|
|
|
nsIFrame* scrolledFrame;
|
2000-01-22 01:16:50 +00:00
|
|
|
frame->FirstChild(aPresContext, nsnull, &scrolledFrame);
|
1999-04-20 21:51:55 +00:00
|
|
|
NS_RELEASE(frameType);
|
|
|
|
if (scrolledFrame) {
|
|
|
|
scrolledFrame->GetFrameType(&frameType);
|
|
|
|
if (nsLayoutAtoms::areaFrame == frameType) {
|
|
|
|
containingBlock = scrolledFrame;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-04-20 22:08:33 +00:00
|
|
|
} else if ((nsLayoutAtoms::areaFrame == frameType) ||
|
|
|
|
(nsLayoutAtoms::positionedInlineFrame == frameType)) {
|
1999-04-20 21:51:55 +00:00
|
|
|
containingBlock = frame;
|
|
|
|
}
|
|
|
|
NS_RELEASE(frameType);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-04-20 21:51:55 +00:00
|
|
|
// See if we found a containing block
|
|
|
|
if (containingBlock) {
|
|
|
|
break;
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If we didn't find an absolutely positioned containing block, then use the
|
|
|
|
// initial containing block
|
1999-04-20 21:51:55 +00:00
|
|
|
if (!containingBlock) {
|
1999-02-05 03:55:18 +00:00
|
|
|
containingBlock = mInitialContainingBlock;
|
|
|
|
}
|
|
|
|
|
|
|
|
return containingBlock;
|
|
|
|
}
|
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
nsIFrame*
|
|
|
|
nsCSSFrameConstructor::GetFloaterContainingBlock(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
NS_PRECONDITION(mInitialContainingBlock, "no initial containing block");
|
|
|
|
|
1999-09-24 17:14:19 +00:00
|
|
|
// Starting with aFrame, look for a frame that is a real block frame,
|
|
|
|
// or a floated inline or absolutely positioned inline frame
|
1999-02-26 17:11:54 +00:00
|
|
|
nsIFrame* containingBlock = aFrame;
|
|
|
|
while (nsnull != containingBlock) {
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
containingBlock->GetStyleData(eStyleStruct_Display,
|
|
|
|
(const nsStyleStruct*&)display);
|
|
|
|
if ((NS_STYLE_DISPLAY_BLOCK == display->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_LIST_ITEM == display->mDisplay)) {
|
|
|
|
break;
|
|
|
|
}
|
1999-09-24 17:14:19 +00:00
|
|
|
else if (NS_STYLE_DISPLAY_INLINE == display->mDisplay) {
|
|
|
|
const nsStylePosition* position;
|
|
|
|
containingBlock->GetStyleData(eStyleStruct_Position,
|
|
|
|
(const nsStyleStruct*&)position);
|
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
if ((NS_STYLE_FLOAT_NONE != display->mFloats) ||
|
1999-09-24 17:14:19 +00:00
|
|
|
(position->IsAbsolutelyPositioned())) {
|
1999-10-29 14:39:48 +00:00
|
|
|
if (NS_STYLE_FLOAT_NONE != display->mFloats) {
|
|
|
|
nsCOMPtr<nsIAtom> frameType;
|
|
|
|
containingBlock->GetFrameType(getter_AddRefs(frameType));
|
|
|
|
if (nsLayoutAtoms::letterFrame != frameType.get()) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
break;
|
|
|
|
}
|
1999-09-24 17:14:19 +00:00
|
|
|
}
|
|
|
|
}
|
1999-02-26 17:11:54 +00:00
|
|
|
|
|
|
|
// Continue walking up the hierarchy
|
|
|
|
containingBlock->GetParent(&containingBlock);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we didn't find a containing block, then use the initial
|
|
|
|
// containing block
|
|
|
|
if (nsnull == containingBlock) {
|
|
|
|
containingBlock = mInitialContainingBlock;
|
|
|
|
}
|
|
|
|
return containingBlock;
|
|
|
|
}
|
|
|
|
|
1999-04-07 03:10:59 +00:00
|
|
|
// Helper function to determine whether a given frame is generated content
|
|
|
|
// for the specified content object. Returns PR_TRUE if the frame is associated
|
|
|
|
// with generated content and PR_FALSE otherwise
|
|
|
|
static inline PRBool
|
1999-09-02 05:21:39 +00:00
|
|
|
IsGeneratedContentFor(nsIContent* aContent, nsIFrame* aFrame, nsIAtom* aPseudoElement)
|
1999-04-07 03:10:59 +00:00
|
|
|
{
|
1999-07-16 02:14:26 +00:00
|
|
|
NS_PRECONDITION(aFrame, "null frame pointer");
|
1999-04-07 03:10:59 +00:00
|
|
|
nsFrameState state;
|
|
|
|
PRBool result = PR_FALSE;
|
|
|
|
|
|
|
|
// First check the frame state bit
|
|
|
|
aFrame->GetFrameState(&state);
|
|
|
|
if (state & NS_FRAME_GENERATED_CONTENT) {
|
|
|
|
nsIContent* content;
|
|
|
|
|
|
|
|
// Check that it has the same content pointer
|
|
|
|
aFrame->GetContent(&content);
|
1999-09-02 05:21:39 +00:00
|
|
|
if (content == aContent) {
|
|
|
|
nsIStyleContext* styleContext;
|
|
|
|
nsIAtom* pseudoType;
|
|
|
|
|
|
|
|
// See if the pseudo element type matches
|
|
|
|
aFrame->GetStyleContext(&styleContext);
|
|
|
|
styleContext->GetPseudoType(pseudoType);
|
|
|
|
result = (pseudoType == aPseudoElement);
|
|
|
|
NS_RELEASE(styleContext);
|
|
|
|
NS_IF_RELEASE(pseudoType);
|
|
|
|
}
|
1999-04-07 03:10:59 +00:00
|
|
|
NS_IF_RELEASE(content);
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
1999-04-06 04:58:05 +00:00
|
|
|
// This function is called by ContentAppended() and ContentInserted() when
|
|
|
|
// appending flowed frames to a parent's principal child list. It handles the
|
|
|
|
// case where the parent frame has :after pseudo-element generated content
|
|
|
|
nsresult
|
1999-08-05 03:09:22 +00:00
|
|
|
nsCSSFrameConstructor::AppendFrames(nsIPresContext* aPresContext,
|
|
|
|
nsIPresShell* aPresShell,
|
|
|
|
nsIFrameManager* aFrameManager,
|
|
|
|
nsIContent* aContainer,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIFrame* aFrameList)
|
1999-04-06 04:58:05 +00:00
|
|
|
{
|
|
|
|
nsIFrame* firstChild;
|
2000-01-22 01:16:50 +00:00
|
|
|
aParentFrame->FirstChild(aPresContext, nsnull, &firstChild);
|
1999-04-06 04:58:05 +00:00
|
|
|
nsFrameList frames(firstChild);
|
|
|
|
nsIFrame* lastChild = frames.LastChild();
|
|
|
|
|
|
|
|
// See if the parent has an :after pseudo-element
|
1999-09-02 05:21:39 +00:00
|
|
|
if (lastChild && IsGeneratedContentFor(aContainer, lastChild, nsCSSAtoms::afterPseudo)) {
|
1999-04-07 03:10:59 +00:00
|
|
|
// Insert the frames before the :after pseudo-element
|
1999-11-24 06:03:41 +00:00
|
|
|
return aFrameManager->InsertFrames(aPresContext, *aPresShell, aParentFrame,
|
1999-08-05 03:09:22 +00:00
|
|
|
nsnull, frames.GetPrevSiblingFor(lastChild),
|
|
|
|
aFrameList);
|
1999-04-06 04:58:05 +00:00
|
|
|
}
|
|
|
|
|
2000-01-28 02:19:45 +00:00
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
1999-12-13 22:56:31 +00:00
|
|
|
// a col group or col appended to a table may result in an insert rather than an append
|
|
|
|
nsIAtom* parentType;
|
|
|
|
aParentFrame->GetFrameType(&parentType);
|
|
|
|
if (nsLayoutAtoms::tableFrame == parentType) {
|
|
|
|
nsTableFrame* tableFrame = (nsTableFrame *)aParentFrame;
|
|
|
|
nsIAtom* childType;
|
|
|
|
aFrameList->GetFrameType(&childType);
|
|
|
|
if (nsLayoutAtoms::tableColFrame == childType) {
|
2000-01-28 02:19:45 +00:00
|
|
|
// table column
|
1999-12-13 22:56:31 +00:00
|
|
|
nsIFrame* parentFrame = aParentFrame;
|
|
|
|
aFrameList->GetParent(&parentFrame);
|
2000-01-28 02:19:45 +00:00
|
|
|
NS_RELEASE(childType);
|
|
|
|
rv = aFrameManager->AppendFrames(aPresContext, *aPresShell, parentFrame,
|
|
|
|
nsLayoutAtoms::colGroupList, aFrameList);
|
1999-12-13 22:56:31 +00:00
|
|
|
}
|
|
|
|
else if (nsLayoutAtoms::tableColGroupFrame == childType) {
|
2000-01-28 02:19:45 +00:00
|
|
|
// table col group
|
1999-12-13 22:56:31 +00:00
|
|
|
nsIFrame* prevSibling;
|
|
|
|
PRBool doAppend = nsTableColGroupFrame::GetLastRealColGroup(tableFrame, &prevSibling);
|
|
|
|
if (doAppend) {
|
2000-01-28 02:19:45 +00:00
|
|
|
rv = aFrameManager->AppendFrames(aPresContext, *aPresShell, aParentFrame,
|
|
|
|
nsLayoutAtoms::colGroupList, aFrameList);
|
1999-12-13 22:56:31 +00:00
|
|
|
}
|
|
|
|
else {
|
2000-01-28 02:19:45 +00:00
|
|
|
rv = aFrameManager->InsertFrames(aPresContext, *aPresShell, aParentFrame,
|
|
|
|
nsLayoutAtoms::colGroupList, prevSibling, aFrameList);
|
1999-12-13 22:56:31 +00:00
|
|
|
}
|
|
|
|
}
|
2000-01-28 02:19:45 +00:00
|
|
|
else if (nsLayoutAtoms::tableCaptionFrame == childType) {
|
|
|
|
// table caption
|
|
|
|
rv = aFrameManager->AppendFrames(aPresContext, *aPresShell, aParentFrame,
|
|
|
|
nsLayoutAtoms::captionList, aFrameList);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
rv = aFrameManager->AppendFrames(aPresContext, *aPresShell, aParentFrame,
|
|
|
|
nsnull, aFrameList);
|
|
|
|
}
|
1999-12-13 22:56:31 +00:00
|
|
|
NS_IF_RELEASE(childType);
|
|
|
|
}
|
2000-01-28 02:19:45 +00:00
|
|
|
else {
|
|
|
|
// Append the frames to the end of the parent's child list
|
|
|
|
rv = aFrameManager->AppendFrames(aPresContext, *aPresShell, aParentFrame,
|
2000-02-04 03:16:47 +00:00
|
|
|
GetChildListFor(aFrameList), aFrameList);
|
2000-01-28 02:19:45 +00:00
|
|
|
}
|
1999-12-13 22:56:31 +00:00
|
|
|
NS_IF_RELEASE(parentType);
|
|
|
|
|
2000-01-28 02:19:45 +00:00
|
|
|
return rv;
|
1999-04-06 04:58:05 +00:00
|
|
|
}
|
|
|
|
|
2000-01-06 08:58:05 +00:00
|
|
|
static nsIFrame*
|
|
|
|
FindPreviousSibling(nsIPresShell* aPresShell,
|
|
|
|
nsIContent* aContainer,
|
|
|
|
PRInt32 aIndexInContainer)
|
|
|
|
{
|
|
|
|
nsIFrame* prevSibling = nsnull;
|
|
|
|
|
|
|
|
// Note: not all content objects are associated with a frame (e.g., if their
|
|
|
|
// 'display' type is 'hidden') so keep looking until we find a previous frame
|
|
|
|
for (PRInt32 i = aIndexInContainer - 1; i >= 0; i--) {
|
|
|
|
nsCOMPtr<nsIContent> precedingContent;
|
|
|
|
|
|
|
|
aContainer->ChildAt(i, *getter_AddRefs(precedingContent));
|
|
|
|
aPresShell->GetPrimaryFrameFor(precedingContent, &prevSibling);
|
|
|
|
|
|
|
|
if (nsnull != prevSibling) {
|
|
|
|
// The frame may have a next-in-flow. Get the last-in-flow
|
|
|
|
nsIFrame* nextInFlow;
|
|
|
|
do {
|
|
|
|
prevSibling->GetNextInFlow(&nextInFlow);
|
|
|
|
if (nsnull != nextInFlow) {
|
|
|
|
prevSibling = nextInFlow;
|
|
|
|
}
|
|
|
|
} while (nsnull != nextInFlow);
|
|
|
|
|
|
|
|
// Did we really get the *right* frame?
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
prevSibling->GetStyleData(eStyleStruct_Display,
|
|
|
|
(const nsStyleStruct*&)display);
|
|
|
|
if (display->IsFloating()) {
|
|
|
|
// Nope. Get the place-holder instead
|
|
|
|
nsIFrame* placeholderFrame;
|
|
|
|
aPresShell->GetPlaceholderFrameFor(prevSibling, &placeholderFrame);
|
|
|
|
NS_ASSERTION(nsnull != placeholderFrame, "yikes");
|
|
|
|
prevSibling = placeholderFrame;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return prevSibling;
|
|
|
|
}
|
|
|
|
|
|
|
|
static nsIFrame*
|
|
|
|
FindNextSibling(nsIPresShell* aPresShell,
|
|
|
|
nsIContent* aContainer,
|
|
|
|
PRInt32 aIndexInContainer)
|
|
|
|
{
|
|
|
|
nsIFrame* nextSibling = nsnull;
|
|
|
|
|
|
|
|
// Note: not all content objects are associated with a frame (e.g., if their
|
|
|
|
// 'display' type is 'hidden') so keep looking until we find a previous frame
|
|
|
|
PRInt32 count;
|
|
|
|
aContainer->ChildCount(count);
|
|
|
|
for (PRInt32 i = aIndexInContainer + 1; i < count; i++) {
|
|
|
|
nsCOMPtr<nsIContent> nextContent;
|
|
|
|
|
|
|
|
aContainer->ChildAt(i, *getter_AddRefs(nextContent));
|
|
|
|
aPresShell->GetPrimaryFrameFor(nextContent, &nextSibling);
|
|
|
|
|
|
|
|
if (nsnull != nextSibling) {
|
|
|
|
// The frame may have a next-in-flow. Get the first-in-flow
|
|
|
|
nsIFrame* prevInFlow;
|
|
|
|
do {
|
|
|
|
nextSibling->GetPrevInFlow(&prevInFlow);
|
|
|
|
if (nsnull != prevInFlow) {
|
|
|
|
nextSibling = prevInFlow;
|
|
|
|
}
|
|
|
|
} while (nsnull != prevInFlow);
|
|
|
|
|
|
|
|
// Did we really get the *right* frame?
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
nextSibling->GetStyleData(eStyleStruct_Display,
|
|
|
|
(const nsStyleStruct*&)display);
|
|
|
|
if (display->IsFloating()) {
|
|
|
|
// Nope. Get the place-holder instead
|
|
|
|
nsIFrame* placeholderFrame;
|
|
|
|
aPresShell->GetPlaceholderFrameFor(nextSibling, &placeholderFrame);
|
|
|
|
NS_ASSERTION(nsnull != placeholderFrame, "yikes");
|
|
|
|
nextSibling = placeholderFrame;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nextSibling;
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::ContentAppended(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContainer,
|
|
|
|
PRInt32 aNewIndexInContainer)
|
|
|
|
{
|
1999-11-01 15:24:57 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (gNoisyContentUpdates) {
|
|
|
|
printf("nsCSSFrameConstructor::ContentAppended container=%p index=%d\n",
|
|
|
|
aContainer, aNewIndexInContainer);
|
|
|
|
if (gReallyNoisyContentUpdates && aContainer) {
|
|
|
|
aContainer->List(stdout, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1999-08-05 03:09:22 +00:00
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
|
|
|
|
1999-07-20 23:24:10 +00:00
|
|
|
#ifdef INCLUDE_XUL
|
|
|
|
if (aContainer) {
|
|
|
|
nsCOMPtr<nsIAtom> tag;
|
|
|
|
aContainer->GetTag(*getter_AddRefs(tag));
|
|
|
|
if (tag && (tag.get() == nsXULAtoms::treechildren ||
|
|
|
|
tag.get() == nsXULAtoms::treeitem)) {
|
|
|
|
// Walk up to the outermost tree row group frame and tell it that
|
|
|
|
// content was added.
|
1999-07-22 10:06:38 +00:00
|
|
|
nsCOMPtr<nsIContent> parent;
|
1999-07-20 23:24:10 +00:00
|
|
|
nsCOMPtr<nsIContent> child = dont_QueryInterface(aContainer);
|
1999-07-22 10:06:38 +00:00
|
|
|
child->GetParent(*getter_AddRefs(parent));
|
1999-07-20 23:24:10 +00:00
|
|
|
while (parent) {
|
|
|
|
parent->GetTag(*getter_AddRefs(tag));
|
|
|
|
if (tag.get() == nsXULAtoms::tree)
|
|
|
|
break;
|
|
|
|
child = parent;
|
1999-07-22 10:06:38 +00:00
|
|
|
child->GetParent(*getter_AddRefs(parent));
|
1999-07-20 23:24:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (parent) {
|
|
|
|
// We found it. Get the primary frame.
|
2000-01-06 08:58:05 +00:00
|
|
|
nsIFrame* outerFrame = GetFrameFor(shell, aPresContext, child);
|
1999-07-20 23:24:10 +00:00
|
|
|
|
|
|
|
// Convert to a tree row group frame.
|
2000-01-06 08:58:05 +00:00
|
|
|
nsTreeRowGroupFrame* treeRowGroup = (nsTreeRowGroupFrame*)outerFrame;
|
2000-01-13 19:20:25 +00:00
|
|
|
if (treeRowGroup) {
|
2000-01-06 08:58:05 +00:00
|
|
|
|
|
|
|
// Get the primary frame for the parent of the child that's being added.
|
|
|
|
nsIFrame* innerFrame = GetFrameFor(shell, aPresContext, aContainer);
|
|
|
|
|
|
|
|
// See if there's a previous sibling.
|
|
|
|
nsIFrame* prevSibling = FindPreviousSibling(shell,
|
|
|
|
aContainer,
|
|
|
|
aNewIndexInContainer);
|
|
|
|
|
2000-01-25 06:35:27 +00:00
|
|
|
// This needs to really be a previous sibling.
|
|
|
|
if (prevSibling && aNewIndexInContainer > 0) {
|
|
|
|
nsCOMPtr<nsIContent> prevContent;
|
|
|
|
nsCOMPtr<nsIContent> frameContent;
|
|
|
|
aContainer->ChildAt(aNewIndexInContainer-1, *getter_AddRefs(prevContent));
|
|
|
|
prevSibling->GetContent(getter_AddRefs(frameContent));
|
|
|
|
if (frameContent.get() != prevContent.get())
|
|
|
|
prevSibling = nsnull;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (prevSibling || (innerFrame && aNewIndexInContainer == 0)) {
|
2000-01-06 08:58:05 +00:00
|
|
|
// We're onscreen. Make sure a full reflow happens.
|
|
|
|
treeRowGroup->OnContentAdded(aPresContext);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// We're going to be offscreen.
|
|
|
|
treeRowGroup->ReflowScrollbar(aPresContext);
|
|
|
|
}
|
|
|
|
|
1999-07-20 23:24:10 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif // INCLUDE_XUL
|
|
|
|
|
1999-09-13 21:25:43 +00:00
|
|
|
// Get the frame associated with the content
|
1999-08-06 21:14:45 +00:00
|
|
|
nsIFrame* parentFrame = GetFrameFor(shell, aPresContext, aContainer);
|
1999-02-05 03:55:18 +00:00
|
|
|
if (nsnull != parentFrame) {
|
1999-11-01 15:24:57 +00:00
|
|
|
|
|
|
|
// If the frame we are manipulating is a special frame then do
|
|
|
|
// something different instead of just appending newly created
|
|
|
|
// frames. Note that only the first-in-flow is marked so we check
|
|
|
|
// before getting to the last-in-flow.
|
|
|
|
if (IsFrameSpecial(aPresContext, parentFrame)) {
|
|
|
|
// We are pretty harsh here (and definitely not optimal) -- we
|
|
|
|
// wipe out the entire containing block and recreate it from
|
|
|
|
// scratch. The reason is that because we know that a special
|
|
|
|
// inline frame has propogated some of its children upward to be
|
|
|
|
// children of the block and that those frames may need to move
|
|
|
|
// around. This logic guarantees a correct answer.
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (gNoisyContentUpdates) {
|
|
|
|
printf("nsCSSFrameConstructor::ContentAppended: parentFrame=");
|
|
|
|
nsFrame::ListTag(stdout, parentFrame);
|
|
|
|
printf(" is special\n");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return ReframeContainingBlock(aPresContext, parentFrame);
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
// Get the parent frame's last-in-flow
|
|
|
|
nsIFrame* nextInFlow = parentFrame;
|
|
|
|
while (nsnull != nextInFlow) {
|
1999-02-24 04:48:08 +00:00
|
|
|
parentFrame->GetNextInFlow(&nextInFlow);
|
1999-02-05 03:55:18 +00:00
|
|
|
if (nsnull != nextInFlow) {
|
|
|
|
parentFrame = nextInFlow;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-09-13 21:25:43 +00:00
|
|
|
// If we didn't process children when we originally created the frame,
|
|
|
|
// then don't do any processing now
|
|
|
|
nsCOMPtr<nsIAtom> frameType;
|
|
|
|
parentFrame->GetFrameType(getter_AddRefs(frameType));
|
|
|
|
if (frameType.get() == nsLayoutAtoms::objectFrame) {
|
|
|
|
// This handles APPLET, EMBED, and OBJECT
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
// Create some new frames
|
1999-04-28 19:08:14 +00:00
|
|
|
PRInt32 count;
|
|
|
|
nsIFrame* firstAppendedFrame = nsnull;
|
|
|
|
nsFrameItems frameItems;
|
1999-08-05 03:09:22 +00:00
|
|
|
nsFrameConstructorState state(aPresContext, mFixedContainingBlock,
|
1999-04-28 19:08:14 +00:00
|
|
|
GetAbsoluteContainingBlock(aPresContext, parentFrame),
|
1999-12-06 07:44:18 +00:00
|
|
|
GetFloaterContainingBlock(aPresContext, parentFrame),
|
|
|
|
nsnull);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
// See if the containing block has :first-letter style applied.
|
1999-10-29 14:39:48 +00:00
|
|
|
PRBool haveFirstLetterStyle, haveFirstLineStyle;
|
1999-08-27 21:46:10 +00:00
|
|
|
nsIFrame* containingBlock = state.mFloatedItems.containingBlock;
|
1999-10-29 14:39:48 +00:00
|
|
|
nsCOMPtr<nsIContent> blockContent;
|
1999-08-27 21:46:10 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> blockSC;
|
|
|
|
containingBlock->GetStyleContext(getter_AddRefs(blockSC));
|
|
|
|
containingBlock->GetContent(getter_AddRefs(blockContent));
|
1999-10-29 14:39:48 +00:00
|
|
|
HaveSpecialBlockStyle(aPresContext, blockContent, blockSC,
|
|
|
|
&haveFirstLetterStyle,
|
|
|
|
&haveFirstLineStyle);
|
|
|
|
|
|
|
|
if (haveFirstLetterStyle) {
|
|
|
|
// Before we get going, remove the current letter frames
|
|
|
|
RemoveLetterFrames(aPresContext, state.mPresShell,
|
|
|
|
state.mFrameManager, containingBlock);
|
1999-08-27 21:46:10 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
PRInt32 i;
|
|
|
|
aContainer->ChildCount(count);
|
1999-08-06 21:43:33 +00:00
|
|
|
for (i = aNewIndexInContainer; i < count; i++) {
|
1999-08-27 21:46:10 +00:00
|
|
|
nsCOMPtr<nsIContent> childContent;
|
|
|
|
aContainer->ChildAt(i, *getter_AddRefs(childContent));
|
1999-08-31 03:09:40 +00:00
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
// Construct a child frame
|
1999-12-04 23:49:50 +00:00
|
|
|
ConstructFrame(shell, aPresContext, state, childContent, parentFrame, frameItems);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
1999-08-31 03:09:40 +00:00
|
|
|
if (haveFirstLineStyle) {
|
|
|
|
// It's possible that some of the new frames go into a
|
|
|
|
// first-line frame. Look at them and see...
|
1999-12-04 23:49:50 +00:00
|
|
|
AppendFirstLineFrames(shell, aPresContext, state, aContainer, parentFrame,
|
1999-08-31 03:09:40 +00:00
|
|
|
frameItems);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Adjust parent frame for table inner/outer frame. We need to do
|
|
|
|
// this here because we need both the parent frame and the
|
|
|
|
// constructed frame
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult result = NS_OK;
|
1999-08-31 03:09:40 +00:00
|
|
|
nsIFrame* adjustedParentFrame = parentFrame;
|
1999-02-05 03:55:18 +00:00
|
|
|
firstAppendedFrame = frameItems.childList;
|
|
|
|
if (nsnull != firstAppendedFrame) {
|
|
|
|
const nsStyleDisplay* firstAppendedFrameDisplay;
|
1999-02-26 17:11:54 +00:00
|
|
|
firstAppendedFrame->GetStyleData(eStyleStruct_Display,
|
|
|
|
(const nsStyleStruct *&)firstAppendedFrameDisplay);
|
2000-01-22 01:16:50 +00:00
|
|
|
result = GetAdjustedParentFrame(aPresContext, parentFrame,
|
1999-02-26 17:11:54 +00:00
|
|
|
firstAppendedFrameDisplay->mDisplay,
|
|
|
|
adjustedParentFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Notify the parent frame passing it the list of new frames
|
1999-08-31 03:09:40 +00:00
|
|
|
if (NS_SUCCEEDED(result) && firstAppendedFrame) {
|
1999-11-01 15:24:57 +00:00
|
|
|
// Perform special check for diddling around with the frames in
|
|
|
|
// a special inline frame.
|
1999-11-09 22:36:05 +00:00
|
|
|
|
|
|
|
// XXX Bug 18366
|
|
|
|
// Although select frame are inline we do not want to call
|
|
|
|
// WipeContainingBlock because it will throw away the entire selct frame and
|
|
|
|
// start over which is something we do not want to do
|
|
|
|
//
|
1999-11-09 23:49:37 +00:00
|
|
|
nsCOMPtr<nsIDOMHTMLSelectElement> selectContent(do_QueryInterface(aContainer));
|
|
|
|
if (!selectContent) {
|
1999-11-09 22:36:05 +00:00
|
|
|
if (WipeContainingBlock(aPresContext, state, blockContent, adjustedParentFrame,
|
|
|
|
frameItems.childList)) {
|
|
|
|
return NS_OK;
|
|
|
|
}
|
1999-11-01 15:24:57 +00:00
|
|
|
}
|
|
|
|
|
1999-04-06 04:58:05 +00:00
|
|
|
// Append the flowed frames to the principal child list
|
1999-08-05 03:09:22 +00:00
|
|
|
AppendFrames(aPresContext, shell, state.mFrameManager, aContainer,
|
|
|
|
adjustedParentFrame, firstAppendedFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// If there are new absolutely positioned child frames, then notify
|
|
|
|
// the parent
|
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
1999-04-28 19:08:14 +00:00
|
|
|
if (state.mAbsoluteItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
state.mAbsoluteItems.containingBlock->AppendFrames(aPresContext, *shell,
|
1999-11-01 15:24:57 +00:00
|
|
|
nsLayoutAtoms::absoluteList,
|
|
|
|
state.mAbsoluteItems.childList);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If there are new fixed positioned child frames, then notify
|
|
|
|
// the parent
|
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
1999-04-28 19:08:14 +00:00
|
|
|
if (state.mFixedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
state.mFixedItems.containingBlock->AppendFrames(aPresContext, *shell,
|
1999-11-01 15:24:57 +00:00
|
|
|
nsLayoutAtoms::fixedList,
|
|
|
|
state.mFixedItems.childList);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-26 17:11:54 +00:00
|
|
|
|
|
|
|
// If there are new floating child frames, then notify
|
|
|
|
// the parent
|
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
1999-04-28 19:08:14 +00:00
|
|
|
if (state.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
state.mFloatedItems.containingBlock->AppendFrames(aPresContext, *shell,
|
1999-11-01 15:24:57 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
|
|
|
state.mFloatedItems.childList);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// Recover first-letter frames
|
|
|
|
if (haveFirstLetterStyle) {
|
1999-12-04 23:49:50 +00:00
|
|
|
RecoverLetterFrames(shell, aPresContext, state, containingBlock);
|
1999-10-29 14:39:48 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-08-06 21:14:45 +00:00
|
|
|
|
1999-08-31 03:09:40 +00:00
|
|
|
// Here we have been notified that content has been appended so if
|
|
|
|
// the select now has a single item we need to go in and removed
|
|
|
|
// the dummy frame.
|
|
|
|
nsCOMPtr<nsIDOMHTMLSelectElement> sel(do_QueryInterface(aContainer));
|
|
|
|
if (sel) {
|
1999-08-06 21:14:45 +00:00
|
|
|
nsCOMPtr<nsIContent> childContent;
|
1999-08-06 21:45:14 +00:00
|
|
|
aContainer->ChildAt(aNewIndexInContainer, *getter_AddRefs(childContent));
|
1999-08-06 21:14:45 +00:00
|
|
|
if (childContent) {
|
1999-08-31 03:09:40 +00:00
|
|
|
RemoveDummyFrameFromSelect(aPresContext, shell, aContainer,
|
|
|
|
childContent, sel);
|
1999-08-06 21:14:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-08-06 21:14:45 +00:00
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::RemoveDummyFrameFromSelect(nsIPresContext* aPresContext,
|
|
|
|
nsIPresShell * aPresShell,
|
|
|
|
nsIContent* aContainer,
|
|
|
|
nsIContent* aChild,
|
|
|
|
nsIDOMHTMLSelectElement * aSelectElement)
|
|
|
|
{
|
|
|
|
//check to see if there is one item,
|
|
|
|
// meaning we need to remove the dummy frame
|
|
|
|
PRUint32 numOptions = 0;
|
|
|
|
nsresult result = aSelectElement->GetLength(&numOptions);
|
1999-08-19 14:29:55 +00:00
|
|
|
if (NS_SUCCEEDED(result)) {
|
|
|
|
if (1 == numOptions) {
|
|
|
|
nsIFrame* parentFrame;
|
|
|
|
nsIFrame* childFrame;
|
|
|
|
// Get the childFrame for the added child (option)
|
|
|
|
// then get the child's parent frame which should be an area frame
|
|
|
|
aPresShell->GetPrimaryFrameFor(aChild, &childFrame);
|
1999-10-20 23:35:38 +00:00
|
|
|
if (nsnull != childFrame) {
|
|
|
|
childFrame->GetParent(&parentFrame);
|
|
|
|
|
|
|
|
// Now loop through all the child looking fr the frame whose content
|
|
|
|
// is equal to the select element's content
|
|
|
|
// this is because when gernated content is created it stuff the parent content
|
|
|
|
// pointer into the generated frame, so in this case it has the select content
|
2000-01-22 01:16:50 +00:00
|
|
|
parentFrame->FirstChild(aPresContext, nsnull, &childFrame);
|
1999-10-20 23:35:38 +00:00
|
|
|
nsCOMPtr<nsIContent> selectContent = do_QueryInterface(aSelectElement);
|
|
|
|
while (nsnull != childFrame) {
|
|
|
|
nsIContent * content;
|
|
|
|
childFrame->GetContent(&content);
|
1999-08-06 21:14:45 +00:00
|
|
|
|
1999-10-20 23:35:38 +00:00
|
|
|
// Found the dummy frame so get the FrameManager and
|
|
|
|
// delete/remove the dummy frame
|
|
|
|
if (selectContent.get() == content) {
|
|
|
|
nsCOMPtr<nsIFrameManager> frameManager;
|
|
|
|
aPresShell->GetFrameManager(getter_AddRefs(frameManager));
|
1999-11-24 06:03:41 +00:00
|
|
|
frameManager->RemoveFrame(aPresContext, *aPresShell, parentFrame, nsnull, childFrame);
|
1999-10-20 23:35:38 +00:00
|
|
|
NS_IF_RELEASE(content);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-08-19 14:29:55 +00:00
|
|
|
NS_IF_RELEASE(content);
|
1999-10-20 23:35:38 +00:00
|
|
|
childFrame->GetNextSibling(&childFrame);
|
1999-08-19 14:29:55 +00:00
|
|
|
}
|
1999-08-06 21:14:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::ContentInserted(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContainer,
|
|
|
|
nsIContent* aChild,
|
1999-12-06 07:44:18 +00:00
|
|
|
PRInt32 aIndexInContainer,
|
|
|
|
nsILayoutHistoryState* aFrameState)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
1999-11-01 15:24:57 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (gNoisyContentUpdates) {
|
|
|
|
printf("nsCSSFrameConstructor::ContentInserted container=%p child=%p index=%d\n",
|
|
|
|
aContainer, aChild, aIndexInContainer);
|
|
|
|
if (gReallyNoisyContentUpdates) {
|
|
|
|
(aContainer ? aContainer : aChild)->List(stdout, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1999-07-20 23:24:10 +00:00
|
|
|
#ifdef INCLUDE_XUL
|
|
|
|
if (aContainer) {
|
|
|
|
nsCOMPtr<nsIAtom> tag;
|
|
|
|
aContainer->GetTag(*getter_AddRefs(tag));
|
|
|
|
if (tag && (tag.get() == nsXULAtoms::treechildren ||
|
1999-11-01 15:24:57 +00:00
|
|
|
tag.get() == nsXULAtoms::treeitem)) {
|
1999-07-20 23:24:10 +00:00
|
|
|
// Walk up to the outermost tree row group frame and tell it that
|
|
|
|
// content was added.
|
1999-07-22 10:06:38 +00:00
|
|
|
nsCOMPtr<nsIContent> parent;
|
1999-07-20 23:24:10 +00:00
|
|
|
nsCOMPtr<nsIContent> child = dont_QueryInterface(aContainer);
|
1999-07-22 10:06:38 +00:00
|
|
|
child->GetParent(*getter_AddRefs(parent));
|
1999-07-20 23:24:10 +00:00
|
|
|
while (parent) {
|
|
|
|
parent->GetTag(*getter_AddRefs(tag));
|
|
|
|
if (tag.get() == nsXULAtoms::tree)
|
|
|
|
break;
|
|
|
|
child = parent;
|
1999-07-22 10:06:38 +00:00
|
|
|
child->GetParent(*getter_AddRefs(parent));
|
1999-07-20 23:24:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (parent) {
|
|
|
|
// We found it. Get the primary frame.
|
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
2000-01-06 08:58:05 +00:00
|
|
|
nsIFrame* outerFrame = GetFrameFor(shell, aPresContext, child);
|
1999-07-20 23:24:10 +00:00
|
|
|
|
|
|
|
// Convert to a tree row group frame.
|
2000-01-06 08:58:05 +00:00
|
|
|
nsTreeRowGroupFrame* treeRowGroup = (nsTreeRowGroupFrame*)outerFrame;
|
2000-01-13 19:20:25 +00:00
|
|
|
if (treeRowGroup) {
|
2000-01-06 08:58:05 +00:00
|
|
|
|
|
|
|
// Get the primary frame for the parent of the child that's being added.
|
|
|
|
nsIFrame* innerFrame = GetFrameFor(shell, aPresContext, aContainer);
|
|
|
|
|
|
|
|
// See if there's a previous sibling.
|
|
|
|
nsIFrame* prevSibling = FindPreviousSibling(shell,
|
|
|
|
aContainer,
|
|
|
|
aIndexInContainer);
|
|
|
|
|
|
|
|
if (prevSibling || innerFrame) {
|
2000-01-06 09:36:36 +00:00
|
|
|
// We're onscreen, but because of the fact that we can be called to
|
|
|
|
// "kill" a displayed frame (e.g., when you close a tree node), we
|
|
|
|
// have to see if this slaying is taking place. If so, then we don't
|
|
|
|
// really want to do anything. Instead we need to add our node to
|
|
|
|
// the undisplayed content list so that the style system will know
|
|
|
|
// to check it for subsequent display changes (e.g., when you next
|
|
|
|
// reopen).
|
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
nsCOMPtr<nsIAtom> tagName;
|
|
|
|
aChild->GetTag(*getter_AddRefs(tagName));
|
|
|
|
ResolveStyleContext(aPresContext, innerFrame, aChild, tagName, getter_AddRefs(styleContext));
|
|
|
|
|
|
|
|
// Pre-check for display "none" - if we find that, don't reflow at all.
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
styleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
if (NS_STYLE_DISPLAY_NONE == display->mDisplay) {
|
|
|
|
|
|
|
|
nsFrameConstructorState state(aPresContext, mFixedContainingBlock,
|
|
|
|
GetAbsoluteContainingBlock(aPresContext, innerFrame),
|
|
|
|
GetFloaterContainingBlock(aPresContext, innerFrame),
|
|
|
|
aFrameState);
|
|
|
|
state.mFrameManager->SetUndisplayedContent(aChild, styleContext);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Good call. Make sure a full reflow happens.
|
2000-01-06 08:58:05 +00:00
|
|
|
nsIFrame* nextSibling = FindNextSibling(shell, aContainer, aIndexInContainer);
|
|
|
|
if(!nextSibling)
|
|
|
|
treeRowGroup->OnContentAdded(aPresContext);
|
|
|
|
else {
|
|
|
|
nsIFrame* frame = GetFrameFor(shell, aPresContext, aContainer);
|
|
|
|
nsTreeRowGroupFrame* frameTreeRowGroup = (nsTreeRowGroupFrame*)frame;
|
|
|
|
if(frameTreeRowGroup)
|
|
|
|
frameTreeRowGroup->OnContentInserted(aPresContext, nextSibling, aIndexInContainer);
|
|
|
|
}
|
|
|
|
}
|
1999-11-01 15:24:57 +00:00
|
|
|
else {
|
2000-01-06 08:58:05 +00:00
|
|
|
// We're going to be offscreen.
|
|
|
|
treeRowGroup->ReflowScrollbar(aPresContext);
|
1999-11-01 15:24:57 +00:00
|
|
|
}
|
2000-01-06 08:58:05 +00:00
|
|
|
|
1999-07-20 23:24:10 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif // INCLUDE_XUL
|
|
|
|
|
1999-02-12 17:45:58 +00:00
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
// If we have a null parent, then this must be the document element
|
|
|
|
// being inserted
|
|
|
|
if (nsnull == aContainer) {
|
|
|
|
NS_PRECONDITION(nsnull == mInitialContainingBlock, "initial containing block already created");
|
1999-02-18 16:56:15 +00:00
|
|
|
nsIContent* docElement = mDocument->GetRootContent();
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-18 16:56:15 +00:00
|
|
|
// Get the style context of the containing block frame
|
1999-02-11 15:54:13 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> containerStyle;
|
|
|
|
mDocElementContainingBlock->GetStyleContext(getter_AddRefs(containerStyle));
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Create frames for the document element and its child elements
|
1999-04-28 19:08:14 +00:00
|
|
|
nsIFrame* docElementFrame;
|
1999-12-06 07:44:18 +00:00
|
|
|
nsFrameConstructorState state(aPresContext, mFixedContainingBlock, nsnull, nsnull, aFrameState);
|
1999-12-04 23:49:50 +00:00
|
|
|
ConstructDocElementFrame(shell, aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
state,
|
1999-02-05 03:55:18 +00:00
|
|
|
docElement,
|
|
|
|
mDocElementContainingBlock,
|
1999-02-18 16:56:15 +00:00
|
|
|
containerStyle,
|
1999-04-28 19:08:14 +00:00
|
|
|
docElementFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
NS_IF_RELEASE(docElement);
|
|
|
|
|
|
|
|
// Set the initial child list for the parent
|
1999-11-24 06:03:41 +00:00
|
|
|
mDocElementContainingBlock->SetInitialChildList(aPresContext,
|
1999-02-05 03:55:18 +00:00
|
|
|
nsnull,
|
|
|
|
docElementFrame);
|
|
|
|
|
|
|
|
// Tell the fixed containing block about its 'fixed' frames
|
1999-04-28 19:08:14 +00:00
|
|
|
if (state.mFixedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
mFixedContainingBlock->SetInitialChildList(aPresContext,
|
1999-02-05 03:55:18 +00:00
|
|
|
nsLayoutAtoms::fixedList,
|
1999-04-28 19:08:14 +00:00
|
|
|
state.mFixedItems.childList);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-08-06 14:11:14 +00:00
|
|
|
|
1999-09-13 21:25:43 +00:00
|
|
|
} else {
|
1999-02-05 03:55:18 +00:00
|
|
|
// Find the frame that precedes the insertion point.
|
|
|
|
nsIFrame* prevSibling = FindPreviousSibling(shell, aContainer, aIndexInContainer);
|
|
|
|
nsIFrame* nextSibling = nsnull;
|
|
|
|
PRBool isAppend = PR_FALSE;
|
|
|
|
|
|
|
|
// If there is no previous sibling, then find the frame that follows
|
|
|
|
if (nsnull == prevSibling) {
|
|
|
|
nextSibling = FindNextSibling(shell, aContainer, aIndexInContainer);
|
|
|
|
}
|
|
|
|
|
1999-08-31 03:09:40 +00:00
|
|
|
// Get the geometric parent. Use the prev sibling if we have it;
|
|
|
|
// otherwise use the next sibling
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* parentFrame;
|
1999-08-31 03:09:40 +00:00
|
|
|
if (nsnull != prevSibling) {
|
|
|
|
prevSibling->GetParent(&parentFrame);
|
|
|
|
}
|
|
|
|
else if (nextSibling) {
|
|
|
|
nextSibling->GetParent(&parentFrame);
|
|
|
|
}
|
|
|
|
else {
|
1999-02-05 03:55:18 +00:00
|
|
|
// No previous or next sibling so treat this like an appended frame.
|
|
|
|
isAppend = PR_TRUE;
|
1999-10-06 23:17:28 +00:00
|
|
|
parentFrame = GetFrameFor(shell, aPresContext, aContainer);
|
1999-09-13 21:25:43 +00:00
|
|
|
|
1999-09-13 22:07:15 +00:00
|
|
|
if (parentFrame) {
|
|
|
|
// If we didn't process children when we originally created the frame,
|
|
|
|
// then don't do any processing now
|
|
|
|
nsCOMPtr<nsIAtom> frameType;
|
|
|
|
parentFrame->GetFrameType(getter_AddRefs(frameType));
|
|
|
|
if (frameType.get() == nsLayoutAtoms::objectFrame) {
|
|
|
|
// This handles APPLET, EMBED, and OBJECT
|
|
|
|
return NS_OK;
|
|
|
|
}
|
1999-09-13 21:25:43 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Construct a new frame
|
|
|
|
if (nsnull != parentFrame) {
|
1999-11-01 15:24:57 +00:00
|
|
|
// If the frame we are manipulating is a special frame then do
|
|
|
|
// something different instead of just inserting newly created
|
|
|
|
// frames.
|
|
|
|
if (IsFrameSpecial(aPresContext, parentFrame)) {
|
|
|
|
// We are pretty harsh here (and definitely not optimal) -- we
|
|
|
|
// wipe out the entire containing block and recreate it from
|
|
|
|
// scratch. The reason is that because we know that a special
|
|
|
|
// inline frame has propogated some of its children upward to be
|
|
|
|
// children of the block and that those frames may need to move
|
|
|
|
// around. This logic guarantees a correct answer.
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (gNoisyContentUpdates) {
|
|
|
|
printf("nsCSSFrameConstructor::ContentInserted: parentFrame=");
|
|
|
|
nsFrame::ListTag(stdout, parentFrame);
|
|
|
|
printf(" is special\n");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return ReframeContainingBlock(aPresContext, parentFrame);
|
|
|
|
}
|
|
|
|
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameItems frameItems;
|
1999-08-05 03:09:22 +00:00
|
|
|
nsFrameConstructorState state(aPresContext, mFixedContainingBlock,
|
1999-04-28 19:08:14 +00:00
|
|
|
GetAbsoluteContainingBlock(aPresContext, parentFrame),
|
1999-12-06 07:44:18 +00:00
|
|
|
GetFloaterContainingBlock(aPresContext, parentFrame),
|
|
|
|
aFrameState);
|
1999-08-27 21:46:10 +00:00
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
// Recover state for the containing block - we need to know if
|
|
|
|
// it has :first-letter or :first-line style applied to it. The
|
|
|
|
// reason we care is that the internal structure in these cases
|
|
|
|
// is not the normal structure and requires custom updating
|
|
|
|
// logic.
|
1999-08-27 21:46:10 +00:00
|
|
|
nsIFrame* containingBlock = state.mFloatedItems.containingBlock;
|
|
|
|
nsCOMPtr<nsIStyleContext> blockSC;
|
|
|
|
nsCOMPtr<nsIContent> blockContent;
|
1999-08-31 03:09:40 +00:00
|
|
|
PRBool haveFirstLetterStyle = PR_FALSE;
|
|
|
|
PRBool haveFirstLineStyle = PR_FALSE;
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// In order to shave off some cycles, we only dig up the
|
|
|
|
// containing block haveFirst* flags if the parent frame where
|
|
|
|
// the insertion/append is occuring is an inline or block
|
|
|
|
// container. For other types of containers this isn't relevant.
|
1999-08-31 03:09:40 +00:00
|
|
|
const nsStyleDisplay* parentDisplay;
|
|
|
|
parentFrame->GetStyleData(eStyleStruct_Display,
|
|
|
|
(const nsStyleStruct*&)parentDisplay);
|
|
|
|
|
|
|
|
// Examine the parentFrame where the insertion is taking
|
|
|
|
// place. If its a certain kind of container then some special
|
|
|
|
// processing is done.
|
|
|
|
if ((NS_STYLE_DISPLAY_BLOCK == parentDisplay->mDisplay) ||
|
1999-10-29 14:39:48 +00:00
|
|
|
(NS_STYLE_DISPLAY_LIST_ITEM == parentDisplay->mDisplay) ||
|
1999-08-31 03:09:40 +00:00
|
|
|
(NS_STYLE_DISPLAY_INLINE == parentDisplay->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_INLINE_BLOCK == parentDisplay->mDisplay)) {
|
1999-10-29 14:39:48 +00:00
|
|
|
// Recover the special style flags for the containing block
|
1999-08-31 03:09:40 +00:00
|
|
|
containingBlock->GetStyleContext(getter_AddRefs(blockSC));
|
|
|
|
containingBlock->GetContent(getter_AddRefs(blockContent));
|
|
|
|
HaveSpecialBlockStyle(aPresContext, blockContent, blockSC,
|
|
|
|
&haveFirstLetterStyle,
|
|
|
|
&haveFirstLineStyle);
|
|
|
|
|
|
|
|
if (haveFirstLetterStyle) {
|
|
|
|
// Get the correct parentFrame and prevSibling - if a
|
|
|
|
// letter-frame is present, use its parent.
|
|
|
|
nsCOMPtr<nsIAtom> parentFrameType;
|
|
|
|
parentFrame->GetFrameType(getter_AddRefs(parentFrameType));
|
1999-08-31 07:14:54 +00:00
|
|
|
if (parentFrameType.get() == nsLayoutAtoms::letterFrame) {
|
1999-08-31 03:09:40 +00:00
|
|
|
if (prevSibling) {
|
|
|
|
prevSibling = parentFrame;
|
|
|
|
}
|
|
|
|
parentFrame->GetParent(&parentFrame);
|
1999-08-27 21:46:10 +00:00
|
|
|
}
|
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
// Remove the old letter frames before doing the insertion
|
|
|
|
RemoveLetterFrames(aPresContext, state.mPresShell,
|
|
|
|
state.mFrameManager,
|
|
|
|
state.mFloatedItems.containingBlock);
|
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-11-01 15:24:57 +00:00
|
|
|
// If the frame we are manipulating is a special inline frame
|
|
|
|
// then do something different instead of just inserting newly
|
|
|
|
// created frames.
|
|
|
|
if (IsFrameSpecial(state.mFrameManager, parentFrame)) {
|
|
|
|
// We are pretty harsh here (and definitely not optimal) -- we
|
|
|
|
// wipe out the entire containing block and recreate it from
|
|
|
|
// scratch. The reason is that because we know that a special
|
|
|
|
// inline frame has propogated some of its children upward to be
|
|
|
|
// children of the block and that those frames may need to move
|
|
|
|
// around. This logic guarantees a correct answer.
|
|
|
|
nsCOMPtr<nsIContent> parentContainer;
|
|
|
|
blockContent->GetParent(*getter_AddRefs(parentContainer));
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (gNoisyContentUpdates) {
|
|
|
|
printf("nsCSSFrameConstructor::ContentInserted: parentFrame=");
|
|
|
|
nsFrame::ListTag(stdout, parentFrame);
|
|
|
|
printf(" is special inline\n");
|
|
|
|
printf(" ==> blockContent=%p, parentContainer=%p\n",
|
|
|
|
blockContent.get(), parentContainer.get());
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
if (parentContainer) {
|
|
|
|
PRInt32 ix;
|
|
|
|
parentContainer->IndexOf(blockContent, ix);
|
|
|
|
ContentReplaced(aPresContext, parentContainer, blockContent, blockContent, ix);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// XXX uh oh. the block that needs reworking has no parent...
|
|
|
|
}
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2000-03-10 01:10:44 +00:00
|
|
|
rv = ConstructFrame(shell, aPresContext, state, aChild, parentFrame, frameItems);
|
1999-11-01 15:24:57 +00:00
|
|
|
|
1999-11-24 21:10:49 +00:00
|
|
|
// XXX Bug 19949
|
|
|
|
// Although select frame are inline we do not want to call
|
|
|
|
// WipeContainingBlock because it will throw away the entire select frame and
|
|
|
|
// start over which is something we do not want to do
|
|
|
|
//
|
|
|
|
nsCOMPtr<nsIDOMHTMLSelectElement> selectContent(do_QueryInterface(aContainer));
|
|
|
|
if (!selectContent) {
|
|
|
|
// Perform special check for diddling around with the frames in
|
|
|
|
// a special inline frame.
|
|
|
|
if (WipeContainingBlock(aPresContext, state, blockContent, parentFrame, frameItems.childList)) {
|
|
|
|
return NS_OK;
|
|
|
|
}
|
1999-11-01 15:24:57 +00:00
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
|
1999-08-31 03:09:40 +00:00
|
|
|
if (haveFirstLineStyle) {
|
|
|
|
// It's possible that the new frame goes into a first-line
|
|
|
|
// frame. Look at it and see...
|
|
|
|
if (isAppend) {
|
|
|
|
// Use append logic when appending
|
1999-12-04 23:49:50 +00:00
|
|
|
AppendFirstLineFrames(shell, aPresContext, state, aContainer, parentFrame,
|
1999-08-31 03:09:40 +00:00
|
|
|
frameItems);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Use more complicated insert logic when inserting
|
|
|
|
InsertFirstLineFrames(aPresContext, state, aContainer,
|
|
|
|
containingBlock, &parentFrame,
|
|
|
|
prevSibling, frameItems);
|
|
|
|
}
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
nsIFrame* newFrame = frameItems.childList;
|
|
|
|
if (NS_SUCCEEDED(rv) && (nsnull != newFrame)) {
|
1999-11-01 15:24:57 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
// Notify the parent frame
|
|
|
|
if (isAppend) {
|
1999-08-31 03:09:40 +00:00
|
|
|
rv = AppendFrames(aPresContext, shell, state.mFrameManager,
|
|
|
|
aContainer, parentFrame, newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
} else {
|
1999-04-06 04:58:05 +00:00
|
|
|
if (!prevSibling) {
|
|
|
|
// We're inserting the new frame as the first child. See if the
|
|
|
|
// parent has a :before pseudo-element
|
|
|
|
nsIFrame* firstChild;
|
2000-01-22 01:16:50 +00:00
|
|
|
parentFrame->FirstChild(aPresContext, nsnull, &firstChild);
|
1999-04-06 04:58:05 +00:00
|
|
|
|
1999-09-02 05:21:39 +00:00
|
|
|
if (firstChild && IsGeneratedContentFor(aContainer, firstChild, nsCSSAtoms::beforePseudo)) {
|
1999-04-07 03:10:59 +00:00
|
|
|
// Insert the new frames after the :before pseudo-element
|
|
|
|
prevSibling = firstChild;
|
1999-04-06 04:58:05 +00:00
|
|
|
}
|
|
|
|
}
|
2000-01-28 02:19:45 +00:00
|
|
|
// check for a table caption which goes on an additional child list
|
|
|
|
nsIAtom* childList = GetChildListFor(newFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mFrameManager->InsertFrames(aPresContext, *shell,
|
1999-08-27 21:46:10 +00:00
|
|
|
parentFrame,
|
2000-01-28 02:19:45 +00:00
|
|
|
childList, prevSibling,
|
1999-08-27 21:46:10 +00:00
|
|
|
newFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If there are new absolutely positioned child frames, then notify
|
|
|
|
// the parent
|
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
1999-04-28 19:08:14 +00:00
|
|
|
if (state.mAbsoluteItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mAbsoluteItems.containingBlock->AppendFrames(aPresContext, *shell,
|
1999-02-05 03:55:18 +00:00
|
|
|
nsLayoutAtoms::absoluteList,
|
1999-04-28 19:08:14 +00:00
|
|
|
state.mAbsoluteItems.childList);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If there are new fixed positioned child frames, then notify
|
|
|
|
// the parent
|
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
1999-04-28 19:08:14 +00:00
|
|
|
if (state.mFixedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mFixedItems.containingBlock->AppendFrames(aPresContext, *shell,
|
1999-02-05 03:55:18 +00:00
|
|
|
nsLayoutAtoms::fixedList,
|
1999-04-28 19:08:14 +00:00
|
|
|
state.mFixedItems.childList);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-26 17:11:54 +00:00
|
|
|
|
|
|
|
// If there are new floating child frames, then notify
|
|
|
|
// the parent
|
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
1999-04-28 19:08:14 +00:00
|
|
|
if (state.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mFloatedItems.containingBlock->AppendFrames(aPresContext, *shell,
|
1999-04-28 19:08:14 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
|
|
|
state.mFloatedItems.childList);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
if (haveFirstLetterStyle) {
|
|
|
|
// Recover the letter frames for the containing block when
|
|
|
|
// it has first-letter style.
|
1999-12-04 23:49:50 +00:00
|
|
|
RecoverLetterFrames(shell, aPresContext, state,
|
1999-10-29 14:39:48 +00:00
|
|
|
state.mFloatedItems.containingBlock);
|
1999-08-27 21:46:10 +00:00
|
|
|
}
|
1999-10-29 14:39:48 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
|
1999-08-06 21:14:45 +00:00
|
|
|
// Here we have been notified that content has been insert
|
|
|
|
// so if the select now has a single item
|
|
|
|
// we need to go in and removed the dummy frame
|
|
|
|
nsCOMPtr<nsIDOMHTMLSelectElement> selectElement;
|
2000-02-02 22:24:56 +00:00
|
|
|
nsresult result = aContainer->QueryInterface(NS_GET_IID(nsIDOMHTMLSelectElement),
|
1999-08-06 21:14:45 +00:00
|
|
|
(void**)getter_AddRefs(selectElement));
|
|
|
|
if (NS_SUCCEEDED(result) && selectElement) {
|
|
|
|
RemoveDummyFrameFromSelect(aPresContext, shell, aContainer, aChild, selectElement);
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::ContentReplaced(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContainer,
|
|
|
|
nsIContent* aOldChild,
|
|
|
|
nsIContent* aNewChild,
|
|
|
|
PRInt32 aIndexInContainer)
|
|
|
|
{
|
|
|
|
// XXX For now, do a brute force remove and insert.
|
|
|
|
nsresult res = ContentRemoved(aPresContext, aContainer,
|
|
|
|
aOldChild, aIndexInContainer);
|
|
|
|
if (NS_OK == res) {
|
|
|
|
res = ContentInserted(aPresContext, aContainer,
|
1999-12-06 07:44:18 +00:00
|
|
|
aNewChild, aIndexInContainer, nsnull);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
1999-04-27 03:10:45 +00:00
|
|
|
// Returns PR_TRUE if aAncestorFrame is an ancestor frame of aFrame
|
|
|
|
static PRBool
|
|
|
|
IsAncestorFrame(nsIFrame* aFrame, nsIFrame* aAncestorFrame)
|
|
|
|
{
|
|
|
|
nsIFrame* parentFrame;
|
|
|
|
aFrame->GetParent(&parentFrame);
|
|
|
|
|
|
|
|
while (parentFrame) {
|
|
|
|
if (parentFrame == aAncestorFrame) {
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
parentFrame->GetParent(&parentFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
1999-07-07 20:33:07 +00:00
|
|
|
/**
|
|
|
|
* Called to delete a frame subtree. Two important things happen:
|
|
|
|
* 1. for each frame in the subtree we remove the mapping from the
|
|
|
|
* content object to its frame
|
|
|
|
* 2. for child frames that have been moved out of the flow we delete
|
|
|
|
* the out-of-flow frame as well
|
|
|
|
*
|
1999-11-21 04:06:44 +00:00
|
|
|
* Note: this function should only be called by DeletingFrameSubtree()
|
|
|
|
*
|
1999-07-07 20:33:07 +00:00
|
|
|
* @param aRemovedFrame this is the frame that was removed from the
|
|
|
|
* content model. As we recurse we need to remember this so we
|
|
|
|
* can check if out-of-flow frames are a descendent of the frame
|
|
|
|
* being removed
|
|
|
|
* @param aFrame the local subtree that is being deleted. This is initially
|
|
|
|
* the same as aRemovedFrame, but as we recurse down the tree
|
|
|
|
* this changes
|
|
|
|
*/
|
1999-04-27 03:10:45 +00:00
|
|
|
static nsresult
|
1999-11-21 04:06:44 +00:00
|
|
|
DoDeletingFrameSubtree(nsIPresContext* aPresContext,
|
|
|
|
nsIPresShell* aPresShell,
|
|
|
|
nsIFrameManager* aFrameManager,
|
|
|
|
nsIFrame* aRemovedFrame,
|
|
|
|
nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
NS_PRECONDITION(aFrameManager, "no frame manager");
|
|
|
|
|
|
|
|
// Remove the mapping from the content object to its frame
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
|
|
aFrame->GetContent(getter_AddRefs(content));
|
|
|
|
aFrameManager->SetPrimaryFrameFor(content, nsnull);
|
|
|
|
aFrameManager->ClearAllUndisplayedContentIn(content);
|
|
|
|
|
|
|
|
// Walk aFrame's child frames
|
|
|
|
nsIAtom* childListName = nsnull;
|
|
|
|
PRInt32 childListIndex = 0;
|
|
|
|
|
|
|
|
do {
|
|
|
|
// Recursively walk aFrame's child frames looking for placeholder frames
|
|
|
|
nsIFrame* childFrame;
|
2000-01-22 01:16:50 +00:00
|
|
|
aFrame->FirstChild(aPresContext, childListName, &childFrame);
|
1999-11-21 04:06:44 +00:00
|
|
|
while (childFrame) {
|
|
|
|
nsIAtom* frameType;
|
|
|
|
PRBool isPlaceholder;
|
|
|
|
|
|
|
|
// See if it's a placeholder frame
|
|
|
|
childFrame->GetFrameType(&frameType);
|
|
|
|
isPlaceholder = (nsLayoutAtoms::placeholderFrame == frameType);
|
|
|
|
NS_IF_RELEASE(frameType);
|
|
|
|
|
|
|
|
if (isPlaceholder) {
|
|
|
|
// Get the out-of-flow frame
|
|
|
|
nsIFrame* outOfFlowFrame = ((nsPlaceholderFrame*)childFrame)->GetOutOfFlowFrame();
|
|
|
|
NS_ASSERTION(outOfFlowFrame, "no out-of-flow frame");
|
|
|
|
|
|
|
|
// Remove the mapping from the out-of-flow frame to its placeholder
|
|
|
|
aFrameManager->SetPlaceholderFrameFor(outOfFlowFrame, nsnull);
|
|
|
|
|
|
|
|
// Recursively find and delete any of its out-of-flow frames, and remove
|
|
|
|
// the mapping from content objects to frames
|
|
|
|
DoDeletingFrameSubtree(aPresContext, aPresShell, aFrameManager, aRemovedFrame,
|
|
|
|
outOfFlowFrame);
|
|
|
|
|
|
|
|
// Don't delete the out-of-flow frame if aRemovedFrame is one of its
|
|
|
|
// ancestor frames, because when aRemovedFrame is deleted it will delete
|
|
|
|
// its child frames including this out-of-flow frame
|
|
|
|
if (!IsAncestorFrame(outOfFlowFrame, aRemovedFrame)) {
|
|
|
|
// Get the out-of-flow frame's parent
|
|
|
|
nsIFrame* parentFrame;
|
|
|
|
outOfFlowFrame->GetParent(&parentFrame);
|
|
|
|
|
|
|
|
// Get the child list name for the out-of-flow frame
|
|
|
|
nsIAtom* listName;
|
2000-01-22 01:16:50 +00:00
|
|
|
GetChildListNameFor(aPresContext, parentFrame, outOfFlowFrame, &listName);
|
1999-11-21 04:06:44 +00:00
|
|
|
|
|
|
|
// Ask the parent to delete the out-of-flow frame
|
1999-11-24 06:03:41 +00:00
|
|
|
aFrameManager->RemoveFrame(aPresContext, *aPresShell, parentFrame,
|
1999-11-21 04:06:44 +00:00
|
|
|
listName, outOfFlowFrame);
|
|
|
|
NS_IF_RELEASE(listName);
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// Recursively find and delete any of its out-of-flow frames, and remove
|
|
|
|
// the mapping from content objects to frames
|
|
|
|
DoDeletingFrameSubtree(aPresContext, aPresShell, aFrameManager, aRemovedFrame,
|
|
|
|
childFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the next sibling child frame
|
|
|
|
childFrame->GetNextSibling(&childFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Look for any additional named child lists
|
|
|
|
NS_IF_RELEASE(childListName);
|
|
|
|
aFrame->GetAdditionalChildListName(childListIndex++, &childListName);
|
|
|
|
} while (childListName);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called to delete a frame subtree. Calls DoDeletingFrameSubtree()
|
|
|
|
* for aFrame and each of its continuing frames
|
|
|
|
*/
|
|
|
|
static nsresult
|
1999-08-05 03:09:22 +00:00
|
|
|
DeletingFrameSubtree(nsIPresContext* aPresContext,
|
|
|
|
nsIPresShell* aPresShell,
|
|
|
|
nsIFrameManager* aFrameManager,
|
|
|
|
nsIFrame* aFrame)
|
1999-04-27 03:10:45 +00:00
|
|
|
{
|
1999-08-05 03:30:09 +00:00
|
|
|
// If there's no frame manager it's probably because the pres shell is
|
|
|
|
// being destroyed
|
|
|
|
if (aFrameManager) {
|
1999-11-04 05:23:11 +00:00
|
|
|
while (aFrame) {
|
1999-11-21 04:06:44 +00:00
|
|
|
DoDeletingFrameSubtree(aPresContext, aPresShell, aFrameManager,
|
|
|
|
aFrame, aFrame);
|
|
|
|
|
|
|
|
// If it's split, then get the continuing frame. Note that we only do
|
|
|
|
// this for the top-most frame being deleted. Don't do it if we're
|
|
|
|
// recursing over a subtree, because those continuing frames should be
|
|
|
|
// found as part of the walk over the top-most frame's continuing frames.
|
|
|
|
// Walking them again will make this an N^2/2 algorithm
|
1999-11-04 05:23:11 +00:00
|
|
|
aFrame->GetNextInFlow(&aFrame);
|
|
|
|
}
|
1999-04-27 03:10:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-07-09 20:58:16 +00:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::RemoveMappingsForFrameSubtree(nsIPresContext* aPresContext,
|
1999-12-06 07:44:18 +00:00
|
|
|
nsIFrame* aRemovedFrame,
|
|
|
|
nsILayoutHistoryState* aFrameState)
|
1999-07-09 20:58:16 +00:00
|
|
|
{
|
|
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(presShell));
|
1999-08-05 03:09:22 +00:00
|
|
|
nsCOMPtr<nsIFrameManager> frameManager;
|
|
|
|
presShell->GetFrameManager(getter_AddRefs(frameManager));
|
1999-12-06 07:44:18 +00:00
|
|
|
|
|
|
|
// Save the frame tree's state before deleting it
|
2000-01-14 09:28:54 +00:00
|
|
|
CaptureStateFor(aPresContext, aRemovedFrame, mTempFrameTreeState);
|
1999-12-06 07:44:18 +00:00
|
|
|
|
1999-11-21 04:06:44 +00:00
|
|
|
return DeletingFrameSubtree(aPresContext, presShell, frameManager, aRemovedFrame);
|
1999-07-09 20:58:16 +00:00
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::ContentRemoved(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContainer,
|
|
|
|
nsIContent* aChild,
|
|
|
|
PRInt32 aIndexInContainer)
|
|
|
|
{
|
1999-11-01 15:24:57 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (gNoisyContentUpdates) {
|
|
|
|
printf("nsCSSFrameConstructor::ContentRemoved container=%p child=%p index=%d\n",
|
|
|
|
aContainer, aChild, aIndexInContainer);
|
|
|
|
if (gReallyNoisyContentUpdates) {
|
|
|
|
aContainer->List(stdout, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1999-08-05 03:09:22 +00:00
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
1999-02-12 17:45:58 +00:00
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
1999-08-05 03:09:22 +00:00
|
|
|
nsCOMPtr<nsIFrameManager> frameManager;
|
|
|
|
shell->GetFrameManager(getter_AddRefs(frameManager));
|
|
|
|
nsresult rv = NS_OK;
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-07-02 04:47:05 +00:00
|
|
|
// Find the child frame that maps the content
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* childFrame;
|
1999-08-06 14:11:14 +00:00
|
|
|
shell->GetPrimaryFrameFor(aChild, &childFrame);
|
|
|
|
|
1999-10-02 04:27:40 +00:00
|
|
|
if (! childFrame) {
|
|
|
|
frameManager->ClearUndisplayedContentIn(aChild, aContainer);
|
|
|
|
}
|
1999-10-29 14:39:48 +00:00
|
|
|
|
1999-08-06 14:11:14 +00:00
|
|
|
// When the last item is removed from a select,
|
|
|
|
// we need to add a pseudo frame so select gets sized as the best it can
|
|
|
|
// so here we see if it is a select and then we get the number of options
|
1999-08-08 19:16:09 +00:00
|
|
|
nsresult result = NS_ERROR_FAILURE;
|
1999-08-31 03:09:40 +00:00
|
|
|
if (aContainer && childFrame) {
|
1999-08-10 21:28:06 +00:00
|
|
|
nsCOMPtr<nsIDOMHTMLSelectElement> selectElement;
|
2000-02-02 22:24:56 +00:00
|
|
|
result = aContainer->QueryInterface(NS_GET_IID(nsIDOMHTMLSelectElement),
|
1999-08-06 14:11:14 +00:00
|
|
|
(void**)getter_AddRefs(selectElement));
|
1999-08-10 21:28:06 +00:00
|
|
|
if (NS_SUCCEEDED(result) && selectElement) {
|
|
|
|
PRUint32 numOptions = 0;
|
|
|
|
result = selectElement->GetLength(&numOptions);
|
1999-08-25 13:42:59 +00:00
|
|
|
nsIFrame * selectFrame; // XXX temp needed only native controls
|
|
|
|
shell->GetPrimaryFrameFor(aContainer, &selectFrame);// XXX temp needed only native controls
|
|
|
|
|
1999-08-10 21:28:06 +00:00
|
|
|
// For "select" add the pseudo frame after the last item is deleted
|
|
|
|
nsIFrame* parentFrame = nsnull;
|
|
|
|
childFrame->GetParent(&parentFrame);
|
1999-08-25 13:42:59 +00:00
|
|
|
if (parentFrame == selectFrame) { // XXX temp needed only native controls
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
1999-08-10 21:28:06 +00:00
|
|
|
if (NS_SUCCEEDED(result) && shell && parentFrame && 1 == numOptions) {
|
1999-08-06 14:11:14 +00:00
|
|
|
|
1999-08-10 21:28:06 +00:00
|
|
|
nsIStyleContext* styleContext = nsnull;
|
|
|
|
nsIFrame* generatedFrame = nsnull;
|
1999-12-06 07:44:18 +00:00
|
|
|
nsFrameConstructorState state(aPresContext, nsnull, nsnull, nsnull, nsnull);
|
1999-08-10 21:28:06 +00:00
|
|
|
|
|
|
|
//shell->GetPrimaryFrameFor(aContainer, &contentFrame);
|
|
|
|
parentFrame->GetStyleContext(&styleContext);
|
1999-12-04 23:49:50 +00:00
|
|
|
if (CreateGeneratedContentFrame(shell, aPresContext, state, parentFrame, aContainer,
|
1999-08-10 21:28:06 +00:00
|
|
|
styleContext, nsLayoutAtoms::dummyOptionPseudo,
|
1999-10-29 14:39:48 +00:00
|
|
|
PR_FALSE, &generatedFrame)) {
|
1999-08-10 21:28:06 +00:00
|
|
|
// Add the generated frame to the child list
|
1999-11-24 06:03:41 +00:00
|
|
|
frameManager->AppendFrames(aPresContext, *shell, parentFrame, nsnull, generatedFrame);
|
1999-08-10 21:28:06 +00:00
|
|
|
}
|
|
|
|
}
|
1999-08-06 14:11:14 +00:00
|
|
|
}
|
1999-08-10 21:28:06 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-07-10 00:51:35 +00:00
|
|
|
#ifdef INCLUDE_XUL
|
1999-07-20 23:24:10 +00:00
|
|
|
if (aContainer) {
|
|
|
|
nsCOMPtr<nsIAtom> tag;
|
|
|
|
aContainer->GetTag(*getter_AddRefs(tag));
|
|
|
|
if (tag.get() == nsXULAtoms::treechildren ||
|
|
|
|
tag.get() == nsXULAtoms::treeitem) {
|
|
|
|
if (childFrame) {
|
1999-07-14 19:21:27 +00:00
|
|
|
// Convert to a tree row group frame.
|
|
|
|
nsIFrame* parentFrame;
|
|
|
|
childFrame->GetParent(&parentFrame);
|
|
|
|
nsTreeRowGroupFrame* treeRowGroup = (nsTreeRowGroupFrame*)parentFrame;
|
2000-01-13 19:20:25 +00:00
|
|
|
if (treeRowGroup) {
|
1999-12-11 02:08:07 +00:00
|
|
|
treeRowGroup->OnContentRemoved(aPresContext, childFrame, aIndexInContainer);
|
1999-07-14 19:21:27 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
1999-07-10 00:51:35 +00:00
|
|
|
}
|
1999-07-20 23:24:10 +00:00
|
|
|
else {
|
|
|
|
// Ensure that we notify the outermost row group that the item
|
|
|
|
// has been removed (so that we can update the scrollbar state).
|
|
|
|
// Walk up to the outermost tree row group frame and tell it that
|
2000-01-06 08:58:05 +00:00
|
|
|
// the scrollbar thumb should be updated.
|
1999-07-22 10:06:38 +00:00
|
|
|
nsCOMPtr<nsIContent> parent;
|
1999-07-20 23:24:10 +00:00
|
|
|
nsCOMPtr<nsIContent> child = dont_QueryInterface(aContainer);
|
1999-07-22 10:06:38 +00:00
|
|
|
child->GetParent(*getter_AddRefs(parent));
|
1999-07-20 23:24:10 +00:00
|
|
|
while (parent) {
|
|
|
|
parent->GetTag(*getter_AddRefs(tag));
|
|
|
|
if (tag.get() == nsXULAtoms::tree)
|
|
|
|
break;
|
|
|
|
child = parent;
|
1999-07-22 10:06:38 +00:00
|
|
|
child->GetParent(*getter_AddRefs(parent));
|
1999-07-20 23:24:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (parent) {
|
|
|
|
// We found it. Get the primary frame.
|
|
|
|
nsIFrame* parentFrame = GetFrameFor(shell, aPresContext, child);
|
|
|
|
|
|
|
|
// Convert to a tree row group frame.
|
|
|
|
nsTreeRowGroupFrame* treeRowGroup = (nsTreeRowGroupFrame*)parentFrame;
|
2000-01-13 19:20:25 +00:00
|
|
|
if (treeRowGroup) {
|
2000-01-06 08:58:05 +00:00
|
|
|
treeRowGroup->ReflowScrollbar(aPresContext);
|
1999-07-20 23:24:10 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
1999-07-10 00:51:35 +00:00
|
|
|
}
|
1999-07-20 23:24:10 +00:00
|
|
|
}
|
1999-07-10 00:51:35 +00:00
|
|
|
#endif // INCLUDE_XUL
|
|
|
|
|
1999-07-20 23:24:10 +00:00
|
|
|
if (childFrame) {
|
1999-08-27 21:46:10 +00:00
|
|
|
// Get the childFrame's parent frame
|
|
|
|
nsIFrame* parentFrame;
|
|
|
|
childFrame->GetParent(&parentFrame);
|
|
|
|
|
1999-11-01 15:24:57 +00:00
|
|
|
// If the frame we are manipulating is a special frame then do
|
|
|
|
// something different instead of just inserting newly created
|
|
|
|
// frames.
|
|
|
|
if (IsFrameSpecial(aPresContext, parentFrame)) {
|
|
|
|
// We are pretty harsh here (and definitely not optimal) -- we
|
|
|
|
// wipe out the entire containing block and recreate it from
|
|
|
|
// scratch. The reason is that because we know that a special
|
|
|
|
// inline frame has propogated some of its children upward to be
|
|
|
|
// children of the block and that those frames may need to move
|
|
|
|
// around. This logic guarantees a correct answer.
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (gNoisyContentUpdates) {
|
|
|
|
printf("nsCSSFrameConstructor::ContentRemoved: parentFrame=");
|
|
|
|
nsFrame::ListTag(stdout, parentFrame);
|
|
|
|
printf(" is special\n");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return ReframeContainingBlock(aPresContext, parentFrame);
|
|
|
|
}
|
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
// Examine the containing-block for the removed content and see if
|
|
|
|
// :first-letter style applies.
|
|
|
|
nsIFrame* containingBlock =
|
|
|
|
GetFloaterContainingBlock(aPresContext, parentFrame);
|
|
|
|
nsCOMPtr<nsIStyleContext> blockSC;
|
|
|
|
containingBlock->GetStyleContext(getter_AddRefs(blockSC));
|
|
|
|
nsCOMPtr<nsIContent> blockContent;
|
|
|
|
containingBlock->GetContent(getter_AddRefs(blockContent));
|
|
|
|
PRBool haveFLS = HaveFirstLetterStyle(aPresContext, blockContent, blockSC);
|
|
|
|
if (haveFLS) {
|
|
|
|
// Trap out to special routine that handles adjusting a blocks
|
|
|
|
// frame tree when first-letter style is present.
|
|
|
|
#ifdef NOISY_FIRST_LETTER
|
|
|
|
printf("ContentRemoved: containingBlock=");
|
|
|
|
nsFrame::ListTag(stdout, containingBlock);
|
|
|
|
printf(" parentFrame=");
|
|
|
|
nsFrame::ListTag(stdout, parentFrame);
|
|
|
|
printf(" childFrame=");
|
|
|
|
nsFrame::ListTag(stdout, childFrame);
|
|
|
|
printf("\n");
|
|
|
|
#endif
|
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
// First update the containing blocks structure by removing the
|
|
|
|
// existing letter frames. This makes the subsequent logic
|
|
|
|
// simpler.
|
|
|
|
RemoveLetterFrames(aPresContext, shell, frameManager, containingBlock);
|
|
|
|
|
|
|
|
// Recover childFrame and parentFrame
|
|
|
|
shell->GetPrimaryFrameFor(aChild, &childFrame);
|
|
|
|
if (!childFrame) {
|
|
|
|
frameManager->ClearUndisplayedContentIn(aChild, aContainer);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
childFrame->GetParent(&parentFrame);
|
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
#ifdef NOISY_FIRST_LETTER
|
1999-10-29 14:39:48 +00:00
|
|
|
printf(" ==> revised parentFrame=");
|
|
|
|
nsFrame::ListTag(stdout, parentFrame);
|
|
|
|
printf(" childFrame=");
|
|
|
|
nsFrame::ListTag(stdout, childFrame);
|
|
|
|
printf("\n");
|
1999-08-27 21:46:10 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
1999-07-07 20:33:07 +00:00
|
|
|
// Walk the frame subtree deleting any out-of-flow frames, and
|
|
|
|
// remove the mapping from content objects to frames
|
1999-11-21 04:06:44 +00:00
|
|
|
DeletingFrameSubtree(aPresContext, shell, frameManager, childFrame);
|
1999-04-27 03:10:45 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
// See if the child frame is a floating frame
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
childFrame->GetStyleData(eStyleStruct_Display,
|
|
|
|
(const nsStyleStruct*&)display);
|
|
|
|
if (display->IsFloating()) {
|
1999-10-29 14:39:48 +00:00
|
|
|
#ifdef NOISY_FIRST_LETTER
|
|
|
|
printf(" ==> child display is still floating!\n");
|
|
|
|
#endif
|
1999-02-06 17:10:42 +00:00
|
|
|
// Get the placeholder frame
|
|
|
|
nsIFrame* placeholderFrame;
|
1999-08-05 03:09:22 +00:00
|
|
|
frameManager->GetPlaceholderFrameFor(childFrame, &placeholderFrame);
|
1999-02-06 17:10:42 +00:00
|
|
|
|
|
|
|
// Remove the mapping from the frame to its placeholder
|
1999-08-05 03:09:22 +00:00
|
|
|
frameManager->SetPlaceholderFrameFor(childFrame, nsnull);
|
1999-02-06 17:10:42 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
// Now we remove the floating frame
|
|
|
|
|
|
|
|
// XXX has to be done first for now: the blocks line list
|
|
|
|
// contains an array of pointers to the placeholder - we have to
|
|
|
|
// remove the floater first (which gets rid of the lines
|
|
|
|
// reference to the placeholder and floater) and then remove the
|
|
|
|
// placeholder
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = frameManager->RemoveFrame(aPresContext, *shell, parentFrame,
|
1999-08-05 03:09:22 +00:00
|
|
|
nsLayoutAtoms::floaterList, childFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
// Remove the placeholder frame first (XXX second for now) (so
|
|
|
|
// that it doesn't retain a dangling pointer to memory)
|
1999-02-05 03:55:18 +00:00
|
|
|
if (nsnull != placeholderFrame) {
|
1999-02-10 01:36:30 +00:00
|
|
|
placeholderFrame->GetParent(&parentFrame);
|
1999-11-21 04:06:44 +00:00
|
|
|
DeletingFrameSubtree(aPresContext, shell, frameManager, placeholderFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = frameManager->RemoveFrame(aPresContext, *shell, parentFrame,
|
1999-08-27 21:46:10 +00:00
|
|
|
nsnull, placeholderFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
|
|
|
else {
|
1999-07-07 20:33:07 +00:00
|
|
|
// See if it's absolutely or fixed positioned
|
1999-02-26 17:11:54 +00:00
|
|
|
const nsStylePosition* position;
|
|
|
|
childFrame->GetStyleData(eStyleStruct_Position, (const nsStyleStruct*&)position);
|
|
|
|
if (position->IsAbsolutelyPositioned()) {
|
|
|
|
// Get the placeholder frame
|
|
|
|
nsIFrame* placeholderFrame;
|
1999-08-05 03:09:22 +00:00
|
|
|
frameManager->GetPlaceholderFrameFor(childFrame, &placeholderFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
|
|
|
|
// Remove the mapping from the frame to its placeholder
|
1999-08-05 03:09:22 +00:00
|
|
|
frameManager->SetPlaceholderFrameFor(childFrame, nsnull);
|
1999-02-26 17:11:54 +00:00
|
|
|
|
|
|
|
// Generate two notifications. First for the absolutely positioned
|
|
|
|
// frame
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = frameManager->RemoveFrame(aPresContext, *shell, parentFrame,
|
1999-07-24 03:58:35 +00:00
|
|
|
(NS_STYLE_POSITION_FIXED == position->mPosition) ?
|
|
|
|
nsLayoutAtoms::fixedList : nsLayoutAtoms::absoluteList, childFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
|
|
|
|
// Now the placeholder frame
|
|
|
|
if (nsnull != placeholderFrame) {
|
|
|
|
placeholderFrame->GetParent(&parentFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = frameManager->RemoveFrame(aPresContext, *shell, parentFrame, nsnull,
|
1999-08-05 03:09:22 +00:00
|
|
|
placeholderFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-26 17:11:54 +00:00
|
|
|
} else {
|
|
|
|
// Notify the parent frame that it should delete the frame
|
2000-01-28 02:19:45 +00:00
|
|
|
nsIAtom* childList = GetChildListFor(childFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = frameManager->RemoveFrame(aPresContext, *shell, parentFrame,
|
2000-01-28 02:19:45 +00:00
|
|
|
childList, childFrame);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mInitialContainingBlock == childFrame) {
|
|
|
|
mInitialContainingBlock = nsnull;
|
|
|
|
}
|
1999-08-06 14:11:14 +00:00
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
if (haveFLS && mInitialContainingBlock) {
|
|
|
|
nsFrameConstructorState state(aPresContext, mFixedContainingBlock,
|
|
|
|
GetAbsoluteContainingBlock(aPresContext,
|
|
|
|
parentFrame),
|
|
|
|
GetFloaterContainingBlock(aPresContext,
|
1999-12-06 07:44:18 +00:00
|
|
|
parentFrame),
|
|
|
|
nsnull);
|
1999-12-04 23:49:50 +00:00
|
|
|
RecoverLetterFrames(shell, aPresContext, state, containingBlock);
|
1999-08-27 21:46:10 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-04-02 22:57:25 +00:00
|
|
|
static void
|
1999-11-24 06:03:41 +00:00
|
|
|
ApplyRenderingChangeToTree(nsIPresContext* aPresContext,
|
1999-09-25 05:02:52 +00:00
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIViewManager* aViewManager);
|
|
|
|
|
|
|
|
static void
|
2000-01-22 01:16:50 +00:00
|
|
|
SyncAndInvalidateView(nsIPresContext* aPresContext,
|
|
|
|
nsIView* aView,
|
|
|
|
nsIFrame* aFrame,
|
1999-09-25 05:02:52 +00:00
|
|
|
nsIViewManager* aViewManager)
|
|
|
|
{
|
|
|
|
const nsStyleColor* color;
|
|
|
|
const nsStyleDisplay* disp;
|
|
|
|
aFrame->GetStyleData(eStyleStruct_Color, (const nsStyleStruct*&) color);
|
|
|
|
aFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct*&) disp);
|
|
|
|
|
|
|
|
aViewManager->SetViewOpacity(aView, color->mOpacity);
|
1999-09-29 03:37:02 +00:00
|
|
|
|
|
|
|
// See if the view should be hidden or visible
|
|
|
|
PRBool viewIsVisible = PR_TRUE;
|
|
|
|
PRBool viewHasTransparentContent = (color->mBackgroundFlags &
|
|
|
|
NS_STYLE_BG_COLOR_TRANSPARENT) == NS_STYLE_BG_COLOR_TRANSPARENT;
|
|
|
|
|
|
|
|
if (NS_STYLE_VISIBILITY_COLLAPSE == disp->mVisible) {
|
|
|
|
viewIsVisible = PR_FALSE;
|
|
|
|
}
|
|
|
|
else if (NS_STYLE_VISIBILITY_HIDDEN == disp->mVisible) {
|
|
|
|
// If it has a widget, hide the view because the widget can't deal with it
|
|
|
|
nsIWidget* widget = nsnull;
|
|
|
|
aView->GetWidget(widget);
|
|
|
|
if (widget) {
|
|
|
|
viewIsVisible = PR_FALSE;
|
|
|
|
NS_RELEASE(widget);
|
1999-09-27 07:28:06 +00:00
|
|
|
}
|
|
|
|
else {
|
1999-09-29 03:37:02 +00:00
|
|
|
// If it's a scroll frame, then hide the view. This means that
|
|
|
|
// child elements can't override their parent's visibility, but
|
|
|
|
// it's not practical to leave it visible in all cases because
|
|
|
|
// the scrollbars will be showing
|
|
|
|
nsIAtom* frameType;
|
|
|
|
aFrame->GetFrameType(&frameType);
|
|
|
|
|
|
|
|
if (frameType == nsLayoutAtoms::scrollFrame) {
|
|
|
|
viewIsVisible = PR_FALSE;
|
|
|
|
} else {
|
|
|
|
// If it's a container element, then leave the view visible, but
|
|
|
|
// mark it as having transparent content. The reason we need to
|
|
|
|
// do this is that child elements can override their parent's
|
|
|
|
// hidden visibility and be visible anyway
|
|
|
|
nsIFrame* firstChild;
|
|
|
|
|
2000-01-22 01:16:50 +00:00
|
|
|
aFrame->FirstChild(aPresContext, nsnull, &firstChild);
|
1999-09-29 03:37:02 +00:00
|
|
|
if (firstChild) {
|
|
|
|
// It's not a left frame, so the view needs to be visible, but
|
|
|
|
// marked as having transparent content
|
|
|
|
viewHasTransparentContent = PR_TRUE;
|
|
|
|
} else {
|
|
|
|
// It's a leaf frame so go ahead and hide the view
|
|
|
|
viewIsVisible = PR_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
NS_IF_RELEASE(frameType);
|
1999-09-25 05:02:52 +00:00
|
|
|
}
|
1999-09-29 03:37:02 +00:00
|
|
|
}
|
|
|
|
|
1999-10-13 01:02:53 +00:00
|
|
|
// If the frame has visible content that overflows the content area, then we
|
|
|
|
// need the view marked as having transparent content
|
|
|
|
if (NS_STYLE_OVERFLOW_VISIBLE == disp->mOverflow) {
|
|
|
|
nsFrameState frameState;
|
|
|
|
|
|
|
|
aFrame->GetFrameState(&frameState);
|
|
|
|
if (frameState & NS_FRAME_OUTSIDE_CHILDREN) {
|
|
|
|
viewHasTransparentContent = PR_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-09-29 03:37:02 +00:00
|
|
|
if (viewIsVisible) {
|
|
|
|
aViewManager->SetViewContentTransparency(aView, viewHasTransparentContent);
|
|
|
|
aViewManager->SetViewVisibility(aView, nsViewVisibility_kShow);
|
1999-11-14 02:51:25 +00:00
|
|
|
aViewManager->UpdateView(aView, NS_VMREFRESH_NO_SYNC);
|
1999-09-25 05:02:52 +00:00
|
|
|
}
|
1999-09-27 07:28:06 +00:00
|
|
|
else {
|
1999-09-29 03:37:02 +00:00
|
|
|
aViewManager->SetViewVisibility(aView, nsViewVisibility_kHide);
|
1999-09-25 05:02:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
1999-11-24 06:03:41 +00:00
|
|
|
UpdateViewsForTree(nsIPresContext* aPresContext, nsIFrame* aFrame,
|
1999-09-25 05:02:52 +00:00
|
|
|
nsIViewManager* aViewManager, nsRect& aBoundsRect)
|
1999-04-02 22:57:25 +00:00
|
|
|
{
|
|
|
|
nsIView* view;
|
1999-11-24 06:03:41 +00:00
|
|
|
aFrame->GetView(aPresContext, &view);
|
1999-04-02 22:57:25 +00:00
|
|
|
|
|
|
|
if (view) {
|
2000-01-22 01:16:50 +00:00
|
|
|
SyncAndInvalidateView(aPresContext, view, aFrame, aViewManager);
|
1999-04-02 22:57:25 +00:00
|
|
|
}
|
|
|
|
|
1999-09-25 05:02:52 +00:00
|
|
|
nsRect bounds;
|
|
|
|
aFrame->GetRect(bounds);
|
|
|
|
nsPoint parentOffset(bounds.x, bounds.y);
|
|
|
|
bounds.x = 0;
|
|
|
|
bounds.y = 0;
|
|
|
|
|
|
|
|
// now do children of frame
|
1999-04-02 22:57:25 +00:00
|
|
|
PRInt32 listIndex = 0;
|
|
|
|
nsIAtom* childList = nsnull;
|
1999-09-25 05:02:52 +00:00
|
|
|
nsIAtom* frameType = nsnull;
|
1999-04-02 22:57:25 +00:00
|
|
|
|
|
|
|
do {
|
|
|
|
nsIFrame* child = nsnull;
|
2000-01-22 01:16:50 +00:00
|
|
|
aFrame->FirstChild(aPresContext, childList, &child);
|
1999-04-02 22:57:25 +00:00
|
|
|
while (child) {
|
1999-09-25 05:02:52 +00:00
|
|
|
nsFrameState childState;
|
|
|
|
child->GetFrameState(&childState);
|
|
|
|
if (NS_FRAME_OUT_OF_FLOW != (childState & NS_FRAME_OUT_OF_FLOW)) {
|
|
|
|
// only do frames that are in flow
|
|
|
|
child->GetFrameType(&frameType);
|
|
|
|
if (nsLayoutAtoms::placeholderFrame == frameType) { // placeholder
|
|
|
|
// get out of flow frame and start over there
|
|
|
|
nsIFrame* outOfFlowFrame = ((nsPlaceholderFrame*)child)->GetOutOfFlowFrame();
|
|
|
|
NS_ASSERTION(outOfFlowFrame, "no out-of-flow frame");
|
|
|
|
|
|
|
|
ApplyRenderingChangeToTree(aPresContext, outOfFlowFrame, aViewManager);
|
|
|
|
}
|
|
|
|
else { // regular frame
|
|
|
|
nsRect childBounds;
|
|
|
|
UpdateViewsForTree(aPresContext, child, aViewManager, childBounds);
|
|
|
|
bounds.UnionRect(bounds, childBounds);
|
|
|
|
}
|
|
|
|
NS_IF_RELEASE(frameType);
|
1999-06-10 05:24:00 +00:00
|
|
|
}
|
1999-04-02 22:57:25 +00:00
|
|
|
child->GetNextSibling(&child);
|
|
|
|
}
|
|
|
|
NS_IF_RELEASE(childList);
|
|
|
|
aFrame->GetAdditionalChildListName(listIndex++, &childList);
|
|
|
|
} while (childList);
|
|
|
|
NS_IF_RELEASE(childList);
|
1999-09-25 05:02:52 +00:00
|
|
|
aBoundsRect = bounds;
|
|
|
|
aBoundsRect += parentOffset;
|
1999-04-02 22:57:25 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
static void
|
1999-11-24 06:03:41 +00:00
|
|
|
ApplyRenderingChangeToTree(nsIPresContext* aPresContext,
|
1999-09-25 05:02:52 +00:00
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIViewManager* aViewManager)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
1999-09-25 05:02:52 +00:00
|
|
|
nsIViewManager* viewManager = aViewManager;
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Trigger rendering updates by damaging this frame and any
|
|
|
|
// continuations of this frame.
|
1999-02-27 07:15:09 +00:00
|
|
|
|
|
|
|
// XXX this needs to detect the need for a view due to an opacity change and deal with it...
|
|
|
|
|
1999-09-25 05:02:52 +00:00
|
|
|
if (viewManager) {
|
|
|
|
NS_ADDREF(viewManager); // add local ref
|
|
|
|
viewManager->BeginUpdateViewBatch();
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
while (nsnull != aFrame) {
|
|
|
|
// Get the frame's bounding rect
|
1999-09-25 05:02:52 +00:00
|
|
|
nsRect invalidRect;
|
|
|
|
nsPoint viewOffset;
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Get view if this frame has one and trigger an update. If the
|
|
|
|
// frame doesn't have a view, find the nearest containing view
|
|
|
|
// (adjusting r's coordinate system to reflect the nesting) and
|
|
|
|
// update there.
|
1999-09-25 05:02:52 +00:00
|
|
|
nsIView* view = nsnull;
|
1999-11-24 06:03:41 +00:00
|
|
|
aFrame->GetView(aPresContext, &view);
|
1999-09-25 05:02:52 +00:00
|
|
|
nsIView* parentView;
|
|
|
|
if (! view) { // XXX can view have children outside it?
|
1999-11-24 06:03:41 +00:00
|
|
|
aFrame->GetOffsetFromView(aPresContext, viewOffset, &parentView);
|
1999-09-25 05:02:52 +00:00
|
|
|
NS_ASSERTION(nsnull != parentView, "no view");
|
|
|
|
if (! viewManager) {
|
|
|
|
parentView->GetViewManager(viewManager);
|
|
|
|
viewManager->BeginUpdateViewBatch();
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-09-25 05:02:52 +00:00
|
|
|
else {
|
|
|
|
if (! viewManager) {
|
|
|
|
view->GetViewManager(viewManager);
|
|
|
|
viewManager->BeginUpdateViewBatch();
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-09-25 05:02:52 +00:00
|
|
|
UpdateViewsForTree(aPresContext, aFrame, viewManager, invalidRect);
|
|
|
|
|
|
|
|
if (! view) { // if frame has view, will already be invalidated
|
|
|
|
// XXX Instead of calling this we should really be calling
|
|
|
|
// Invalidate on on the nsFrame (which does this)
|
|
|
|
const nsStyleSpacing* spacing;
|
|
|
|
aFrame->GetStyleData(eStyleStruct_Spacing, (const nsStyleStruct*&)spacing);
|
|
|
|
nscoord width;
|
|
|
|
spacing->GetOutlineWidth(width);
|
|
|
|
if (width > 0) {
|
|
|
|
invalidRect.Inflate(width, width);
|
|
|
|
}
|
|
|
|
nsPoint frameOrigin;
|
|
|
|
aFrame->GetOrigin(frameOrigin);
|
|
|
|
invalidRect -= frameOrigin;
|
|
|
|
invalidRect += viewOffset;
|
|
|
|
viewManager->UpdateView(parentView, invalidRect, NS_VMREFRESH_NO_SYNC);
|
1999-08-19 14:29:55 +00:00
|
|
|
}
|
|
|
|
|
1999-02-24 04:48:08 +00:00
|
|
|
aFrame->GetNextInFlow(&aFrame);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
1999-09-25 05:02:52 +00:00
|
|
|
if (viewManager) {
|
2000-02-09 15:48:01 +00:00
|
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
|
|
nsresult rv = aPresContext->GetShell(getter_AddRefs(presShell));
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
PRBool isReflowLocked = PR_FALSE;
|
|
|
|
presShell->IsReflowLocked(&isReflowLocked);
|
|
|
|
if (isReflowLocked) {
|
|
|
|
viewManager->EndUpdateViewBatch(NS_VMREFRESH_NO_SYNC);
|
|
|
|
} else {
|
|
|
|
viewManager->EndUpdateViewBatch(NS_VMREFRESH_IMMEDIATE);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
viewManager->EndUpdateViewBatch(NS_VMREFRESH_NO_SYNC);
|
|
|
|
}
|
1999-09-25 05:02:52 +00:00
|
|
|
// viewManager->Composite();
|
1999-02-05 03:55:18 +00:00
|
|
|
NS_RELEASE(viewManager);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
StyleChangeReflow(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIAtom * aAttribute)
|
|
|
|
{
|
1999-02-12 17:45:58 +00:00
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIReflowCommand* reflowCmd;
|
|
|
|
nsresult rv = NS_NewHTMLReflowCommand(&reflowCmd, aFrame,
|
|
|
|
nsIReflowCommand::StyleChanged,
|
|
|
|
nsnull,
|
|
|
|
aAttribute);
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
shell->AppendReflowCommand(reflowCmd);
|
|
|
|
NS_RELEASE(reflowCmd);
|
|
|
|
}
|
1) implememted box reflow coelescing.
2) implemented gfx scrollbars for list boxes
3) fixed progess meter to be an animated gif
4) fixed bugs 23521, 24721, 19114, 20546, 24385, 24457, 23156, 20226, 22543
-r hyatt, troy, rod
2000-02-09 22:02:40 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
nsFrameState state;
|
|
|
|
aFrame->GetFrameState(&state);
|
|
|
|
state |= NS_FRAME_IS_DIRTY;
|
|
|
|
aFrame->SetFrameState(state);
|
|
|
|
nsIFrame* parent;
|
|
|
|
aFrame->GetParent(&parent);
|
|
|
|
parent->ReflowDirtyChild(shell, aFrame);
|
|
|
|
*/
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::ContentChanged(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsISupports* aSubContent)
|
|
|
|
{
|
1999-02-12 17:45:58 +00:00
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
// Find the child frame
|
|
|
|
nsIFrame* frame;
|
1999-02-12 17:45:58 +00:00
|
|
|
shell->GetPrimaryFrameFor(aContent, &frame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
// Notify the first frame that maps the content. It will generate a reflow
|
|
|
|
// command
|
|
|
|
|
|
|
|
// It's possible the frame whose content changed isn't inserted into the
|
|
|
|
// frame hierarchy yet, or that there is no frame that maps the content
|
|
|
|
if (nsnull != frame) {
|
|
|
|
#if 0
|
|
|
|
NS_FRAME_LOG(NS_FRAME_TRACE_CALLS,
|
|
|
|
("nsHTMLStyleSheet::ContentChanged: content=%p[%s] subcontent=%p frame=%p",
|
|
|
|
aContent, ContentTag(aContent, 0),
|
|
|
|
aSubContent, frame));
|
|
|
|
#endif
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// Special check for text content that is a child of a letter
|
|
|
|
// frame. There are two interesting cases that we have to handle
|
|
|
|
// carefully: text content that is going empty (which means we
|
|
|
|
// should select a new text node as the first-letter text) or text
|
|
|
|
// content that empty but is no longer empty (it might be the
|
|
|
|
// first-letter text but isn't currently).
|
|
|
|
//
|
|
|
|
// To deal with both of these we make a simple change: map a
|
|
|
|
// ContentChanged into a ContentReplaced when we are changing text
|
|
|
|
// that is part of a first-letter situation.
|
|
|
|
PRBool doContentChanged = PR_TRUE;
|
|
|
|
nsCOMPtr<nsITextContent> textContent(do_QueryInterface(aContent));
|
|
|
|
if (textContent) {
|
|
|
|
// Ok, it's text content. Now do some real work...
|
|
|
|
nsIFrame* block = GetFloaterContainingBlock(aPresContext, frame);
|
|
|
|
if (block) {
|
|
|
|
// See if the block has first-letter style applied to it.
|
|
|
|
nsCOMPtr<nsIContent> blockContent;
|
|
|
|
block->GetContent(getter_AddRefs(blockContent));
|
|
|
|
nsCOMPtr<nsIStyleContext> blockSC;
|
|
|
|
block->GetStyleContext(getter_AddRefs(blockSC));
|
|
|
|
PRBool haveFirstLetterStyle =
|
|
|
|
HaveFirstLetterStyle(aPresContext, blockContent, blockSC);
|
|
|
|
if (haveFirstLetterStyle) {
|
|
|
|
// The block has first-letter style. Use content-replaced to
|
|
|
|
// repair the blocks frame structure properly.
|
|
|
|
nsCOMPtr<nsIContent> container;
|
|
|
|
aContent->GetParent(*getter_AddRefs(container));
|
|
|
|
if (container) {
|
|
|
|
PRInt32 ix;
|
|
|
|
container->IndexOf(aContent, ix);
|
|
|
|
doContentChanged = PR_FALSE;
|
|
|
|
rv = ContentReplaced(aPresContext, container,
|
|
|
|
aContent, aContent, ix);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (doContentChanged) {
|
|
|
|
frame->ContentChanged(aPresContext, aContent, aSubContent);
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
1999-02-27 07:15:09 +00:00
|
|
|
|
1999-03-25 06:41:43 +00:00
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::ProcessRestyledFrames(nsStyleChangeList& aChangeList,
|
|
|
|
nsIPresContext* aPresContext)
|
|
|
|
{
|
|
|
|
PRInt32 count = aChangeList.Count();
|
|
|
|
while (0 < count--) {
|
|
|
|
nsIFrame* frame;
|
1999-10-02 04:27:40 +00:00
|
|
|
nsIContent* content;
|
1999-03-25 06:41:43 +00:00
|
|
|
PRInt32 hint;
|
1999-10-02 04:27:40 +00:00
|
|
|
aChangeList.ChangeAt(count, frame, content, hint);
|
1999-03-25 06:41:43 +00:00
|
|
|
switch (hint) {
|
|
|
|
case NS_STYLE_HINT_RECONSTRUCT_ALL:
|
|
|
|
NS_ERROR("This shouldn't happen");
|
|
|
|
break;
|
|
|
|
case NS_STYLE_HINT_FRAMECHANGE:
|
|
|
|
RecreateFramesForContent(aPresContext, content);
|
|
|
|
break;
|
|
|
|
case NS_STYLE_HINT_REFLOW:
|
|
|
|
StyleChangeReflow(aPresContext, frame, nsnull);
|
|
|
|
break;
|
|
|
|
case NS_STYLE_HINT_VISUAL:
|
1999-11-24 06:03:41 +00:00
|
|
|
ApplyRenderingChangeToTree(aPresContext, frame, nsnull);
|
1999-03-25 06:41:43 +00:00
|
|
|
break;
|
|
|
|
case NS_STYLE_HINT_CONTENT:
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
aChangeList.Clear();
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-02-27 07:15:09 +00:00
|
|
|
NS_IMETHODIMP
|
1999-04-20 00:05:14 +00:00
|
|
|
nsCSSFrameConstructor::ContentStatesChanged(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent1,
|
|
|
|
nsIContent* aContent2)
|
1999-02-27 07:15:09 +00:00
|
|
|
{
|
|
|
|
nsresult result = NS_OK;
|
|
|
|
|
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
|
|
|
|
1999-04-20 00:05:14 +00:00
|
|
|
NS_ASSERTION(shell, "couldn't get pres shell");
|
|
|
|
if (shell) {
|
|
|
|
nsIStyleSet* styleSet;
|
|
|
|
shell->GetStyleSet(&styleSet);
|
|
|
|
|
|
|
|
NS_ASSERTION(styleSet, "couldn't get style set");
|
|
|
|
if (styleSet) { // test if any style rules exist which are dependent on content state
|
|
|
|
nsIFrame* primaryFrame1 = nsnull;
|
|
|
|
nsIFrame* primaryFrame2 = nsnull;
|
|
|
|
if (aContent1 && (NS_OK == styleSet->HasStateDependentStyle(aPresContext, aContent1))) {
|
|
|
|
shell->GetPrimaryFrameFor(aContent1, &primaryFrame1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
aContent1 = nsnull;
|
|
|
|
}
|
1999-02-27 07:15:09 +00:00
|
|
|
|
1999-04-20 00:05:14 +00:00
|
|
|
if (aContent2 && (aContent2 != aContent1) &&
|
|
|
|
(NS_OK == styleSet->HasStateDependentStyle(aPresContext, aContent2))) {
|
|
|
|
shell->GetPrimaryFrameFor(aContent2, &primaryFrame2);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
aContent2 = nsnull;
|
|
|
|
}
|
|
|
|
NS_RELEASE(styleSet);
|
|
|
|
|
|
|
|
if (primaryFrame1 && primaryFrame2) { // detect if one is parent of other, skip child
|
|
|
|
nsIFrame* parent;
|
|
|
|
primaryFrame1->GetParent(&parent);
|
|
|
|
while (parent) {
|
|
|
|
if (parent == primaryFrame2) { // frame2 is frame1's parent, skip frame1
|
|
|
|
primaryFrame1 = nsnull;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
parent->GetParent(&parent);
|
|
|
|
}
|
|
|
|
if (primaryFrame1) {
|
|
|
|
primaryFrame2->GetParent(&parent);
|
|
|
|
while (parent) {
|
|
|
|
if (parent == primaryFrame1) { // frame1 is frame2's parent, skip frame2
|
|
|
|
primaryFrame2 = nsnull;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
parent->GetParent(&parent);
|
1999-03-25 06:41:43 +00:00
|
|
|
}
|
1999-02-27 07:15:09 +00:00
|
|
|
}
|
1999-04-20 00:05:14 +00:00
|
|
|
}
|
|
|
|
|
1999-09-21 07:55:37 +00:00
|
|
|
nsCOMPtr<nsIFrameManager> frameManager;
|
|
|
|
shell->GetFrameManager(getter_AddRefs(frameManager));
|
1999-04-20 00:05:14 +00:00
|
|
|
|
|
|
|
if (primaryFrame1) {
|
|
|
|
nsStyleChangeList changeList1;
|
|
|
|
nsStyleChangeList changeList2;
|
|
|
|
PRInt32 frameChange1 = NS_STYLE_HINT_NONE;
|
|
|
|
PRInt32 frameChange2 = NS_STYLE_HINT_NONE;
|
1999-11-24 06:03:41 +00:00
|
|
|
frameManager->ComputeStyleChangeFor(aPresContext, primaryFrame1,
|
1999-10-15 23:14:44 +00:00
|
|
|
kNameSpaceID_Unknown, nsnull,
|
|
|
|
changeList1, NS_STYLE_HINT_NONE, frameChange1);
|
1999-04-20 00:05:14 +00:00
|
|
|
|
|
|
|
if ((frameChange1 != NS_STYLE_HINT_RECONSTRUCT_ALL) && (primaryFrame2)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
frameManager->ComputeStyleChangeFor(aPresContext, primaryFrame2,
|
1999-10-15 23:14:44 +00:00
|
|
|
kNameSpaceID_Unknown, nsnull,
|
|
|
|
changeList2, NS_STYLE_HINT_NONE, frameChange2);
|
1999-04-20 00:05:14 +00:00
|
|
|
}
|
1999-02-27 07:15:09 +00:00
|
|
|
|
1999-04-20 00:05:14 +00:00
|
|
|
if ((frameChange1 == NS_STYLE_HINT_RECONSTRUCT_ALL) ||
|
|
|
|
(frameChange2 == NS_STYLE_HINT_RECONSTRUCT_ALL)) {
|
|
|
|
result = ReconstructDocElementHierarchy(aPresContext);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
switch (frameChange1) {
|
|
|
|
case NS_STYLE_HINT_FRAMECHANGE:
|
|
|
|
result = RecreateFramesForContent(aPresContext, aContent1);
|
1999-04-20 14:23:18 +00:00
|
|
|
changeList1.Clear();
|
|
|
|
break;
|
1999-04-20 00:05:14 +00:00
|
|
|
case NS_STYLE_HINT_REFLOW:
|
|
|
|
case NS_STYLE_HINT_VISUAL:
|
|
|
|
case NS_STYLE_HINT_CONTENT:
|
|
|
|
// let primary frame deal with it
|
1999-09-10 05:54:00 +00:00
|
|
|
result = primaryFrame1->ContentStateChanged(aPresContext, aContent1, frameChange1);
|
1999-04-20 00:05:14 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
switch (frameChange2) {
|
|
|
|
case NS_STYLE_HINT_FRAMECHANGE:
|
|
|
|
result = RecreateFramesForContent(aPresContext, aContent2);
|
1999-04-20 14:23:18 +00:00
|
|
|
changeList2.Clear();
|
|
|
|
break;
|
1999-04-20 00:05:14 +00:00
|
|
|
case NS_STYLE_HINT_REFLOW:
|
|
|
|
case NS_STYLE_HINT_VISUAL:
|
|
|
|
case NS_STYLE_HINT_CONTENT:
|
|
|
|
// let primary frame deal with it
|
1999-09-10 05:54:00 +00:00
|
|
|
result = primaryFrame2->ContentStateChanged(aPresContext, aContent2, frameChange2);
|
1999-04-20 00:05:14 +00:00
|
|
|
// then process any children that need it
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
1999-04-20 14:23:18 +00:00
|
|
|
ProcessRestyledFrames(changeList1, aPresContext);
|
|
|
|
ProcessRestyledFrames(changeList2, aPresContext);
|
1999-04-20 00:05:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (primaryFrame2) {
|
|
|
|
nsStyleChangeList changeList;
|
|
|
|
PRInt32 frameChange = NS_STYLE_HINT_NONE;
|
1999-11-24 06:03:41 +00:00
|
|
|
frameManager->ComputeStyleChangeFor(aPresContext, primaryFrame2,
|
1999-10-15 23:14:44 +00:00
|
|
|
kNameSpaceID_Unknown, nsnull,
|
|
|
|
changeList, NS_STYLE_HINT_NONE, frameChange);
|
1999-04-20 00:05:14 +00:00
|
|
|
|
|
|
|
switch (frameChange) { // max change needed for top level frames
|
|
|
|
case NS_STYLE_HINT_RECONSTRUCT_ALL:
|
|
|
|
result = ReconstructDocElementHierarchy(aPresContext);
|
1999-04-20 14:23:18 +00:00
|
|
|
changeList.Clear();
|
|
|
|
break;
|
1999-04-20 00:05:14 +00:00
|
|
|
case NS_STYLE_HINT_FRAMECHANGE:
|
|
|
|
result = RecreateFramesForContent(aPresContext, aContent2);
|
1999-04-20 14:23:18 +00:00
|
|
|
changeList.Clear();
|
|
|
|
break;
|
1999-04-20 00:05:14 +00:00
|
|
|
case NS_STYLE_HINT_REFLOW:
|
|
|
|
case NS_STYLE_HINT_VISUAL:
|
|
|
|
case NS_STYLE_HINT_CONTENT:
|
|
|
|
// let primary frame deal with it
|
1999-09-10 05:54:00 +00:00
|
|
|
result = primaryFrame2->ContentStateChanged(aPresContext, aContent2, frameChange);
|
1999-04-20 00:05:14 +00:00
|
|
|
// then process any children that need it
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
1999-04-20 14:23:18 +00:00
|
|
|
ProcessRestyledFrames(changeList, aPresContext);
|
1999-04-20 00:05:14 +00:00
|
|
|
}
|
|
|
|
else { // no frames, reconstruct for content
|
|
|
|
if (aContent1) {
|
|
|
|
result = RecreateFramesForContent(aPresContext, aContent1);
|
|
|
|
}
|
|
|
|
if (aContent2) {
|
|
|
|
result = RecreateFramesForContent(aPresContext, aContent2);
|
|
|
|
}
|
|
|
|
}
|
1999-02-27 07:15:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::AttributeChanged(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent,
|
1999-10-15 23:14:44 +00:00
|
|
|
PRInt32 aNameSpaceID,
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIAtom* aAttribute,
|
|
|
|
PRInt32 aHint)
|
|
|
|
{
|
|
|
|
nsresult result = NS_OK;
|
|
|
|
|
1999-02-12 17:45:58 +00:00
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
1999-03-25 06:41:43 +00:00
|
|
|
nsIFrame* primaryFrame;
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-03-25 06:41:43 +00:00
|
|
|
shell->GetPrimaryFrameFor(aContent, &primaryFrame);
|
1999-03-01 16:57:35 +00:00
|
|
|
|
2000-01-06 08:58:05 +00:00
|
|
|
#ifdef INCLUDE_XUL
|
|
|
|
// The following tree widget trap prevents offscreen tree widget
|
|
|
|
// content from being removed and re-inserted (which is what would
|
|
|
|
// happen otherwise).
|
|
|
|
if (!primaryFrame) {
|
|
|
|
nsCOMPtr<nsIAtom> tag;
|
|
|
|
aContent->GetTag(*getter_AddRefs(tag));
|
|
|
|
if (tag && (tag.get() == nsXULAtoms::treechildren ||
|
|
|
|
tag.get() == nsXULAtoms::treeitem))
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
#endif // INCLUDE_XUL
|
|
|
|
|
1999-03-01 16:57:35 +00:00
|
|
|
PRBool reconstruct = PR_FALSE;
|
1999-02-05 03:55:18 +00:00
|
|
|
PRBool restyle = PR_FALSE;
|
|
|
|
PRBool reframe = PR_FALSE;
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
NS_FRAME_LOG(NS_FRAME_TRACE_CALLS,
|
|
|
|
("HTMLStyleSheet::AttributeChanged: content=%p[%s] frame=%p",
|
|
|
|
aContent, ContentTag(aContent, 0), frame));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// the style tag has its own interpretation based on aHint
|
1999-02-27 01:34:32 +00:00
|
|
|
if (NS_STYLE_HINT_UNKNOWN == aHint) {
|
|
|
|
nsIStyledContent* styledContent;
|
2000-02-02 22:24:56 +00:00
|
|
|
result = aContent->QueryInterface(NS_GET_IID(nsIStyledContent), (void**)&styledContent);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
if (NS_OK == result) {
|
|
|
|
// Get style hint from HTML content object.
|
1999-07-07 01:28:43 +00:00
|
|
|
styledContent->GetMappedAttributeImpact(aAttribute, aHint);
|
1999-02-27 01:34:32 +00:00
|
|
|
NS_RELEASE(styledContent);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (aHint) {
|
|
|
|
default:
|
1999-03-01 16:57:35 +00:00
|
|
|
case NS_STYLE_HINT_RECONSTRUCT_ALL:
|
|
|
|
reconstruct = PR_TRUE;
|
1999-02-05 03:55:18 +00:00
|
|
|
case NS_STYLE_HINT_FRAMECHANGE:
|
|
|
|
reframe = PR_TRUE;
|
|
|
|
case NS_STYLE_HINT_REFLOW:
|
|
|
|
case NS_STYLE_HINT_VISUAL:
|
1999-03-25 06:41:43 +00:00
|
|
|
case NS_STYLE_HINT_UNKNOWN:
|
1999-02-05 03:55:18 +00:00
|
|
|
case NS_STYLE_HINT_CONTENT:
|
|
|
|
case NS_STYLE_HINT_AURAL:
|
|
|
|
restyle = PR_TRUE;
|
|
|
|
break;
|
|
|
|
case NS_STYLE_HINT_NONE:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// apply changes
|
1999-03-01 16:57:35 +00:00
|
|
|
if (PR_TRUE == reconstruct) {
|
1999-03-25 06:41:43 +00:00
|
|
|
result = ReconstructDocElementHierarchy(aPresContext);
|
1999-03-01 16:57:35 +00:00
|
|
|
}
|
|
|
|
else if (PR_TRUE == reframe) {
|
1999-03-25 06:41:43 +00:00
|
|
|
result = RecreateFramesForContent(aPresContext, aContent);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
else if (PR_TRUE == restyle) {
|
|
|
|
// If there is no frame then there is no point in re-styling it,
|
|
|
|
// is there?
|
1999-03-25 06:41:43 +00:00
|
|
|
if (primaryFrame) {
|
|
|
|
PRInt32 maxHint = aHint;
|
|
|
|
nsStyleChangeList changeList;
|
|
|
|
// put primary frame on list to deal with, re-resolve may update or add next in flows
|
1999-10-02 04:27:40 +00:00
|
|
|
changeList.AppendChange(primaryFrame, aContent, maxHint);
|
1999-09-21 07:55:37 +00:00
|
|
|
nsCOMPtr<nsIFrameManager> frameManager;
|
|
|
|
shell->GetFrameManager(getter_AddRefs(frameManager));
|
1999-11-24 06:03:41 +00:00
|
|
|
frameManager->ComputeStyleChangeFor(aPresContext, primaryFrame,
|
1999-10-15 23:14:44 +00:00
|
|
|
aNameSpaceID, aAttribute,
|
|
|
|
changeList, aHint, maxHint);
|
1999-03-25 06:41:43 +00:00
|
|
|
|
1999-04-20 14:23:18 +00:00
|
|
|
switch (maxHint) { // maxHint is hint for primary only
|
1999-03-25 06:41:43 +00:00
|
|
|
case NS_STYLE_HINT_RECONSTRUCT_ALL:
|
|
|
|
result = ReconstructDocElementHierarchy(aPresContext);
|
1999-04-20 14:23:18 +00:00
|
|
|
changeList.Clear();
|
1999-03-25 06:41:43 +00:00
|
|
|
break;
|
|
|
|
case NS_STYLE_HINT_FRAMECHANGE:
|
|
|
|
result = RecreateFramesForContent(aPresContext, aContent);
|
1999-04-20 14:23:18 +00:00
|
|
|
changeList.Clear();
|
1999-03-25 06:41:43 +00:00
|
|
|
break;
|
|
|
|
case NS_STYLE_HINT_REFLOW:
|
|
|
|
case NS_STYLE_HINT_VISUAL:
|
|
|
|
case NS_STYLE_HINT_CONTENT:
|
|
|
|
// let the frame deal with it, since we don't know how to
|
1999-10-15 23:14:44 +00:00
|
|
|
result = primaryFrame->AttributeChanged(aPresContext, aContent, aNameSpaceID, aAttribute, maxHint);
|
1999-03-25 06:41:43 +00:00
|
|
|
default:
|
|
|
|
break;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-04-20 14:23:18 +00:00
|
|
|
// handle any children (primary may be on list too)
|
|
|
|
ProcessRestyledFrames(changeList, aPresContext);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
1999-03-25 06:41:43 +00:00
|
|
|
else { // no frame now, possibly genetate one with new style data
|
|
|
|
result = RecreateFramesForContent(aPresContext, aContent);
|
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Style change notifications
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::StyleRuleChanged(nsIPresContext* aPresContext,
|
|
|
|
nsIStyleSheet* aStyleSheet,
|
|
|
|
nsIStyleRule* aStyleRule,
|
|
|
|
PRInt32 aHint)
|
|
|
|
{
|
1999-03-25 06:41:43 +00:00
|
|
|
nsresult result = NS_OK;
|
1999-02-12 17:45:58 +00:00
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* frame;
|
1999-02-12 17:45:58 +00:00
|
|
|
shell->GetRootFrame(&frame);
|
1999-02-05 03:55:18 +00:00
|
|
|
|
|
|
|
PRBool reframe = PR_FALSE;
|
|
|
|
PRBool reflow = PR_FALSE;
|
|
|
|
PRBool render = PR_FALSE;
|
|
|
|
PRBool restyle = PR_FALSE;
|
|
|
|
switch (aHint) {
|
|
|
|
default:
|
|
|
|
case NS_STYLE_HINT_UNKNOWN:
|
|
|
|
case NS_STYLE_HINT_FRAMECHANGE:
|
|
|
|
reframe = PR_TRUE;
|
|
|
|
case NS_STYLE_HINT_REFLOW:
|
|
|
|
reflow = PR_TRUE;
|
|
|
|
case NS_STYLE_HINT_VISUAL:
|
|
|
|
render = PR_TRUE;
|
|
|
|
case NS_STYLE_HINT_CONTENT:
|
|
|
|
case NS_STYLE_HINT_AURAL:
|
|
|
|
restyle = PR_TRUE;
|
|
|
|
break;
|
|
|
|
case NS_STYLE_HINT_NONE:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
1999-09-22 01:18:45 +00:00
|
|
|
if (restyle) {
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIStyleContext* sc;
|
1999-02-10 00:42:56 +00:00
|
|
|
frame->GetStyleContext(&sc);
|
1999-02-05 03:55:18 +00:00
|
|
|
sc->RemapStyle(aPresContext);
|
|
|
|
NS_RELEASE(sc);
|
1999-09-22 01:18:45 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-09-22 01:18:45 +00:00
|
|
|
if (reframe) {
|
|
|
|
result = ReconstructDocElementHierarchy(aPresContext);
|
|
|
|
}
|
|
|
|
else {
|
1999-02-05 03:55:18 +00:00
|
|
|
// XXX hack, skip the root and scrolling frames
|
2000-01-22 01:16:50 +00:00
|
|
|
frame->FirstChild(aPresContext, nsnull, &frame);
|
|
|
|
frame->FirstChild(aPresContext, nsnull, &frame);
|
1999-03-25 06:41:43 +00:00
|
|
|
if (reflow) {
|
1999-02-05 03:55:18 +00:00
|
|
|
StyleChangeReflow(aPresContext, frame, nsnull);
|
|
|
|
}
|
|
|
|
else if (render) {
|
1999-11-24 06:03:41 +00:00
|
|
|
ApplyRenderingChangeToTree(aPresContext, frame, nsnull);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-03-25 06:41:43 +00:00
|
|
|
return result;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::StyleRuleAdded(nsIPresContext* aPresContext,
|
|
|
|
nsIStyleSheet* aStyleSheet,
|
|
|
|
nsIStyleRule* aStyleRule)
|
|
|
|
{
|
1999-03-25 06:41:43 +00:00
|
|
|
// XXX TBI: should query rule for impact and do minimal work
|
|
|
|
ReconstructDocElementHierarchy(aPresContext);
|
1999-02-05 03:55:18 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::StyleRuleRemoved(nsIPresContext* aPresContext,
|
|
|
|
nsIStyleSheet* aStyleSheet,
|
|
|
|
nsIStyleRule* aStyleRule)
|
|
|
|
{
|
1999-03-25 06:41:43 +00:00
|
|
|
// XXX TBI: should query rule for impact and do minimal work
|
|
|
|
ReconstructDocElementHierarchy(aPresContext);
|
1999-02-05 03:55:18 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2000-03-14 05:30:00 +00:00
|
|
|
static void
|
|
|
|
GetAlternateTextFor(nsIContent* aContent,
|
|
|
|
nsIAtom* aTag, // content object's tag
|
|
|
|
nsString& aAltText)
|
1999-02-06 03:45:11 +00:00
|
|
|
{
|
2000-03-14 05:30:00 +00:00
|
|
|
nsresult rv;
|
1999-02-06 03:45:11 +00:00
|
|
|
|
1999-12-22 21:26:14 +00:00
|
|
|
// The "alt" attribute specifies alternate text that is rendered
|
|
|
|
// when the image can not be displayed
|
2000-03-14 05:30:00 +00:00
|
|
|
rv = aContent->GetAttribute(kNameSpaceID_HTML, nsHTMLAtoms::alt, aAltText);
|
1999-12-22 21:26:14 +00:00
|
|
|
if (NS_CONTENT_ATTR_NOT_THERE == rv) {
|
|
|
|
// If there's no "alt" attribute, then use the value of the "title"
|
|
|
|
// attribute. Note that this is not the same as a value of ""
|
2000-03-14 05:30:00 +00:00
|
|
|
rv = aContent->GetAttribute(kNameSpaceID_HTML, nsHTMLAtoms::title, aAltText);
|
1999-12-22 21:26:14 +00:00
|
|
|
}
|
|
|
|
if (NS_CONTENT_ATTR_NOT_THERE == rv) {
|
2000-03-14 05:30:00 +00:00
|
|
|
// If there's no "title" attribute, then what we do depends on the type
|
|
|
|
// of element
|
|
|
|
if (nsHTMLAtoms::img == aTag) {
|
|
|
|
// Use the filename minus the extension
|
|
|
|
aContent->GetAttribute(kNameSpaceID_HTML, nsHTMLAtoms::src, aAltText);
|
|
|
|
if (aAltText.Length() > 0) {
|
|
|
|
// The path is a hierarchical structure of segments. Get the last substring
|
|
|
|
// in the path
|
|
|
|
PRInt32 offset = aAltText.RFindChar('/');
|
|
|
|
if (offset >= 0) {
|
|
|
|
aAltText.Cut(0, offset + 1);
|
|
|
|
}
|
1999-02-06 03:45:11 +00:00
|
|
|
|
2000-03-14 05:30:00 +00:00
|
|
|
// Ignore fragment identifiers ('#' delimiter), query strings ('?'
|
|
|
|
// delimiter), and anything beginning with ';'
|
|
|
|
offset = aAltText.FindCharInSet("#?;");
|
|
|
|
if (offset >= 0) {
|
|
|
|
aAltText.Truncate(offset);
|
|
|
|
}
|
2000-02-11 01:00:17 +00:00
|
|
|
|
2000-03-14 05:30:00 +00:00
|
|
|
// Trim off any extension (including the '.')
|
|
|
|
offset = aAltText.RFindChar('.');
|
|
|
|
if (offset >= 0) {
|
|
|
|
aAltText.Truncate(offset);
|
|
|
|
}
|
1999-02-06 03:45:11 +00:00
|
|
|
}
|
2000-03-14 05:30:00 +00:00
|
|
|
|
|
|
|
} else if (nsHTMLAtoms::input == aTag) {
|
|
|
|
// Use the valuf of the "name" attribute
|
|
|
|
aContent->GetAttribute(kNameSpaceID_HTML, nsHTMLAtoms::name, aAltText);
|
1999-02-06 03:45:11 +00:00
|
|
|
}
|
1999-12-22 21:26:14 +00:00
|
|
|
}
|
2000-03-14 05:30:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Construct an alternate frame to use when the image can't be rendered
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::ConstructAlternateImageFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIFrame*& aFrame)
|
|
|
|
{
|
|
|
|
nsAutoString altText;
|
|
|
|
|
|
|
|
// Initialize OUT parameter
|
|
|
|
aFrame = nsnull;
|
|
|
|
|
|
|
|
// Get the alternate text to use
|
|
|
|
GetAlternateTextFor(aContent, nsHTMLAtoms::img, altText);
|
1999-04-14 04:00:16 +00:00
|
|
|
|
1999-12-22 21:26:14 +00:00
|
|
|
// Create a text content element for the alternate text
|
|
|
|
nsCOMPtr<nsIContent> altTextContent;
|
|
|
|
NS_NewTextNode(getter_AddRefs(altTextContent));
|
1999-02-06 03:45:11 +00:00
|
|
|
|
1999-12-22 21:26:14 +00:00
|
|
|
// Set the content's text
|
|
|
|
nsIDOMCharacterData* domData;
|
|
|
|
altTextContent->QueryInterface(kIDOMCharacterDataIID, (void**)&domData);
|
|
|
|
domData->SetData(altText);
|
|
|
|
NS_RELEASE(domData);
|
|
|
|
|
|
|
|
// Set aContent as the parent content and set the document object
|
|
|
|
nsCOMPtr<nsIDocument> document;
|
|
|
|
aContent->GetDocument(*getter_AddRefs(document));
|
|
|
|
altTextContent->SetParent(aContent);
|
|
|
|
altTextContent->SetDocument(document, PR_TRUE);
|
|
|
|
|
|
|
|
// Create either an inline frame, block frame, or area frame
|
|
|
|
nsIFrame* containerFrame;
|
2000-01-22 01:16:50 +00:00
|
|
|
PRBool isOutOfFlow = PR_FALSE;
|
1999-12-22 21:26:14 +00:00
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
const nsStylePosition* position = (const nsStylePosition*)
|
|
|
|
aStyleContext->GetStyleData(eStyleStruct_Position);
|
1999-04-20 03:57:01 +00:00
|
|
|
|
1999-12-22 21:26:14 +00:00
|
|
|
if (position->IsAbsolutelyPositioned()) {
|
|
|
|
NS_NewAbsoluteItemWrapperFrame(aPresShell, &containerFrame);
|
2000-01-22 01:16:50 +00:00
|
|
|
isOutOfFlow = PR_TRUE;
|
|
|
|
} else if (display->IsFloating()) {
|
|
|
|
NS_NewFloatingItemWrapperFrame(aPresShell, &containerFrame);
|
|
|
|
isOutOfFlow = PR_TRUE;
|
|
|
|
} else if (NS_STYLE_DISPLAY_BLOCK == display->mDisplay) {
|
1999-12-22 21:26:14 +00:00
|
|
|
NS_NewBlockFrame(aPresShell, &containerFrame);
|
|
|
|
} else {
|
|
|
|
NS_NewInlineFrame(aPresShell, &containerFrame);
|
|
|
|
}
|
|
|
|
containerFrame->Init(aPresContext, aContent, aParentFrame, aStyleContext, nsnull);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, containerFrame,
|
|
|
|
aStyleContext, PR_FALSE);
|
1999-04-20 03:57:01 +00:00
|
|
|
|
2000-01-22 01:16:50 +00:00
|
|
|
// If the frame is out-of-flow, then mark it as such
|
2000-03-17 05:10:37 +00:00
|
|
|
if (isOutOfFlow) {
|
|
|
|
nsFrameState frameState;
|
|
|
|
containerFrame->GetFrameState(&frameState);
|
|
|
|
containerFrame->SetFrameState(frameState | NS_FRAME_OUT_OF_FLOW);
|
|
|
|
}
|
2000-01-22 01:16:50 +00:00
|
|
|
|
1999-12-22 21:26:14 +00:00
|
|
|
// Create a text frame to display the alt-text. It gets a pseudo-element
|
|
|
|
// style context
|
|
|
|
nsIFrame* textFrame;
|
|
|
|
nsIStyleContext* textStyleContext;
|
1999-04-20 03:57:01 +00:00
|
|
|
|
1999-12-22 21:26:14 +00:00
|
|
|
NS_NewTextFrame(aPresShell, &textFrame);
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::textPseudo,
|
|
|
|
aStyleContext, PR_FALSE,
|
|
|
|
&textStyleContext);
|
1999-04-20 03:57:01 +00:00
|
|
|
|
1999-12-22 21:26:14 +00:00
|
|
|
textFrame->Init(aPresContext, altTextContent, containerFrame,
|
|
|
|
textStyleContext, nsnull);
|
|
|
|
NS_RELEASE(textStyleContext);
|
|
|
|
containerFrame->SetInitialChildList(aPresContext, nsnull, textFrame);
|
|
|
|
|
|
|
|
// Return the container frame
|
|
|
|
aFrame = containerFrame;
|
1999-02-06 03:45:11 +00:00
|
|
|
|
2000-03-14 05:30:00 +00:00
|
|
|
return NS_OK;
|
1999-02-06 03:45:11 +00:00
|
|
|
}
|
|
|
|
|
1999-09-15 04:03:08 +00:00
|
|
|
#ifdef NS_DEBUG
|
|
|
|
static PRBool
|
|
|
|
IsPlaceholderFrame(nsIFrame* aFrame)
|
1999-09-07 02:42:00 +00:00
|
|
|
{
|
1999-09-15 04:03:08 +00:00
|
|
|
nsIAtom* frameType;
|
|
|
|
PRBool result;
|
1999-09-07 02:42:00 +00:00
|
|
|
|
1999-09-15 04:03:08 +00:00
|
|
|
aFrame->GetFrameType(&frameType);
|
|
|
|
result = frameType == nsLayoutAtoms::placeholderFrame;
|
|
|
|
NS_IF_RELEASE(frameType);
|
|
|
|
return result;
|
1999-09-07 02:42:00 +00:00
|
|
|
}
|
1999-09-15 04:03:08 +00:00
|
|
|
#endif
|
1999-02-06 03:45:11 +00:00
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
NS_IMETHODIMP
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::CantRenderReplacedElement(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-02-05 03:55:18 +00:00
|
|
|
nsIFrame* aFrame)
|
|
|
|
{
|
1999-04-27 03:58:46 +00:00
|
|
|
nsIFrame* parentFrame;
|
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
|
|
nsCOMPtr<nsIAtom> tag;
|
|
|
|
nsresult rv = NS_OK;
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-02-10 01:36:30 +00:00
|
|
|
aFrame->GetParent(&parentFrame);
|
1999-04-27 03:58:46 +00:00
|
|
|
aFrame->GetStyleContext(getter_AddRefs(styleContext));
|
1999-02-05 03:55:18 +00:00
|
|
|
|
1999-04-27 03:58:46 +00:00
|
|
|
// Get aFrame's content object and the tag name
|
1999-02-11 15:54:13 +00:00
|
|
|
aFrame->GetContent(getter_AddRefs(content));
|
|
|
|
NS_ASSERTION(content, "null content object");
|
|
|
|
content->GetTag(*getter_AddRefs(tag));
|
1999-04-27 03:58:46 +00:00
|
|
|
|
|
|
|
// Get the child list name that the frame is contained in
|
|
|
|
nsCOMPtr<nsIAtom> listName;
|
2000-01-22 01:16:50 +00:00
|
|
|
GetChildListNameFor(aPresContext, parentFrame, aFrame, getter_AddRefs(listName));
|
1999-04-14 04:00:16 +00:00
|
|
|
|
|
|
|
// If the frame is out of the flow, then it has a placeholder frame.
|
|
|
|
nsIFrame* placeholderFrame = nsnull;
|
|
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(presShell));
|
|
|
|
if (listName) {
|
|
|
|
presShell->GetPlaceholderFrameFor(aFrame, &placeholderFrame);
|
|
|
|
}
|
1999-02-05 18:24:48 +00:00
|
|
|
|
1999-02-17 17:02:27 +00:00
|
|
|
// Get the previous sibling frame
|
|
|
|
nsIFrame* firstChild;
|
2000-01-22 01:16:50 +00:00
|
|
|
parentFrame->FirstChild(aPresContext, listName, &firstChild);
|
1999-02-17 17:02:27 +00:00
|
|
|
nsFrameList frameList(firstChild);
|
|
|
|
|
1999-02-05 18:24:48 +00:00
|
|
|
// See whether it's an IMG or an OBJECT element
|
1999-02-17 04:39:22 +00:00
|
|
|
if (nsHTMLAtoms::img == tag.get()) {
|
1999-02-06 03:45:11 +00:00
|
|
|
// It's an IMG element. Try and construct an alternate frame to use when the
|
|
|
|
// image can't be rendered
|
|
|
|
nsIFrame* newFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructAlternateImageFrame(aPresShell, aPresContext, content, styleContext,
|
1999-04-14 04:00:16 +00:00
|
|
|
parentFrame, newFrame);
|
1999-02-06 03:45:11 +00:00
|
|
|
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-08-05 03:09:22 +00:00
|
|
|
nsCOMPtr<nsIFrameManager> frameManager;
|
|
|
|
presShell->GetFrameManager(getter_AddRefs(frameManager));
|
1999-04-14 04:00:16 +00:00
|
|
|
|
1999-09-07 02:42:00 +00:00
|
|
|
// Replace the old frame with the new frame
|
1999-09-15 04:03:08 +00:00
|
|
|
// Reset the primary frame mapping
|
|
|
|
frameManager->SetPrimaryFrameFor(content, newFrame);
|
|
|
|
|
|
|
|
if (placeholderFrame) {
|
|
|
|
// Reuse the existing placeholder frame, and add an association to the
|
|
|
|
// new frame
|
|
|
|
frameManager->SetPlaceholderFrameFor(newFrame, placeholderFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Replace the old frame with the new frame
|
1999-11-24 06:03:41 +00:00
|
|
|
frameManager->ReplaceFrame(aPresContext, *presShell, parentFrame,
|
1999-09-15 04:03:08 +00:00
|
|
|
listName, aFrame, newFrame);
|
1999-09-21 05:20:11 +00:00
|
|
|
|
|
|
|
// Now that we've replaced the primary frame, if there's a placeholder
|
|
|
|
// frame then complete the transition from image frame to new frame
|
|
|
|
if (placeholderFrame) {
|
|
|
|
// Remove the association between the old frame and its placeholder
|
|
|
|
frameManager->SetPlaceholderFrameFor(aFrame, nsnull);
|
|
|
|
|
|
|
|
// Placeholder frames have a pointer back to the out-of-flow frame.
|
|
|
|
// Make sure that's correct, too.
|
|
|
|
((nsPlaceholderFrame*)placeholderFrame)->SetOutOfFlowFrame(newFrame);
|
2000-01-22 01:16:50 +00:00
|
|
|
|
|
|
|
// XXX Work around a bug in the block code where the floater won't get
|
|
|
|
// reflowed unless the line containing the placeholder frame is reflowed...
|
|
|
|
nsIFrame* placeholderParentFrame;
|
|
|
|
placeholderFrame->GetParent(&placeholderParentFrame);
|
|
|
|
placeholderParentFrame->ReflowDirtyChild(aPresShell, placeholderFrame);
|
1999-09-21 05:20:11 +00:00
|
|
|
}
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
1999-03-24 01:13:59 +00:00
|
|
|
} else if ((nsHTMLAtoms::object == tag.get()) ||
|
1999-04-07 05:16:47 +00:00
|
|
|
(nsHTMLAtoms::embed == tag.get()) ||
|
1999-03-24 01:13:59 +00:00
|
|
|
(nsHTMLAtoms::applet == tag.get())) {
|
1999-09-15 04:03:08 +00:00
|
|
|
|
|
|
|
// It's an OBJECT, EMBED, or APPLET, so we should display the contents
|
1999-09-07 02:42:00 +00:00
|
|
|
// instead
|
1999-09-15 04:03:08 +00:00
|
|
|
nsIFrame* absoluteContainingBlock;
|
|
|
|
nsIFrame* floaterContainingBlock;
|
|
|
|
nsIFrame* inFlowParent = parentFrame;
|
|
|
|
|
|
|
|
// If the OBJECT frame is out-of-flow, then get the placeholder frame's
|
|
|
|
// parent and use that when determining the absolute containing block and
|
|
|
|
// floater containing block
|
|
|
|
if (placeholderFrame) {
|
|
|
|
placeholderFrame->GetParent(&inFlowParent);
|
|
|
|
}
|
|
|
|
|
|
|
|
absoluteContainingBlock = GetAbsoluteContainingBlock(aPresContext, inFlowParent),
|
|
|
|
floaterContainingBlock = GetFloaterContainingBlock(aPresContext, inFlowParent);
|
|
|
|
|
|
|
|
#ifdef NS_DEBUG
|
|
|
|
// Verify that we calculated the same containing block
|
|
|
|
if (listName.get() == nsLayoutAtoms::absoluteList) {
|
|
|
|
NS_ASSERTION(absoluteContainingBlock == parentFrame,
|
|
|
|
"wrong absolute containing block");
|
|
|
|
} else if (listName.get() == nsLayoutAtoms::floaterList) {
|
|
|
|
NS_ASSERTION(floaterContainingBlock == parentFrame,
|
|
|
|
"wrong floater containing block");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Now initialize the frame construction state
|
1999-08-05 03:09:22 +00:00
|
|
|
nsFrameConstructorState state(aPresContext, mFixedContainingBlock,
|
1999-12-06 07:44:18 +00:00
|
|
|
absoluteContainingBlock, floaterContainingBlock, nsnull);
|
1999-04-28 19:08:14 +00:00
|
|
|
nsFrameItems frameItems;
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
1999-04-27 03:58:46 +00:00
|
|
|
styleContext->GetStyleData(eStyleStruct_Display);
|
1999-02-17 17:02:27 +00:00
|
|
|
|
1999-09-15 04:03:08 +00:00
|
|
|
// Create a new frame based on the display type.
|
|
|
|
// Note: if the old frame was out-of-flow, then so will the new frame
|
|
|
|
// and we'll get a new placeholder frame
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructFrameByDisplayType(aPresShell, aPresContext, state, display, content,
|
1999-11-01 15:24:57 +00:00
|
|
|
inFlowParent, styleContext, frameItems);
|
1999-02-17 17:02:27 +00:00
|
|
|
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-09-07 02:42:00 +00:00
|
|
|
nsIFrame* newFrame = frameItems.childList;
|
|
|
|
|
1999-09-15 04:03:08 +00:00
|
|
|
if (placeholderFrame) {
|
|
|
|
// Remove the association between the old frame and its placeholder
|
|
|
|
// Note: ConstructFrameByDisplayType() will already have added an
|
|
|
|
// association for the new placeholder frame
|
|
|
|
state.mFrameManager->SetPlaceholderFrameFor(aFrame, nsnull);
|
|
|
|
|
|
|
|
// Verify that the new frame is also a placeholder frame
|
|
|
|
NS_ASSERTION(IsPlaceholderFrame(newFrame), "unexpected frame type");
|
|
|
|
|
|
|
|
// Replace the old placeholder frame with the new placeholder frame
|
1999-11-24 06:03:41 +00:00
|
|
|
state.mFrameManager->ReplaceFrame(aPresContext, *presShell, inFlowParent,
|
1999-09-15 04:03:08 +00:00
|
|
|
nsnull, placeholderFrame, newFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Replace the primary frame
|
|
|
|
if (listName.get() == nsLayoutAtoms::absoluteList) {
|
|
|
|
newFrame = state.mAbsoluteItems.childList;
|
|
|
|
state.mAbsoluteItems.childList = nsnull;
|
|
|
|
} else if (listName.get() == nsLayoutAtoms::fixedList) {
|
|
|
|
newFrame = state.mFixedItems.childList;
|
|
|
|
state.mFixedItems.childList = nsnull;
|
|
|
|
} else if (listName.get() == nsLayoutAtoms::floaterList) {
|
|
|
|
newFrame = state.mFloatedItems.childList;
|
|
|
|
state.mFloatedItems.childList = nsnull;
|
|
|
|
}
|
1999-11-24 06:03:41 +00:00
|
|
|
state.mFrameManager->ReplaceFrame(aPresContext, *presShell, parentFrame,
|
1999-09-15 04:03:08 +00:00
|
|
|
listName, aFrame, newFrame);
|
|
|
|
|
|
|
|
// Reset the primary frame mapping. Don't assume that
|
|
|
|
// ConstructFrameByDisplayType() has done this
|
|
|
|
state.mFrameManager->SetPrimaryFrameFor(content, newFrame);
|
|
|
|
|
1999-02-17 17:02:27 +00:00
|
|
|
// If there are new absolutely positioned child frames, then notify
|
|
|
|
// the parent
|
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
1999-04-28 19:08:14 +00:00
|
|
|
if (state.mAbsoluteItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mAbsoluteItems.containingBlock->AppendFrames(aPresContext, *presShell,
|
1999-09-15 04:03:08 +00:00
|
|
|
nsLayoutAtoms::absoluteList,
|
|
|
|
state.mAbsoluteItems.childList);
|
1999-02-17 17:02:27 +00:00
|
|
|
}
|
1999-09-07 02:42:00 +00:00
|
|
|
|
1999-02-17 17:02:27 +00:00
|
|
|
// If there are new fixed positioned child frames, then notify
|
|
|
|
// the parent
|
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
1999-04-28 19:08:14 +00:00
|
|
|
if (state.mFixedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mFixedItems.containingBlock->AppendFrames(aPresContext, *presShell,
|
1999-09-15 04:03:08 +00:00
|
|
|
nsLayoutAtoms::fixedList,
|
|
|
|
state.mFixedItems.childList);
|
1999-02-17 17:02:27 +00:00
|
|
|
}
|
1999-09-07 02:42:00 +00:00
|
|
|
|
1999-09-15 04:03:08 +00:00
|
|
|
// If there are new floating child frames, then notify the parent
|
1999-02-26 17:11:54 +00:00
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
1999-04-28 19:08:14 +00:00
|
|
|
if (state.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mFloatedItems.containingBlock->AppendFrames(aPresContext,
|
1999-04-28 19:08:14 +00:00
|
|
|
*presShell,
|
1999-02-26 17:11:54 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
1999-04-28 19:08:14 +00:00
|
|
|
state.mFloatedItems.childList);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-02-17 17:02:27 +00:00
|
|
|
}
|
|
|
|
|
1999-04-20 03:57:01 +00:00
|
|
|
} else if (nsHTMLAtoms::input == tag.get()) {
|
|
|
|
// XXX image INPUT elements are also image frames, but don't throw away the
|
|
|
|
// image frame, because the frame class has extra logic that is specific to
|
|
|
|
// INPUT elements
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
} else {
|
|
|
|
NS_ASSERTION(PR_FALSE, "unexpected tag");
|
|
|
|
}
|
|
|
|
|
1999-02-06 03:45:11 +00:00
|
|
|
return rv;
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
|
1999-02-26 03:35:22 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::CreateContinuingOuterTableFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-02-26 03:35:22 +00:00
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame** aContinuingFrame)
|
|
|
|
{
|
|
|
|
nsIFrame* newFrame;
|
|
|
|
nsresult rv;
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewTableOuterFrame(aPresShell, &newFrame);
|
1999-02-26 03:35:22 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, aContent, aParentFrame, aStyleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-26 03:35:22 +00:00
|
|
|
aStyleContext, PR_FALSE);
|
|
|
|
|
|
|
|
// Create a continuing inner table frame, and if there's a caption then
|
|
|
|
// replicate the caption
|
|
|
|
nsIFrame* childFrame;
|
|
|
|
nsFrameItems newChildFrames;
|
|
|
|
|
2000-01-22 01:16:50 +00:00
|
|
|
aFrame->FirstChild(aPresContext, nsnull, &childFrame);
|
1999-02-26 03:35:22 +00:00
|
|
|
while (childFrame) {
|
|
|
|
nsIAtom* tableType;
|
|
|
|
|
|
|
|
// See if it's the inner table frame
|
|
|
|
childFrame->GetFrameType(&tableType);
|
|
|
|
if (nsLayoutAtoms::tableFrame == tableType) {
|
|
|
|
nsIFrame* continuingTableFrame;
|
|
|
|
|
|
|
|
// It's the inner table frame, so create a continuing frame
|
1999-12-04 23:49:50 +00:00
|
|
|
CreateContinuingFrame(aPresShell, aPresContext, childFrame, newFrame, &continuingTableFrame);
|
1999-02-26 03:35:22 +00:00
|
|
|
newChildFrames.AddChild(continuingTableFrame);
|
|
|
|
} else {
|
|
|
|
nsIContent* caption;
|
|
|
|
nsIStyleContext* captionStyle;
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
|
|
|
|
childFrame->GetContent(&caption);
|
|
|
|
childFrame->GetStyleContext(&captionStyle);
|
|
|
|
display = (const nsStyleDisplay*)captionStyle->GetStyleData(eStyleStruct_Display);
|
|
|
|
NS_ASSERTION(NS_STYLE_DISPLAY_TABLE_CAPTION == display->mDisplay, "expected caption");
|
|
|
|
|
|
|
|
// Replicate the caption frame
|
|
|
|
// XXX We have to do it this way instead of calling ConstructFrameByDisplayType(),
|
|
|
|
// because of a bug in the way ConstructTableFrame() handles the initial child
|
|
|
|
// list...
|
1999-04-28 19:08:14 +00:00
|
|
|
nsIFrame* captionFrame;
|
|
|
|
nsFrameItems childItems;
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewTableCaptionFrame(aPresShell, &captionFrame);
|
1999-08-05 03:09:22 +00:00
|
|
|
nsFrameConstructorState state(aPresContext, mFixedContainingBlock,
|
1999-04-28 19:08:14 +00:00
|
|
|
GetAbsoluteContainingBlock(aPresContext, newFrame),
|
1999-12-06 07:44:18 +00:00
|
|
|
captionFrame, nsnull);
|
1999-11-24 06:03:41 +00:00
|
|
|
captionFrame->Init(aPresContext, caption, newFrame, captionStyle, nsnull);
|
1999-12-04 23:49:50 +00:00
|
|
|
ProcessChildren(aPresShell, aPresContext, state, caption, captionFrame,
|
1999-08-27 21:46:10 +00:00
|
|
|
PR_TRUE, childItems, PR_TRUE);
|
1999-11-24 06:03:41 +00:00
|
|
|
captionFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-03-03 16:33:57 +00:00
|
|
|
// XXX Deal with absolute and fixed frames...
|
1999-04-28 19:08:14 +00:00
|
|
|
if (state.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
captionFrame->SetInitialChildList(aPresContext,
|
1999-02-26 17:11:54 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
1999-04-28 19:08:14 +00:00
|
|
|
state.mFloatedItems.childList);
|
1999-02-26 17:11:54 +00:00
|
|
|
}
|
1999-02-26 03:35:22 +00:00
|
|
|
newChildFrames.AddChild(captionFrame);
|
|
|
|
NS_RELEASE(caption);
|
|
|
|
NS_RELEASE(captionStyle);
|
|
|
|
}
|
|
|
|
NS_IF_RELEASE(tableType);
|
|
|
|
childFrame->GetNextSibling(&childFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the outer table's initial child list
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, newChildFrames.childList);
|
1999-02-26 03:35:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
*aContinuingFrame = newFrame;
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-03-03 16:33:57 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::CreateContinuingTableFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-03-03 16:33:57 +00:00
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame** aContinuingFrame)
|
|
|
|
{
|
|
|
|
nsIFrame* newFrame;
|
|
|
|
nsresult rv;
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewTableFrame(aPresShell, &newFrame);
|
1999-03-03 16:33:57 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, aContent, aParentFrame, aStyleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-03-03 16:33:57 +00:00
|
|
|
aStyleContext, PR_FALSE);
|
|
|
|
|
|
|
|
// Replicate any header/footer frames
|
|
|
|
nsIFrame* rowGroupFrame;
|
|
|
|
nsFrameItems childFrames;
|
|
|
|
|
2000-01-22 01:16:50 +00:00
|
|
|
aFrame->FirstChild(aPresContext, nsnull, &rowGroupFrame);
|
1999-03-03 16:33:57 +00:00
|
|
|
while (rowGroupFrame) {
|
|
|
|
// See if it's a header/footer
|
|
|
|
nsIStyleContext* rowGroupStyle;
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
|
|
|
|
rowGroupFrame->GetStyleContext(&rowGroupStyle);
|
|
|
|
display = (const nsStyleDisplay*)rowGroupStyle->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
if ((NS_STYLE_DISPLAY_TABLE_HEADER_GROUP == display->mDisplay) ||
|
|
|
|
(NS_STYLE_DISPLAY_TABLE_FOOTER_GROUP == display->mDisplay)) {
|
|
|
|
|
|
|
|
// Replicate the header/footer frame
|
1999-04-28 19:08:14 +00:00
|
|
|
nsIFrame* headerFooterFrame;
|
|
|
|
nsFrameItems childItems;
|
|
|
|
nsIContent* headerFooter;
|
1999-08-05 03:09:22 +00:00
|
|
|
nsFrameConstructorState state(aPresContext, mFixedContainingBlock,
|
1999-04-28 19:08:14 +00:00
|
|
|
GetAbsoluteContainingBlock(aPresContext, newFrame),
|
1999-12-06 07:44:18 +00:00
|
|
|
nsnull, nsnull);
|
1999-03-03 16:33:57 +00:00
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewTableRowGroupFrame(aPresShell, &headerFooterFrame);
|
1999-03-03 16:33:57 +00:00
|
|
|
rowGroupFrame->GetContent(&headerFooter);
|
1999-11-24 06:03:41 +00:00
|
|
|
headerFooterFrame->Init(aPresContext, headerFooter, newFrame,
|
1999-03-03 16:33:57 +00:00
|
|
|
rowGroupStyle, nsnull);
|
1999-12-04 23:49:50 +00:00
|
|
|
ProcessChildren(aPresShell, aPresContext, state, headerFooter, headerFooterFrame,
|
1999-08-27 21:46:10 +00:00
|
|
|
PR_FALSE, childItems, PR_FALSE);
|
1999-04-28 19:08:14 +00:00
|
|
|
NS_ASSERTION(!state.mFloatedItems.childList, "unexpected floated element");
|
1999-03-03 16:33:57 +00:00
|
|
|
NS_RELEASE(headerFooter);
|
1999-11-24 06:03:41 +00:00
|
|
|
headerFooterFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-03-03 16:33:57 +00:00
|
|
|
|
|
|
|
// Table specific initialization
|
|
|
|
((nsTableRowGroupFrame*)headerFooterFrame)->InitRepeatedFrame
|
2000-01-22 01:16:50 +00:00
|
|
|
(aPresContext, (nsTableRowGroupFrame*)rowGroupFrame);
|
1999-03-03 16:33:57 +00:00
|
|
|
|
|
|
|
// XXX Deal with absolute and fixed frames...
|
|
|
|
childFrames.AddChild(headerFooterFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_RELEASE(rowGroupStyle);
|
1999-03-05 06:00:40 +00:00
|
|
|
|
|
|
|
// Header and footer must be first, and then the body row groups.
|
|
|
|
// So if we found a body row group, then stop looking for header and
|
|
|
|
// footer elements
|
|
|
|
if (NS_STYLE_DISPLAY_TABLE_ROW_GROUP == display->mDisplay) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the next row group frame
|
1999-03-03 16:33:57 +00:00
|
|
|
rowGroupFrame->GetNextSibling(&rowGroupFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the table frame's initial child list
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, childFrames.childList);
|
1999-03-03 16:33:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
*aContinuingFrame = newFrame;
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-02-24 04:03:50 +00:00
|
|
|
NS_IMETHODIMP
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::CreateContinuingFrame(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-02-24 04:03:50 +00:00
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIFrame** aContinuingFrame)
|
|
|
|
{
|
1999-02-25 05:31:15 +00:00
|
|
|
nsIAtom* frameType;
|
|
|
|
nsIContent* content;
|
|
|
|
nsIStyleContext* styleContext;
|
|
|
|
nsIFrame* newFrame = nsnull;
|
|
|
|
nsresult rv;
|
1999-02-24 17:16:04 +00:00
|
|
|
|
1999-02-25 05:31:15 +00:00
|
|
|
// Use the frame type to determine what type of frame to create
|
1999-02-24 17:16:04 +00:00
|
|
|
aFrame->GetFrameType(&frameType);
|
1999-02-25 05:31:15 +00:00
|
|
|
aFrame->GetContent(&content);
|
|
|
|
aFrame->GetStyleContext(&styleContext);
|
1999-02-24 17:16:04 +00:00
|
|
|
|
|
|
|
if (nsLayoutAtoms::textFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewContinuingTextFrame(aPresShell, &newFrame);
|
1999-02-25 05:31:15 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, content, aParentFrame, styleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-25 05:31:15 +00:00
|
|
|
styleContext, PR_FALSE);
|
|
|
|
}
|
1999-02-24 17:16:04 +00:00
|
|
|
|
1999-04-20 18:23:13 +00:00
|
|
|
} else if (nsLayoutAtoms::inlineFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewInlineFrame(aPresShell, &newFrame);
|
1999-02-25 05:31:15 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, content, aParentFrame, styleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-25 05:31:15 +00:00
|
|
|
styleContext, PR_FALSE);
|
|
|
|
}
|
1999-02-24 17:16:04 +00:00
|
|
|
|
1999-04-20 18:23:13 +00:00
|
|
|
} else if (nsLayoutAtoms::blockFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewBlockFrame(aPresShell, &newFrame);
|
1999-02-25 05:31:15 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, content, aParentFrame, styleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-25 05:31:15 +00:00
|
|
|
styleContext, PR_FALSE);
|
|
|
|
}
|
1999-02-24 17:16:04 +00:00
|
|
|
|
|
|
|
} else if (nsLayoutAtoms::areaFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewAreaFrame(aPresShell, &newFrame);
|
1999-02-25 05:31:15 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, content, aParentFrame, styleContext,
|
1999-05-11 22:03:29 +00:00
|
|
|
aFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-25 05:31:15 +00:00
|
|
|
styleContext, PR_FALSE);
|
|
|
|
}
|
1999-02-24 17:16:04 +00:00
|
|
|
|
1999-04-20 20:00:52 +00:00
|
|
|
} else if (nsLayoutAtoms::positionedInlineFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewPositionedInlineFrame(aPresShell, &newFrame);
|
1999-04-20 20:00:52 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, content, aParentFrame, styleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-04-20 20:00:52 +00:00
|
|
|
styleContext, PR_FALSE);
|
|
|
|
}
|
|
|
|
|
1999-02-24 17:16:04 +00:00
|
|
|
} else if (nsLayoutAtoms::pageFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewPageFrame(aPresShell, &newFrame);
|
1999-02-25 05:31:15 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, content, aParentFrame, styleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-25 05:31:15 +00:00
|
|
|
styleContext, PR_TRUE);
|
|
|
|
}
|
1999-02-24 17:16:04 +00:00
|
|
|
|
|
|
|
} else if (nsLayoutAtoms::tableOuterFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = CreateContinuingOuterTableFrame(aPresShell, aPresContext, aFrame, aParentFrame,
|
1999-02-26 03:35:22 +00:00
|
|
|
content, styleContext, &newFrame);
|
1999-02-24 17:16:04 +00:00
|
|
|
|
|
|
|
} else if (nsLayoutAtoms::tableFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = CreateContinuingTableFrame(aPresShell, aPresContext, aFrame, aParentFrame,
|
1999-03-03 16:33:57 +00:00
|
|
|
content, styleContext, &newFrame);
|
1999-02-24 17:16:04 +00:00
|
|
|
|
|
|
|
} else if (nsLayoutAtoms::tableRowGroupFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewTableRowGroupFrame(aPresShell, &newFrame);
|
1999-02-25 05:31:15 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, content, aParentFrame, styleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-25 05:31:15 +00:00
|
|
|
styleContext, PR_FALSE);
|
|
|
|
}
|
1999-02-24 17:16:04 +00:00
|
|
|
|
|
|
|
} else if (nsLayoutAtoms::tableRowFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewTableRowFrame(aPresShell, &newFrame);
|
1999-02-25 05:31:15 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, content, aParentFrame, styleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-25 05:31:15 +00:00
|
|
|
styleContext, PR_FALSE);
|
1999-02-24 17:16:04 +00:00
|
|
|
|
1999-02-25 05:31:15 +00:00
|
|
|
// Create a continuing frame for each table cell frame
|
|
|
|
nsIFrame* cellFrame;
|
|
|
|
nsFrameItems newChildList;
|
1999-02-24 17:16:04 +00:00
|
|
|
|
2000-01-22 01:16:50 +00:00
|
|
|
aFrame->FirstChild(aPresContext, nsnull, &cellFrame);
|
1999-02-25 05:31:15 +00:00
|
|
|
while (cellFrame) {
|
|
|
|
nsIAtom* tableType;
|
|
|
|
|
|
|
|
// See if it's a table cell frame
|
|
|
|
cellFrame->GetFrameType(&tableType);
|
|
|
|
if (nsLayoutAtoms::tableCellFrame == tableType) {
|
|
|
|
nsIFrame* continuingCellFrame;
|
1999-02-24 17:16:04 +00:00
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
CreateContinuingFrame(aPresShell, aPresContext, cellFrame, newFrame, &continuingCellFrame);
|
1999-02-25 05:31:15 +00:00
|
|
|
newChildList.AddChild(continuingCellFrame);
|
|
|
|
}
|
1999-02-24 17:16:04 +00:00
|
|
|
|
1999-02-25 05:31:15 +00:00
|
|
|
NS_IF_RELEASE(tableType);
|
|
|
|
cellFrame->GetNextSibling(&cellFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the table cell's initial child list
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, newChildList.childList);
|
1999-02-25 05:31:15 +00:00
|
|
|
}
|
1999-02-24 17:16:04 +00:00
|
|
|
|
1999-02-25 05:31:15 +00:00
|
|
|
} else if (nsLayoutAtoms::tableCellFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewTableCellFrame(aPresShell, &newFrame);
|
1999-02-25 05:31:15 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, content, aParentFrame, styleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-02-25 05:31:15 +00:00
|
|
|
styleContext, PR_FALSE);
|
1999-02-24 17:16:04 +00:00
|
|
|
|
1999-02-25 05:31:15 +00:00
|
|
|
// Create a continuing area frame
|
|
|
|
nsIFrame* areaFrame;
|
|
|
|
nsIFrame* continuingAreaFrame;
|
2000-01-22 01:16:50 +00:00
|
|
|
aFrame->FirstChild(aPresContext, nsnull, &areaFrame);
|
1999-12-04 23:49:50 +00:00
|
|
|
CreateContinuingFrame(aPresShell, aPresContext, areaFrame, newFrame, &continuingAreaFrame);
|
1999-02-24 17:16:04 +00:00
|
|
|
|
1999-02-25 05:31:15 +00:00
|
|
|
// Set the table cell's initial child list
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->SetInitialChildList(aPresContext, nsnull, continuingAreaFrame);
|
1999-02-24 17:16:04 +00:00
|
|
|
}
|
1999-04-27 22:14:54 +00:00
|
|
|
|
|
|
|
} else if (nsLayoutAtoms::lineFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewFirstLineFrame(aPresShell, &newFrame);
|
1999-04-27 22:14:54 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, content, aParentFrame, styleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-04-27 22:14:54 +00:00
|
|
|
styleContext, PR_FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
} else if (nsLayoutAtoms::letterFrame == frameType) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewFirstLetterFrame(aPresShell, &newFrame);
|
1999-04-27 22:14:54 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-11-24 06:03:41 +00:00
|
|
|
newFrame->Init(aPresContext, content, aParentFrame, styleContext, aFrame);
|
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, newFrame,
|
1999-04-27 22:14:54 +00:00
|
|
|
styleContext, PR_FALSE);
|
|
|
|
}
|
1999-02-25 05:31:15 +00:00
|
|
|
|
|
|
|
} else {
|
|
|
|
NS_ASSERTION(PR_FALSE, "unexpected frame type");
|
|
|
|
rv = NS_ERROR_UNEXPECTED;
|
1999-02-24 17:16:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
*aContinuingFrame = newFrame;
|
1999-02-25 05:31:15 +00:00
|
|
|
NS_RELEASE(styleContext);
|
|
|
|
NS_IF_RELEASE(content);
|
|
|
|
NS_IF_RELEASE(frameType);
|
1999-02-24 17:16:04 +00:00
|
|
|
return rv;
|
1999-02-24 04:03:50 +00:00
|
|
|
}
|
|
|
|
|
1999-07-02 04:47:05 +00:00
|
|
|
// Helper function that searches the immediate child frames for a frame that
|
1999-07-06 03:52:33 +00:00
|
|
|
// maps the specified content object
|
1999-07-02 04:47:05 +00:00
|
|
|
static nsIFrame*
|
2000-01-22 01:16:50 +00:00
|
|
|
FindFrameWithContent(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIContent* aParentContent,
|
|
|
|
nsIContent* aContent)
|
1999-07-02 04:47:05 +00:00
|
|
|
{
|
1999-07-06 03:52:33 +00:00
|
|
|
NS_PRECONDITION(aParentFrame, "No frame to search!");
|
|
|
|
if (!aParentFrame) {
|
|
|
|
return nsnull;
|
1999-07-02 04:47:05 +00:00
|
|
|
}
|
|
|
|
|
1999-07-06 03:52:33 +00:00
|
|
|
keepLooking:
|
|
|
|
// Search for the frame in each child list that aParentFrame supports
|
|
|
|
nsIAtom* listName = nsnull;
|
|
|
|
PRInt32 listIndex = 0;
|
|
|
|
do {
|
|
|
|
nsIFrame* kidFrame;
|
2000-01-22 01:16:50 +00:00
|
|
|
aParentFrame->FirstChild(aPresContext, listName, &kidFrame);
|
1999-07-06 03:52:33 +00:00
|
|
|
while (kidFrame) {
|
|
|
|
nsCOMPtr<nsIContent> kidContent;
|
|
|
|
|
|
|
|
// See if the child frame points to the content object we're
|
|
|
|
// looking for
|
|
|
|
kidFrame->GetContent(getter_AddRefs(kidContent));
|
|
|
|
if (kidContent.get() == aContent) {
|
|
|
|
nsCOMPtr<nsIAtom> frameType;
|
|
|
|
|
|
|
|
// We found a match. See if it's a placeholder frame
|
|
|
|
kidFrame->GetFrameType(getter_AddRefs(frameType));
|
|
|
|
if (nsLayoutAtoms::placeholderFrame == frameType.get()) {
|
|
|
|
// Ignore the placeholder and return the out-of-flow frame instead
|
|
|
|
return ((nsPlaceholderFrame*)kidFrame)->GetOutOfFlowFrame();
|
|
|
|
} else {
|
|
|
|
// Return the matching child frame
|
|
|
|
return kidFrame;
|
|
|
|
}
|
|
|
|
}
|
1999-07-02 04:47:05 +00:00
|
|
|
|
1999-07-06 03:52:33 +00:00
|
|
|
// We search the immediate children only, but if the child frame has
|
|
|
|
// the same content pointer as its parent then we need to search its
|
|
|
|
// child frames, too
|
|
|
|
if (kidContent.get() == aParentContent) {
|
2000-01-22 01:16:50 +00:00
|
|
|
nsIFrame* matchingFrame = FindFrameWithContent(aPresContext, kidFrame, aParentContent,
|
1999-07-06 03:52:33 +00:00
|
|
|
aContent);
|
1999-07-02 04:47:05 +00:00
|
|
|
|
1999-07-06 03:52:33 +00:00
|
|
|
if (matchingFrame) {
|
|
|
|
return matchingFrame;
|
|
|
|
}
|
1999-07-02 04:47:05 +00:00
|
|
|
}
|
1999-07-06 03:52:33 +00:00
|
|
|
|
|
|
|
// Get the next sibling frame
|
|
|
|
kidFrame->GetNextSibling(&kidFrame);
|
1999-07-02 04:47:05 +00:00
|
|
|
}
|
|
|
|
|
1999-07-06 03:52:33 +00:00
|
|
|
NS_IF_RELEASE(listName);
|
|
|
|
aParentFrame->GetAdditionalChildListName(listIndex++, &listName);
|
|
|
|
} while(listName);
|
|
|
|
|
|
|
|
// We didn't find a matching frame. If aFrame has a next-in-flow,
|
|
|
|
// then continue looking there
|
|
|
|
aParentFrame->GetNextInFlow(&aParentFrame);
|
|
|
|
if (aParentFrame) {
|
|
|
|
goto keepLooking;
|
1999-07-02 04:47:05 +00:00
|
|
|
}
|
|
|
|
|
1999-07-06 03:52:33 +00:00
|
|
|
// No matching frame
|
1999-07-02 04:47:05 +00:00
|
|
|
return nsnull;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Request to find the primary frame associated with a given content object.
|
|
|
|
// This is typically called by the pres shell when there is no mapping in
|
|
|
|
// the pres shell hash table
|
|
|
|
NS_IMETHODIMP
|
1999-08-05 03:09:22 +00:00
|
|
|
nsCSSFrameConstructor::FindPrimaryFrameFor(nsIPresContext* aPresContext,
|
|
|
|
nsIFrameManager* aFrameManager,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame** aFrame)
|
1999-07-02 04:47:05 +00:00
|
|
|
{
|
1999-07-06 03:52:33 +00:00
|
|
|
*aFrame = nsnull; // initialize OUT parameter
|
1999-07-02 04:47:05 +00:00
|
|
|
|
1999-07-06 03:52:33 +00:00
|
|
|
// Get the pres shell
|
|
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(presShell));
|
1999-07-02 04:47:05 +00:00
|
|
|
|
1999-07-06 03:52:33 +00:00
|
|
|
// We want to be able to quickly map from a content object to its frame,
|
|
|
|
// but we also want to keep the hash table small. Therefore, many frames
|
|
|
|
// are not added to the hash table when they're first created:
|
|
|
|
// - text frames
|
|
|
|
// - inline frames (often things like FONT and B)
|
|
|
|
// - BR frames
|
|
|
|
// - internal table frames (row-group, row, cell, col-group, col)
|
|
|
|
//
|
|
|
|
// That means we need to need to search for the frame
|
2000-03-10 01:10:44 +00:00
|
|
|
nsCOMPtr<nsIContent> parentContent; // we get this one time
|
|
|
|
nsIFrame* parentFrame; // this pointer is used to iterate across all frames that map to parentContent
|
1999-07-06 03:52:33 +00:00
|
|
|
|
|
|
|
// Get the frame that corresponds to the parent content object.
|
|
|
|
// Note that this may recurse indirectly, because the pres shell will
|
|
|
|
// call us back if there is no mapping in the hash table
|
|
|
|
aContent->GetParent(*getter_AddRefs(parentContent));
|
|
|
|
if (parentContent.get()) {
|
1999-08-05 03:09:22 +00:00
|
|
|
aFrameManager->GetPrimaryFrameFor(parentContent, &parentFrame);
|
2000-03-10 01:10:44 +00:00
|
|
|
while (parentFrame) {
|
1999-07-06 03:52:33 +00:00
|
|
|
// Search the child frames for a match
|
2000-01-22 01:16:50 +00:00
|
|
|
*aFrame = FindFrameWithContent(aPresContext, parentFrame, parentContent.get(), aContent);
|
1999-07-02 04:47:05 +00:00
|
|
|
|
1999-07-06 03:52:33 +00:00
|
|
|
// If we found a match, then add a mapping to the hash table so
|
|
|
|
// next time this will be quick
|
|
|
|
if (*aFrame) {
|
1999-08-05 03:09:22 +00:00
|
|
|
aFrameManager->SetPrimaryFrameFor(aContent, *aFrame);
|
2000-03-10 01:10:44 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// We need to check parentFrame's sibling frame as well
|
|
|
|
parentFrame->GetNextSibling(&parentFrame);
|
|
|
|
if (!parentFrame) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
nsCOMPtr<nsIContent>nextParentContent;
|
|
|
|
parentFrame->GetContent(getter_AddRefs(nextParentContent));
|
|
|
|
if (parentContent!=nextParentContent) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
1999-07-06 03:52:33 +00:00
|
|
|
}
|
|
|
|
}
|
1999-07-02 04:47:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
// Capture state for the frame tree rooted at the frame associated with the
|
|
|
|
// content object, aContent
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::CaptureStateForFramesOf(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsILayoutHistoryState* aHistoryState)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
rv = aPresContext->GetShell(getter_AddRefs(presShell));
|
|
|
|
if (NS_SUCCEEDED(rv) && presShell) {
|
|
|
|
nsIFrame* frame;
|
|
|
|
rv = presShell->GetPrimaryFrameFor(aContent, &frame);
|
|
|
|
if (NS_SUCCEEDED(rv) && frame) {
|
|
|
|
CaptureStateFor(aPresContext, frame, aHistoryState);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Capture state for the frame tree rooted at aFrame.
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::CaptureStateFor(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsILayoutHistoryState* aHistoryState)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
if (aFrame && aPresContext && aHistoryState) {
|
|
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
|
|
rv = aPresContext->GetShell(getter_AddRefs(presShell));
|
|
|
|
if (NS_SUCCEEDED(rv) && presShell) {
|
|
|
|
nsCOMPtr<nsIFrameManager> frameManager;
|
|
|
|
rv = presShell->GetFrameManager(getter_AddRefs(frameManager));
|
|
|
|
if (NS_SUCCEEDED(rv) && frameManager) {
|
1999-12-06 09:03:16 +00:00
|
|
|
rv = frameManager->CaptureFrameState(aPresContext, aFrame, aHistoryState);
|
1999-12-06 07:44:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-02-05 03:55:18 +00:00
|
|
|
nsresult
|
1999-02-27 07:15:09 +00:00
|
|
|
nsCSSFrameConstructor::RecreateFramesForContent(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent)
|
1999-02-05 03:55:18 +00:00
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
1999-12-06 07:44:18 +00:00
|
|
|
nsIContent* container;
|
1999-03-25 19:27:11 +00:00
|
|
|
rv = aContent->GetParent(container);
|
1999-12-06 07:44:18 +00:00
|
|
|
if (NS_SUCCEEDED(rv) && container) {
|
|
|
|
PRInt32 indexInContainer;
|
1999-03-25 19:27:11 +00:00
|
|
|
rv = container->IndexOf(aContent, indexInContainer);
|
1999-10-12 00:32:13 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-12-06 07:44:18 +00:00
|
|
|
// Before removing the frames associated with the content object, ask them to save their
|
|
|
|
// state onto a temporary state object.
|
|
|
|
CaptureStateForFramesOf(aPresContext, aContent, mTempFrameTreeState);
|
|
|
|
|
|
|
|
// Remove the frames associated with the content object on which the
|
1999-03-25 19:27:11 +00:00
|
|
|
// attribute change occurred.
|
|
|
|
rv = ContentRemoved(aPresContext, container, aContent, indexInContainer);
|
|
|
|
|
1999-10-12 00:32:13 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
1999-03-25 19:27:11 +00:00
|
|
|
// Now, recreate the frames associated with this content object.
|
1999-12-06 07:44:18 +00:00
|
|
|
rv = ContentInserted(aPresContext, container, aContent, indexInContainer, mTempFrameTreeState);
|
|
|
|
}
|
1999-03-25 19:27:11 +00:00
|
|
|
}
|
1999-12-06 07:44:18 +00:00
|
|
|
NS_RELEASE(container);
|
1999-02-05 03:55:18 +00:00
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-04-27 22:14:54 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// Block frame construction code
|
|
|
|
|
|
|
|
nsIStyleContext*
|
|
|
|
nsCSSFrameConstructor::GetFirstLetterStyle(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIStyleContext* aStyleContext)
|
|
|
|
{
|
|
|
|
nsIStyleContext* fls = nsnull;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent,
|
|
|
|
nsHTMLAtoms::firstLetterPseudo,
|
|
|
|
aStyleContext, PR_FALSE, &fls);
|
|
|
|
return fls;
|
|
|
|
}
|
|
|
|
|
1999-08-31 03:09:40 +00:00
|
|
|
nsIStyleContext*
|
|
|
|
nsCSSFrameConstructor::GetFirstLineStyle(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIStyleContext* aStyleContext)
|
|
|
|
{
|
|
|
|
nsIStyleContext* fls = nsnull;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent,
|
|
|
|
nsHTMLAtoms::firstLinePseudo,
|
|
|
|
aStyleContext, PR_FALSE, &fls);
|
|
|
|
return fls;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Predicate to see if a given content (block element) has
|
|
|
|
// first-letter style applied to it.
|
|
|
|
PRBool
|
|
|
|
nsCSSFrameConstructor::HaveFirstLetterStyle(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIStyleContext* aStyleContext)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIStyleContext> fls;
|
|
|
|
aPresContext->ProbePseudoStyleContextFor(aContent,
|
|
|
|
nsHTMLAtoms::firstLetterPseudo,
|
|
|
|
aStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(fls));
|
|
|
|
PRBool result = PR_FALSE;
|
|
|
|
if (fls) {
|
|
|
|
result = PR_TRUE;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsCSSFrameConstructor::HaveFirstLineStyle(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIStyleContext* aStyleContext)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIStyleContext> fls;
|
|
|
|
aPresContext->ProbePseudoStyleContextFor(aContent,
|
|
|
|
nsHTMLAtoms::firstLinePseudo,
|
|
|
|
aStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(fls));
|
|
|
|
PRBool result = PR_FALSE;
|
|
|
|
if (fls) {
|
|
|
|
result = PR_TRUE;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsCSSFrameConstructor::HaveSpecialBlockStyle(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
PRBool* aHaveFirstLetterStyle,
|
|
|
|
PRBool* aHaveFirstLineStyle)
|
|
|
|
{
|
|
|
|
*aHaveFirstLetterStyle =
|
|
|
|
HaveFirstLetterStyle(aPresContext, aContent, aStyleContext);
|
|
|
|
*aHaveFirstLineStyle =
|
|
|
|
HaveFirstLineStyle(aPresContext, aContent, aStyleContext);
|
|
|
|
}
|
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
/**
|
|
|
|
* Request to process the child content elements and create frames.
|
|
|
|
*
|
|
|
|
* @param aContent the content object whose child elements to process
|
|
|
|
* @param aFrame the the associated with aContent. This will be the
|
|
|
|
* parent frame (both content and geometric) for the flowed
|
|
|
|
* child frames
|
|
|
|
*/
|
1999-04-27 22:14:54 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ProcessChildren(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-08-27 21:46:10 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
PRBool aCanHaveGeneratedContent,
|
|
|
|
nsFrameItems& aFrameItems,
|
|
|
|
PRBool aParentIsBlock)
|
1999-04-27 22:14:54 +00:00
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
1999-08-27 21:46:10 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
1999-04-27 22:14:54 +00:00
|
|
|
|
|
|
|
if (aCanHaveGeneratedContent) {
|
|
|
|
// Probe for generated content before
|
|
|
|
nsIFrame* generatedFrame;
|
1999-08-27 21:46:10 +00:00
|
|
|
aFrame->GetStyleContext(getter_AddRefs(styleContext));
|
1999-12-04 23:49:50 +00:00
|
|
|
if (CreateGeneratedContentFrame(aPresShell, aPresContext, aState, aFrame, aContent,
|
1999-04-27 22:14:54 +00:00
|
|
|
styleContext, nsCSSAtoms::beforePseudo,
|
1999-08-27 21:46:10 +00:00
|
|
|
aParentIsBlock, &generatedFrame)) {
|
1999-04-27 22:14:54 +00:00
|
|
|
// Add the generated frame to the child list
|
|
|
|
aFrameItems.AddChild(generatedFrame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate the child content objects and construct frames
|
|
|
|
PRInt32 count;
|
|
|
|
aContent->ChildCount(count);
|
|
|
|
for (PRInt32 i = 0; i < count; i++) {
|
|
|
|
nsCOMPtr<nsIContent> childContent;
|
|
|
|
if (NS_SUCCEEDED(aContent->ChildAt(i, *getter_AddRefs(childContent)))) {
|
|
|
|
// Construct a child frame
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructFrame(aPresShell, aPresContext, aState, childContent, aFrame, aFrameItems);
|
1999-04-27 22:14:54 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aCanHaveGeneratedContent) {
|
|
|
|
// Probe for generated content after
|
|
|
|
nsIFrame* generatedFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
if (CreateGeneratedContentFrame(aPresShell, aPresContext, aState, aFrame, aContent,
|
1999-04-27 22:14:54 +00:00
|
|
|
styleContext, nsCSSAtoms::afterPseudo,
|
1999-08-27 21:46:10 +00:00
|
|
|
aParentIsBlock, &generatedFrame)) {
|
1999-04-27 22:14:54 +00:00
|
|
|
// Add the generated frame to the child list
|
|
|
|
aFrameItems.AddChild(generatedFrame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-11-01 15:24:57 +00:00
|
|
|
if (aParentIsBlock) {
|
|
|
|
if (aState.mFirstLetterStyle) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = WrapFramesInFirstLetterFrame(aPresShell, aPresContext, aState, aContent, aFrame, aFrameItems);
|
1999-11-01 15:24:57 +00:00
|
|
|
}
|
|
|
|
if (aState.mFirstLineStyle) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = WrapFramesInFirstLineFrame(aPresShell, aPresContext, aState, aContent, aFrame, aFrameItems);
|
1999-11-01 15:24:57 +00:00
|
|
|
}
|
1999-08-31 03:09:40 +00:00
|
|
|
}
|
|
|
|
|
1999-04-27 22:14:54 +00:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-08-31 03:09:40 +00:00
|
|
|
//----------------------------------------------------------------------
|
|
|
|
|
|
|
|
// Support for :first-line style
|
|
|
|
|
|
|
|
// XXX this predicate and its cousins need to migrated to a single
|
|
|
|
// place in layout - something in nsStyleDisplay maybe?
|
|
|
|
static PRBool
|
|
|
|
IsInlineFrame(nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
aFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct*&) display);
|
|
|
|
switch (display->mDisplay) {
|
|
|
|
case NS_STYLE_DISPLAY_INLINE:
|
|
|
|
case NS_STYLE_DISPLAY_INLINE_BLOCK:
|
|
|
|
case NS_STYLE_DISPLAY_INLINE_TABLE:
|
|
|
|
return PR_TRUE;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
ReparentFrame(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aNewParentFrame,
|
|
|
|
nsIStyleContext* aParentStyleContext,
|
|
|
|
nsIFrame* aFrame)
|
|
|
|
{
|
1999-09-03 23:37:37 +00:00
|
|
|
aPresContext->ReParentStyleContext(aFrame, aParentStyleContext);
|
1999-08-31 03:09:40 +00:00
|
|
|
aFrame->SetParent(aNewParentFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Special routine to handle placing a list of frames into a block
|
|
|
|
// frame that has first-line style. The routine ensures that the first
|
|
|
|
// collection of inline frames end up in a first-line frame.
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::WrapFramesInFirstLineFrame(
|
1999-12-04 23:49:50 +00:00
|
|
|
nsIPresShell* aPresShell,
|
1999-08-31 03:09:40 +00:00
|
|
|
nsIPresContext* aPresContext,
|
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsFrameItems& aFrameItems)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
// Find the first and last inline frame in aFrameItems
|
|
|
|
nsIFrame* kid = aFrameItems.childList;
|
|
|
|
nsIFrame* firstInlineFrame = nsnull;
|
|
|
|
nsIFrame* lastInlineFrame = nsnull;
|
|
|
|
while (kid) {
|
|
|
|
if (IsInlineFrame(kid)) {
|
|
|
|
if (!firstInlineFrame) firstInlineFrame = kid;
|
|
|
|
lastInlineFrame = kid;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
kid->GetNextSibling(&kid);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we don't find any inline frames, then there is nothing to do
|
|
|
|
if (!firstInlineFrame) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create line frame
|
|
|
|
nsCOMPtr<nsIStyleContext> parentStyle;
|
|
|
|
aFrame->GetStyleContext(getter_AddRefs(parentStyle));
|
|
|
|
nsCOMPtr<nsIStyleContext> firstLineStyle(
|
|
|
|
getter_AddRefs(GetFirstLineStyle(aPresContext, aContent, parentStyle))
|
|
|
|
);
|
|
|
|
nsIFrame* lineFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = NS_NewFirstLineFrame(aPresShell, &lineFrame);
|
1999-08-31 03:09:40 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
// Initialize the line frame
|
1999-12-06 07:44:18 +00:00
|
|
|
rv = InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aFrame, firstLineStyle, nsnull, lineFrame);
|
1999-08-31 03:09:40 +00:00
|
|
|
|
|
|
|
// Mangle the list of frames we are giving to the block: first
|
|
|
|
// chop the list in two after lastInlineFrame
|
|
|
|
nsIFrame* secondBlockFrame;
|
|
|
|
lastInlineFrame->GetNextSibling(&secondBlockFrame);
|
|
|
|
lastInlineFrame->SetNextSibling(nsnull);
|
|
|
|
|
|
|
|
// The lineFrame will be the block's first child; the rest of the
|
|
|
|
// frame list (after lastInlineFrame) will be the second and
|
|
|
|
// subsequent children; join the list together and reset
|
|
|
|
// aFrameItems appropriately.
|
|
|
|
if (secondBlockFrame) {
|
|
|
|
lineFrame->SetNextSibling(secondBlockFrame);
|
|
|
|
}
|
|
|
|
if (aFrameItems.childList == lastInlineFrame) {
|
|
|
|
// Just in case the block had exactly one inline child
|
|
|
|
aFrameItems.lastChild = lineFrame;
|
|
|
|
}
|
|
|
|
aFrameItems.childList = lineFrame;
|
|
|
|
|
|
|
|
// Give the inline frames to the lineFrame <b>after</b> reparenting them
|
|
|
|
kid = firstInlineFrame;
|
|
|
|
while (kid) {
|
|
|
|
ReparentFrame(aPresContext, lineFrame, firstLineStyle, kid);
|
|
|
|
kid->GetNextSibling(&kid);
|
|
|
|
}
|
1999-11-24 06:03:41 +00:00
|
|
|
lineFrame->SetInitialChildList(aPresContext, nsnull, firstInlineFrame);
|
1999-08-31 03:09:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Special routine to handle appending a new frame to a block frame's
|
|
|
|
// child list. Takes care of placing the new frame into the right
|
|
|
|
// place when first-line style is present.
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::AppendFirstLineFrames(
|
1999-12-04 23:49:50 +00:00
|
|
|
nsIPresShell* aPresShell,
|
1999-08-31 03:09:40 +00:00
|
|
|
nsIPresContext* aPresContext,
|
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aBlockFrame,
|
|
|
|
nsFrameItems& aFrameItems)
|
|
|
|
{
|
|
|
|
// It's possible that aBlockFrame needs to have a first-line frame
|
|
|
|
// created because it doesn't currently have any children.
|
|
|
|
nsIFrame* blockKid;
|
2000-01-22 01:16:50 +00:00
|
|
|
aBlockFrame->FirstChild(aPresContext, nsnull, &blockKid);
|
1999-08-31 03:09:40 +00:00
|
|
|
if (!blockKid) {
|
1999-12-04 23:49:50 +00:00
|
|
|
return WrapFramesInFirstLineFrame(aPresShell, aPresContext, aState, aContent,
|
1999-08-31 03:09:40 +00:00
|
|
|
aBlockFrame, aFrameItems);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Examine the last block child - if it's a first-line frame then
|
|
|
|
// appended frames need special treatment.
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
nsFrameList blockFrames(blockKid);
|
|
|
|
nsIFrame* lastBlockKid = blockFrames.LastChild();
|
|
|
|
nsCOMPtr<nsIAtom> frameType;
|
|
|
|
lastBlockKid->GetFrameType(getter_AddRefs(frameType));
|
1999-08-31 07:14:54 +00:00
|
|
|
if (frameType.get() != nsLayoutAtoms::lineFrame) {
|
1999-08-31 03:09:40 +00:00
|
|
|
// No first-line frame at the end of the list, therefore there is
|
|
|
|
// an interveening block between any first-line frame the frames
|
|
|
|
// we are appending. Therefore, we don't need any special
|
|
|
|
// treatment of the appended frames.
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
nsIFrame* lineFrame = lastBlockKid;
|
|
|
|
nsCOMPtr<nsIStyleContext> firstLineStyle;
|
|
|
|
lineFrame->GetStyleContext(getter_AddRefs(firstLineStyle));
|
|
|
|
|
|
|
|
// Find the first and last inline frame in aFrameItems
|
|
|
|
nsIFrame* kid = aFrameItems.childList;
|
|
|
|
nsIFrame* firstInlineFrame = nsnull;
|
|
|
|
nsIFrame* lastInlineFrame = nsnull;
|
|
|
|
while (kid) {
|
|
|
|
if (IsInlineFrame(kid)) {
|
|
|
|
if (!firstInlineFrame) firstInlineFrame = kid;
|
|
|
|
lastInlineFrame = kid;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
kid->GetNextSibling(&kid);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we don't find any inline frames, then there is nothing to do
|
|
|
|
if (!firstInlineFrame) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The inline frames get appended to the lineFrame. Make sure they
|
|
|
|
// are reparented properly.
|
|
|
|
nsIFrame* remainingFrames;
|
|
|
|
lastInlineFrame->GetNextSibling(&remainingFrames);
|
|
|
|
lastInlineFrame->SetNextSibling(nsnull);
|
|
|
|
kid = firstInlineFrame;
|
|
|
|
while (kid) {
|
|
|
|
ReparentFrame(aPresContext, lineFrame, firstLineStyle, kid);
|
|
|
|
kid->GetNextSibling(&kid);
|
|
|
|
}
|
1999-11-24 06:03:41 +00:00
|
|
|
aState.mFrameManager->AppendFrames(aPresContext, *aState.mPresShell,
|
1999-08-31 03:09:40 +00:00
|
|
|
lineFrame, nsnull, firstInlineFrame);
|
|
|
|
|
|
|
|
// The remaining frames get appended to the block frame
|
|
|
|
if (remainingFrames) {
|
|
|
|
aFrameItems.childList = remainingFrames;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
aFrameItems.childList = nsnull;
|
|
|
|
aFrameItems.lastChild = nsnull;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Special routine to handle inserting a new frame into a block
|
|
|
|
// frame's child list. Takes care of placing the new frame into the
|
|
|
|
// right place when first-line style is present.
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::InsertFirstLineFrames(
|
|
|
|
nsIPresContext* aPresContext,
|
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aBlockFrame,
|
|
|
|
nsIFrame** aParentFrame,
|
|
|
|
nsIFrame* aPrevSibling,
|
|
|
|
nsFrameItems& aFrameItems)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
#if 0
|
|
|
|
nsIFrame* parentFrame = *aParentFrame;
|
|
|
|
nsIFrame* newFrame = aFrameItems.childList;
|
|
|
|
PRBool isInline = IsInlineFrame(newFrame);
|
|
|
|
|
|
|
|
if (!aPrevSibling) {
|
|
|
|
// Insertion will become the first frame. Two cases: we either
|
|
|
|
// already have a first-line frame or we don't.
|
|
|
|
nsIFrame* firstBlockKid;
|
|
|
|
aBlockFrame->FirstChild(nsnull, &firstBlockKid);
|
|
|
|
nsCOMPtr<nsIAtom> frameType;
|
|
|
|
firstBlockKid->GetFrameType(getter_AddRefs(frameType));
|
1999-09-01 01:00:25 +00:00
|
|
|
if (frameType.get() == nsLayoutAtoms::lineFrame) {
|
1999-08-31 03:09:40 +00:00
|
|
|
// We already have a first-line frame
|
|
|
|
nsIFrame* lineFrame = firstBlockKid;
|
|
|
|
nsCOMPtr<nsIStyleContext> firstLineStyle;
|
|
|
|
lineFrame->GetStyleContext(getter_AddRefs(firstLineStyle));
|
|
|
|
|
|
|
|
if (isInline) {
|
|
|
|
// Easy case: the new inline frame will go into the lineFrame.
|
|
|
|
ReparentFrame(aPresContext, lineFrame, firstLineStyle, newFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
aState.mFrameManager->InsertFrames(aPresContext, *aState.mPresShell,
|
1999-08-31 03:09:40 +00:00
|
|
|
lineFrame, nsnull, nsnull,
|
|
|
|
newFrame);
|
|
|
|
|
|
|
|
// Since the frame is going into the lineFrame, don't let it
|
|
|
|
// go into the block too.
|
|
|
|
aFrameItems.childList = nsnull;
|
|
|
|
aFrameItems.lastChild = nsnull;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Harder case: We are about to insert a block level element
|
|
|
|
// before the first-line frame.
|
|
|
|
// XXX need a method to steal away frames from the line-frame
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// We do not have a first-line frame
|
|
|
|
if (isInline) {
|
|
|
|
// We now need a first-line frame to contain the inline frame.
|
|
|
|
nsIFrame* lineFrame;
|
|
|
|
rv = NS_NewFirstLineFrame(&lineFrame);
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
// Lookup first-line style context
|
|
|
|
nsCOMPtr<nsIStyleContext> parentStyle;
|
|
|
|
aBlockFrame->GetStyleContext(getter_AddRefs(parentStyle));
|
|
|
|
nsCOMPtr<nsIStyleContext> firstLineStyle(
|
|
|
|
getter_AddRefs(GetFirstLineStyle(aPresContext, aContent,
|
|
|
|
parentStyle))
|
|
|
|
);
|
|
|
|
|
|
|
|
// Initialize the line frame
|
1999-12-06 07:44:18 +00:00
|
|
|
rv = InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aBlockFrame, firstLineStyle, nsnull, lineFrame);
|
1999-08-31 03:09:40 +00:00
|
|
|
|
|
|
|
// Make sure the caller inserts the lineFrame into the
|
|
|
|
// blocks list of children.
|
|
|
|
aFrameItems.childList = lineFrame;
|
|
|
|
aFrameItems.lastChild = lineFrame;
|
|
|
|
|
|
|
|
// Give the inline frames to the lineFrame <b>after</b>
|
|
|
|
// reparenting them
|
|
|
|
ReparentFrame(aPresContext, lineFrame, firstLineStyle, newFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
lineFrame->SetInitialChildList(aPresContext, nsnull, newFrame);
|
1999-08-31 03:09:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Easy case: the regular insertion logic can insert the new
|
|
|
|
// frame because its a block frame.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Insertion will not be the first frame.
|
|
|
|
nsIFrame* prevSiblingParent;
|
|
|
|
aPrevSibling->GetParent(&prevSiblingParent);
|
|
|
|
if (prevSiblingParent == aBlockFrame) {
|
|
|
|
// Easy case: The prev-siblings parent is the block
|
|
|
|
// frame. Therefore the prev-sibling is not currently in a
|
|
|
|
// line-frame. Therefore the new frame which is going after it,
|
|
|
|
// regardless of type, is not going into a line-frame.
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// If the prevSiblingParent is not the block-frame then it must
|
|
|
|
// be a line-frame (if it were a letter-frame, that logic would
|
|
|
|
// already have adjusted the prev-sibling to be the
|
|
|
|
// letter-frame).
|
|
|
|
if (isInline) {
|
|
|
|
// Easy case: the insertion can go where the caller thinks it
|
|
|
|
// should go (which is into prevSiblingParent).
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Block elements don't end up in line-frames, therefore
|
|
|
|
// change the insertion point to aBlockFrame. However, there
|
|
|
|
// might be more inline elements following aPrevSibling that
|
|
|
|
// need to be pulled out of the line-frame and become children
|
|
|
|
// of the block.
|
|
|
|
nsIFrame* nextSibling;
|
|
|
|
aPrevSibling->GetNextSibling(&nextSibling);
|
|
|
|
nsIFrame* nextLineFrame;
|
|
|
|
prevSiblingParent->GetNextInFlow(&nextLineFrame);
|
|
|
|
if (nextSibling || nextLineFrame) {
|
|
|
|
// Oy. We have work to do. Create a list of the new frames
|
|
|
|
// that are going into the block by stripping them away from
|
|
|
|
// the line-frame(s).
|
|
|
|
nsFrameList list(nextSibling);
|
|
|
|
if (nextSibling) {
|
|
|
|
nsLineFrame* lineFrame = (nsLineFrame*) prevSiblingParent;
|
|
|
|
lineFrame->StealFramesFrom(nextSibling);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsLineFrame* nextLineFrame = (nsLineFrame*) lineFrame;
|
|
|
|
for (;;) {
|
|
|
|
nextLineFrame->GetNextInFlow(&nextLineFrame);
|
|
|
|
if (!nextLineFrame) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
nsIFrame* kids;
|
|
|
|
nextLineFrame->FirstChild(nsnull, &kids);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// We got lucky: aPrevSibling was the last inline frame in
|
|
|
|
// the line-frame.
|
|
|
|
ReparentFrame(aPresContext, aBlockFrame, firstLineStyle, newFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
aState.mFrameManager->InsertFrames(aPresContext, *aState.mPresShell,
|
1999-08-31 03:09:40 +00:00
|
|
|
aBlockFrame, nsnull,
|
|
|
|
prevSiblingParent, newFrame);
|
|
|
|
aFrameItems.childList = nsnull;
|
|
|
|
aFrameItems.lastChild = nsnull;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
// First-letter support
|
|
|
|
|
1999-08-27 21:46:10 +00:00
|
|
|
// Determine how many characters in the text fragment apply to the
|
|
|
|
// first letter
|
1999-04-27 22:14:54 +00:00
|
|
|
static PRInt32
|
1999-10-15 23:36:07 +00:00
|
|
|
FirstLetterCount(const nsTextFragment* aFragment)
|
1999-04-27 22:14:54 +00:00
|
|
|
{
|
|
|
|
PRInt32 count = 0;
|
|
|
|
PRInt32 firstLetterLength = 0;
|
|
|
|
PRBool done = PR_FALSE;
|
1999-10-15 23:36:07 +00:00
|
|
|
|
|
|
|
PRInt32 i, n = aFragment->GetLength();
|
|
|
|
for (i = 0; i < n; i++) {
|
|
|
|
PRUnichar ch = aFragment->CharAt(i);
|
|
|
|
if (XP_IS_SPACE(ch)) {
|
|
|
|
if (firstLetterLength) {
|
|
|
|
done = PR_TRUE;
|
|
|
|
break;
|
1999-04-27 22:14:54 +00:00
|
|
|
}
|
1999-10-15 23:36:07 +00:00
|
|
|
count++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// XXX I18n
|
|
|
|
if ((ch == '\'') || (ch == '\"')) {
|
|
|
|
if (firstLetterLength) {
|
1999-04-27 22:14:54 +00:00
|
|
|
done = PR_TRUE;
|
|
|
|
break;
|
|
|
|
}
|
1999-10-15 23:36:07 +00:00
|
|
|
// keep looping
|
|
|
|
firstLetterLength = 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
count++;
|
|
|
|
done = PR_TRUE;
|
|
|
|
break;
|
1999-04-27 22:14:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-10-15 23:36:07 +00:00
|
|
|
return count;
|
1999-04-27 22:14:54 +00:00
|
|
|
}
|
|
|
|
|
1999-08-31 03:09:40 +00:00
|
|
|
static PRBool
|
|
|
|
NeedFirstLetterContinuation(nsIContent* aContent)
|
1999-04-27 22:14:54 +00:00
|
|
|
{
|
|
|
|
NS_PRECONDITION(aContent, "null ptr");
|
|
|
|
|
|
|
|
PRBool result = PR_FALSE;
|
|
|
|
if (aContent) {
|
1999-08-31 03:09:40 +00:00
|
|
|
nsCOMPtr<nsITextContent> tc(do_QueryInterface(aContent));
|
|
|
|
if (tc) {
|
1999-10-15 23:36:07 +00:00
|
|
|
const nsTextFragment* frag = nsnull;
|
|
|
|
tc->GetText(&frag);
|
|
|
|
PRInt32 flc = FirstLetterCount(frag);
|
|
|
|
PRInt32 tl = frag->GetLength();
|
1999-04-27 22:14:54 +00:00
|
|
|
if (flc < tl) {
|
|
|
|
result = PR_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
static PRBool IsFirstLetterContent(nsIContent* aContent)
|
|
|
|
{
|
|
|
|
PRBool result = PR_FALSE;
|
|
|
|
|
|
|
|
nsCOMPtr<nsITextContent> textContent = do_QueryInterface(aContent);
|
|
|
|
if (textContent) {
|
|
|
|
PRInt32 textLength;
|
|
|
|
textContent->GetTextLength(&textLength);
|
|
|
|
if (textLength) {
|
|
|
|
PRBool onlyWhiteSpace;
|
|
|
|
textContent->IsOnlyWhitespace(&onlyWhiteSpace);
|
|
|
|
result = !onlyWhiteSpace;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a letter frame, only make it a floating frame.
|
|
|
|
*/
|
1999-04-27 22:14:54 +00:00
|
|
|
void
|
1999-10-29 14:39:48 +00:00
|
|
|
nsCSSFrameConstructor::CreateFloatingLetterFrame(
|
1999-12-04 23:49:50 +00:00
|
|
|
nsIPresShell* aPresShell,
|
1999-04-27 22:14:54 +00:00
|
|
|
nsIPresContext* aPresContext,
|
1999-10-29 14:39:48 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aTextContent,
|
1999-04-27 22:14:54 +00:00
|
|
|
nsIFrame* aTextFrame,
|
1999-10-29 14:39:48 +00:00
|
|
|
nsIContent* aBlockContent,
|
1999-04-27 22:14:54 +00:00
|
|
|
nsIFrame* aParentFrame,
|
1999-10-29 14:39:48 +00:00
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsFrameItems& aResult)
|
1999-04-27 22:14:54 +00:00
|
|
|
{
|
1999-10-29 14:39:48 +00:00
|
|
|
// Create the first-letter-frame
|
|
|
|
nsIFrame* letterFrame;
|
1999-12-06 07:44:18 +00:00
|
|
|
|
|
|
|
NS_NewFirstLetterFrame(aPresShell, &letterFrame);
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aTextContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, letterFrame);
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// Init the text frame to refer to the letter frame. Make sure we
|
|
|
|
// get a proper style context for it (the one passed in is for the
|
|
|
|
// letter frame and will have the float property set on it; the text
|
|
|
|
// frame shouldn't have that set).
|
|
|
|
nsCOMPtr<nsIStyleContext> textSC;
|
|
|
|
aPresContext->ResolveStyleContextFor(aTextContent, aStyleContext, PR_FALSE,
|
1999-12-06 07:44:18 +00:00
|
|
|
getter_AddRefs(textSC));
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aTextContent,
|
|
|
|
letterFrame, textSC, nsnull, aTextFrame);
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// And then give the text frame to the letter frame
|
1999-11-24 06:03:41 +00:00
|
|
|
letterFrame->SetInitialChildList(aPresContext, nsnull, aTextFrame);
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// Now make the placeholder
|
|
|
|
nsIFrame* placeholderFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
CreatePlaceholderFrameFor(aPresShell, aPresContext, aState.mFrameManager, aTextContent,
|
1999-10-29 14:39:48 +00:00
|
|
|
letterFrame, aStyleContext, aParentFrame,
|
|
|
|
&placeholderFrame);
|
|
|
|
|
1999-04-27 22:14:54 +00:00
|
|
|
// See if we will need to continue the text frame (does it contain
|
|
|
|
// more than just the first-letter text or not?) If it does, then we
|
|
|
|
// create (in advance) a continuation frame for it.
|
|
|
|
nsIFrame* nextTextFrame = nsnull;
|
1999-10-29 14:39:48 +00:00
|
|
|
if (NeedFirstLetterContinuation(aTextContent)) {
|
1999-04-27 22:14:54 +00:00
|
|
|
// Create continuation
|
1999-12-04 23:49:50 +00:00
|
|
|
CreateContinuingFrame(aPresShell, aPresContext, aTextFrame, aParentFrame,
|
1999-04-27 22:14:54 +00:00
|
|
|
&nextTextFrame);
|
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
// Repair the continuations style context
|
|
|
|
nsCOMPtr<nsIStyleContext> parentStyleContext;
|
|
|
|
parentStyleContext = getter_AddRefs(aStyleContext->GetParent());
|
|
|
|
if (parentStyleContext) {
|
|
|
|
nsCOMPtr<nsIStyleContext> newSC;
|
|
|
|
aPresContext->ResolveStyleContextFor(aTextContent, parentStyleContext,
|
|
|
|
PR_FALSE, getter_AddRefs(newSC));
|
|
|
|
if (newSC) {
|
|
|
|
nextTextFrame->SetStyleContext(aPresContext, newSC);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
1999-04-27 22:14:54 +00:00
|
|
|
|
|
|
|
// Update the child lists for the frame containing the floating first
|
|
|
|
// letter frame.
|
1999-10-29 14:39:48 +00:00
|
|
|
aState.mFloatedItems.AddChild(letterFrame);
|
|
|
|
aResult.childList = aResult.lastChild = placeholderFrame;
|
1999-04-27 22:14:54 +00:00
|
|
|
if (nextTextFrame) {
|
1999-10-29 14:39:48 +00:00
|
|
|
aResult.AddChild(nextTextFrame);
|
1999-04-27 22:14:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
/**
|
|
|
|
* Create a new letter frame for aTextFrame. The letter frame will be
|
|
|
|
* a child of aParentFrame.
|
|
|
|
*/
|
1999-04-27 22:14:54 +00:00
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::CreateLetterFrame(nsIPresShell* aPresShell, nsIPresContext* aPresContext,
|
1999-10-29 14:39:48 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aTextContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsFrameItems& aResult)
|
1999-04-27 22:14:54 +00:00
|
|
|
{
|
1999-10-29 14:39:48 +00:00
|
|
|
nsCOMPtr<nsIContent> parentContent;
|
|
|
|
aParentFrame->GetContent(getter_AddRefs(parentContent));
|
|
|
|
|
1999-04-27 22:14:54 +00:00
|
|
|
// Get style context for the first-letter-frame
|
1999-10-29 14:39:48 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> parentStyleContext;
|
|
|
|
aParentFrame->GetStyleContext(getter_AddRefs(parentStyleContext));
|
|
|
|
if (parentStyleContext) {
|
|
|
|
// Use content from containing block so that we can actually
|
|
|
|
// find a matching style rule.
|
|
|
|
nsCOMPtr<nsIContent> blockContent;
|
|
|
|
aState.mFloatedItems.containingBlock->GetContent(
|
|
|
|
getter_AddRefs(blockContent));
|
|
|
|
|
|
|
|
// Create first-letter style rule
|
|
|
|
nsCOMPtr<nsIStyleContext> sc = getter_AddRefs(
|
|
|
|
GetFirstLetterStyle(aPresContext, blockContent, parentStyleContext));
|
1999-04-27 22:14:54 +00:00
|
|
|
if (sc) {
|
1999-10-29 14:39:48 +00:00
|
|
|
// Create a new text frame (the original one will be discarded)
|
|
|
|
nsIFrame* textFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewTextFrame(aPresShell, &textFrame);
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// Create the right type of first-letter frame
|
1999-04-27 22:14:54 +00:00
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
sc->GetStyleData(eStyleStruct_Display);
|
|
|
|
if (display->IsFloating()) {
|
1999-10-29 14:39:48 +00:00
|
|
|
// Make a floating first-letter frame
|
1999-12-04 23:49:50 +00:00
|
|
|
CreateFloatingLetterFrame(aPresShell, aPresContext, aState,
|
1999-10-29 14:39:48 +00:00
|
|
|
aTextContent, textFrame,
|
|
|
|
blockContent, aParentFrame,
|
|
|
|
sc, aResult);
|
1999-04-27 22:14:54 +00:00
|
|
|
}
|
|
|
|
else {
|
1999-10-29 14:39:48 +00:00
|
|
|
// Make an inflow first-letter frame
|
|
|
|
nsIFrame* letterFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
nsresult rv = NS_NewFirstLetterFrame(aPresShell, &letterFrame);
|
1999-04-27 22:14:54 +00:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
// Initialize the first-letter-frame.
|
1999-11-24 06:03:41 +00:00
|
|
|
letterFrame->Init(aPresContext, aTextContent, aParentFrame,
|
1999-10-29 14:39:48 +00:00
|
|
|
sc, nsnull);
|
|
|
|
nsCOMPtr<nsIStyleContext> textSC;
|
|
|
|
aPresContext->ResolveStyleContextFor(aTextContent, sc, PR_FALSE,
|
|
|
|
getter_AddRefs(textSC));
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aTextContent,
|
|
|
|
letterFrame, textSC, nsnull, textFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
letterFrame->SetInitialChildList(aPresContext, nsnull, textFrame);
|
1999-10-29 14:39:48 +00:00
|
|
|
aResult.childList = aResult.lastChild = letterFrame;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::WrapFramesInFirstLetterFrame(
|
1999-12-04 23:49:50 +00:00
|
|
|
nsIPresShell* aPresShell,
|
1999-10-29 14:39:48 +00:00
|
|
|
nsIPresContext* aPresContext,
|
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aBlockContent,
|
|
|
|
nsIFrame* aBlockFrame,
|
|
|
|
nsFrameItems& aBlockFrames)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
nsIFrame* parentFrame = nsnull;
|
|
|
|
nsIFrame* textFrame = nsnull;
|
|
|
|
nsIFrame* prevFrame = nsnull;
|
|
|
|
nsFrameItems letterFrames;
|
|
|
|
PRBool stopLooking = PR_FALSE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = WrapFramesInFirstLetterFrame(aPresShell, aPresContext, aState,
|
1999-10-29 14:39:48 +00:00
|
|
|
aBlockFrame, aBlockFrames.childList,
|
|
|
|
&parentFrame, &textFrame, &prevFrame,
|
|
|
|
letterFrames, &stopLooking);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
if (parentFrame) {
|
|
|
|
if (parentFrame == aBlockFrame) {
|
|
|
|
// Text textFrame out of the blocks frame list and substitute the
|
|
|
|
// letter frame(s) instead.
|
|
|
|
nsIFrame* nextSibling;
|
|
|
|
textFrame->GetNextSibling(&nextSibling);
|
|
|
|
textFrame->SetNextSibling(nsnull);
|
|
|
|
if (prevFrame) {
|
|
|
|
prevFrame->SetNextSibling(letterFrames.childList);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
aBlockFrames.childList = letterFrames.childList;
|
|
|
|
}
|
|
|
|
letterFrames.lastChild->SetNextSibling(nextSibling);
|
|
|
|
|
|
|
|
// Destroy the old textFrame
|
1999-11-24 06:03:41 +00:00
|
|
|
textFrame->Destroy(aPresContext);
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// Repair lastChild; the only time this needs to happen is when
|
|
|
|
// the block had one child (the text frame).
|
|
|
|
if (!nextSibling) {
|
|
|
|
aBlockFrames.lastChild = letterFrames.lastChild;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Take the old textFrame out of the inline parents child list
|
1999-11-24 06:03:41 +00:00
|
|
|
parentFrame->RemoveFrame(aPresContext, *aState.mPresShell.get(),
|
1999-10-29 14:39:48 +00:00
|
|
|
nsnull, textFrame);
|
|
|
|
|
|
|
|
// Insert in the letter frame(s)
|
1999-11-24 06:03:41 +00:00
|
|
|
parentFrame->InsertFrames(aPresContext, *aState.mPresShell.get(),
|
1999-10-29 14:39:48 +00:00
|
|
|
nsnull, prevFrame, letterFrames.childList);
|
|
|
|
}
|
|
|
|
}
|
1999-04-27 22:14:54 +00:00
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::WrapFramesInFirstLetterFrame(
|
1999-12-04 23:49:50 +00:00
|
|
|
nsIPresShell* aPresShell,
|
1999-10-29 14:39:48 +00:00
|
|
|
nsIPresContext* aPresContext,
|
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIFrame* aParentFrameList,
|
|
|
|
nsIFrame** aModifiedParent,
|
|
|
|
nsIFrame** aTextFrame,
|
|
|
|
nsIFrame** aPrevFrame,
|
|
|
|
nsFrameItems& aLetterFrames,
|
|
|
|
PRBool* aStopLooking)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
nsIFrame* prevFrame = nsnull;
|
|
|
|
nsIFrame* frame = aParentFrameList;
|
|
|
|
|
|
|
|
while (frame) {
|
|
|
|
nsIFrame* nextFrame;
|
|
|
|
frame->GetNextSibling(&nextFrame);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIAtom> frameType;
|
|
|
|
frame->GetFrameType(getter_AddRefs(frameType));
|
1999-10-29 15:20:20 +00:00
|
|
|
if (nsLayoutAtoms::textFrame == frameType.get()) {
|
1999-10-29 14:39:48 +00:00
|
|
|
// Wrap up first-letter content in a letter frame
|
|
|
|
nsCOMPtr<nsIContent> textContent;
|
|
|
|
frame->GetContent(getter_AddRefs(textContent));
|
|
|
|
if (IsFirstLetterContent(textContent)) {
|
|
|
|
// Create letter frame to wrap up the text
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = CreateLetterFrame(aPresShell, aPresContext, aState, textContent,
|
1999-10-29 14:39:48 +00:00
|
|
|
aParentFrame, aLetterFrames);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
return rv;
|
1999-04-27 22:14:54 +00:00
|
|
|
}
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// Provide adjustment information for parent
|
|
|
|
*aModifiedParent = aParentFrame;
|
|
|
|
*aTextFrame = frame;
|
|
|
|
*aPrevFrame = prevFrame;
|
|
|
|
*aStopLooking = PR_TRUE;
|
|
|
|
return NS_OK;
|
1999-04-27 22:14:54 +00:00
|
|
|
}
|
1999-10-29 14:39:48 +00:00
|
|
|
}
|
1999-10-29 15:20:20 +00:00
|
|
|
else if ((nsLayoutAtoms::inlineFrame == frameType.get()) ||
|
|
|
|
(nsLayoutAtoms::lineFrame == frameType.get())) {
|
1999-10-29 14:39:48 +00:00
|
|
|
nsIFrame* kids;
|
2000-01-22 01:16:50 +00:00
|
|
|
frame->FirstChild(aPresContext, nsnull, &kids);
|
1999-12-04 23:49:50 +00:00
|
|
|
WrapFramesInFirstLetterFrame(aPresShell, aPresContext, aState, frame, kids,
|
1999-10-29 14:39:48 +00:00
|
|
|
aModifiedParent, aTextFrame,
|
|
|
|
aPrevFrame, aLetterFrames, aStopLooking);
|
|
|
|
if (*aStopLooking) {
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// This will stop us looking to create more letter frames. For
|
|
|
|
// example, maybe the frame-type is "letterFrame" or
|
|
|
|
// "placeholderFrame". This keeps us from creating extra letter
|
|
|
|
// frames, and also prevents us from creating letter frames when
|
|
|
|
// the first real content child of a block is not text (e.g. an
|
|
|
|
// image, hr, etc.)
|
|
|
|
*aStopLooking = PR_TRUE;
|
|
|
|
break;
|
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
prevFrame = frame;
|
|
|
|
frame = nextFrame;
|
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
return rv;
|
|
|
|
}
|
1999-08-27 21:46:10 +00:00
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::RemoveFloatingFirstLetterFrames(
|
|
|
|
nsIPresContext* aPresContext,
|
|
|
|
nsIPresShell* aPresShell,
|
|
|
|
nsIFrameManager* aFrameManager,
|
|
|
|
nsIFrame* aBlockFrame,
|
|
|
|
PRBool* aStopLooking)
|
|
|
|
{
|
|
|
|
// First look for the floater frame that is a letter frame
|
|
|
|
nsIFrame* floater;
|
2000-01-22 01:16:50 +00:00
|
|
|
aBlockFrame->FirstChild(aPresContext, nsLayoutAtoms::floaterList, &floater);
|
1999-10-29 14:39:48 +00:00
|
|
|
while (floater) {
|
|
|
|
// See if we found a floating letter frame
|
|
|
|
nsCOMPtr<nsIAtom> frameType;
|
|
|
|
floater->GetFrameType(getter_AddRefs(frameType));
|
|
|
|
if (nsLayoutAtoms::letterFrame == frameType.get()) {
|
|
|
|
break;
|
1999-04-27 22:14:54 +00:00
|
|
|
}
|
1999-10-29 14:39:48 +00:00
|
|
|
floater->GetNextSibling(&floater);
|
|
|
|
}
|
|
|
|
if (!floater) {
|
|
|
|
// No such frame
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Take the text frame away from the letter frame (so it isn't
|
|
|
|
// destroyed when we destroy the letter frame).
|
|
|
|
nsIFrame* textFrame;
|
2000-01-22 01:16:50 +00:00
|
|
|
floater->FirstChild(aPresContext, nsnull, &textFrame);
|
1999-10-29 14:39:48 +00:00
|
|
|
if (!textFrame) {
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Discover the placeholder frame for the letter frame
|
|
|
|
nsIFrame* parentFrame;
|
|
|
|
nsIFrame* placeholderFrame;
|
|
|
|
aFrameManager->GetPlaceholderFrameFor(floater, &placeholderFrame);
|
|
|
|
if (!placeholderFrame) {
|
|
|
|
// Somethings really wrong
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
placeholderFrame->GetParent(&parentFrame);
|
|
|
|
if (!parentFrame) {
|
|
|
|
// Somethings really wrong
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a new text frame with the right style context that maps
|
|
|
|
// all of the content that was previously part of the letter frame
|
|
|
|
// (and probably continued elsewhere).
|
|
|
|
nsCOMPtr<nsIStyleContext> parentSC;
|
|
|
|
parentFrame->GetStyleContext(getter_AddRefs(parentSC));
|
|
|
|
if (!parentSC) {
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
nsCOMPtr<nsIContent> textContent;
|
|
|
|
textFrame->GetContent(getter_AddRefs(textContent));
|
|
|
|
if (!textContent) {
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
nsCOMPtr<nsIStyleContext> newSC;
|
|
|
|
aPresContext->ResolveStyleContextFor(textContent, parentSC,
|
|
|
|
PR_FALSE,
|
|
|
|
getter_AddRefs(newSC));
|
|
|
|
if (!newSC) {
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
nsIFrame* newTextFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
nsresult rv = NS_NewTextFrame(aPresShell, &newTextFrame);
|
1999-10-29 14:39:48 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
return rv;
|
|
|
|
}
|
1999-11-24 06:03:41 +00:00
|
|
|
newTextFrame->Init(aPresContext, textContent, parentFrame, newSC, nsnull);
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// Destroy the old text frame's continuations (the old text frame
|
|
|
|
// will be destroyed when its letter frame is destroyed).
|
|
|
|
nsIFrame* nextTextFrame;
|
|
|
|
textFrame->GetNextInFlow(&nextTextFrame);
|
|
|
|
if (nextTextFrame) {
|
|
|
|
nsIFrame* nextTextParent;
|
|
|
|
nextTextFrame->GetParent(&nextTextParent);
|
|
|
|
if (nextTextParent) {
|
|
|
|
nsSplittableFrame::BreakFromPrevFlow(nextTextFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
aFrameManager->RemoveFrame(aPresContext, *aPresShell, nextTextParent,
|
1999-10-29 14:39:48 +00:00
|
|
|
nsnull, nextTextFrame);
|
1999-04-28 00:55:11 +00:00
|
|
|
}
|
1999-04-27 22:14:54 +00:00
|
|
|
}
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// First find out where (in the content) the placeholder frames
|
|
|
|
// text is and its previous sibling frame, if any.
|
|
|
|
nsIFrame* prevSibling = nsnull;
|
|
|
|
|
|
|
|
nsCOMPtr<nsIContent> container;
|
|
|
|
parentFrame->GetContent(getter_AddRefs(container));
|
|
|
|
if (container.get() && textContent.get()) {
|
|
|
|
PRInt32 ix = 0;
|
|
|
|
container->IndexOf(textContent, ix);
|
|
|
|
prevSibling = FindPreviousSibling(aPresShell, container, ix);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now that everything is set...
|
|
|
|
#ifdef NOISY_FIRST_LETTER
|
|
|
|
printf("RemoveFloatingFirstLetterFrames: textContent=%p oldTextFrame=%p newTextFrame=%p\n",
|
|
|
|
textContent.get(), textFrame, newTextFrame);
|
|
|
|
#endif
|
|
|
|
aFrameManager->SetPlaceholderFrameFor(floater, nsnull);
|
|
|
|
aFrameManager->SetPrimaryFrameFor(textContent, nsnull);
|
|
|
|
|
|
|
|
// Remove the floater frame
|
1999-11-24 06:03:41 +00:00
|
|
|
aFrameManager->RemoveFrame(aPresContext, *aPresShell,
|
1999-10-29 14:39:48 +00:00
|
|
|
aBlockFrame, nsLayoutAtoms::floaterList,
|
|
|
|
floater);
|
|
|
|
|
|
|
|
// Remove placeholder frame
|
1999-11-24 06:03:41 +00:00
|
|
|
aFrameManager->RemoveFrame(aPresContext, *aPresShell,
|
1999-10-29 14:39:48 +00:00
|
|
|
parentFrame, nsnull, placeholderFrame);
|
|
|
|
|
|
|
|
// Insert text frame in its place
|
1999-11-24 06:03:41 +00:00
|
|
|
aFrameManager->InsertFrames(aPresContext, *aPresShell, parentFrame, nsnull,
|
1999-10-29 14:39:48 +00:00
|
|
|
prevSibling, newTextFrame);
|
|
|
|
|
1999-04-27 22:14:54 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
1999-04-28 19:08:14 +00:00
|
|
|
|
1999-10-29 14:39:48 +00:00
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::RemoveFirstLetterFrames(nsIPresContext* aPresContext,
|
|
|
|
nsIPresShell* aPresShell,
|
|
|
|
nsIFrameManager* aFrameManager,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
PRBool* aStopLooking)
|
|
|
|
{
|
|
|
|
nsIFrame* kid;
|
|
|
|
nsIFrame* prevSibling = nsnull;
|
2000-01-22 01:16:50 +00:00
|
|
|
aFrame->FirstChild(aPresContext, nsnull, &kid);
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
while (kid) {
|
|
|
|
nsCOMPtr<nsIAtom> frameType;
|
|
|
|
kid->GetFrameType(getter_AddRefs(frameType));
|
1999-10-29 15:20:20 +00:00
|
|
|
if (nsLayoutAtoms::letterFrame == frameType.get()) {
|
1999-10-29 14:39:48 +00:00
|
|
|
// Bingo. Found it. First steal away the text frame.
|
|
|
|
nsIFrame* textFrame;
|
2000-01-22 01:16:50 +00:00
|
|
|
kid->FirstChild(aPresContext, nsnull, &textFrame);
|
1999-10-29 14:39:48 +00:00
|
|
|
if (!textFrame) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a new textframe
|
|
|
|
nsCOMPtr<nsIStyleContext> parentSC;
|
|
|
|
aFrame->GetStyleContext(getter_AddRefs(parentSC));
|
|
|
|
if (!parentSC) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
nsCOMPtr<nsIContent> textContent;
|
|
|
|
textFrame->GetContent(getter_AddRefs(textContent));
|
|
|
|
if (!textContent) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
nsCOMPtr<nsIStyleContext> newSC;
|
|
|
|
aPresContext->ResolveStyleContextFor(textContent, parentSC,
|
|
|
|
PR_FALSE,
|
|
|
|
getter_AddRefs(newSC));
|
|
|
|
if (!newSC) {
|
|
|
|
break;
|
|
|
|
}
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewTextFrame(aPresShell, &textFrame);
|
1999-11-24 06:03:41 +00:00
|
|
|
textFrame->Init(aPresContext, textContent, aFrame, newSC, nsnull);
|
1999-10-29 14:39:48 +00:00
|
|
|
|
|
|
|
// Next rip out the kid and replace it with the text frame
|
|
|
|
nsIFrameManager* frameManager = aFrameManager;
|
|
|
|
frameManager->SetPrimaryFrameFor(textContent, nsnull);
|
1999-11-24 06:03:41 +00:00
|
|
|
frameManager->RemoveFrame(aPresContext, *aPresShell,
|
1999-10-29 14:39:48 +00:00
|
|
|
aFrame, nsnull, kid);
|
|
|
|
|
|
|
|
// Insert text frame in its place
|
1999-11-24 06:03:41 +00:00
|
|
|
frameManager->InsertFrames(aPresContext, *aPresShell, aFrame, nsnull,
|
1999-10-29 14:39:48 +00:00
|
|
|
prevSibling, textFrame);
|
|
|
|
|
|
|
|
*aStopLooking = PR_TRUE;
|
|
|
|
break;
|
|
|
|
}
|
1999-10-29 15:20:20 +00:00
|
|
|
else if ((nsLayoutAtoms::inlineFrame == frameType.get()) ||
|
|
|
|
(nsLayoutAtoms::lineFrame == frameType.get())) {
|
1999-10-29 14:39:48 +00:00
|
|
|
// Look inside child inline frame for the letter frame
|
|
|
|
RemoveFirstLetterFrames(aPresContext, aPresShell, aFrameManager, kid,
|
|
|
|
aStopLooking);
|
|
|
|
if (*aStopLooking) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
prevSibling = kid;
|
|
|
|
kid->GetNextSibling(&kid);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::RemoveLetterFrames(nsIPresContext* aPresContext,
|
|
|
|
nsIPresShell* aPresShell,
|
|
|
|
nsIFrameManager* aFrameManager,
|
|
|
|
nsIFrame* aBlockFrame)
|
|
|
|
{
|
|
|
|
PRBool stopLooking = PR_FALSE;
|
|
|
|
nsresult rv = RemoveFloatingFirstLetterFrames(aPresContext, aPresShell,
|
|
|
|
aFrameManager,
|
|
|
|
aBlockFrame, &stopLooking);
|
|
|
|
if (NS_SUCCEEDED(rv) && !stopLooking) {
|
|
|
|
rv = RemoveFirstLetterFrames(aPresContext, aPresShell, aFrameManager,
|
|
|
|
aBlockFrame, &stopLooking);
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fixup the letter frame situation for the given block
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::RecoverLetterFrames(nsIPresShell* aPresShell, nsIPresContext* aPresContext,
|
1999-10-29 14:39:48 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIFrame* aBlockFrame)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
nsIFrame* blockKids;
|
2000-01-22 01:16:50 +00:00
|
|
|
aBlockFrame->FirstChild(aPresContext, nsnull, &blockKids);
|
1999-10-29 14:39:48 +00:00
|
|
|
nsIFrame* parentFrame = nsnull;
|
|
|
|
nsIFrame* textFrame = nsnull;
|
|
|
|
nsIFrame* prevFrame = nsnull;
|
|
|
|
nsFrameItems letterFrames;
|
|
|
|
PRBool stopLooking = PR_FALSE;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = WrapFramesInFirstLetterFrame(aPresShell, aPresContext, aState,
|
1999-10-29 14:39:48 +00:00
|
|
|
aBlockFrame, blockKids,
|
|
|
|
&parentFrame, &textFrame, &prevFrame,
|
|
|
|
letterFrames, &stopLooking);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
if (parentFrame) {
|
|
|
|
// Take the old textFrame out of the parents child list
|
1999-11-24 06:03:41 +00:00
|
|
|
parentFrame->RemoveFrame(aPresContext, *aState.mPresShell.get(),
|
1999-10-29 14:39:48 +00:00
|
|
|
nsnull, textFrame);
|
|
|
|
|
|
|
|
// Insert in the letter frame(s)
|
1999-11-24 06:03:41 +00:00
|
|
|
parentFrame->InsertFrames(aPresContext, *aState.mPresShell.get(),
|
1999-10-29 14:39:48 +00:00
|
|
|
nsnull, prevFrame, letterFrames.childList);
|
|
|
|
|
|
|
|
// Insert in floaters too if needed
|
|
|
|
if (aState.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
aBlockFrame->AppendFrames(aPresContext, *aState.mPresShell.get(),
|
1999-10-29 14:39:48 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
|
|
|
aState.mFloatedItems.childList);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
|
1999-06-23 03:02:21 +00:00
|
|
|
// Tree Widget Routines
|
1999-10-29 14:39:48 +00:00
|
|
|
|
1999-06-23 03:02:21 +00:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsCSSFrameConstructor::CreateTreeWidgetContent(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aParentFrame,
|
1999-06-28 08:35:10 +00:00
|
|
|
nsIFrame* aPrevFrame,
|
1999-06-23 03:02:21 +00:00
|
|
|
nsIContent* aChild,
|
1999-06-28 08:35:10 +00:00
|
|
|
nsIFrame** aNewFrame,
|
1999-06-29 20:20:40 +00:00
|
|
|
PRBool aIsAppend,
|
1999-12-06 07:44:18 +00:00
|
|
|
PRBool aIsScrollbar,
|
|
|
|
nsILayoutHistoryState* aFrameState)
|
1999-06-23 03:02:21 +00:00
|
|
|
{
|
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
// Construct a new frame
|
|
|
|
if (nsnull != aParentFrame) {
|
|
|
|
nsFrameItems frameItems;
|
1999-08-05 03:09:22 +00:00
|
|
|
nsFrameConstructorState state(aPresContext, mFixedContainingBlock,
|
1999-06-23 03:02:21 +00:00
|
|
|
GetAbsoluteContainingBlock(aPresContext, aParentFrame),
|
1999-12-06 07:44:18 +00:00
|
|
|
GetFloaterContainingBlock(aPresContext, aParentFrame),
|
2000-01-14 09:28:54 +00:00
|
|
|
mTempFrameTreeState);
|
1999-12-01 11:13:06 +00:00
|
|
|
|
|
|
|
// Get the element's tag
|
|
|
|
nsCOMPtr<nsIAtom> tag;
|
|
|
|
aChild->GetTag(*getter_AddRefs(tag));
|
|
|
|
|
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
rv = ResolveStyleContext(aPresContext, aParentFrame, aChild, tag, getter_AddRefs(styleContext));
|
|
|
|
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
// Pre-check for display "none" - only if we find that, do we create
|
|
|
|
// any frame at all
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
styleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
if (NS_STYLE_DISPLAY_NONE == display->mDisplay) {
|
|
|
|
*aNewFrame = nsnull;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructFrame(shell, aPresContext, state, aChild, aParentFrame, frameItems);
|
1999-06-23 03:02:21 +00:00
|
|
|
|
|
|
|
nsIFrame* newFrame = frameItems.childList;
|
|
|
|
*aNewFrame = newFrame;
|
|
|
|
|
|
|
|
if (NS_SUCCEEDED(rv) && (nsnull != newFrame)) {
|
|
|
|
// Notify the parent frame
|
1999-06-29 20:20:40 +00:00
|
|
|
if (aIsScrollbar)
|
2000-01-22 01:16:50 +00:00
|
|
|
((nsTreeRowGroupFrame*)aParentFrame)->SetScrollbarFrame(aPresContext, newFrame);
|
1999-06-29 20:20:40 +00:00
|
|
|
else if (aIsAppend)
|
1999-06-28 08:35:10 +00:00
|
|
|
rv = ((nsTreeRowGroupFrame*)aParentFrame)->TreeAppendFrames(newFrame);
|
|
|
|
else
|
|
|
|
rv = ((nsTreeRowGroupFrame*)aParentFrame)->TreeInsertFrames(aPrevFrame, newFrame);
|
|
|
|
|
1999-06-23 03:02:21 +00:00
|
|
|
// If there are new absolutely positioned child frames, then notify
|
|
|
|
// the parent
|
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
|
|
|
if (state.mAbsoluteItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mAbsoluteItems.containingBlock->AppendFrames(aPresContext, *shell,
|
1999-06-23 03:02:21 +00:00
|
|
|
nsLayoutAtoms::absoluteList,
|
|
|
|
state.mAbsoluteItems.childList);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there are new fixed positioned child frames, then notify
|
|
|
|
// the parent
|
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
|
|
|
if (state.mFixedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mFixedItems.containingBlock->AppendFrames(aPresContext, *shell,
|
1999-06-23 03:02:21 +00:00
|
|
|
nsLayoutAtoms::fixedList,
|
|
|
|
state.mFixedItems.childList);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there are new floating child frames, then notify
|
|
|
|
// the parent
|
|
|
|
// XXX We can't just assume these frames are being appended, we need to
|
|
|
|
// determine where in the list they should be inserted...
|
|
|
|
if (state.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
rv = state.mFloatedItems.containingBlock->AppendFrames(aPresContext, *shell,
|
1999-06-23 03:02:21 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
|
|
|
state.mFloatedItems.childList);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-11-01 15:24:57 +00:00
|
|
|
//----------------------------------------------------------------------
|
|
|
|
|
|
|
|
// Block/inline frame construction logic. We maintain a few invariants here:
|
|
|
|
//
|
|
|
|
// 1. Block frames contain block and inline frames.
|
|
|
|
//
|
|
|
|
// 2. Inline frames only contain inline frames. If an inline parent has a block
|
|
|
|
// child then the block child is migrated upward until it lands in a block
|
|
|
|
// parent (the inline frames containing block is where it will end up).
|
|
|
|
|
|
|
|
// XXX consolidate these things
|
|
|
|
static PRBool
|
|
|
|
IsBlockFrame(nsIPresContext* aPresContext, nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
aFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct*&) display);
|
|
|
|
if (NS_STYLE_DISPLAY_INLINE == display->mDisplay) {
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static nsIFrame*
|
|
|
|
FindFirstBlock(nsIPresContext* aPresContext, nsIFrame* aKid, nsIFrame** aPrevKid)
|
|
|
|
{
|
|
|
|
nsIFrame* prevKid = nsnull;
|
|
|
|
while (aKid) {
|
|
|
|
if (IsBlockFrame(aPresContext, aKid)) {
|
|
|
|
*aPrevKid = prevKid;
|
|
|
|
return aKid;
|
|
|
|
}
|
|
|
|
prevKid = aKid;
|
|
|
|
aKid->GetNextSibling(&aKid);
|
|
|
|
}
|
|
|
|
*aPrevKid = nsnull;
|
|
|
|
return nsnull;
|
|
|
|
}
|
1999-06-23 03:02:21 +00:00
|
|
|
|
1999-11-01 15:24:57 +00:00
|
|
|
static nsIFrame*
|
|
|
|
FindLastBlock(nsIPresContext* aPresContext, nsIFrame* aKid)
|
|
|
|
{
|
|
|
|
nsIFrame* lastBlock = nsnull;
|
|
|
|
while (aKid) {
|
|
|
|
if (IsBlockFrame(aPresContext, aKid)) {
|
|
|
|
lastBlock = aKid;
|
|
|
|
}
|
|
|
|
aKid->GetNextSibling(&aKid);
|
|
|
|
}
|
|
|
|
return lastBlock;
|
|
|
|
}
|
|
|
|
|
|
|
|
static nsresult
|
|
|
|
MoveChildrenTo(nsIPresContext* aPresContext,
|
|
|
|
nsIStyleContext* aNewParentSC,
|
|
|
|
nsIFrame* aNewParent,
|
|
|
|
nsIFrame* aFrameList)
|
|
|
|
{
|
|
|
|
while (aFrameList) {
|
|
|
|
aFrameList->SetParent(aNewParent);
|
|
|
|
aFrameList->GetNextSibling(&aFrameList);
|
|
|
|
}
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructBlock(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-11-01 15:24:57 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
const nsStyleDisplay* aDisplay,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame* aNewFrame)
|
|
|
|
{
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, aNewFrame);
|
1999-11-01 15:24:57 +00:00
|
|
|
|
|
|
|
// See if we need to create a view, e.g. the frame is absolutely positioned
|
1999-11-24 06:03:41 +00:00
|
|
|
nsHTMLContainerFrame::CreateViewForFrame(aPresContext, aNewFrame,
|
1999-11-01 15:24:57 +00:00
|
|
|
aStyleContext, PR_FALSE);
|
|
|
|
|
|
|
|
// See if the block has first-letter style applied to it...
|
|
|
|
PRBool haveFirstLetterStyle, haveFirstLineStyle;
|
|
|
|
HaveSpecialBlockStyle(aPresContext, aContent, aStyleContext,
|
|
|
|
&haveFirstLetterStyle, &haveFirstLineStyle);
|
|
|
|
|
|
|
|
// Process the child content
|
|
|
|
nsFrameItems childItems;
|
|
|
|
nsFrameConstructorSaveState floaterSaveState;
|
|
|
|
aState.PushFloaterContainingBlock(aNewFrame, floaterSaveState,
|
|
|
|
haveFirstLetterStyle,
|
|
|
|
haveFirstLineStyle);
|
1999-12-04 23:49:50 +00:00
|
|
|
nsresult rv = ProcessBlockChildren(aPresShell, aPresContext, aState, aContent, aNewFrame,
|
1999-11-01 15:24:57 +00:00
|
|
|
PR_TRUE, childItems, PR_TRUE);
|
|
|
|
|
|
|
|
// Set the frame's initial child list
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-11-01 15:24:57 +00:00
|
|
|
|
|
|
|
// Set the frame's floater list if there were any floated children
|
|
|
|
if (aState.mFloatedItems.childList) {
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewFrame->SetInitialChildList(aPresContext,
|
1999-11-01 15:24:57 +00:00
|
|
|
nsLayoutAtoms::floaterList,
|
|
|
|
aState.mFloatedItems.childList);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ProcessBlockChildren(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-11-01 15:24:57 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
PRBool aCanHaveGeneratedContent,
|
|
|
|
nsFrameItems& aFrameItems,
|
|
|
|
PRBool aParentIsBlock)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
|
|
|
|
if (aCanHaveGeneratedContent) {
|
|
|
|
// Probe for generated content before
|
|
|
|
nsIFrame* generatedFrame;
|
|
|
|
aFrame->GetStyleContext(getter_AddRefs(styleContext));
|
1999-12-04 23:49:50 +00:00
|
|
|
if (CreateGeneratedContentFrame(aPresShell, aPresContext, aState, aFrame, aContent,
|
1999-11-01 15:24:57 +00:00
|
|
|
styleContext, nsCSSAtoms::beforePseudo,
|
|
|
|
aParentIsBlock, &generatedFrame)) {
|
|
|
|
// Add the generated frame to the child list
|
|
|
|
aFrameItems.AddChild(generatedFrame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate the child content objects and construct frames
|
|
|
|
PRInt32 count;
|
|
|
|
aContent->ChildCount(count);
|
|
|
|
for (PRInt32 i = 0; i < count; i++) {
|
|
|
|
nsCOMPtr<nsIContent> childContent;
|
|
|
|
if (NS_SUCCEEDED(aContent->ChildAt(i, *getter_AddRefs(childContent)))) {
|
|
|
|
// Construct a child frame
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructFrame(aPresShell, aPresContext, aState, childContent, aFrame, aFrameItems);
|
1999-11-01 15:24:57 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aCanHaveGeneratedContent) {
|
|
|
|
// Probe for generated content after
|
|
|
|
nsIFrame* generatedFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
if (CreateGeneratedContentFrame(aPresShell, aPresContext, aState, aFrame, aContent,
|
1999-11-01 15:24:57 +00:00
|
|
|
styleContext, nsCSSAtoms::afterPseudo,
|
|
|
|
aParentIsBlock, &generatedFrame)) {
|
|
|
|
// Add the generated frame to the child list
|
|
|
|
aFrameItems.AddChild(generatedFrame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aParentIsBlock) {
|
|
|
|
if (aState.mFirstLetterStyle) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = WrapFramesInFirstLetterFrame(aPresShell, aPresContext, aState, aContent, aFrame, aFrameItems);
|
1999-11-01 15:24:57 +00:00
|
|
|
}
|
|
|
|
if (aState.mFirstLineStyle) {
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = WrapFramesInFirstLineFrame(aPresShell, aPresContext, aState, aContent, aFrame, aFrameItems);
|
1999-11-01 15:24:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// When inline frames get weird and have block frames in them, we
|
|
|
|
// annotate them to help us respond to incremental content changes
|
|
|
|
// more easily.
|
|
|
|
|
|
|
|
static void
|
|
|
|
DestroyInlineFrameAnnotation(nsIPresContext* aPresContext,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIAtom* aPropertyName,
|
|
|
|
void* aPropertyValue)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsCSSFrameConstructor::IsFrameSpecial(nsIFrameManager* aFrameManager, nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
void* value;
|
|
|
|
nsresult rv = aFrameManager->GetFrameProperty(aFrame, nsLayoutAtoms::inlineFrameAnnotation, 0, &value);
|
|
|
|
if (NS_OK == rv) {
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsCSSFrameConstructor::IsFrameSpecial(nsIPresContext* aPresContext, nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
// Get to aFrame's first-in-flow; only the first-in-flow is marked
|
|
|
|
// with an annotation.
|
|
|
|
nsSplittableType splits;
|
|
|
|
aFrame->IsSplittable(splits);
|
|
|
|
if (splits != NS_FRAME_NOT_SPLITTABLE) {
|
|
|
|
nsIFrame* prevInFlow = aFrame;
|
|
|
|
while (prevInFlow) {
|
|
|
|
aFrame = prevInFlow;
|
|
|
|
prevInFlow->GetPrevInFlow(&prevInFlow);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool result = PR_FALSE;
|
|
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
aPresContext->GetShell(getter_AddRefs(shell));
|
|
|
|
if (shell) {
|
|
|
|
nsCOMPtr<nsIFrameManager> frameManager;
|
|
|
|
shell->GetFrameManager(getter_AddRefs(frameManager));
|
|
|
|
if (frameManager) {
|
|
|
|
void* value;
|
|
|
|
nsresult rv = frameManager->GetFrameProperty(aFrame, nsLayoutAtoms::inlineFrameAnnotation,
|
|
|
|
0, &value);
|
|
|
|
if (NS_OK == rv) {
|
|
|
|
result = PR_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsCSSFrameConstructor::SetFrameIsSpecial(nsIFrameManager* aFrameManager, nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
aFrameManager->SetFrameProperty(aFrame, nsLayoutAtoms::inlineFrameAnnotation,
|
|
|
|
(void*) PR_TRUE, DestroyInlineFrameAnnotation);
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsCSSFrameConstructor::AreAllKidsInline(nsIFrame* aFrameList)
|
|
|
|
{
|
|
|
|
nsIFrame* kid = aFrameList;
|
|
|
|
while (kid) {
|
|
|
|
if (!IsInlineFrame(kid)) {
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
kid->GetNextSibling(&kid);
|
|
|
|
}
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ConstructInline(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-11-01 15:24:57 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
const nsStyleDisplay* aDisplay,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParentFrame,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame* aNewFrame,
|
|
|
|
nsIFrame** aNewBlockFrame,
|
|
|
|
nsIFrame** aNextInlineFrame)
|
|
|
|
{
|
|
|
|
// Initialize the frame
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, aNewFrame);
|
1999-11-01 15:24:57 +00:00
|
|
|
|
|
|
|
// Process the child content
|
|
|
|
nsFrameItems childItems;
|
|
|
|
PRBool kidsAllInline;
|
1999-12-04 23:49:50 +00:00
|
|
|
nsresult rv = ProcessInlineChildren(aPresShell, aPresContext, aState, aContent,
|
1999-11-01 15:24:57 +00:00
|
|
|
aNewFrame, PR_TRUE, childItems, &kidsAllInline);
|
|
|
|
if (kidsAllInline) {
|
|
|
|
// Set the inline frame's initial child list
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewFrame->SetInitialChildList(aPresContext, nsnull, childItems.childList);
|
1999-11-01 15:24:57 +00:00
|
|
|
*aNewBlockFrame = nsnull;
|
|
|
|
*aNextInlineFrame = nsnull;
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This inline frame contains several types of children. Therefore
|
|
|
|
// this frame has to be chopped into several pieces. We will produce
|
|
|
|
// as a result of this 3 lists of children. The first list contains
|
|
|
|
// all of the inline children that preceed the first block child
|
|
|
|
// (and may be empty). The second list contains all of the block
|
|
|
|
// children and any inlines that are between them (and must not be
|
|
|
|
// empty, otherwise - why are we here?). The final list contains all
|
|
|
|
// of the inline children that follow the final block child.
|
|
|
|
|
|
|
|
// Find the first block child which defines list1 and list2
|
|
|
|
nsIFrame* list1 = childItems.childList;
|
|
|
|
nsIFrame* prevToFirstBlock;
|
|
|
|
nsIFrame* list2 = FindFirstBlock(aPresContext, list1, &prevToFirstBlock);
|
|
|
|
if (prevToFirstBlock) {
|
|
|
|
prevToFirstBlock->SetNextSibling(nsnull);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
list1 = nsnull;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the last block child which defines the end of list2 and the
|
|
|
|
// start of list3
|
|
|
|
nsIFrame* afterFirstBlock;
|
|
|
|
list2->GetNextSibling(&afterFirstBlock);
|
|
|
|
nsIFrame* list3 = nsnull;
|
|
|
|
nsIFrame* lastBlock = FindLastBlock(aPresContext, afterFirstBlock);
|
|
|
|
if (!lastBlock) {
|
|
|
|
lastBlock = list2;
|
|
|
|
}
|
|
|
|
lastBlock->GetNextSibling(&list3);
|
|
|
|
lastBlock->SetNextSibling(nsnull);
|
|
|
|
|
|
|
|
// list1's frames belong to this inline frame so go ahead and take them
|
1999-11-24 06:03:41 +00:00
|
|
|
aNewFrame->SetInitialChildList(aPresContext, nsnull, list1);
|
1999-11-01 15:24:57 +00:00
|
|
|
|
|
|
|
// list2's frames belong to an anonymous block that we create right
|
|
|
|
// now. The anonymous block will be the parent of the block children
|
|
|
|
// of the inline.
|
|
|
|
nsIFrame* blockFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewBlockFrame(aPresShell, &blockFrame);
|
1999-11-01 15:24:57 +00:00
|
|
|
nsCOMPtr<nsIStyleContext> blockSC;
|
|
|
|
aPresContext->ResolvePseudoStyleContextFor(aContent, nsHTMLAtoms::mozAnonymousBlock,
|
|
|
|
aStyleContext, PR_FALSE,
|
|
|
|
getter_AddRefs(blockSC));
|
1999-12-06 07:44:18 +00:00
|
|
|
|
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, blockSC, nsnull, blockFrame);
|
|
|
|
|
1999-11-01 15:24:57 +00:00
|
|
|
MoveChildrenTo(aPresContext, blockSC, blockFrame, list2);
|
1999-11-24 06:03:41 +00:00
|
|
|
blockFrame->SetInitialChildList(aPresContext, nsnull, list2);
|
1999-11-01 15:24:57 +00:00
|
|
|
|
|
|
|
// list3's frames belong to another inline frame
|
|
|
|
nsIFrame* inlineFrame = nsnull;
|
1999-12-04 23:49:50 +00:00
|
|
|
NS_NewInlineFrame(aPresShell, &inlineFrame);
|
1999-12-06 07:44:18 +00:00
|
|
|
InitAndRestoreFrame(aPresContext, aState, aContent,
|
|
|
|
aParentFrame, aStyleContext, nsnull, inlineFrame);
|
1999-11-01 15:24:57 +00:00
|
|
|
if (list3) {
|
|
|
|
// Reparent (cheaply) the frames in list3 - we don't have to futz
|
|
|
|
// with their style context because they already have the right one.
|
|
|
|
nsFrameList list;
|
|
|
|
list.AppendFrames(inlineFrame, list3);
|
|
|
|
}
|
1999-11-24 06:03:41 +00:00
|
|
|
inlineFrame->SetInitialChildList(aPresContext, nsnull, list3);
|
1999-11-01 15:24:57 +00:00
|
|
|
|
|
|
|
// Mark the 3 frames as special. That way if any of the
|
|
|
|
// append/insert/remove methods try to fiddle with the children, the
|
|
|
|
// containing block will be reframed instead.
|
|
|
|
SetFrameIsSpecial(aState.mFrameManager, aNewFrame);
|
|
|
|
SetFrameIsSpecial(aState.mFrameManager, blockFrame);
|
|
|
|
SetFrameIsSpecial(aState.mFrameManager, inlineFrame);
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (gNoisyInlineConstruction) {
|
1999-11-02 06:06:46 +00:00
|
|
|
nsIFrameDebug* frameDebug;
|
|
|
|
|
1999-11-01 15:24:57 +00:00
|
|
|
printf("nsCSSFrameConstructor::ConstructInline:\n");
|
2000-02-02 22:24:56 +00:00
|
|
|
if (NS_SUCCEEDED(aNewFrame->QueryInterface(NS_GET_IID(nsIFrameDebug), (void**)&frameDebug))) {
|
1999-11-02 06:06:46 +00:00
|
|
|
printf(" ==> leading inline frame:\n");
|
|
|
|
frameDebug->List(aPresContext, stdout, 2);
|
|
|
|
}
|
2000-02-02 22:24:56 +00:00
|
|
|
if (NS_SUCCEEDED(blockFrame->QueryInterface(NS_GET_IID(nsIFrameDebug), (void**)&frameDebug))) {
|
1999-11-02 06:06:46 +00:00
|
|
|
printf(" ==> block frame:\n");
|
|
|
|
frameDebug->List(aPresContext, stdout, 2);
|
|
|
|
}
|
2000-02-02 22:24:56 +00:00
|
|
|
if (NS_SUCCEEDED(inlineFrame->QueryInterface(NS_GET_IID(nsIFrameDebug), (void**)&frameDebug))) {
|
1999-11-02 06:06:46 +00:00
|
|
|
printf(" ==> trailing inline frame:\n");
|
|
|
|
frameDebug->List(aPresContext, stdout, 2);
|
|
|
|
}
|
1999-11-01 15:24:57 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
*aNewBlockFrame = blockFrame;
|
|
|
|
*aNextInlineFrame = inlineFrame;
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1999-12-04 23:49:50 +00:00
|
|
|
nsCSSFrameConstructor::ProcessInlineChildren(nsIPresShell* aPresShell,
|
|
|
|
nsIPresContext* aPresContext,
|
1999-11-01 15:24:57 +00:00
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
PRBool aCanHaveGeneratedContent,
|
|
|
|
nsFrameItems& aFrameItems,
|
|
|
|
PRBool* aKidsAllInline)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
nsCOMPtr<nsIStyleContext> styleContext;
|
|
|
|
|
|
|
|
if (aCanHaveGeneratedContent) {
|
|
|
|
// Probe for generated content before
|
|
|
|
nsIFrame* generatedFrame;
|
|
|
|
aFrame->GetStyleContext(getter_AddRefs(styleContext));
|
1999-12-04 23:49:50 +00:00
|
|
|
if (CreateGeneratedContentFrame(aPresShell, aPresContext, aState, aFrame, aContent,
|
1999-11-01 15:24:57 +00:00
|
|
|
styleContext, nsCSSAtoms::beforePseudo,
|
|
|
|
PR_FALSE, &generatedFrame)) {
|
|
|
|
// Add the generated frame to the child list
|
|
|
|
aFrameItems.AddChild(generatedFrame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate the child content objects and construct frames
|
|
|
|
PRBool allKidsInline = PR_TRUE;
|
|
|
|
PRInt32 count;
|
|
|
|
aContent->ChildCount(count);
|
|
|
|
for (PRInt32 i = 0; i < count; i++) {
|
|
|
|
nsCOMPtr<nsIContent> childContent;
|
|
|
|
if (NS_SUCCEEDED(aContent->ChildAt(i, *getter_AddRefs(childContent)))) {
|
|
|
|
// Construct a child frame
|
|
|
|
nsIFrame* oldLastChild = aFrameItems.lastChild;
|
1999-12-04 23:49:50 +00:00
|
|
|
rv = ConstructFrame(aPresShell, aPresContext, aState, childContent, aFrame, aFrameItems);
|
1999-11-01 15:24:57 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Examine newly added children (we may have added more than one
|
|
|
|
// child if the child was another inline frame that ends up
|
|
|
|
// being carved in 3 pieces) to maintain the allKidsInline flag.
|
|
|
|
if (allKidsInline) {
|
|
|
|
nsIFrame* kid;
|
|
|
|
if (oldLastChild) {
|
|
|
|
oldLastChild->GetNextSibling(&kid);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
kid = aFrameItems.childList;
|
|
|
|
}
|
|
|
|
while (kid) {
|
|
|
|
if (!IsInlineFrame(kid)) {
|
|
|
|
allKidsInline = PR_FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
kid->GetNextSibling(&kid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aCanHaveGeneratedContent) {
|
|
|
|
// Probe for generated content after
|
|
|
|
nsIFrame* generatedFrame;
|
1999-12-04 23:49:50 +00:00
|
|
|
if (CreateGeneratedContentFrame(aPresShell, aPresContext, aState, aFrame, aContent,
|
1999-11-01 15:24:57 +00:00
|
|
|
styleContext, nsCSSAtoms::afterPseudo,
|
|
|
|
PR_FALSE, &generatedFrame)) {
|
|
|
|
// Add the generated frame to the child list
|
|
|
|
aFrameItems.AddChild(generatedFrame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*aKidsAllInline = allKidsInline;
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1999-11-30 23:23:50 +00:00
|
|
|
// Helper function that recursively removes content to frame mappings and
|
|
|
|
// undisplayed content mappings.
|
|
|
|
// This differs from DeletingFrameSubtree() because the frames have not yet been
|
1999-11-11 21:08:32 +00:00
|
|
|
// added to the frame hierarchy
|
|
|
|
static void
|
2000-01-22 01:16:50 +00:00
|
|
|
DoCleanupFrameReferences(nsIPresContext* aPresContext,
|
|
|
|
nsIFrameManager* aFrameManager,
|
1999-11-30 23:23:50 +00:00
|
|
|
nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
|
|
aFrame->GetContent(getter_AddRefs(content));
|
|
|
|
|
|
|
|
// Remove the mapping from the content object to its frame
|
|
|
|
aFrameManager->SetPrimaryFrameFor(content, nsnull);
|
|
|
|
aFrameManager->ClearAllUndisplayedContentIn(content);
|
|
|
|
|
|
|
|
// Recursively walk the child frames.
|
|
|
|
// Note: we only need to look at the principal child list
|
|
|
|
nsIFrame* childFrame;
|
2000-01-22 01:16:50 +00:00
|
|
|
aFrame->FirstChild(aPresContext, nsnull, &childFrame);
|
1999-11-30 23:23:50 +00:00
|
|
|
while (childFrame) {
|
2000-01-22 01:16:50 +00:00
|
|
|
DoCleanupFrameReferences(aPresContext, aFrameManager, childFrame);
|
1999-11-30 23:23:50 +00:00
|
|
|
|
|
|
|
// Get the next sibling child frame
|
|
|
|
childFrame->GetNextSibling(&childFrame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Helper function that walks a frame list and calls DoCleanupFrameReference()
|
|
|
|
static void
|
2000-01-22 01:16:50 +00:00
|
|
|
CleanupFrameReferences(nsIPresContext* aPresContext,
|
|
|
|
nsIFrameManager* aFrameManager,
|
1999-11-11 21:08:32 +00:00
|
|
|
nsIFrame* aFrameList)
|
|
|
|
{
|
|
|
|
while (aFrameList) {
|
2000-01-22 01:16:50 +00:00
|
|
|
DoCleanupFrameReferences(aPresContext, aFrameManager, aFrameList);
|
1999-11-11 21:08:32 +00:00
|
|
|
|
|
|
|
// Get the sibling frame
|
|
|
|
aFrameList->GetNextSibling(&aFrameList);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-11-01 15:24:57 +00:00
|
|
|
PRBool
|
|
|
|
nsCSSFrameConstructor::WipeContainingBlock(nsIPresContext* aPresContext,
|
|
|
|
nsFrameConstructorState& aState,
|
|
|
|
nsIContent* aBlockContent,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIFrame* aFrameList)
|
|
|
|
{
|
|
|
|
// Before we go and append the frames, check for a special
|
|
|
|
// situation: an inline frame that will now contain block
|
|
|
|
// frames. This is a no-no and the frame construction logic knows
|
|
|
|
// how to fix this.
|
|
|
|
const nsStyleDisplay* parentDisplay;
|
|
|
|
aFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct *&) parentDisplay);
|
|
|
|
if (NS_STYLE_DISPLAY_INLINE == parentDisplay->mDisplay) {
|
|
|
|
if (!AreAllKidsInline(aFrameList)) {
|
|
|
|
// Ok, reverse tracks: wipe out the frames we just created
|
1999-11-11 21:08:32 +00:00
|
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
|
|
nsCOMPtr<nsIFrameManager> frameManager;
|
|
|
|
|
|
|
|
aPresContext->GetShell(getter_AddRefs(presShell));
|
|
|
|
presShell->GetFrameManager(getter_AddRefs(frameManager));
|
|
|
|
|
|
|
|
// Destroy the frames. As we do make sure any content to frame mappings
|
|
|
|
// or entries in the undisplayed content map are removed
|
2000-01-22 01:16:50 +00:00
|
|
|
CleanupFrameReferences(aPresContext, frameManager, aFrameList);
|
1999-11-01 15:24:57 +00:00
|
|
|
nsFrameList tmp(aFrameList);
|
1999-11-24 06:03:41 +00:00
|
|
|
tmp.DestroyFrames(aPresContext);
|
1999-11-01 15:24:57 +00:00
|
|
|
if (aState.mAbsoluteItems.childList) {
|
2000-01-22 01:16:50 +00:00
|
|
|
CleanupFrameReferences(aPresContext, frameManager, aState.mAbsoluteItems.childList);
|
1999-11-01 15:24:57 +00:00
|
|
|
tmp.SetFrames(aState.mAbsoluteItems.childList);
|
1999-11-24 06:03:41 +00:00
|
|
|
tmp.DestroyFrames(aPresContext);
|
1999-11-01 15:24:57 +00:00
|
|
|
}
|
|
|
|
if (aState.mFixedItems.childList) {
|
2000-01-22 01:16:50 +00:00
|
|
|
CleanupFrameReferences(aPresContext, frameManager, aState.mFixedItems.childList);
|
1999-11-01 15:24:57 +00:00
|
|
|
tmp.SetFrames(aState.mFixedItems.childList);
|
1999-11-24 06:03:41 +00:00
|
|
|
tmp.DestroyFrames(aPresContext);
|
1999-11-01 15:24:57 +00:00
|
|
|
}
|
|
|
|
if (aState.mFloatedItems.childList) {
|
2000-01-22 01:16:50 +00:00
|
|
|
CleanupFrameReferences(aPresContext, frameManager, aState.mFloatedItems.childList);
|
1999-11-01 15:24:57 +00:00
|
|
|
tmp.SetFrames(aState.mFloatedItems.childList);
|
1999-11-24 06:03:41 +00:00
|
|
|
tmp.DestroyFrames(aPresContext);
|
1999-11-01 15:24:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Tell parent of the containing block to reformulate the
|
|
|
|
// entire block. This is painful and definitely not optimal
|
|
|
|
// but it will *always* get the right answer.
|
|
|
|
nsCOMPtr<nsIContent> parentContainer;
|
|
|
|
aBlockContent->GetParent(*getter_AddRefs(parentContainer));
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (gNoisyContentUpdates) {
|
|
|
|
printf("nsCSSFrameConstructor::WipeContainingBlock: aBlockContent=%p parentContainer=%p\n",
|
|
|
|
aBlockContent, parentContainer.get());
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
if (parentContainer) {
|
|
|
|
PRInt32 ix;
|
|
|
|
parentContainer->IndexOf(aBlockContent, ix);
|
|
|
|
ContentReplaced(aPresContext, parentContainer, aBlockContent, aBlockContent, ix);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// XXX uh oh. the block we need to reframe has no parent!
|
|
|
|
}
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::ReframeContainingBlock(nsIPresContext* aPresContext, nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
// Get the parent of the target frame. From there we look for the
|
|
|
|
// containing block in case the target frame is already a block
|
|
|
|
// (which can happen when an inline frame wraps some of its content
|
|
|
|
// in an anonymous block; see ConstructInline)
|
|
|
|
nsIFrame* parentFrame;
|
|
|
|
aFrame->GetParent(&parentFrame);
|
|
|
|
if (!parentFrame) {
|
|
|
|
return RecreateEntireFrameTree(aPresContext);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now find the containing block
|
|
|
|
nsIFrame* containingBlock = GetFloaterContainingBlock(aPresContext, parentFrame);
|
|
|
|
if (!containingBlock) {
|
|
|
|
return RecreateEntireFrameTree(aPresContext);
|
|
|
|
}
|
|
|
|
|
|
|
|
// And get the containingBlock's content
|
|
|
|
nsCOMPtr<nsIContent> blockContent;
|
|
|
|
containingBlock->GetContent(getter_AddRefs(blockContent));
|
|
|
|
if (!blockContent) {
|
|
|
|
return RecreateEntireFrameTree(aPresContext);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now find the containingBlock's content's parent
|
|
|
|
nsCOMPtr<nsIContent> parentContainer;
|
|
|
|
blockContent->GetParent(*getter_AddRefs(parentContainer));
|
|
|
|
if (!parentContainer) {
|
|
|
|
return RecreateEntireFrameTree(aPresContext);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (gNoisyContentUpdates) {
|
|
|
|
printf(" ==> blockContent=%p, parentContainer=%p\n",
|
|
|
|
blockContent.get(), parentContainer.get());
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
PRInt32 ix;
|
|
|
|
parentContainer->IndexOf(blockContent, ix);
|
|
|
|
nsresult rv = ContentReplaced(aPresContext, parentContainer, blockContent, blockContent, ix);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsCSSFrameConstructor::RecreateEntireFrameTree(nsIPresContext* aPresContext)
|
|
|
|
{
|
|
|
|
// XXX write me some day
|
|
|
|
return NS_OK;
|
|
|
|
}
|