/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- * * The contents of this file are subject to the Netscape Public License * Version 1.0 (the "NPL"); you may not use this file except in * compliance with the NPL. You may obtain a copy of the NPL at * http://www.mozilla.org/NPL/ * * Software distributed under the NPL is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL * for the specific language governing rights and limitations under the * NPL. * * The Initial Developer of this code under the NPL is Netscape * Communications Corporation. Portions created by Netscape are * Copyright (C) 1998 Netscape Communications Corporation. All Rights * Reserved. */ #include "nsFrameSetFrame.h" #include "nsHTMLContainer.h" #include "nsLeafFrame.h" #include "nsHTMLContainerFrame.h" #include "nsIWebShell.h" #include "nsIPresContext.h" #include "nsIPresShell.h" #include "nsHTMLIIDs.h" #include "nsRepository.h" #include "nsIStreamListener.h" #include "nsIURL.h" #include "nsIDocument.h" #include "nsIView.h" #include "nsIViewManager.h" #include "nsWidgetsCID.h" #include "nsViewsCID.h" #include "nsHTMLAtoms.h" #include "nsIScrollableView.h" #include "nsStyleCoord.h" #include "nsStyleConsts.h" #include "nsIStyleContext.h" #include "nsCSSLayout.h" #include "nsHTMLBase.h" #include "nsIDocumentLoader.h" #include "nsGenericHTMLElement.h" #include "nsHTMLParts.h" // masks for mEdgeVisibility #define LEFT_VIS 0x0001 #define RIGHT_VIS 0x0002 #define TOP_VIS 0x0004 #define BOTTOM_VIS 0x0008 #define ALL_VIS 0x000F #define NONE_VIS 0x0000 static nsPoint NULL_POINT(-1000000,1000000); static PRInt32 LEFT_EDGE = -1; static PRInt32 RIGHT_EDGE = 1000000; static NS_DEFINE_IID(kIFramesetFrameIID, NS_IFRAMESETFRAME_IID); /******************************************************************************* * nsHTMLFramesetBorderFrame ******************************************************************************/ class nsHTMLFramesetBorderFrame : public nsLeafFrame { public: NS_IMETHOD ListTag(FILE* out = stdout) const; NS_IMETHOD HandleEvent(nsIPresContext& aPresContext, nsGUIEvent* aEvent, nsEventStatus& aEventStatus); NS_IMETHOD Paint(nsIPresContext& aPresContext, nsIRenderingContext& aRenderingContext, const nsRect& aDirtyRect); NS_IMETHOD Reflow(nsIPresContext& aPresContext, nsReflowMetrics& aDesiredSize, const nsReflowState& aReflowState, nsReflowStatus& aStatus); PRBool GetVisibility() { return mVisibility; } void SetVisibility(PRBool aVisibility); void SetColor(nscolor aColor); protected: nsHTMLFramesetBorderFrame(nsIContent* aContent, nsIFrame* aParentFrame, PRInt32 aWidth, PRBool aVertical, PRBool aVisible); virtual ~nsHTMLFramesetBorderFrame(); virtual void GetDesiredSize(nsIPresContext* aPresContext, const nsReflowState& aReflowState, nsReflowMetrics& aDesiredSize); PRInt32 mWidth; PRBool mVertical; PRBool mVisibility; nscolor mColor; // the prev and next neighbors are indexes into the row (for a horizontal border) or col (for // a vertical border) of nsHTMLFramesetFrames or nsHTMLFrames PRInt32 mPrevNeighbor; PRInt32 mNextNeighbor; PRBool mCanResize; friend class nsHTMLFramesetFrame; }; /******************************************************************************* * nsHTMLFramesetBlankFrame ******************************************************************************/ class nsHTMLFramesetBlankFrame : public nsLeafFrame { public: NS_IMETHOD List(FILE* out = stdout, PRInt32 aIndent = 0) const; NS_IMETHOD Paint(nsIPresContext& aPresContext, nsIRenderingContext& aRenderingContext, const nsRect& aDirtyRect); NS_IMETHOD Reflow(nsIPresContext& aPresContext, nsReflowMetrics& aDesiredSize, const nsReflowState& aReflowState, nsReflowStatus& aStatus); protected: nsHTMLFramesetBlankFrame(nsIContent* aContent, nsIFrame* aParentFrame); virtual ~nsHTMLFramesetBlankFrame(); virtual void GetDesiredSize(nsIPresContext* aPresContext, const nsReflowState& aReflowState, nsReflowMetrics& aDesiredSize); friend class nsHTMLFramesetFrame; friend class nsHTMLFrameset; }; /******************************************************************************* * nsHTMLFramesetFrame ******************************************************************************/ PRInt32 nsHTMLFramesetFrame::gMaxNumRowColSpecs = 25; PRBool nsHTMLFramesetFrame::gDragInProgress = PR_FALSE; nsHTMLFramesetFrame::nsHTMLFramesetFrame(nsIContent* aContent, nsIFrame* aParent) : nsHTMLContainerFrame(aContent, aParent) { mNumRows = 0; mRowSpecs = nsnull; mRowSizes = nsnull; mNumCols = 0; mColSpecs = nsnull; mColSizes = nsnull; mEdgeVisibility = 0; mEdgeColors.Set(NO_COLOR); mParentFrameborder = eFrameborder_Yes; // default mParentBorderWidth = -1; // default not set mParentBorderColor = NO_COLOR; // default not set mLastDragPoint.x = mLastDragPoint.y = 0; mMinDrag = 0; } nsHTMLFramesetFrame::~nsHTMLFramesetFrame() { printf("nsFramesetFrame destructor %X \n", this); if (mRowSizes) delete [] mRowSizes; if (mRowSpecs) delete [] mRowSpecs; if (mColSizes) delete [] mColSizes; if (mColSpecs) delete [] mColSpecs; mRowSizes = mColSizes = nsnull; mRowSpecs = mColSpecs = nsnull; } nsresult nsHTMLFramesetFrame::QueryInterface(const nsIID& aIID, void** aInstancePtr) { if (NULL == aInstancePtr) { return NS_ERROR_NULL_POINTER; } else if (aIID.Equals(kIFramesetFrameIID)) { *aInstancePtr = (void*)this; return NS_OK; } return nsHTMLContainerFrame::QueryInterface(aIID, aInstancePtr); } /** * Translate the rows/cols specs into an array of integer sizes for * each cell in the frameset. The sizes are computed by first * rationalizing the spec into a set of percentages that total * 100. Then the cell sizes are computed. */ // XXX This doesn't do a good job of translating "*,625,*" specs void nsHTMLFramesetFrame::CalculateRowCol(nsIPresContext* aPresContext, nscoord aSize, PRInt32 aNumSpecs, nsFramesetSpec* aSpecs, nscoord* aValues) { // Total up the three different type of grid cell // percentages. Transfer the resulting values into result for // temporary storage (we don't overwrite the cell specs) PRInt32 free = 0; PRInt32 percent = 0; PRInt32 pixel = 0; float p2t = aPresContext->GetPixelsToTwips(); int i; // feeble compiler for (i = 0; i < aNumSpecs; i++) { switch (aSpecs[i].mUnit) { case eFramesetUnit_Pixel: // Now that we know the size we are laying out in, turn fixed // pixel dimensions into percents. // XXX maybe use UnitConverter for proper rounding? MMP aValues[i] = NSToCoordRound(100 * p2t * aSpecs[i].mValue / aSize); if (aValues[i] < 1) aValues[i] = 1; pixel += aValues[i]; break; case eFramesetUnit_Percent: aValues[i] = aSpecs[i].mValue; percent += aValues[i]; break; case eFramesetUnit_Free: aValues[i] = aSpecs[i].mValue; free += aValues[i]; break; } } if (percent + pixel < 100) { // The user didn't explicitly use up all the space. Spread the // left over space to the free percentage cells first, then to // the normal percentage cells second, and finally to the fixed // cells as a last resort. if (free != 0) { // We have some free percentage cells that want to soak up the // excess space. Spread out the left over space to those cells. int used = 0; int last = -1; int leftOver = 100 - (percent + pixel); for (int i = 0; i < aNumSpecs; i++) { if (eFramesetUnit_Free == aSpecs[i].mUnit) { aValues[i] = aValues[i] * leftOver / free; if (aValues[i] < 1) aValues[i] = 1; used += aValues[i]; last = i; } } int remainder = 100 - percent - pixel - used; if ((remainder > 0) && (last >= 0)) { // Slop the extra space into the last qualifying cell aValues[last] += remainder; } } else if (percent != 0) { // We have no free cells but have some normal percentage // cells. Distribute the extra space among them. int used = 0; int last = -1; int leftOver = 100 - pixel; for (int i = 0; i < aNumSpecs; i++) { if (eFramesetUnit_Percent == aSpecs[i].mUnit) { aValues[i] = aValues[i] * leftOver / percent; used += aValues[i]; last = i; } } if ((used < leftOver) && (last >= 0)) { // Slop the extra space into the last qualifying cell aValues[last] = aValues[last] + (leftOver - used); } } else { // Give the leftover space to the fixed percentage // cells. Recall that at the start of this method we converted // the fixed pixel values into percentages. int used = 0; for (int i = 0; i < aNumSpecs; i++) { aValues[i] = aValues[i] * 100 / pixel; used += aValues[i]; } if ((used < 100) && (aNumSpecs > 0)) { // Slop the extra space into the last cell aValues[aNumSpecs - 1] += (100 - used); } } } else if (percent + pixel > 100) { // The user overallocated the space. We need to shrink something. // If there is not too much fixed percentage added, we can // just shrink the normal percentage cells to make things fit. if (pixel <= 100) { int used = 0; int last = -1; int val = 100 - pixel; for (int i = 0; i < aNumSpecs; i++) { if (eFramesetUnit_Percent == aSpecs[i].mUnit) { aValues[i] = aValues[i] * val / percent; used += aValues[i]; last = i; } } // Since integer division always truncates, we either made // it fit exactly, or overcompensated and made it too small. if ((used < val) && (last >= 0)) { aValues[last] += (val - used); } } else { // There is too much fixed percentage as well. We will just // shrink all the cells. int used = 0; for (int i = 0; i < aNumSpecs; i++) { if (eFramesetUnit_Free == aSpecs[i].mUnit) { aValues[i] = 0; } else { aValues[i] = aValues[i] * 100 / (percent + pixel); } used += aValues[i]; } if ((used < 100) && (aNumSpecs > 0)) { aValues[aNumSpecs-1] += (100 - used); } } } // Now map the result which contains nothing but percentages into // fixed pixel values. int sum = 0; for (i = 0; i < aNumSpecs; i++) { aValues[i] = (aValues[i] * aSize) / 100; sum += aValues[i]; } //Assert.Assertion(sum <= aSize); if ((sum < aSize) && (aNumSpecs > 0)) { // Any remaining pixels (from roundoff) go into the last cell aValues[aNumSpecs-1] += aSize - sum; } } PRInt32 nsHTMLFramesetFrame::GetBorderWidth(nsIPresContext* aPresContext) { float p2t = aPresContext->GetPixelsToTwips(); nsHTMLValue htmlVal; PRInt32 intVal; nsIHTMLContent* content = nsnull; mContent->QueryInterface(kIHTMLContentIID, (void**)&content); if (nsnull != content) { if (NS_CONTENT_ATTR_HAS_VALUE == (content->GetAttribute(nsHTMLAtoms::border, htmlVal))) { if (eHTMLUnit_Pixel == htmlVal.GetUnit()) { intVal = htmlVal.GetPixelValue(); } else { intVal = htmlVal.GetIntValue(); } if (intVal < 0) { intVal = 0; } NS_RELEASE(content); return NSIntPixelsToTwips(intVal, p2t); } NS_RELEASE(content); } if (mParentBorderWidth >= 0) { return mParentBorderWidth; } return NSIntPixelsToTwips(6, p2t); } PRIntn nsHTMLFramesetFrame::GetSkipSides() const { return 0; } void nsHTMLFramesetFrame::GetDesiredSize(nsIPresContext* aPresContext, const nsReflowState& aReflowState, nsReflowMetrics& aDesiredSize) { nsHTMLFramesetFrame* framesetParent = GetFramesetParent(this); if (nsnull == framesetParent) { nsRect area; aPresContext->GetVisibleArea(area); aDesiredSize.width = area.width; aDesiredSize.height= area.height; } else { nsSize size; framesetParent->GetSizeOfChild(this, size); aDesiredSize.width = size.width; aDesiredSize.height = size.height; } aDesiredSize.ascent = aDesiredSize.height; aDesiredSize.descent = 0; } nsHTMLFramesetFrame* nsHTMLFramesetFrame::GetFramesetParent(nsIFrame* aChild) { nsHTMLFramesetFrame* parent = nsnull; nsIContent* content = nsnull; aChild->GetContent(content); if (nsnull != content) { nsIContent* contentParent = nsnull; content->GetParent(contentParent); if (nsnull != contentParent) { nsIHTMLContent* contentParent2 = nsnull; contentParent->QueryInterface(kIHTMLContentIID, (void**)&contentParent2); if (nsnull != contentParent2) { nsIAtom* tag; contentParent2->GetTag(tag); if (nsHTMLAtoms::frameset == tag) { aChild->GetGeometricParent((nsIFrame*&)parent); } NS_IF_RELEASE(tag); NS_RELEASE(contentParent2); } NS_RELEASE(contentParent); } NS_RELEASE(content); } return parent; } // only valid for non border children void nsHTMLFramesetFrame::GetSizeOfChildAt(PRInt32 aIndexInParent, nsSize& aSize, nsPoint& aCellIndex) { PRInt32 row = aIndexInParent / mNumCols; PRInt32 col = aIndexInParent - (row * mNumCols); // remainder from dividing index by mNumCols if ((row < mNumRows) && (col < mNumCols)) { aSize.width = mColSizes[col]; aSize.height = mRowSizes[row]; aCellIndex.x = col; aCellIndex.y = row; } else { aSize.width = aSize.height = aCellIndex.x = aCellIndex.y = 0; } } // only valid for non border children void nsHTMLFramesetFrame::GetSizeOfChild(nsIFrame* aChild, nsSize& aSize) { // Reflow only creates children frames for and content. // this assumption is used here int i = 0; for (nsIFrame* child = mFirstChild; child; child->GetNextSibling(child)) { if (aChild == child) { nsPoint ignore; GetSizeOfChildAt(i, aSize, ignore); return; } i++; } aSize.width = 0; aSize.height = 0; } NS_METHOD nsHTMLFramesetFrame::HandleEvent(nsIPresContext& aPresContext, nsGUIEvent* aEvent, nsEventStatus& aEventStatus) { nsIView* view; GetView(view); if (view) { nsIViewManager* viewMan; view->GetViewManager(viewMan); if (viewMan) { nsIView* grabber; viewMan->GetMouseEventGrabber(grabber); if (grabber == view) { // the nsFramesetBorderFrame has captured NS_MOUSE_DOWN and changed the cursor to a resize switch (aEvent->message) { case NS_MOUSE_MOVE: MouseDrag(aPresContext, aEvent); break; case NS_MOUSE_LEFT_BUTTON_UP: nsIWidget* window; GetWindow(window); if (window) { window->SetCursor(eCursor_standard); NS_RELEASE(window); } EndMouseDrag(); break; } aEventStatus = nsEventStatus_eConsumeNoDefault; NS_RELEASE(viewMan); return NS_OK; } NS_RELEASE(viewMan); } } // to avoid processing events after destruction of this or children, call super as follows // switch (aEvent->message) { // case NS_MOUSE_MOVE: // case NS_MOUSE_ENTER: // case NS_MOUSE_EXIT: // case NS_MOUSE_LEFT_BUTTON_UP: // case NS_MOUSE_LEFT_BUTTON_DOWN: // aEventStatus = nsEventStatus_eIgnore; if (nsnull == mFirstChild) { // XXX see corresponding hack in nsHTMLContainerFrame::DeleteFrame aEventStatus = nsEventStatus_eConsumeNoDefault; return NS_OK; } else { return nsHTMLContainerFrame::HandleEvent(aPresContext, aEvent, aEventStatus); } } NS_IMETHODIMP nsHTMLFramesetFrame::Paint(nsIPresContext& aPresContext, nsIRenderingContext& aRenderingContext, const nsRect& aDirtyRect) { //printf("frameset paint %X (%d,%d,%d,%d) \n", this, aDirtyRect.x, aDirtyRect.y, aDirtyRect.width, aDirtyRect.height); return nsHTMLContainerFrame::Paint(aPresContext, aRenderingContext, aDirtyRect); } void nsHTMLFramesetFrame::ParseRowCol(nsIAtom* aAttrType, PRInt32& aNumSpecs, nsFramesetSpec** aSpecs) { nsHTMLValue value; nsAutoString rowsCols; nsIHTMLContent* content = nsnull; nsresult result = mContent->QueryInterface(kIHTMLContentIID, (void**)&content); if (nsnull != content) { if (NS_CONTENT_ATTR_HAS_VALUE == content->GetAttribute(aAttrType, value)) { if (eHTMLUnit_String == value.GetUnit()) { value.GetStringValue(rowsCols); nsFramesetSpec* specs = new nsFramesetSpec[gMaxNumRowColSpecs]; aNumSpecs = ParseRowColSpec(rowsCols, gMaxNumRowColSpecs, specs); *aSpecs = new nsFramesetSpec[aNumSpecs]; for (int i = 0; i < aNumSpecs; i++) { (*aSpecs)[i] = specs[i]; } delete [] specs; NS_RELEASE(content); return; } } NS_RELEASE(content); } aNumSpecs = 1; *aSpecs = new nsFramesetSpec[1]; aSpecs[0]->mUnit = eFramesetUnit_Free; aSpecs[0]->mValue = 1; } /** * Translate a "rows" or "cols" spec into an array of nsFramesetSpecs */ PRInt32 nsHTMLFramesetFrame::ParseRowColSpec(nsString& aSpec, PRInt32 aMaxNumValues, nsFramesetSpec* aSpecs) { static const PRUnichar ASTER('*'); static const PRUnichar PERCENT('%'); static const PRUnichar COMMA(','); // Count the commas PRInt32 commaX = aSpec.Find(COMMA); PRInt32 count = 1; while (commaX >= 0) { count++; commaX = aSpec.Find(COMMA, commaX + 1); } if (count > aMaxNumValues) { NS_ASSERTION(0, "Not enough space for values"); count = aMaxNumValues; } // Parse each comma separated token PRInt32 start = 0; PRInt32 specLen = aSpec.Length(); for (PRInt32 i = 0; i < count; i++) { // Find our comma commaX = aSpec.Find(COMMA, start); PRInt32 end = (commaX < 0) ? specLen : commaX; // Note: If end == start then it means that the token has no // data in it other than a terminating comma (or the end of the spec) aSpecs[i].mUnit = eFramesetUnit_Pixel; if (end > start) { PRInt32 numberEnd = end - 1; PRUnichar ch = aSpec.CharAt(numberEnd); if (ASTER == ch) { aSpecs[i].mUnit = eFramesetUnit_Free; numberEnd--; } else if (PERCENT == ch) { aSpecs[i].mUnit = eFramesetUnit_Percent; numberEnd--; } // Translate value to an integer nsString token(""); aSpec.Mid(token, start, 1 + numberEnd - start); //aValues[i] = nsCRT::atoi(token); XXX this is broken, consequently the next 3 lines? char* tokenIso = token.ToNewCString(); aSpecs[i].mValue = atoi(tokenIso); delete [] tokenIso; if (eFramesetUnit_Percent == aSpecs[i].mUnit) { if (aSpecs[i].mValue <= 0) { aSpecs[i].mValue = 100 / count; } } if (aSpecs[i].mValue < 1) aSpecs[i].mValue = 1; start = end + 1; } } return count; } void nsHTMLFramesetFrame::ReflowPlaceChild(nsIFrame* aChild, nsIPresContext& aPresContext, const nsReflowState& aReflowState, nsPoint& aOffset, nsSize& aSize, nsFramesetDrag* aDrag, nsPoint* aCellIndex) { PRBool needFramesetReflow = PR_FALSE; nsFramesetDrag* childDrag = nsnull; if (aDrag && aCellIndex && ChildIsFrameset(aChild)) { nsIContent* childContent; aChild->GetContent(childContent); if (childContent) { nsIAtom* tag; childContent->GetTag(tag); if (nsHTMLAtoms::frameset == tag) { nsHTMLFramesetFrame* childFS = (nsHTMLFramesetFrame*)aChild; if (!aDrag->mConsumed) { PRBool vertical = PR_TRUE; PRInt32 side = RIGHT_EDGE; if (aDrag->mVertical) { if (aDrag->mIndex == aCellIndex->x) { childFS->mColSizes[childFS->mNumCols-1] += aDrag->mChange; } else if (aDrag->mIndex == RIGHT_EDGE) { //aSize.width += aDrag->mChange; childFS->mColSizes[childFS->mNumCols-1] += aDrag->mChange; } else if (aDrag->mIndex+1 == aCellIndex->x) { side = LEFT_EDGE; childFS->mColSizes[0] -= aDrag->mChange; } else if (aDrag->mIndex == LEFT_EDGE) { side = LEFT_EDGE; //aSize.width -= aDrag->mChange; childFS->mColSizes[0] -= aDrag->mChange; //aOffset.x += aDrag->mChange; } } else { vertical = PR_FALSE; if (aDrag->mIndex == aCellIndex->y) { childFS->mRowSizes[childFS->mNumRows-1] += aDrag->mChange; } else if (aDrag->mIndex == RIGHT_EDGE) { //aSize.height += aDrag->mChange; childFS->mRowSizes[childFS->mNumRows-1] += aDrag->mChange; } else if (aDrag->mIndex+1 == aCellIndex->y) { side = LEFT_EDGE; childFS->mRowSizes[0] -= aDrag->mChange; } else if (aDrag->mIndex == LEFT_EDGE) { side = LEFT_EDGE; //aSize.height -= aDrag->mChange; childFS->mRowSizes[0] -= aDrag->mChange; //aOffset.y += aDrag->mChange; } } childDrag = new nsFramesetDrag(vertical, side, aDrag->mChange, aDrag->mSource); } else { childDrag = new nsFramesetDrag(); // consumed drag } needFramesetReflow = PR_TRUE; } NS_IF_RELEASE(tag); NS_RELEASE(childContent); } } nsReflowState reflowState(aChild, aReflowState, aSize); aChild->WillReflow(aPresContext); nsReflowMetrics metrics(nsnull); metrics.width = aSize.width; metrics.height= aSize.height; nsReflowStatus status; if (needFramesetReflow) { status = ((nsHTMLFramesetFrame*)aChild)->Reflow(aPresContext, childDrag, metrics, reflowState, status); } else { status = ReflowChild(aChild, &aPresContext, metrics, reflowState); } NS_ASSERTION(NS_FRAME_IS_COMPLETE(status), "bad status"); if (childDrag) { delete childDrag; } // Place and size the child nsRect rect(aOffset.x, aOffset.y, aSize.width, aSize.height); aChild->SetFrameState(0); aChild->SetRect(rect); aChild->DidReflow(aPresContext, NS_FRAME_REFLOW_FINISHED); // XXX do we need this } nsFrameborder GetFrameBorderHelper(nsIHTMLContent* aContent, PRBool aStandardMode) { if (nsnull != aContent) { nsHTMLValue value; if (NS_CONTENT_ATTR_HAS_VALUE == (aContent->GetAttribute(nsHTMLAtoms::frameborder, value))) { if (eHTMLUnit_Enumerated == value.GetUnit()) { PRInt32 intValue; intValue = value.GetIntValue(); if (!aStandardMode) { if (NS_STYLE_FRAME_YES == intValue) { intValue = NS_STYLE_FRAME_1; } else if (NS_STYLE_FRAME_NO == intValue) { intValue = NS_STYLE_FRAME_0; } } if (NS_STYLE_FRAME_0 == intValue) { return eFrameborder_No; } else if (NS_STYLE_FRAME_1 == intValue) { return eFrameborder_Yes; } } } } return eFrameborder_Notset; } nsFrameborder nsHTMLFramesetFrame::GetFrameBorder(PRBool aStandardMode) { nsFrameborder result = eFrameborder_Notset; nsIHTMLContent* content = nsnull; mContent->QueryInterface(kIHTMLContentIID, (void**) &content); if (nsnull != content) { result = GetFrameBorderHelper(content, aStandardMode); NS_RELEASE(content); } // XXX if we get here, check for nsIDOMFRAMEElement interface if (eFrameborder_Notset == result) { return mParentFrameborder; } return result; } nsFrameborder nsHTMLFramesetFrame::GetFrameBorder(nsIContent* aContent, PRBool aStandardMode) { nsFrameborder result = eFrameborder_Notset; nsIHTMLContent* content = nsnull; aContent->QueryInterface(kIHTMLContentIID, (void**) &content); if (nsnull != content) { result = GetFrameBorderHelper(content, aStandardMode); NS_RELEASE(content); } // XXX if we get here, check for nsIDOMFRAMEElement interface if (eFrameborder_Notset == result) { return GetFrameBorder(aStandardMode); } return result; } nscolor nsHTMLFramesetFrame::GetBorderColor() { nscolor result = NO_COLOR; nsIHTMLContent* content = nsnull; mContent->QueryInterface(kIHTMLContentIID, (void**)&content); if (nsnull != content) { nsHTMLValue value; if (NS_CONTENT_ATTR_HAS_VALUE == (content->GetAttribute(nsHTMLAtoms::bordercolor, value))) { if (value.GetUnit() == eHTMLUnit_Color) { result = value.GetColorValue(); } else if (value.GetUnit() == eHTMLUnit_String) { nsAutoString buffer; value.GetStringValue(buffer); char cbuf[40]; buffer.ToCString(cbuf, sizeof(cbuf)); NS_ColorNameToRGB(cbuf, &result); } } NS_RELEASE(content); } if (NO_COLOR == result) { return mParentBorderColor; } return result; } nscolor nsHTMLFramesetFrame::GetBorderColor(nsIContent* aContent) { nscolor result = NO_COLOR; nsIHTMLContent* content = nsnull; aContent->QueryInterface(kIHTMLContentIID, (void**)&content); if (nsnull != content) { nsHTMLValue value; if (NS_CONTENT_ATTR_HAS_VALUE == (content->GetAttribute(nsHTMLAtoms::bordercolor, value))) { if (value.GetUnit() == eHTMLUnit_Color) { result = value.GetColorValue(); } else if (value.GetUnit() == eHTMLUnit_String) { nsAutoString buffer; value.GetStringValue(buffer); char cbuf[40]; buffer.ToCString(cbuf, sizeof(cbuf)); NS_ColorNameToRGB(cbuf, &result); } } NS_RELEASE(content); } if (NO_COLOR == result) { return GetBorderColor(); } return result; } #define FRAMESET 0 #define FRAME 1 #define BLANK 2 NS_IMETHODIMP nsHTMLFramesetFrame::Reflow(nsIPresContext& aPresContext, nsReflowMetrics& aDesiredSize, const nsReflowState& aReflowState, nsReflowStatus& aStatus) { //printf("FramesetFrame::Reflow %X (%d,%d) \n", this, aReflowState.maxSize.width, aReflowState.maxSize.height); return Reflow(aPresContext, nsnull, aDesiredSize, aReflowState, aStatus); } static NS_DEFINE_IID(kViewCID, NS_VIEW_CID); static NS_DEFINE_IID(kIViewIID, NS_IVIEW_IID); NS_IMETHODIMP nsHTMLFramesetFrame::Reflow(nsIPresContext& aPresContext, nsFramesetDrag* aDrag, nsReflowMetrics& aDesiredSize, const nsReflowState& aReflowState, nsReflowStatus& aStatus) { //printf("FramesetFrame2::Reflow %X (%d,%d) \n", this, aReflowState.maxSize.width, aReflowState.maxSize.height); // Always get the size so that the caller knows how big we are GetDesiredSize(&aPresContext, aReflowState, aDesiredSize); PRBool firstTime = (0 == mNumRows); if (firstTime) { // create the view. a view is needed since it needs to be a mouse grabber // make sure the style context is set if (nsnull == mStyleContext) { GetStyleContext(&aPresContext, mStyleContext); } nsIView* view; nsresult result = nsRepository::CreateInstance(kViewCID, nsnull, kIViewIID, (void **)&view); nsIPresShell *presShell = aPresContext.GetShell(); nsIViewManager *viewMan = presShell->GetViewManager(); NS_RELEASE(presShell); nsIFrame* parWithView; nsIView *parView; GetParentWithView(parWithView); parWithView->GetView(parView); nsRect boundBox(0, 0, aDesiredSize.width, aDesiredSize.height); result = view->Init(viewMan, boundBox, parView, nsnull); viewMan->InsertChild(parView, view, 0); SetView(view); NS_RELEASE(viewMan); // parse the rows= cols= data ParseRowCol(nsHTMLAtoms::rows, mNumRows, &mRowSpecs); ParseRowCol(nsHTMLAtoms::cols, mNumCols, &mColSpecs); mRowSizes = new nscoord[mNumRows]; mColSizes = new nscoord[mNumCols]; } PRInt32 numCells = mNumRows*mNumCols; nscoord width = (aDesiredSize.width <= aReflowState.maxSize.width) ? aDesiredSize.width : aReflowState.maxSize.width; nscoord height = (aDesiredSize.height <= aReflowState.maxSize.height) ? aDesiredSize.height : aReflowState.maxSize.height; // subtract out the width of all of the potential borders. There are // only borders between s. There are none on the edges (e.g the // leftmost has no left border). PRInt32 borderWidth = GetBorderWidth(&aPresContext); width -= (mNumCols - 1) * borderWidth; height -= (mNumRows - 1) * borderWidth; if (!aDrag && ( (firstTime) || ( (mRect.width != 0) && (mRect.height != 0) && (aDesiredSize.width != 0) && (aDesiredSize.height != 0) && ((aDesiredSize.width != mRect.width) || (aDesiredSize.height != mRect.height))) ) ) { CalculateRowCol(&aPresContext, width, mNumCols, mColSpecs, mColSizes); CalculateRowCol(&aPresContext, height, mNumRows, mRowSpecs, mRowSizes); } PRBool* verBordersVis; // vertical borders visibility nsHTMLFramesetBorderFrame** verBorders; // vertical borders nscolor* verBorderColors; PRBool* horBordersVis; // horizontal borders visibility nsHTMLFramesetBorderFrame** horBorders; // horizontal borders nscolor* horBorderColors; PRInt32* childTypes; // frameset/frame distinction of children nsFrameborder* childFrameborder; // the frameborder attr of children nsFrameborder frameborder; nsBorderColor* childBorderColors; nscolor borderColor; if (firstTime) { verBorders = new nsHTMLFramesetBorderFrame*[mNumCols]; // 1 more than number of ver borders verBordersVis = new PRBool[mNumCols]; verBorderColors = new nscolor[mNumCols]; for (int verX = 0; verX < mNumCols; verX++) { verBorders[verX] = nsnull; verBordersVis[verX] = PR_FALSE; verBorderColors[verX] = NO_COLOR; } horBorders = new nsHTMLFramesetBorderFrame*[mNumRows]; // 1 more than number of hor borders horBordersVis = new PRBool[mNumRows]; horBorderColors = new nscolor[mNumRows]; for (int horX = 0; horX < mNumRows; horX++) { horBorders[horX] = nsnull; horBordersVis[horX] = PR_FALSE; horBorderColors[horX] = NO_COLOR; } childTypes = new PRInt32[numCells]; childFrameborder = new nsFrameborder[numCells]; frameborder = GetFrameBorder(PR_FALSE); childBorderColors = new nsBorderColor[numCells]; borderColor = GetBorderColor(); } // create the children frames; skip content which isn't or nsIFrame* lastChild = nsnull; if (firstTime) { mChildCount = 0; if (nsnull != mContent) { nsIFrame* frame; PRInt32 numChildren; mContent->ChildCount(numChildren); for (int childX = 0; childX < numChildren; childX++) { nsIContent* childCon; mContent->ChildAt(childX, childCon); nsIHTMLContent* child = nsnull; nsresult result = childCon->QueryInterface(kIHTMLContentIID, (void**)&child); if (nsnull == child) { continue; } nsIAtom* tag; child->GetTag(tag); if ((nsHTMLAtoms::frameset == tag) || (nsHTMLAtoms::frame == tag)) { nsIStyleContext* kidSC = aPresContext.ResolveStyleContextFor(child, this); nsresult result; if (nsHTMLAtoms::frameset == tag) { result = NS_NewHTMLFramesetFrame(child, this, frame); childTypes[mChildCount] = FRAMESET; nsHTMLFramesetFrame* childFrame = (nsHTMLFramesetFrame*)frame; childFrame->SetParentFrameborder(frameborder); childFrame->SetParentBorderWidth(borderWidth); childFrame->SetParentBorderColor(borderColor); childBorderColors[mChildCount].Set(childFrame->GetBorderColor()); } else { // frame result = NS_NewHTMLFrameOuterFrame(child, this, frame); childTypes[mChildCount] = FRAME; // childFrameborder[mChildCount] = GetFrameBorder(child, PR_FALSE); childBorderColors[mChildCount].Set(GetBorderColor(child)); } if (NS_OK != result) { NS_RELEASE(tag); NS_RELEASE(child); NS_RELEASE(kidSC); return result; } // Set the style context frame->SetStyleContext(&aPresContext, kidSC); NS_RELEASE(kidSC); if (nsnull == lastChild) { mFirstChild = frame; } else { lastChild->SetNextSibling(frame); } lastChild = frame; mChildCount++; } NS_RELEASE(child); NS_IF_RELEASE(tag); } mNonBlankChildCount = mChildCount; // add blank frames for frameset cells that had no content provided for (int blankX = mChildCount; blankX < numCells; blankX++) { // XXX the blank frame is using the content of its parent - at some point it should just have null content nsHTMLFramesetBlankFrame* blankFrame = new nsHTMLFramesetBlankFrame(mContent, this); //GetStyleContext(&aPresContext, blankFrame->mStyleContext); // set the blank frame's style context if (nsnull == lastChild) { mFirstChild = blankFrame; } else { lastChild->SetNextSibling(blankFrame); } lastChild = blankFrame; childTypes[mChildCount] = BLANK; childBorderColors[mChildCount].Set(NO_COLOR); mChildCount++; } mNonBorderChildCount = mChildCount; } } // reflow the children PRInt32 lastRow = 0; PRInt32 lastCol = 0; PRInt32 borderChildX = mNonBorderChildCount; // index of border children nsHTMLFramesetBorderFrame* borderFrame = nsnull; nsPoint offset(0,0); nsSize size, lastSize; nsIFrame* child = mFirstChild; for (PRInt32 childX = 0; childX < mNonBorderChildCount; childX++) { nsPoint cellIndex; GetSizeOfChildAt(childX, size, cellIndex); if (lastRow != cellIndex.y) { // changed to next row offset.x = 0; offset.y += lastSize.height; if (borderWidth > 0) { if (firstTime) { // create horizontal border borderFrame = new nsHTMLFramesetBorderFrame(mContent, this, borderWidth, PR_FALSE, PR_FALSE); mChildCount++; lastChild->SetNextSibling(borderFrame); lastChild = borderFrame; horBorders[cellIndex.y-1] = borderFrame; // set the neighbors for determining drag boundaries borderFrame->mPrevNeighbor = lastRow; borderFrame->mNextNeighbor = cellIndex.y; } else { borderFrame = (nsHTMLFramesetBorderFrame*)FrameAt(mFirstChild, borderChildX); borderChildX++; } nsSize borderSize(aDesiredSize.width, borderWidth); ReflowPlaceChild(borderFrame, aPresContext, aReflowState, offset, borderSize); offset.y += borderWidth; } } else { if ((cellIndex.x > 0) && (borderWidth > 0)) { // moved to next col in same row if (0 == cellIndex.y) { // in 1st row if (firstTime) { // create vertical border borderFrame = new nsHTMLFramesetBorderFrame(mContent, this, borderWidth, PR_TRUE, PR_FALSE); mChildCount++; lastChild->SetNextSibling(borderFrame); lastChild = borderFrame; verBorders[cellIndex.x-1] = borderFrame; // set the neighbors for determining drag boundaries borderFrame->mPrevNeighbor = lastCol; borderFrame->mNextNeighbor = cellIndex.x; } else { borderFrame = (nsHTMLFramesetBorderFrame*)FrameAt(mFirstChild, borderChildX); borderChildX++; } nsSize borderSize(borderWidth, aDesiredSize.height); ReflowPlaceChild(borderFrame, aPresContext, aReflowState, offset, borderSize); } offset.x += borderWidth; } } ReflowPlaceChild(child, aPresContext, aReflowState, offset, size, aDrag, &cellIndex); if (firstTime) { PRInt32 childVis; if (FRAMESET == childTypes[childX]) { nsHTMLFramesetFrame* childFS = (nsHTMLFramesetFrame*)child; childVis = childFS->mEdgeVisibility; childBorderColors[childX] = childFS->mEdgeColors; } else if (FRAME == childTypes[childX]) { if (eFrameborder_Yes == childFrameborder[childX]) { childVis = ALL_VIS; } else if (eFrameborder_No == childFrameborder[childX]) { childVis = NONE_VIS; } else { // notset childVis = (eFrameborder_No == frameborder) ? NONE_VIS : ALL_VIS; } } else { // blank childVis = NONE_VIS; } nsBorderColor childColors = childBorderColors[childX]; // set the visibility, color of our edge borders based on children if (0 == cellIndex.x) { if (!(mEdgeVisibility & LEFT_VIS)) { mEdgeVisibility |= (LEFT_VIS & childVis); } if (NO_COLOR == mEdgeColors.mLeft) { mEdgeColors.mLeft = childColors.mLeft; } } if (0 == cellIndex.y) { if (!(mEdgeVisibility & TOP_VIS)) { mEdgeVisibility |= (TOP_VIS & childVis); } if (NO_COLOR == mEdgeColors.mTop) { mEdgeColors.mTop = childColors.mTop; } } if (mNumCols-1 == cellIndex.x) { if (!(mEdgeVisibility & RIGHT_VIS)) { mEdgeVisibility |= (RIGHT_VIS & childVis); } if (NO_COLOR == mEdgeColors.mRight) { mEdgeColors.mRight = childColors.mRight; } } if (mNumRows-1 == cellIndex.y) { if (!(mEdgeVisibility & BOTTOM_VIS)) { mEdgeVisibility |= (BOTTOM_VIS & childVis); } if (NO_COLOR == mEdgeColors.mBottom) { mEdgeColors.mBottom = childColors.mBottom; } } // set the visibility of borders that the child may affect if (childVis & RIGHT_VIS) { verBordersVis[cellIndex.x] = PR_TRUE; } if (childVis & BOTTOM_VIS) { horBordersVis[cellIndex.y] = PR_TRUE; } if ((cellIndex.x > 0) && (childVis & LEFT_VIS)) { verBordersVis[cellIndex.x-1] = PR_TRUE; } if ((cellIndex.y > 0) && (childVis & TOP_VIS)) { horBordersVis[cellIndex.y-1] = PR_TRUE; } // set the colors of borders that the child may affect if (NO_COLOR == verBorderColors[cellIndex.x]) { verBorderColors[cellIndex.x] = childBorderColors[childX].mRight; } if (NO_COLOR == horBorderColors[cellIndex.y]) { horBorderColors[cellIndex.y] = childBorderColors[childX].mBottom; } if ((cellIndex.x > 0) && (NO_COLOR == verBorderColors[cellIndex.x-1])) { verBorderColors[cellIndex.x-1] = childBorderColors[childX].mLeft; } if ((cellIndex.y > 0) && (NO_COLOR == horBorderColors[cellIndex.y-1])) { horBorderColors[cellIndex.y-1] = childBorderColors[childX].mTop; } } lastRow = cellIndex.y; lastCol = cellIndex.x; lastSize = size; offset.x += size.width; child->GetNextSibling(child); } if (firstTime) { nscolor childColor; // set the visibility, color, mouse sensitivity of borders for (int verX = 0; verX < mNumCols-1; verX++) { if (verBorders[verX]) { verBorders[verX]->SetVisibility(verBordersVis[verX]); SetBorderResize(childTypes, verBorders[verX]); childColor = (NO_COLOR == verBorderColors[verX]) ? borderColor : verBorderColors[verX]; verBorders[verX]->SetColor(childColor); } } for (int horX = 0; horX < mNumRows-1; horX++) { if (horBorders[horX]) { horBorders[horX]->SetVisibility(horBordersVis[horX]); SetBorderResize(childTypes, horBorders[horX]); childColor = (NO_COLOR == horBorderColors[horX]) ? borderColor : horBorderColors[horX]; horBorders[horX]->SetColor(childColor); } } delete[] verBordersVis; delete[] verBorders; delete[] verBorderColors; delete[] horBordersVis; delete[] horBorders; delete[] horBorderColors; delete[] childTypes; delete[] childFrameborder; delete[] childBorderColors; } if (nsnull != aDesiredSize.maxElementSize) { aDesiredSize.maxElementSize->width = aDesiredSize.width; aDesiredSize.maxElementSize->height = aDesiredSize.height; } aStatus = NS_FRAME_COMPLETE; return NS_OK; } PRBool nsHTMLFramesetFrame::ChildIsFrameset(nsIFrame* aChild) { nsIFrame* childFrame = nsnull; nsresult result = aChild->QueryInterface(kIFramesetFrameIID, (void**)&childFrame); if (childFrame) { return PR_TRUE; } return PR_FALSE; } PRBool nsHTMLFramesetFrame::CanResize(PRBool aVertical, PRBool aLeft) { nsIFrame* child; PRInt32 childX; PRInt32 startX; if (aVertical) { startX = (aLeft) ? 0 : mNumCols-1; for (childX = startX; childX < mNonBorderChildCount; childX += mNumCols) { child = FrameAt(mFirstChild, childX); if (!CanChildResize(aVertical, aLeft, childX, ChildIsFrameset(child))) { return PR_FALSE; } } } else { startX = (aLeft) ? 0 : (mNumRows - 1) * mNumCols; PRInt32 endX = startX + mNumCols; for (childX = startX; childX < endX; childX++) { child = FrameAt(mFirstChild, childX); if (!CanChildResize(aVertical, aLeft, childX, ChildIsFrameset(child))) { return PR_FALSE; } } } return PR_TRUE; } PRBool nsHTMLFramesetFrame::GetNoResize(nsIFrame* aChildFrame) { PRBool result = PR_FALSE; nsIContent* content = nsnull; aChildFrame->GetContent(content); if (nsnull != content) { nsIHTMLContent* htmlContent = nsnull; content->QueryInterface(kIHTMLContentIID, (void**)&htmlContent); if (nsnull != htmlContent) { nsHTMLValue value; if (NS_CONTENT_ATTR_HAS_VALUE == htmlContent->GetAttribute(nsHTMLAtoms::noresize, value)) { result = PR_TRUE; } NS_RELEASE(htmlContent); } NS_RELEASE(content); } return result; } PRBool nsHTMLFramesetFrame::CanChildResize(PRBool aVertical, PRBool aLeft, PRInt32 aChildX, PRBool aFrameset) { nsIFrame* child = FrameAt(mFirstChild, aChildX); if (aFrameset) { return ((nsHTMLFramesetFrame*)child)->CanResize(aVertical, aLeft); } else { return !GetNoResize(child); } } void nsHTMLFramesetFrame::SetBorderResize(PRInt32* aChildTypes, nsHTMLFramesetBorderFrame* aBorderFrame) { if (aBorderFrame->mVertical) { for (int rowX = 0; rowX < mNumRows; rowX++) { PRInt32 childX = aBorderFrame->mPrevNeighbor + (rowX * mNumCols); if (!CanChildResize(PR_TRUE, PR_FALSE, childX, (FRAMESET == aChildTypes[childX])) || !CanChildResize(PR_TRUE, PR_TRUE, childX+1,(FRAMESET == aChildTypes[childX]))) { aBorderFrame->mCanResize = PR_FALSE; } } } else { PRInt32 childX = aBorderFrame->mPrevNeighbor * mNumCols; PRInt32 endX = childX + mNumCols; for (; childX < endX; childX++) { if (!CanChildResize(PR_FALSE, PR_FALSE, childX, (FRAMESET == aChildTypes[childX]))) { aBorderFrame->mCanResize = PR_FALSE; } } endX = endX + mNumCols; for (; childX < endX; childX++) { if (!CanChildResize(PR_FALSE, PR_TRUE, childX, (FRAMESET == aChildTypes[childX]))) { aBorderFrame->mCanResize = PR_FALSE; } } } } NS_IMETHODIMP nsHTMLFramesetFrame::VerifyTree() const { // XXX Completely disabled for now; once pseud-frames are reworked // then we can turn it back on. return NS_OK; } void nsHTMLFramesetFrame::StartMouseDrag(nsIPresContext& aPresContext, nsHTMLFramesetBorderFrame* aBorder, nsGUIEvent* aEvent) { if (mMinDrag == 0) { float p2t = aPresContext.GetPixelsToTwips(); mMinDrag = NSIntPixelsToTwips(2, p2t); // set min drag and min frame size to 2 pixels } #if 0 PRInt32 index; IndexOf(aBorder, index); NS_ASSERTION((nsnull != aBorder) && (index >= 0), "invalid dragger"); #endif nsIView* view; GetView(view); if (view) { nsIViewManager* viewMan; view->GetViewManager(viewMan); if (viewMan) { PRBool ignore; viewMan->GrabMouseEvents(view, ignore); NS_RELEASE(viewMan); mDragger = aBorder; mLastDragPoint = NULL_POINT; // can't set it to this event's point - it is not in framesetframe coords gDragInProgress = PR_TRUE; } } } void nsHTMLFramesetFrame::MouseDrag(nsIPresContext& aPresContext, nsGUIEvent* aEvent) { if (NULL_POINT == mLastDragPoint) { mLastDragPoint.x = aEvent->point.x; mLastDragPoint.y = aEvent->point.y; return; } PRInt32 change; // measured positive from left-to-right or top-to-bottom if (mDragger->mVertical) { change = aEvent->point.x - mLastDragPoint.x; if (change < 0) { if (mMinDrag >= -change) { return; } PRInt32 index = mDragger->mPrevNeighbor; if (mColSizes[index] + change < mMinDrag) { change = mMinDrag - mColSizes[index]; } } else if (change > 0) { if (mMinDrag >= change) { return; } PRInt32 index = mDragger->mNextNeighbor; if (mColSizes[index] - change < mMinDrag) { change = mColSizes[index] - mMinDrag; } } mColSizes[mDragger->mPrevNeighbor] += change; mColSizes[mDragger->mNextNeighbor] -= change; } else { change = aEvent->point.y - mLastDragPoint.y; if (change < 0) { if (mMinDrag >= -change) { return; } PRInt32 index = mDragger->mPrevNeighbor; if (mRowSizes[index] + change < mMinDrag) { change = mMinDrag - mRowSizes[index]; } } else if (change > 0) { if (mMinDrag >= change) { return; } PRInt32 index = mDragger->mNextNeighbor; if (mRowSizes[index] - change < mMinDrag) { change = mRowSizes[index] - mMinDrag; } } mRowSizes[mDragger->mPrevNeighbor] += change; mRowSizes[mDragger->mNextNeighbor] -= change; } if (change != 0) { nsReflowMetrics metrics(nsnull); nsSize size; GetSize(size); nsReflowState state(this, eReflowReason_Initial, size); state.reason = eReflowReason_Incremental; nsReflowStatus status; nsDidReflowStatus didStatus; WillReflow(aPresContext); nsFramesetDrag drag(mDragger->mVertical, mDragger->mPrevNeighbor, change, this); Reflow(aPresContext, &drag, metrics, state, status); DidReflow(aPresContext, didStatus); } mLastDragPoint.x = aEvent->point.x; mLastDragPoint.y = aEvent->point.y; } void nsHTMLFramesetFrame::EndMouseDrag() { nsIView* view; GetView(view); if (view) { nsIViewManager* viewMan; view->GetViewManager(viewMan); if (viewMan) { mDragger = nsnull; PRBool ignore; viewMan->GrabMouseEvents(nsnull, ignore); NS_RELEASE(viewMan); } } gDragInProgress = PR_FALSE; } nsresult NS_NewHTMLFramesetFrame(nsIContent* aContent, nsIFrame* aParentFrame, nsIFrame*& aResult) { nsIFrame* frame = new nsHTMLFramesetFrame(aContent, aParentFrame); if (nsnull == frame) { return NS_ERROR_OUT_OF_MEMORY; } aResult = frame; return NS_OK; } /******************************************************************************* * nsHTMLFramesetBorderFrame ******************************************************************************/ nsHTMLFramesetBorderFrame::nsHTMLFramesetBorderFrame(nsIContent* aContent, nsIFrame* aParent, PRInt32 aWidth, PRBool aVertical, PRBool aVisibility) : nsLeafFrame(aContent, aParent), mWidth(aWidth), mVertical(aVertical), mVisibility(aVisibility) { mCanResize = PR_TRUE; mColor = NO_COLOR; } nsHTMLFramesetBorderFrame::~nsHTMLFramesetBorderFrame() { printf("nsHTMLFramesetBorderFrame destructor %X \n", this); } void nsHTMLFramesetBorderFrame::GetDesiredSize(nsIPresContext* aPresContext, const nsReflowState& aReflowState, nsReflowMetrics& aDesiredSize) { aDesiredSize.width = aReflowState.maxSize.width; aDesiredSize.height = aReflowState.maxSize.height; aDesiredSize.ascent = aDesiredSize.width; aDesiredSize.descent = 0; } void nsHTMLFramesetBorderFrame::SetVisibility(PRBool aVisibility) { mVisibility = aVisibility; } void nsHTMLFramesetBorderFrame::SetColor(nscolor aColor) { mColor = aColor; } NS_IMETHODIMP nsHTMLFramesetBorderFrame::Reflow(nsIPresContext& aPresContext, nsReflowMetrics& aDesiredSize, const nsReflowState& aReflowState, nsReflowStatus& aStatus) { //printf("BorderFrame::Reflow %X (%d,%d) \n", this, aReflowState.maxSize.width, aReflowState.maxSize.height); GetDesiredSize(&aPresContext, aReflowState, aDesiredSize); aStatus = NS_FRAME_COMPLETE; return NS_OK; } NS_METHOD nsHTMLFramesetBorderFrame::Paint(nsIPresContext& aPresContext, nsIRenderingContext& aRenderingContext, const nsRect& aDirtyRect) { //printf("border frame paint %X (%d,%d,%d,%d) \n", this, aDirtyRect.x, aDirtyRect.y, aDirtyRect.width, aDirtyRect.height); static nscolor GREY = NS_RGB(200, 200, 200); static nscolor WHITE = NS_RGB(255, 255, 255); static nscolor BLACK = NS_RGB(0, 0, 0); static nscolor DARK_GREY = NS_RGB(130, 130, 130); nscoord widthInPixels = NSTwipsToIntPixels(mWidth, aPresContext.GetTwipsToPixels()); nscoord pixelWidth = NSIntPixelsToTwips(1, aPresContext.GetPixelsToTwips()); if (widthInPixels <= 0) { return NS_OK; } nscoord x0 = 0; nscoord y0 = 0; nscoord x1 = (mVertical) ? x0 : mRect.width; nscoord y1 = (mVertical) ? mRect.height : x0; nscolor color = WHITE; if (mVisibility) { color = (NO_COLOR == mColor) ? GREY : mColor; } aRenderingContext.SetColor(color); // draw grey or white first for (int i = 0; i < widthInPixels; i++) { aRenderingContext.DrawLine (x0, y0, x1, y1); if (mVertical) { x0 += pixelWidth; x1 = x0; } else { y0 += pixelWidth; y1 = y0; } } if (!mVisibility) { return NS_OK; } if (widthInPixels >= 5) { aRenderingContext.SetColor(WHITE); x0 = (mVertical) ? pixelWidth : 0; y0 = (mVertical) ? 0 : pixelWidth; x1 = (mVertical) ? x0 : mRect.width; y1 = (mVertical) ? mRect.height : y0; aRenderingContext.DrawLine(x0, y0, x1, y1); } if (widthInPixels >= 2) { aRenderingContext.SetColor(DARK_GREY); x0 = (mVertical) ? mRect.width - (2 * pixelWidth) : 0; y0 = (mVertical) ? 0 : mRect.height - (2 * pixelWidth); x1 = (mVertical) ? x0 : mRect.width; y1 = (mVertical) ? mRect.height : y0; aRenderingContext.DrawLine(x0, y0, x1, y1); } if (widthInPixels >= 1) { aRenderingContext.SetColor(BLACK); x0 = (mVertical) ? mRect.width - pixelWidth : 0; y0 = (mVertical) ? 0 : mRect.height - pixelWidth; x1 = (mVertical) ? x0 : mRect.width; y1 = (mVertical) ? mRect.height : y0; aRenderingContext.DrawLine(x0, y0, x1, y1); } return NS_OK; } NS_METHOD nsHTMLFramesetBorderFrame::HandleEvent(nsIPresContext& aPresContext, nsGUIEvent* aEvent, nsEventStatus& aEventStatus) { if (!mCanResize) { aEventStatus = nsEventStatus_eIgnore; return NS_OK; } aEventStatus = nsEventStatus_eConsumeNoDefault; nsIWidget* window; nsCursor cursor; switch (aEvent->message) { case NS_MOUSE_ENTER: case NS_MOUSE_MOVE: GetWindow(window); if (window) { cursor = (mVertical) ? eCursor_sizeWE : eCursor_sizeNS; window->SetCursor(eCursor_standard); // XXX why is this necessary window->SetCursor(cursor); NS_RELEASE(window); } break; case NS_MOUSE_EXIT: GetWindow(window); if (window) { cursor = (mVertical) ? eCursor_sizeWE : eCursor_sizeNS; window->SetCursor(cursor); // XXX why is this necessary window->SetCursor(eCursor_standard); NS_RELEASE(window); } break; case NS_MOUSE_LEFT_BUTTON_DOWN: nsHTMLFramesetFrame* parentFrame = nsnull; GetGeometricParent((nsIFrame*&)parentFrame); parentFrame->StartMouseDrag(aPresContext, this, aEvent); break; } return NS_OK; } NS_IMETHODIMP nsHTMLFramesetBorderFrame::ListTag(FILE* out) const { nsLeafFrame::ListTag(out); fputs(" (BORDER)", out); return NS_OK; } /******************************************************************************* * nsHTMLFramesetBlankFrame ******************************************************************************/ nsHTMLFramesetBlankFrame::nsHTMLFramesetBlankFrame(nsIContent* aContent, nsIFrame* aParent) : nsLeafFrame(aContent, aParent) { } nsHTMLFramesetBlankFrame::~nsHTMLFramesetBlankFrame() { printf("nsHTMLFramesetBlankFrame destructor %X \n", this); } void nsHTMLFramesetBlankFrame::GetDesiredSize(nsIPresContext* aPresContext, const nsReflowState& aReflowState, nsReflowMetrics& aDesiredSize) { aDesiredSize.width = aReflowState.maxSize.width; aDesiredSize.height = aReflowState.maxSize.height; aDesiredSize.ascent = aDesiredSize.width; aDesiredSize.descent = 0; } NS_IMETHODIMP nsHTMLFramesetBlankFrame::Reflow(nsIPresContext& aPresContext, nsReflowMetrics& aDesiredSize, const nsReflowState& aReflowState, nsReflowStatus& aStatus) { GetDesiredSize(&aPresContext, aReflowState, aDesiredSize); aStatus = NS_FRAME_COMPLETE; return NS_OK; } NS_METHOD nsHTMLFramesetBlankFrame::Paint(nsIPresContext& aPresContext, nsIRenderingContext& aRenderingContext, const nsRect& aDirtyRect) { nscolor white = NS_RGB(255,255,255); aRenderingContext.SetColor (white); // XXX FillRect doesn't seem to work //aRenderingContext.FillRect (mRect); float p2t = aPresContext.GetPixelsToTwips(); nscoord x0 = 0; nscoord y0 = 0; nscoord x1 = x0; nscoord y1 = mRect.height; nscoord pixel = NSIntPixelsToTwips(1, p2t); aRenderingContext.SetColor(white); for (int i = 0; i < mRect.width; i += pixel) { aRenderingContext.DrawLine (x0, y0, x1, y1); x0 += NSIntPixelsToTwips(1, p2t); x1 = x0; } return NS_OK; } NS_IMETHODIMP nsHTMLFramesetBlankFrame::List(FILE* out, PRInt32 aIndent) const { for (PRInt32 i = aIndent; --i >= 0; ) fputs(" ", out); // Indent fprintf(out, "%X BLANK \n", this); return nsLeafFrame::List(out, aIndent); }