/* -*- 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.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/ * * 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. * * The Original Code is mozilla.org code. * * The Initial Developer of the Original Code is Netscape * Communications Corporation. Portions created by Netscape are * Copyright (C) 1998 Netscape Communications Corporation. All * Rights Reserved. * * Contributor(s): */ #include "nsInlineFrame.h" #include "nsIPresShell.h" #include "nsFormControlFrame.h" #include "nsIDOMHTMLAnchorElement.h" typedef nsInlineFrame nsLabelFrameSuper; class nsLabelFrame : public nsLabelFrameSuper { public: nsLabelFrame(); virtual ~nsLabelFrame(); NS_IMETHOD Destroy(nsIPresContext *aPresContext); NS_IMETHOD Reflow(nsIPresContext* aPresContext, nsHTMLReflowMetrics& aDesiredSize, const nsHTMLReflowState& aReflowState, nsReflowStatus& aStatus); NS_IMETHOD GetFrameForPoint(nsIPresContext* aPresContext, const nsPoint& aPoint, nsFramePaintLayer aWhichLayer, nsIFrame** aFrame); #ifdef DEBUG NS_IMETHOD GetFrameName(nsString& aResult) const { return MakeFrameName("Label", aResult); } #endif }; nsresult NS_NewLabelFrame(nsIPresShell* aPresShell, nsIFrame** aNewFrame, PRUint32 aStateFlags) { NS_PRECONDITION(aNewFrame, "null OUT ptr"); if (nsnull == aNewFrame) { return NS_ERROR_NULL_POINTER; } nsLabelFrame* it = new (aPresShell) nsLabelFrame; if (!it) { return NS_ERROR_OUT_OF_MEMORY; } // set the state flags (if any are provided) nsFrameState state; it->GetFrameState( &state ); state |= aStateFlags; it->SetFrameState( state ); *aNewFrame = it; return NS_OK; } nsLabelFrame::nsLabelFrame() : nsLabelFrameSuper() { } nsLabelFrame::~nsLabelFrame() { } NS_IMETHODIMP nsLabelFrame::Destroy(nsIPresContext *aPresContext) { nsFormControlFrame::RegUnRegAccessKey(aPresContext, NS_STATIC_CAST(nsIFrame*, this), PR_FALSE); return nsLabelFrameSuper::Destroy(aPresContext); } NS_IMETHODIMP nsLabelFrame::Reflow(nsIPresContext* aPresContext, nsHTMLReflowMetrics& aDesiredSize, const nsHTMLReflowState& aReflowState, nsReflowStatus& aStatus) { if (eReflowReason_Initial == aReflowState.reason) nsFormControlFrame::RegUnRegAccessKey(aPresContext, NS_STATIC_CAST(nsIFrame*, this), PR_TRUE); return nsLabelFrameSuper::Reflow(aPresContext, aDesiredSize, aReflowState, aStatus); } NS_IMETHODIMP nsLabelFrame::GetFrameForPoint(nsIPresContext* aPresContext, const nsPoint& aPoint, nsFramePaintLayer aWhichLayer, nsIFrame** aFrame) { nsresult rv = nsLabelFrameSuper::GetFrameForPoint(aPresContext, aPoint, aWhichLayer, aFrame); if (rv == NS_OK) { nsCOMPtr controlFrame = do_QueryInterface(*aFrame); if (!controlFrame) { // if the hit frame isn't a form control then // check to see if it is an anchor // XXX It could be something else that should get the event. Perhaps // this is better handled by event bubbling? nsIFrame * parent; (*aFrame)->GetParent(&parent); while (parent != this && parent != nsnull) { nsCOMPtr content; parent->GetContent(getter_AddRefs(content)); nsCOMPtr anchorElement(do_QueryInterface(content)); if (anchorElement) { nsIStyleContext *psc; parent->GetStyleContext(&psc); if (psc) { const nsStyleVisibility* vis = (const nsStyleVisibility*)psc->GetStyleData(eStyleStruct_Visibility); if (vis->IsVisible()) { *aFrame = parent; return NS_OK; } } } parent->GetParent(&parent); } const nsStyleVisibility* vis = (const nsStyleVisibility*)mStyleContext->GetStyleData(eStyleStruct_Visibility); if (vis->IsVisible()) { *aFrame = this; return NS_OK; } } } return rv; }