1999-04-20 21:39:52 +00:00
|
|
|
/* -*- 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
|
1999-06-15 04:02:43 +00:00
|
|
|
* Version 1.0 (the "License"); you may not use this file except in
|
|
|
|
* compliance with the License. You may obtain a copy of the License at
|
1999-04-20 21:39:52 +00:00
|
|
|
* http://www.mozilla.org/NPL/
|
|
|
|
*
|
1999-06-15 04:02:43 +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-04-20 21:39:52 +00:00
|
|
|
*
|
1999-06-15 04:02:43 +00:00
|
|
|
* The Original Code is Mozilla Communicator client 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.
|
1999-04-20 21:39:52 +00:00
|
|
|
*/
|
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
//
|
|
|
|
// Eric Vaughan
|
|
|
|
// Netscape Communications
|
|
|
|
//
|
|
|
|
// See documentation in associated header file
|
|
|
|
//
|
1999-04-20 21:39:52 +00:00
|
|
|
|
1999-04-20 21:43:09 +00:00
|
|
|
#include "nsSliderFrame.h"
|
1999-04-20 21:39:52 +00:00
|
|
|
#include "nsIStyleContext.h"
|
1999-06-15 04:02:43 +00:00
|
|
|
#include "nsIPresContext.h"
|
|
|
|
#include "nsIContent.h"
|
|
|
|
#include "nsCOMPtr.h"
|
|
|
|
#include "nsHTMLIIDs.h"
|
|
|
|
#include "nsUnitConversion.h"
|
1999-04-20 21:39:52 +00:00
|
|
|
#include "nsINameSpaceManager.h"
|
1999-06-15 04:02:43 +00:00
|
|
|
#include "nsXULAtoms.h"
|
|
|
|
#include "nsHTMLAtoms.h"
|
|
|
|
#include "nsIReflowCommand.h"
|
|
|
|
#include "nsHTMLParts.h"
|
|
|
|
#include "nsIPresShell.h"
|
|
|
|
#include "nsStyleChangeList.h"
|
|
|
|
#include "nsCSSRendering.h"
|
|
|
|
#include "nsHTMLAtoms.h"
|
|
|
|
#include "nsIDOMEventReceiver.h"
|
|
|
|
#include "nsIViewManager.h"
|
|
|
|
#include "nsIDOMElement.h"
|
|
|
|
#include "nsIDOMDocument.h"
|
|
|
|
#include "nsIDOMUIEvent.h"
|
|
|
|
#include "nsDocument.h"
|
|
|
|
#include "nsTitledButtonFrame.h"
|
|
|
|
#include "nsScrollbarButtonFrame.h"
|
1999-06-23 17:00:17 +00:00
|
|
|
#include "nsIScrollbarListener.h"
|
1999-06-30 22:17:43 +00:00
|
|
|
#include "nsISupportsArray.h"
|
|
|
|
|
|
|
|
static NS_DEFINE_IID(kIAnonymousContentCreatorIID, NS_IANONYMOUS_CONTENT_CREATOR_IID);
|
1999-04-20 21:39:52 +00:00
|
|
|
|
|
|
|
nsresult
|
1999-06-30 22:17:43 +00:00
|
|
|
NS_NewSliderFrame ( nsIFrame** aNewFrame)
|
1999-04-20 21:39:52 +00:00
|
|
|
{
|
1999-05-11 22:03:29 +00:00
|
|
|
NS_PRECONDITION(aNewFrame, "null OUT ptr");
|
|
|
|
if (nsnull == aNewFrame) {
|
|
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
}
|
1999-06-30 22:17:43 +00:00
|
|
|
nsSliderFrame* it = new nsSliderFrame();
|
1999-06-15 04:02:43 +00:00
|
|
|
if (nsnull == it)
|
1999-04-20 21:39:52 +00:00
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
1999-06-15 04:02:43 +00:00
|
|
|
|
1999-05-11 22:03:29 +00:00
|
|
|
*aNewFrame = it;
|
1999-04-20 21:39:52 +00:00
|
|
|
return NS_OK;
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
} // NS_NewSliderFrame
|
1999-04-20 21:39:52 +00:00
|
|
|
|
1999-04-20 21:43:09 +00:00
|
|
|
nsSliderFrame::nsSliderFrame()
|
1999-06-23 17:00:17 +00:00
|
|
|
:mScrollbarListener(nsnull), mCurPos(0)
|
1999-04-20 21:39:52 +00:00
|
|
|
{
|
1999-06-15 04:02:43 +00:00
|
|
|
}
|
1999-04-20 21:39:52 +00:00
|
|
|
|
1999-06-30 22:17:43 +00:00
|
|
|
/**
|
|
|
|
* Anonymous interface
|
|
|
|
*/
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsSliderFrame::CreateAnonymousContent(nsISupportsArray& aAnonymousChildren)
|
|
|
|
{
|
|
|
|
// supply anonymous content if there is no content
|
|
|
|
PRInt32 count = 0;
|
|
|
|
mContent->ChildCount(count);
|
|
|
|
if (count == 0)
|
|
|
|
{
|
|
|
|
// get the document
|
|
|
|
nsCOMPtr<nsIDocument> idocument;
|
|
|
|
mContent->GetDocument(*getter_AddRefs(idocument));
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMDocument> document(do_QueryInterface(idocument));
|
|
|
|
|
|
|
|
// create a thumb
|
|
|
|
nsCOMPtr<nsIDOMElement> node;
|
|
|
|
document->CreateElement("thumb",getter_AddRefs(node));
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
|
|
content = do_QueryInterface(node);
|
|
|
|
|
|
|
|
// if we are not in a scrollbar default our thumbs flex to be
|
|
|
|
// flexible.
|
|
|
|
nsIContent* scrollbar = GetScrollBar();
|
|
|
|
|
|
|
|
if (scrollbar)
|
|
|
|
content->SetAttribute(kNameSpaceID_None, nsXULAtoms::flex, "100%", PR_TRUE);
|
|
|
|
|
|
|
|
aAnonymousChildren.AppendElement(content);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
1999-04-20 21:39:52 +00:00
|
|
|
|
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsSliderFrame::Init(nsIPresContext& aPresContext,
|
|
|
|
nsIContent* aContent,
|
|
|
|
nsIFrame* aParent,
|
|
|
|
nsIStyleContext* aContext,
|
|
|
|
nsIFrame* aPrevInFlow)
|
1999-04-20 21:39:52 +00:00
|
|
|
{
|
1999-06-15 04:02:43 +00:00
|
|
|
nsresult rv = nsHTMLContainerFrame::Init(aPresContext, aContent, aParent, aContext, aPrevInFlow);
|
|
|
|
CreateViewForFrame(aPresContext,this,aContext,PR_TRUE);
|
|
|
|
return rv;
|
1999-04-20 21:39:52 +00:00
|
|
|
}
|
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
PRInt32
|
|
|
|
nsSliderFrame::GetCurrentPosition(nsIContent* content)
|
|
|
|
{
|
|
|
|
return GetIntegerAttribute(content, nsXULAtoms::curpos, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
PRInt32
|
|
|
|
nsSliderFrame::GetMaxPosition(nsIContent* content)
|
|
|
|
{
|
|
|
|
return GetIntegerAttribute(content, nsXULAtoms::maxpos, 100);
|
|
|
|
}
|
|
|
|
|
|
|
|
PRInt32
|
|
|
|
nsSliderFrame::GetIncrement(nsIContent* content)
|
|
|
|
{
|
|
|
|
return GetIntegerAttribute(content, nsXULAtoms::increment, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
PRInt32
|
|
|
|
nsSliderFrame::GetPageIncrement(nsIContent* content)
|
|
|
|
{
|
|
|
|
return GetIntegerAttribute(content, nsXULAtoms::pageincrement, 10);
|
|
|
|
}
|
|
|
|
|
|
|
|
PRInt32
|
|
|
|
nsSliderFrame::GetIntegerAttribute(nsIContent* content, nsIAtom* atom, PRInt32 defaultValue)
|
|
|
|
{
|
|
|
|
nsString value;
|
|
|
|
if (NS_CONTENT_ATTR_HAS_VALUE == content->GetAttribute(kNameSpaceID_None, atom, value))
|
|
|
|
{
|
|
|
|
PRInt32 error;
|
|
|
|
|
|
|
|
// convert it to an integer
|
|
|
|
defaultValue = value.ToInteger(&error);
|
|
|
|
}
|
|
|
|
|
|
|
|
return defaultValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsSliderFrame::AttributeChanged(nsIPresContext* aPresContext,
|
|
|
|
nsIContent* aChild,
|
|
|
|
nsIAtom* aAttribute,
|
|
|
|
PRInt32 aHint)
|
|
|
|
{
|
|
|
|
nsresult rv = nsHTMLContainerFrame::AttributeChanged(aPresContext, aChild,
|
|
|
|
aAttribute, aHint);
|
|
|
|
// if the current position changes
|
|
|
|
if (aAttribute == nsXULAtoms::curpos) {
|
|
|
|
CurrentPositionChanged(aPresContext);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsSliderFrame::Paint(nsIPresContext& aPresContext,
|
|
|
|
nsIRenderingContext& aRenderingContext,
|
|
|
|
const nsRect& aDirtyRect,
|
|
|
|
nsFramePaintLayer aWhichLayer)
|
|
|
|
{
|
|
|
|
return nsHTMLContainerFrame::Paint(aPresContext, aRenderingContext, aDirtyRect, aWhichLayer);
|
|
|
|
}
|
1999-04-20 21:39:52 +00:00
|
|
|
|
|
|
|
void
|
1999-06-15 04:02:43 +00:00
|
|
|
nsSliderFrame::ReflowThumb(nsIPresContext& aPresContext,
|
|
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
|
|
const nsHTMLReflowState& aReflowState,
|
|
|
|
nsReflowStatus& aStatus,
|
|
|
|
nsIFrame* thumbFrame,
|
|
|
|
nsSize available,
|
|
|
|
nsSize computed)
|
1999-04-20 21:39:52 +00:00
|
|
|
{
|
1999-06-15 04:02:43 +00:00
|
|
|
nsHTMLReflowState thumbReflowState(aPresContext, aReflowState,
|
|
|
|
thumbFrame, available);
|
|
|
|
|
|
|
|
// always give the thumb as much size as it needs
|
|
|
|
thumbReflowState.computedWidth = computed.width;
|
|
|
|
thumbReflowState.computedHeight = computed.height;
|
|
|
|
|
|
|
|
// subtract out the childs margin and border if computed
|
|
|
|
const nsStyleSpacing* spacing;
|
|
|
|
nsresult rv = thumbFrame->GetStyleData(eStyleStruct_Spacing,
|
|
|
|
(const nsStyleStruct*&) spacing);
|
|
|
|
|
|
|
|
nsMargin margin;
|
|
|
|
spacing->GetMargin(margin);
|
|
|
|
nsMargin border;
|
|
|
|
spacing->GetBorderPadding(border);
|
|
|
|
nsMargin total = margin + border;
|
1999-04-20 21:39:52 +00:00
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
if (thumbReflowState.computedWidth != NS_INTRINSICSIZE)
|
|
|
|
thumbReflowState.computedWidth -= total.left + total.right;
|
|
|
|
|
|
|
|
if (thumbReflowState.computedHeight != NS_INTRINSICSIZE)
|
|
|
|
thumbReflowState.computedHeight -= total.top + total.bottom;
|
|
|
|
|
|
|
|
ReflowChild(thumbFrame, aPresContext, aDesiredSize, thumbReflowState, aStatus);
|
|
|
|
|
|
|
|
// add the margin back in
|
|
|
|
aDesiredSize.width += margin.left + margin.right;
|
|
|
|
aDesiredSize.height += margin.top + margin.bottom;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsSliderFrame::IsHorizontal(nsIContent* scrollbar)
|
|
|
|
{
|
|
|
|
PRBool isHorizontal = PR_TRUE;
|
|
|
|
nsString value;
|
|
|
|
if (NS_CONTENT_ATTR_HAS_VALUE == scrollbar->GetAttribute(kNameSpaceID_None, nsHTMLAtoms::align, value))
|
|
|
|
{
|
|
|
|
if (value=="vertical")
|
|
|
|
isHorizontal = PR_FALSE;
|
1999-04-20 21:39:52 +00:00
|
|
|
}
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
return isHorizontal;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsSliderFrame::Reflow(nsIPresContext& aPresContext,
|
|
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
|
|
const nsHTMLReflowState& aReflowState,
|
|
|
|
nsReflowStatus& aStatus)
|
|
|
|
{
|
|
|
|
nsIContent* scrollbar = GetScrollBar();
|
|
|
|
|
|
|
|
PRBool isHorizontal = IsHorizontal(scrollbar);
|
|
|
|
|
|
|
|
// flow our thumb with our computed width and its intrinsic height
|
|
|
|
nsIFrame* thumbFrame = mFrames.FirstChild();
|
|
|
|
NS_ASSERTION(thumbFrame,"Slider does not have a thumb!!!!");
|
|
|
|
|
|
|
|
nsHTMLReflowMetrics thumbSize(nsnull);
|
|
|
|
|
|
|
|
nsSize availableSize(isHorizontal ? NS_INTRINSICSIZE: aReflowState.computedWidth, isHorizontal ? aReflowState.computedHeight : NS_INTRINSICSIZE);
|
|
|
|
nsSize computedSize(isHorizontal ? NS_INTRINSICSIZE: aReflowState.computedWidth, isHorizontal ? aReflowState.computedHeight : NS_INTRINSICSIZE);
|
|
|
|
|
|
|
|
ReflowThumb(aPresContext, thumbSize, aReflowState, aStatus, thumbFrame, availableSize, computedSize);
|
|
|
|
|
|
|
|
// get our current position and max position from our content node
|
|
|
|
PRInt32 curpos = GetCurrentPosition(scrollbar);
|
|
|
|
PRInt32 maxpos = GetMaxPosition(scrollbar);
|
|
|
|
|
|
|
|
if (curpos < 0)
|
|
|
|
curpos = 0;
|
|
|
|
else if (curpos > maxpos)
|
|
|
|
curpos = maxpos;
|
|
|
|
|
|
|
|
// if the computed height we are given is intrinsic then set it to some default height
|
|
|
|
float p2t;
|
|
|
|
aPresContext.GetScaledPixelsToTwips(&p2t);
|
|
|
|
nscoord onePixel = NSIntPixelsToTwips(1, p2t);
|
|
|
|
|
|
|
|
if (aReflowState.computedHeight == NS_INTRINSICSIZE)
|
|
|
|
aDesiredSize.height = isHorizontal ? thumbSize.height : 200*onePixel;
|
1999-04-20 21:39:52 +00:00
|
|
|
else {
|
1999-06-15 04:02:43 +00:00
|
|
|
aDesiredSize.height = aReflowState.computedHeight;
|
|
|
|
if (aDesiredSize.height < thumbSize.height)
|
|
|
|
aDesiredSize.height = thumbSize.height;
|
1999-04-20 21:39:52 +00:00
|
|
|
}
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
// set the width to the computed or if intrinsic then the width of the thumb.
|
|
|
|
if (aReflowState.computedWidth == NS_INTRINSICSIZE)
|
|
|
|
aDesiredSize.width = isHorizontal ? 200*onePixel : thumbSize.width;
|
1999-04-20 21:39:52 +00:00
|
|
|
else {
|
1999-06-15 04:02:43 +00:00
|
|
|
aDesiredSize.width = aReflowState.computedWidth;
|
|
|
|
if (aDesiredSize.width < thumbSize.width)
|
|
|
|
aDesiredSize.width = thumbSize.width;
|
1999-04-20 21:39:52 +00:00
|
|
|
}
|
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
// convert the set max size to pixels
|
|
|
|
nscoord maxpospx = maxpos * onePixel;
|
|
|
|
|
|
|
|
// get our maxpos in pixels. This is the space we have left over in the scrollbar
|
|
|
|
// after the height of the thumb has been removed
|
|
|
|
nscoord& desiredcoord = isHorizontal ? aDesiredSize.width : aDesiredSize.height;
|
|
|
|
nscoord& thumbcoord = isHorizontal ? thumbSize.width : thumbSize.height;
|
|
|
|
|
|
|
|
nscoord ourmaxpospx = desiredcoord - thumbcoord; // maxpos in pixels
|
|
|
|
mRatio = 1.0;
|
|
|
|
|
|
|
|
// now if we remove the max positon we now know the difference in pixels between the
|
|
|
|
// 2 and can create a ratio between them.
|
|
|
|
nscoord diffpx = ourmaxpospx - maxpospx;
|
|
|
|
|
|
|
|
// create a variable called ratio set it to 1.
|
|
|
|
// 1) If the above subtraction is 0 do nothing
|
|
|
|
// 2) If less than 0 then make the ratio squash it
|
|
|
|
// 3) If greater than 0 try to reflow the thumb to that height. Take what ever
|
|
|
|
// height that was returned and us it to calculate the ratio.
|
|
|
|
if (diffpx < 0)
|
|
|
|
mRatio = float(ourmaxpospx)/float(maxpospx);
|
|
|
|
else if (diffpx > 0) {
|
|
|
|
// if the thumb is flexible make the thumb bigger.
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
|
|
thumbFrame->GetContent(getter_AddRefs(content));
|
|
|
|
|
|
|
|
PRInt32 error;
|
|
|
|
nsString value;
|
|
|
|
if (NS_CONTENT_ATTR_HAS_VALUE == content->GetAttribute(kNameSpaceID_None, nsXULAtoms::flex, value))
|
|
|
|
{
|
|
|
|
value.Trim("%");
|
|
|
|
// convert to a percent.
|
|
|
|
if (value.ToFloat(&error) > 0.0) {
|
|
|
|
if (isHorizontal)
|
|
|
|
computedSize.width = diffpx+thumbcoord;
|
|
|
|
else
|
|
|
|
computedSize.height = diffpx+thumbcoord;
|
|
|
|
|
|
|
|
ReflowThumb(aPresContext, thumbSize, aReflowState, aStatus, thumbFrame, availableSize, computedSize);
|
|
|
|
ourmaxpospx = desiredcoord - thumbcoord; // recalc ourmaxpospx
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mRatio = float(ourmaxpospx)/float(maxpospx);
|
1999-04-20 21:39:52 +00:00
|
|
|
}
|
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
// get our border
|
|
|
|
const nsMargin& borderPadding = aReflowState.mComputedBorderPadding;
|
|
|
|
|
|
|
|
nscoord curpospx = curpos*onePixel;
|
|
|
|
|
|
|
|
// set the thumbs y coord to be the current pos * the ratio.
|
|
|
|
nscoord pospx = nscoord(float(curpospx)*mRatio);
|
|
|
|
nsRect thumbRect(borderPadding.left, borderPadding.top, thumbSize.width, thumbSize.height);
|
|
|
|
|
|
|
|
if (isHorizontal)
|
|
|
|
thumbRect.x += pospx;
|
|
|
|
else
|
|
|
|
thumbRect.y += pospx;
|
|
|
|
|
|
|
|
thumbFrame->SetRect(thumbRect);
|
|
|
|
|
|
|
|
// add in our border
|
|
|
|
aDesiredSize.width += borderPadding.left + borderPadding.right;
|
|
|
|
aDesiredSize.height += borderPadding.top + borderPadding.bottom;
|
|
|
|
|
|
|
|
aDesiredSize.ascent = aDesiredSize.height;
|
|
|
|
aDesiredSize.descent = 0;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsSliderFrame::HandleEvent(nsIPresContext& aPresContext,
|
|
|
|
nsGUIEvent* aEvent,
|
|
|
|
nsEventStatus& aEventStatus)
|
|
|
|
{
|
|
|
|
nsIContent* scrollbar = GetScrollBar();
|
|
|
|
|
|
|
|
PRBool isHorizontal = IsHorizontal(scrollbar);
|
|
|
|
|
|
|
|
if (isCapturingEvents())
|
|
|
|
{
|
|
|
|
switch (aEvent->message) {
|
|
|
|
case NS_MOUSE_MOVE: {
|
|
|
|
// convert coord to pixels
|
|
|
|
nscoord pos = isHorizontal ? aEvent->point.x : aEvent->point.y;
|
|
|
|
|
1999-06-28 19:57:28 +00:00
|
|
|
// mDragStartPx is in pixels and is in our client areas coordinate system.
|
|
|
|
// so we need to first convert it so twips and then get it into our coordinate system.
|
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
// convert start to twips
|
|
|
|
nscoord startpx = mDragStartPx;
|
|
|
|
|
|
|
|
float p2t;
|
|
|
|
aPresContext.GetScaledPixelsToTwips(&p2t);
|
|
|
|
nscoord onePixel = NSIntPixelsToTwips(1, p2t);
|
|
|
|
nscoord start = startpx*onePixel;
|
|
|
|
|
|
|
|
nsIFrame* thumbFrame = mFrames.FirstChild();
|
1999-06-28 19:57:28 +00:00
|
|
|
|
|
|
|
// get it into our coordintate system by subtracting our parents offsets.
|
|
|
|
nsIFrame* parent = this;
|
|
|
|
while(parent != nsnull)
|
|
|
|
{
|
|
|
|
nsRect r;
|
|
|
|
parent->GetRect(r);
|
|
|
|
isHorizontal ? start -= r.x : start -= r.y;
|
|
|
|
parent->GetParent(&parent);
|
|
|
|
}
|
|
|
|
|
|
|
|
//printf("Translated to start=%d\n",start);
|
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
start -= mThumbStart;
|
|
|
|
|
|
|
|
// take our current position and substract the start location
|
|
|
|
pos -= start;
|
|
|
|
|
|
|
|
// convert to pixels
|
|
|
|
nscoord pospx = pos/onePixel;
|
|
|
|
|
|
|
|
// convert to our internal coordinate system
|
|
|
|
pospx = nscoord(pospx/mRatio);
|
|
|
|
|
|
|
|
// set it
|
|
|
|
SetCurrentPosition(aPresContext, scrollbar, thumbFrame, pospx);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NS_MOUSE_LEFT_BUTTON_UP:
|
|
|
|
// stop capturing
|
|
|
|
//printf("stop capturing\n");
|
|
|
|
nsIFrame* thumbFrame = mFrames.FirstChild();
|
|
|
|
AddListener();
|
|
|
|
CaptureMouseEvents(PR_FALSE);
|
|
|
|
}
|
|
|
|
return nsFrame::HandleEvent(aPresContext, aEvent, aEventStatus);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIFrame* thumbFrame = mFrames.FirstChild();
|
|
|
|
nsRect thumbRect;
|
|
|
|
thumbFrame->GetRect(thumbRect);
|
|
|
|
|
|
|
|
if (thumbRect.Contains(aEvent->point))
|
|
|
|
{
|
|
|
|
switch (aEvent->message) {
|
|
|
|
case NS_MOUSE_LEFT_BUTTON_DOWN:
|
|
|
|
// on a mouse press grab the cursor
|
|
|
|
|
|
|
|
// if the cursor moves up or down by n pixels divide them by the ratio and add or subtract them from from
|
|
|
|
// the current pos. Do bounds checking, make sure the current pos is greater or equal to 0 and less that max pos.
|
|
|
|
// redraw
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return nsHTMLContainerFrame::HandleEvent(aPresContext, aEvent, aEventStatus);
|
|
|
|
// on release stop grabbing.
|
|
|
|
} else if ((isHorizontal && aEvent->point.x < thumbRect.x) || (!isHorizontal && aEvent->point.y < thumbRect.y)) {
|
|
|
|
// if the mouse is above the thumb
|
|
|
|
// on a left mouse press
|
|
|
|
switch (aEvent->message) {
|
|
|
|
case NS_MOUSE_LEFT_BUTTON_DOWN:
|
|
|
|
// decrease the current pos by the page size
|
|
|
|
PageUpDown(aPresContext, thumbFrame, -1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
switch (aEvent->message) {
|
|
|
|
case NS_MOUSE_LEFT_BUTTON_DOWN:
|
|
|
|
// increase the current pos by the page size
|
|
|
|
PageUpDown(aPresContext, thumbFrame, 1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// don't invoke any sort of dragging or selection code.
|
|
|
|
return NS_OK; //nsFrame::HandleEvent(aPresContext, aEvent, aEventStatus);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsSliderFrame::HandleDrag(nsIPresContext& aPresContext,
|
|
|
|
nsGUIEvent* aEvent,
|
|
|
|
nsEventStatus& aEventStatus)
|
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIContent*
|
|
|
|
nsSliderFrame::GetScrollBar()
|
|
|
|
{
|
|
|
|
// if we are in a scrollbar then return the scrollbar's content node
|
|
|
|
// if we are not then return ours.
|
|
|
|
nsIFrame* scrollbar;
|
|
|
|
nsScrollbarButtonFrame::GetParentWithTag(nsXULAtoms::scrollbar, this, scrollbar);
|
|
|
|
|
|
|
|
if (scrollbar == nsnull)
|
|
|
|
scrollbar = this;
|
|
|
|
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
|
|
scrollbar->GetContent(getter_AddRefs(content));
|
|
|
|
|
|
|
|
return content;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsSliderFrame::PageUpDown(nsIPresContext& aPresContext, nsIFrame* aThumbFrame, nscoord change)
|
|
|
|
{
|
|
|
|
// on a page up or down get our page increment. We get this by getting the scrollbar we are in and
|
|
|
|
// asking it for the current position and the page increment. If we are not in a scrollbar we will
|
|
|
|
// get the values from our own node.
|
|
|
|
nsIContent* scrollbar = GetScrollBar();
|
1999-06-23 17:00:17 +00:00
|
|
|
|
|
|
|
if (mScrollbarListener)
|
|
|
|
mScrollbarListener->PagedUpDown(); // Let the listener decide our increment.
|
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
nscoord pageIncrement = GetPageIncrement(scrollbar);
|
|
|
|
PRInt32 curpos = GetCurrentPosition(scrollbar);
|
|
|
|
SetCurrentPosition(aPresContext, scrollbar, aThumbFrame, curpos + change*pageIncrement);
|
|
|
|
}
|
|
|
|
|
|
|
|
// called when the current position changed and we need to update the thumb's location
|
|
|
|
void
|
|
|
|
nsSliderFrame::CurrentPositionChanged(nsIPresContext* aPresContext)
|
|
|
|
{
|
|
|
|
nsIContent* scrollbar = GetScrollBar();
|
|
|
|
|
|
|
|
PRBool isHorizontal = IsHorizontal(scrollbar);
|
|
|
|
|
|
|
|
// get the current position
|
|
|
|
PRInt32 curpos = GetCurrentPosition(scrollbar);
|
|
|
|
|
|
|
|
// get our current position and max position from our content node
|
|
|
|
PRInt32 maxpos = GetMaxPosition(scrollbar);
|
|
|
|
|
|
|
|
if (curpos < 0)
|
|
|
|
curpos = 0;
|
|
|
|
else if (curpos > maxpos)
|
|
|
|
curpos = maxpos;
|
|
|
|
|
|
|
|
// convert to pixels
|
|
|
|
float p2t;
|
|
|
|
aPresContext->GetScaledPixelsToTwips(&p2t);
|
|
|
|
nscoord onePixel = NSIntPixelsToTwips(1, p2t);
|
|
|
|
|
|
|
|
nscoord curpospx = curpos*onePixel;
|
|
|
|
|
|
|
|
// get the thumb's rect
|
|
|
|
nsIFrame* thumbFrame = mFrames.FirstChild();
|
|
|
|
nsRect thumbRect;
|
|
|
|
thumbFrame->GetRect(thumbRect);
|
|
|
|
|
|
|
|
// get our border and padding
|
|
|
|
const nsStyleSpacing* spacing;
|
|
|
|
nsresult rv = GetStyleData(eStyleStruct_Spacing,
|
|
|
|
(const nsStyleStruct*&) spacing);
|
|
|
|
|
|
|
|
nsMargin borderPadding;
|
|
|
|
spacing->GetBorderPadding(borderPadding);
|
|
|
|
|
|
|
|
// figure out the new rect
|
|
|
|
nsRect newThumbRect(thumbRect);
|
|
|
|
|
|
|
|
if (isHorizontal)
|
|
|
|
newThumbRect.x = borderPadding.left + nscoord(float(curpospx)*mRatio);
|
|
|
|
else
|
|
|
|
newThumbRect.y = borderPadding.top + nscoord(float(curpospx)*mRatio);
|
|
|
|
|
|
|
|
// set the rect
|
|
|
|
thumbFrame->SetRect(newThumbRect);
|
|
|
|
|
|
|
|
// figure out the union of the rect so we know what to redraw
|
|
|
|
nsRect changeRect;
|
|
|
|
changeRect.UnionRect(thumbRect, newThumbRect);
|
|
|
|
|
|
|
|
// redraw just the change
|
|
|
|
Invalidate(changeRect, PR_TRUE);
|
1999-06-23 17:00:17 +00:00
|
|
|
|
|
|
|
if (mScrollbarListener)
|
1999-06-25 01:28:43 +00:00
|
|
|
mScrollbarListener->PositionChanged(*aPresContext, mCurPos, curpos);
|
1999-06-23 17:00:17 +00:00
|
|
|
|
|
|
|
mCurPos = curpos;
|
|
|
|
|
1999-06-15 04:02:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsSliderFrame::SetCurrentPosition(nsIPresContext& aPresContext, nsIContent* scrollbar, nsIFrame* aThumbFrame, nscoord newpos)
|
|
|
|
{
|
|
|
|
|
|
|
|
float p2t;
|
|
|
|
aPresContext.GetScaledPixelsToTwips(&p2t);
|
|
|
|
nscoord onePixel = NSIntPixelsToTwips(1, p2t);
|
|
|
|
|
|
|
|
// get our current position and max position from our content node
|
1999-06-23 17:00:17 +00:00
|
|
|
PRInt32 curpos = GetCurrentPosition(scrollbar);
|
1999-06-15 04:02:43 +00:00
|
|
|
PRInt32 maxpos = GetMaxPosition(scrollbar);
|
|
|
|
|
|
|
|
// get the new position and make sure it is in bounds
|
|
|
|
if (newpos > maxpos)
|
|
|
|
newpos = maxpos;
|
|
|
|
else if (newpos < 0)
|
|
|
|
newpos = 0;
|
|
|
|
|
|
|
|
char ch[100];
|
|
|
|
sprintf(ch,"%d", newpos);
|
|
|
|
|
|
|
|
// set the new position
|
|
|
|
scrollbar->SetAttribute(kNameSpaceID_None, nsXULAtoms::curpos, nsString(ch), PR_TRUE);
|
|
|
|
|
1999-06-28 19:57:28 +00:00
|
|
|
//printf("Current Pos=%d\n",newpos);
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsSliderFrame::GetFrameForPoint(const nsPoint& aPoint,
|
|
|
|
nsIFrame** aFrame)
|
|
|
|
{
|
|
|
|
if (isCapturingEvents())
|
|
|
|
{
|
|
|
|
*aFrame = this;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIFrame* thumbFrame = mFrames.FirstChild();
|
|
|
|
nsRect thumbRect;
|
|
|
|
thumbFrame->GetRect(thumbRect);
|
|
|
|
|
|
|
|
if (thumbRect.Contains(aPoint)) {
|
|
|
|
*aFrame = thumbFrame;
|
|
|
|
} else {
|
|
|
|
// always return us
|
|
|
|
*aFrame = this;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
//return nsHTMLContainerFrame::GetFrameForPoint(aPoint, aFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static NS_DEFINE_IID(kIDOMMouseListenerIID, NS_IDOMMOUSELISTENER_IID);
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsSliderFrame::SetInitialChildList(nsIPresContext& aPresContext,
|
|
|
|
nsIAtom* aListName,
|
|
|
|
nsIFrame* aChildList)
|
|
|
|
{
|
|
|
|
nsresult r = nsHTMLContainerFrame::SetInitialChildList(aPresContext, aListName, aChildList);
|
|
|
|
|
|
|
|
AddListener();
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsSliderFrame::RemoveFrame(nsIPresContext& aPresContext,
|
|
|
|
nsIPresShell& aPresShell,
|
|
|
|
nsIAtom* aListName,
|
|
|
|
nsIFrame* aOldFrame)
|
|
|
|
{
|
|
|
|
// remove the child frame
|
|
|
|
nsresult rv = nsHTMLContainerFrame::RemoveFrame(aPresContext, aPresShell, aListName, aOldFrame);
|
|
|
|
mFrames.DeleteFrame(aPresContext, aOldFrame);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsSliderFrame::InsertFrames(nsIPresContext& aPresContext,
|
|
|
|
nsIPresShell& aPresShell,
|
|
|
|
nsIAtom* aListName,
|
|
|
|
nsIFrame* aPrevFrame,
|
|
|
|
nsIFrame* aFrameList)
|
|
|
|
{
|
|
|
|
mFrames.InsertFrames(nsnull, aPrevFrame, aFrameList);
|
|
|
|
return nsHTMLContainerFrame::InsertFrames(aPresContext, aPresShell, aListName, aPrevFrame, aFrameList);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsSliderFrame::AppendFrames(nsIPresContext& aPresContext,
|
|
|
|
nsIPresShell& aPresShell,
|
|
|
|
nsIAtom* aListName,
|
|
|
|
nsIFrame* aFrameList)
|
|
|
|
{
|
|
|
|
mFrames.AppendFrames(nsnull, aFrameList);
|
|
|
|
return nsHTMLContainerFrame::AppendFrames(aPresContext, aPresShell, aListName, aFrameList);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsSliderFrame::MouseDown(nsIDOMEvent* aMouseEvent)
|
|
|
|
{
|
|
|
|
//printf("Begin dragging\n");
|
|
|
|
|
|
|
|
nsIContent* scrollbar = GetScrollBar();
|
|
|
|
PRBool isHorizontal = IsHorizontal(scrollbar);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMUIEvent> uiEvent(do_QueryInterface(aMouseEvent));
|
|
|
|
|
|
|
|
RemoveListener();
|
|
|
|
CaptureMouseEvents(PR_TRUE);
|
|
|
|
PRInt32 c = 0;
|
|
|
|
if (isHorizontal)
|
|
|
|
uiEvent->GetClientX(&c);
|
|
|
|
else
|
|
|
|
uiEvent->GetClientY(&c);
|
|
|
|
|
|
|
|
mDragStartPx = c;
|
|
|
|
nsIFrame* thumbFrame = mFrames.FirstChild();
|
|
|
|
nsRect thumbRect;
|
|
|
|
thumbFrame->GetRect(thumbRect);
|
|
|
|
|
|
|
|
if (isHorizontal)
|
|
|
|
mThumbStart = thumbRect.x;
|
|
|
|
else
|
|
|
|
mThumbStart = thumbRect.y;
|
|
|
|
|
1999-06-28 19:57:28 +00:00
|
|
|
//printf("Pressed mDragStartPx=%d\n",mDragStartPx);
|
1999-06-15 04:02:43 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsSliderFrame::MouseUp(nsIDOMEvent* aMouseEvent)
|
|
|
|
{
|
|
|
|
// printf("Finish dragging\n");
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsSliderFrame :: CaptureMouseEvents(PRBool aGrabMouseEvents)
|
|
|
|
{
|
|
|
|
// get its view
|
|
|
|
nsIView* view = nsnull;
|
|
|
|
GetView(&view);
|
|
|
|
nsCOMPtr<nsIViewManager> viewMan;
|
|
|
|
PRBool result;
|
|
|
|
|
|
|
|
if (view) {
|
|
|
|
view->GetViewManager(*getter_AddRefs(viewMan));
|
|
|
|
|
|
|
|
if (viewMan) {
|
|
|
|
if (aGrabMouseEvents) {
|
|
|
|
viewMan->GrabMouseEvents(view,result);
|
|
|
|
} else {
|
|
|
|
viewMan->GrabMouseEvents(nsnull,result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsSliderFrame :: isCapturingEvents()
|
|
|
|
{
|
|
|
|
// get its view
|
|
|
|
nsIView* view = nsnull;
|
|
|
|
GetView(&view);
|
|
|
|
nsCOMPtr<nsIViewManager> viewMan;
|
|
|
|
|
|
|
|
if (view) {
|
|
|
|
view->GetViewManager(*getter_AddRefs(viewMan));
|
|
|
|
|
|
|
|
if (viewMan) {
|
|
|
|
nsIView* grabbingView;
|
|
|
|
viewMan->GetMouseEventGrabber(grabbingView);
|
|
|
|
if (grabbingView == view)
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsSliderFrame::AddListener()
|
|
|
|
{
|
|
|
|
nsIFrame* thumbFrame = mFrames.FirstChild();
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
|
|
thumbFrame->GetContent(getter_AddRefs(content));
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMEventReceiver> reciever(do_QueryInterface(content));
|
|
|
|
|
|
|
|
reciever->AddEventListenerByIID(this, kIDOMMouseListenerIID);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsSliderFrame::RemoveListener()
|
|
|
|
{
|
|
|
|
nsIFrame* thumbFrame = mFrames.FirstChild();
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
|
|
thumbFrame->GetContent(getter_AddRefs(content));
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMEventReceiver> reciever(do_QueryInterface(content));
|
|
|
|
|
|
|
|
reciever->RemoveEventListenerByIID(this,kIDOMMouseListenerIID);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsSliderFrame::QueryInterface(REFNSIID aIID, void** aInstancePtr)
|
|
|
|
{
|
|
|
|
if (NULL == aInstancePtr) {
|
|
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
}
|
|
|
|
|
|
|
|
*aInstancePtr = NULL;
|
|
|
|
|
1999-06-30 22:17:43 +00:00
|
|
|
if (aIID.Equals(kIAnonymousContentCreatorIID)) {
|
|
|
|
*aInstancePtr = (void*)(nsIAnonymousContentCreator*) this;
|
|
|
|
NS_ADDREF_THIS();
|
|
|
|
return NS_OK;
|
|
|
|
} else if (aIID.Equals(kIDOMMouseListenerIID)) {
|
1999-06-15 04:02:43 +00:00
|
|
|
*aInstancePtr = (void*)(nsIDOMMouseListener*) this;
|
|
|
|
NS_ADDREF_THIS();
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nsHTMLContainerFrame::QueryInterface(aIID, aInstancePtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP_(nsrefcnt)
|
|
|
|
nsSliderFrame::AddRef(void)
|
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP_(nsrefcnt)
|
|
|
|
nsSliderFrame::Release(void)
|
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
1999-06-23 17:00:17 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
nsSliderFrame::SetScrollbarListener(nsIScrollbarListener* aListener)
|
|
|
|
{
|
|
|
|
// Don't addref/release this, since it's actually a frame.
|
|
|
|
mScrollbarListener = aListener;
|
|
|
|
}
|