2007-03-22 10:30:00 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2012-05-21 04:12:37 -07:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/* utility functions for drawing borders and backgrounds */
|
|
|
|
|
|
|
|
#ifndef nsCSSRendering_h___
|
|
|
|
#define nsCSSRendering_h___
|
|
|
|
|
2008-02-16 07:31:27 -08:00
|
|
|
#include "nsStyleConsts.h"
|
2008-09-25 12:53:09 -07:00
|
|
|
#include "gfxBlur.h"
|
2007-08-06 01:15:00 -07:00
|
|
|
#include "gfxContext.h"
|
2008-06-12 15:02:32 -07:00
|
|
|
#include "gfxImageSurface.h"
|
2012-05-03 13:11:57 -07:00
|
|
|
#include "nsLayoutUtils.h"
|
2008-09-25 12:53:09 -07:00
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
struct nsPoint;
|
|
|
|
class nsStyleContext;
|
|
|
|
class nsPresContext;
|
2011-04-07 18:04:40 -07:00
|
|
|
class nsRenderingContext;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2012-05-03 13:11:57 -07:00
|
|
|
/**
|
|
|
|
* This is a small wrapper class to encapsulate image drawing that can draw an
|
|
|
|
* nsStyleImage image, which may internally be a real image, a sub image, or a
|
|
|
|
* CSS gradient.
|
|
|
|
*
|
|
|
|
* @note Always call the member functions in the order of PrepareImage(),
|
|
|
|
* ComputeSize(), and Draw().
|
|
|
|
*/
|
|
|
|
class nsImageRenderer {
|
|
|
|
public:
|
|
|
|
typedef mozilla::layers::ImageContainer ImageContainer;
|
|
|
|
|
|
|
|
enum {
|
|
|
|
FLAG_SYNC_DECODE_IMAGES = 0x01
|
|
|
|
};
|
|
|
|
nsImageRenderer(nsIFrame* aForFrame, const nsStyleImage* aImage, PRUint32 aFlags);
|
|
|
|
~nsImageRenderer();
|
|
|
|
/**
|
|
|
|
* Populates member variables to get ready for rendering.
|
|
|
|
* @return true iff the image is ready, and there is at least a pixel to
|
|
|
|
* draw.
|
|
|
|
*/
|
|
|
|
bool PrepareImage();
|
|
|
|
/**
|
|
|
|
* @return the image size in appunits when rendered, after accounting for the
|
|
|
|
* background positioning area, background-size, and the image's intrinsic
|
|
|
|
* dimensions (if any).
|
|
|
|
*/
|
|
|
|
nsSize ComputeSize(const nsStyleBackground::Size& aLayerSize,
|
|
|
|
const nsSize& aBgPositioningArea);
|
|
|
|
/**
|
|
|
|
* Draws the image to the target rendering context.
|
|
|
|
* @see nsLayoutUtils::DrawImage() for other parameters
|
|
|
|
*/
|
|
|
|
void Draw(nsPresContext* aPresContext,
|
|
|
|
nsRenderingContext& aRenderingContext,
|
|
|
|
const nsRect& aDest,
|
|
|
|
const nsRect& aFill,
|
|
|
|
const nsPoint& aAnchor,
|
|
|
|
const nsRect& aDirty);
|
|
|
|
|
|
|
|
|
|
|
|
bool IsRasterImage();
|
|
|
|
already_AddRefed<ImageContainer> GetContainer();
|
|
|
|
private:
|
|
|
|
/*
|
|
|
|
* Compute the "unscaled" dimensions of the image in aUnscaled{Width,Height}
|
|
|
|
* and aRatio. Whether the image has a height and width are indicated by
|
|
|
|
* aHaveWidth and aHaveHeight. If the image doesn't have a ratio, aRatio will
|
|
|
|
* be (0, 0).
|
|
|
|
*/
|
|
|
|
void ComputeUnscaledDimensions(const nsSize& aBgPositioningArea,
|
|
|
|
nscoord& aUnscaledWidth, bool& aHaveWidth,
|
|
|
|
nscoord& aUnscaledHeight, bool& aHaveHeight,
|
|
|
|
nsSize& aRatio);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Using the previously-computed unscaled width and height (if each are
|
|
|
|
* valid, as indicated by aHaveWidth/aHaveHeight), compute the size at which
|
|
|
|
* the image should actually render.
|
|
|
|
*/
|
|
|
|
nsSize
|
|
|
|
ComputeDrawnSize(const nsStyleBackground::Size& aLayerSize,
|
|
|
|
const nsSize& aBgPositioningArea,
|
|
|
|
nscoord aUnscaledWidth, bool aHaveWidth,
|
|
|
|
nscoord aUnscaledHeight, bool aHaveHeight,
|
|
|
|
const nsSize& aIntrinsicRatio);
|
|
|
|
|
|
|
|
nsIFrame* mForFrame;
|
|
|
|
const nsStyleImage* mImage;
|
|
|
|
nsStyleImageType mType;
|
|
|
|
nsCOMPtr<imgIContainer> mImageContainer;
|
|
|
|
nsRefPtr<nsStyleGradient> mGradientData;
|
|
|
|
nsIFrame* mPaintServerFrame;
|
|
|
|
nsLayoutUtils::SurfaceFromElementResult mImageElementSurface;
|
|
|
|
bool mIsReady;
|
|
|
|
nsSize mSize; // unscaled size of the image, in app units
|
|
|
|
PRUint32 mFlags;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A struct representing all the information needed to paint a background
|
|
|
|
* image to some target, taking into account all CSS background-* properties.
|
|
|
|
* See PrepareBackgroundLayer.
|
|
|
|
*/
|
|
|
|
struct nsBackgroundLayerState {
|
|
|
|
/**
|
|
|
|
* @param aFlags some combination of nsCSSRendering::PAINTBG_* flags
|
|
|
|
*/
|
|
|
|
nsBackgroundLayerState(nsIFrame* aForFrame, const nsStyleImage* aImage, PRUint32 aFlags)
|
|
|
|
: mImageRenderer(aForFrame, aImage, aFlags) {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The nsImageRenderer that will be used to draw the background.
|
|
|
|
*/
|
|
|
|
nsImageRenderer mImageRenderer;
|
|
|
|
/**
|
|
|
|
* A rectangle that one copy of the image tile is mapped onto. Same
|
|
|
|
* coordinate system as aBorderArea/aBGClipRect passed into
|
|
|
|
* PrepareBackgroundLayer.
|
|
|
|
*/
|
|
|
|
nsRect mDestArea;
|
|
|
|
/**
|
|
|
|
* The actual rectangle that should be filled with (complete or partial)
|
|
|
|
* image tiles. Same coordinate system as aBorderArea/aBGClipRect passed into
|
|
|
|
* PrepareBackgroundLayer.
|
|
|
|
*/
|
|
|
|
nsRect mFillArea;
|
|
|
|
/**
|
|
|
|
* The anchor point that should be snapped to a pixel corner. Same
|
|
|
|
* coordinate system as aBorderArea/aBGClipRect passed into
|
|
|
|
* PrepareBackgroundLayer.
|
|
|
|
*/
|
|
|
|
nsPoint mAnchor;
|
|
|
|
};
|
|
|
|
|
2008-09-25 09:03:15 -07:00
|
|
|
struct nsCSSRendering {
|
2007-03-22 10:30:00 -07:00
|
|
|
/**
|
|
|
|
* Initialize any static variables used by nsCSSRendering.
|
|
|
|
*/
|
2011-06-02 05:56:46 -07:00
|
|
|
static void Init();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Clean up any static variables used by nsCSSRendering.
|
|
|
|
*/
|
|
|
|
static void Shutdown();
|
|
|
|
|
2009-02-10 00:45:13 -08:00
|
|
|
static void PaintBoxShadowInner(nsPresContext* aPresContext,
|
2011-04-07 18:04:40 -07:00
|
|
|
nsRenderingContext& aRenderingContext,
|
2009-02-10 00:45:13 -08:00
|
|
|
nsIFrame* aForFrame,
|
|
|
|
const nsRect& aFrameArea,
|
|
|
|
const nsRect& aDirtyRect);
|
|
|
|
|
|
|
|
static void PaintBoxShadowOuter(nsPresContext* aPresContext,
|
2011-04-07 18:04:40 -07:00
|
|
|
nsRenderingContext& aRenderingContext,
|
2009-02-10 00:45:13 -08:00
|
|
|
nsIFrame* aForFrame,
|
|
|
|
const nsRect& aFrameArea,
|
|
|
|
const nsRect& aDirtyRect);
|
2008-07-07 17:57:47 -07:00
|
|
|
|
2010-09-07 15:20:35 -07:00
|
|
|
static void ComputePixelRadii(const nscoord *aAppUnitsRadii,
|
|
|
|
nscoord aAppUnitsPerPixel,
|
|
|
|
gfxCornerSizes *oBorderRadii);
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/**
|
|
|
|
* Render the border for an element using css rendering rules
|
|
|
|
* for borders. aSkipSides is a bitmask of the sides to skip
|
|
|
|
* when rendering. If 0 then no sides are skipped.
|
|
|
|
*/
|
|
|
|
static void PaintBorder(nsPresContext* aPresContext,
|
2011-04-07 18:04:40 -07:00
|
|
|
nsRenderingContext& aRenderingContext,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIFrame* aForFrame,
|
|
|
|
const nsRect& aDirtyRect,
|
|
|
|
const nsRect& aBorderArea,
|
|
|
|
nsStyleContext* aStyleContext,
|
2008-08-06 03:33:18 -07:00
|
|
|
PRIntn aSkipSides = 0);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-04-02 18:58:26 -07:00
|
|
|
/**
|
|
|
|
* Like PaintBorder, but taking an nsStyleBorder argument instead of
|
|
|
|
* getting it from aStyleContext.
|
|
|
|
*/
|
|
|
|
static void PaintBorderWithStyleBorder(nsPresContext* aPresContext,
|
2011-04-07 18:04:40 -07:00
|
|
|
nsRenderingContext& aRenderingContext,
|
2010-04-02 18:58:26 -07:00
|
|
|
nsIFrame* aForFrame,
|
|
|
|
const nsRect& aDirtyRect,
|
|
|
|
const nsRect& aBorderArea,
|
|
|
|
const nsStyleBorder& aBorderStyle,
|
|
|
|
nsStyleContext* aStyleContext,
|
|
|
|
PRIntn aSkipSides = 0);
|
|
|
|
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/**
|
|
|
|
* Render the outline for an element using css rendering rules
|
|
|
|
* for borders. aSkipSides is a bitmask of the sides to skip
|
|
|
|
* when rendering. If 0 then no sides are skipped.
|
|
|
|
*/
|
|
|
|
static void PaintOutline(nsPresContext* aPresContext,
|
2011-04-07 18:04:40 -07:00
|
|
|
nsRenderingContext& aRenderingContext,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIFrame* aForFrame,
|
|
|
|
const nsRect& aDirtyRect,
|
|
|
|
const nsRect& aBorderArea,
|
2008-08-06 03:33:18 -07:00
|
|
|
nsStyleContext* aStyleContext);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Render keyboard focus on an element.
|
|
|
|
* |aFocusRect| is the outer rectangle of the focused element.
|
|
|
|
* Uses a fixed style equivalent to "1px dotted |aColor|".
|
|
|
|
* Not used for controls, because the native theme may differ.
|
|
|
|
*/
|
|
|
|
static void PaintFocus(nsPresContext* aPresContext,
|
2011-04-07 18:04:40 -07:00
|
|
|
nsRenderingContext& aRenderingContext,
|
2008-08-06 03:33:18 -07:00
|
|
|
const nsRect& aFocusRect,
|
|
|
|
nscolor aColor);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2009-08-01 08:53:40 -07:00
|
|
|
/**
|
|
|
|
* Render a gradient for an element.
|
|
|
|
*/
|
|
|
|
static void PaintGradient(nsPresContext* aPresContext,
|
2011-04-07 18:04:40 -07:00
|
|
|
nsRenderingContext& aRenderingContext,
|
2009-08-01 08:53:40 -07:00
|
|
|
nsStyleGradient* aGradient,
|
|
|
|
const nsRect& aDirtyRect,
|
|
|
|
const nsRect& aOneCellArea,
|
2009-11-02 11:36:43 -08:00
|
|
|
const nsRect& aFillArea);
|
2009-08-01 08:53:40 -07:00
|
|
|
|
2009-04-10 06:41:59 -07:00
|
|
|
/**
|
2009-12-01 09:21:00 -08:00
|
|
|
* Find the frame whose background style should be used to draw the
|
|
|
|
* canvas background. aForFrame must be the frame for the root element
|
|
|
|
* whose background style should be used. This function will return
|
|
|
|
* aForFrame unless the <body> background should be propagated, in
|
|
|
|
* which case we return the frame associated with the <body>'s background.
|
2009-04-10 06:41:59 -07:00
|
|
|
*/
|
2009-12-01 09:21:00 -08:00
|
|
|
static nsIFrame* FindBackgroundStyleFrame(nsIFrame* aForFrame);
|
2009-04-10 06:41:59 -07:00
|
|
|
|
2009-04-06 14:00:29 -07:00
|
|
|
/**
|
2011-10-17 07:59:28 -07:00
|
|
|
* @return true if |aFrame| is a canvas frame, in the CSS sense.
|
2009-02-15 17:11:34 -08:00
|
|
|
*/
|
2011-09-28 23:19:26 -07:00
|
|
|
static bool IsCanvasFrame(nsIFrame* aFrame);
|
2009-02-15 17:11:34 -08:00
|
|
|
|
|
|
|
/**
|
2010-04-02 18:58:26 -07:00
|
|
|
* Fill in an aBackgroundSC to be used to paint the background
|
2009-02-15 17:11:34 -08:00
|
|
|
* for an element. This applies the rules for propagating
|
2007-03-22 10:30:00 -07:00
|
|
|
* backgrounds between BODY, the root element, and the canvas.
|
2011-10-17 07:59:28 -07:00
|
|
|
* @return true if there is some meaningful background.
|
2007-03-22 10:30:00 -07:00
|
|
|
*/
|
2011-09-28 23:19:26 -07:00
|
|
|
static bool FindBackground(nsPresContext* aPresContext,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIFrame* aForFrame,
|
2010-04-02 18:58:26 -07:00
|
|
|
nsStyleContext** aBackgroundSC);
|
2009-02-15 17:11:34 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* As FindBackground, but the passed-in frame is known to be a root frame
|
|
|
|
* (returned from nsCSSFrameConstructor::GetRootElementStyleFrame())
|
|
|
|
* and there is always some meaningful background returned.
|
|
|
|
*/
|
2010-04-02 18:58:26 -07:00
|
|
|
static nsStyleContext* FindRootFrameBackground(nsIFrame* aForFrame);
|
2009-02-15 17:11:34 -08:00
|
|
|
|
2009-10-22 16:46:08 -07:00
|
|
|
/**
|
|
|
|
* Returns background style information for the canvas.
|
|
|
|
*
|
|
|
|
* @param aForFrame
|
|
|
|
* the frame used to represent the canvas, in the CSS sense (i.e.
|
|
|
|
* nsCSSRendering::IsCanvasFrame(aForFrame) must be true)
|
|
|
|
* @param aRootElementFrame
|
|
|
|
* the frame representing the root element of the document
|
|
|
|
* @param aBackground
|
|
|
|
* contains background style information for the canvas on return
|
|
|
|
*/
|
2010-04-02 18:58:26 -07:00
|
|
|
static nsStyleContext*
|
2009-10-22 16:46:08 -07:00
|
|
|
FindCanvasBackground(nsIFrame* aForFrame, nsIFrame* aRootElementFrame)
|
|
|
|
{
|
|
|
|
NS_ABORT_IF_FALSE(IsCanvasFrame(aForFrame), "not a canvas frame");
|
|
|
|
if (aRootElementFrame)
|
|
|
|
return FindRootFrameBackground(aRootElementFrame);
|
|
|
|
|
|
|
|
// This should always give transparent, so we'll fill it in with the
|
|
|
|
// default color if needed. This seems to happen a bit while a page is
|
|
|
|
// being loaded.
|
2010-04-02 18:58:26 -07:00
|
|
|
return aForFrame->GetStyleContext();
|
2009-10-22 16:46:08 -07:00
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/**
|
2010-04-08 05:44:57 -07:00
|
|
|
* Find a frame which draws a non-transparent background,
|
2009-01-08 16:29:38 -08:00
|
|
|
* for various table-related and HR-related backwards-compatibility hacks.
|
2010-04-08 05:44:57 -07:00
|
|
|
* This function will also stop if it finds themed frame which might draw
|
|
|
|
* background.
|
2009-01-08 16:29:38 -08:00
|
|
|
*
|
|
|
|
* Be very hesitant if you're considering calling this function -- it's
|
|
|
|
* usually not what you want.
|
2007-03-22 10:30:00 -07:00
|
|
|
*/
|
2010-04-08 05:44:57 -07:00
|
|
|
static nsIFrame*
|
|
|
|
FindNonTransparentBackgroundFrame(nsIFrame* aFrame,
|
2011-09-28 23:19:26 -07:00
|
|
|
bool aStartAtParent = false);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2009-07-04 02:30:59 -07:00
|
|
|
/**
|
|
|
|
* Determine the background color to draw taking into account print settings.
|
|
|
|
*/
|
|
|
|
static nscolor
|
|
|
|
DetermineBackgroundColor(nsPresContext* aPresContext,
|
2010-04-02 18:58:26 -07:00
|
|
|
nsStyleContext* aStyleContext,
|
2012-05-03 13:11:57 -07:00
|
|
|
nsIFrame* aFrame,
|
|
|
|
bool& aDrawBackgroundImage,
|
|
|
|
bool& aDrawBackgroundColor);
|
|
|
|
|
|
|
|
static nsBackgroundLayerState
|
|
|
|
PrepareBackgroundLayer(nsPresContext* aPresContext,
|
|
|
|
nsIFrame* aForFrame,
|
|
|
|
PRUint32 aFlags,
|
|
|
|
const nsRect& aBorderArea,
|
|
|
|
const nsRect& aBGClipRect,
|
|
|
|
const nsStyleBackground& aBackground,
|
|
|
|
const nsStyleBackground::Layer& aLayer);
|
2009-07-04 02:30:59 -07:00
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/**
|
|
|
|
* Render the background for an element using css rendering rules
|
|
|
|
* for backgrounds.
|
|
|
|
*/
|
2009-05-27 22:01:42 -07:00
|
|
|
enum {
|
|
|
|
/**
|
|
|
|
* When this flag is passed, the element's nsDisplayBorder will be
|
|
|
|
* painted immediately on top of this background.
|
|
|
|
*/
|
2009-09-12 15:44:18 -07:00
|
|
|
PAINTBG_WILL_PAINT_BORDER = 0x01,
|
|
|
|
/**
|
2010-10-25 07:38:09 -07:00
|
|
|
* When this flag is passed, images are synchronously decoded.
|
|
|
|
*/
|
|
|
|
PAINTBG_SYNC_DECODE_IMAGES = 0x02,
|
|
|
|
/**
|
|
|
|
* When this flag is passed, painting will go to the screen so we can
|
|
|
|
* take advantage of the fact that it will be clipped to the viewport.
|
|
|
|
*/
|
|
|
|
PAINTBG_TO_WINDOW = 0x04
|
2009-05-27 22:01:42 -07:00
|
|
|
};
|
2007-03-22 10:30:00 -07:00
|
|
|
static void PaintBackground(nsPresContext* aPresContext,
|
2011-04-07 18:04:40 -07:00
|
|
|
nsRenderingContext& aRenderingContext,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIFrame* aForFrame,
|
|
|
|
const nsRect& aDirtyRect,
|
|
|
|
const nsRect& aBorderArea,
|
2009-05-27 22:01:42 -07:00
|
|
|
PRUint32 aFlags,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsRect* aBGClipRect = nsnull);
|
|
|
|
|
|
|
|
/**
|
2008-09-23 21:07:22 -07:00
|
|
|
* Same as |PaintBackground|, except using the provided style structs.
|
|
|
|
* This short-circuits the code that ensures that the root element's
|
2007-03-22 10:30:00 -07:00
|
|
|
* background is drawn on the canvas.
|
|
|
|
*/
|
|
|
|
static void PaintBackgroundWithSC(nsPresContext* aPresContext,
|
2011-04-07 18:04:40 -07:00
|
|
|
nsRenderingContext& aRenderingContext,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIFrame* aForFrame,
|
|
|
|
const nsRect& aDirtyRect,
|
|
|
|
const nsRect& aBorderArea,
|
2010-04-02 18:58:26 -07:00
|
|
|
nsStyleContext *aStyleContext,
|
2007-03-22 10:30:00 -07:00
|
|
|
const nsStyleBorder& aBorder,
|
2009-05-27 22:01:42 -07:00
|
|
|
PRUint32 aFlags,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsRect* aBGClipRect = nsnull);
|
|
|
|
|
2011-01-02 17:48:09 -08:00
|
|
|
/**
|
|
|
|
* Returns the rectangle covered by the given background layer image, taking
|
|
|
|
* into account background positioning, sizing, and repetition, but not
|
|
|
|
* clipping.
|
|
|
|
*/
|
|
|
|
static nsRect GetBackgroundLayerRect(nsPresContext* aPresContext,
|
|
|
|
nsIFrame* aForFrame,
|
|
|
|
const nsRect& aBorderArea,
|
|
|
|
const nsStyleBackground& aBackground,
|
|
|
|
const nsStyleBackground::Layer& aLayer);
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/**
|
|
|
|
* Called by the presShell when painting is finished, so we can clear our
|
|
|
|
* inline background data cache.
|
|
|
|
*/
|
|
|
|
static void DidPaint();
|
|
|
|
|
2008-09-25 09:03:15 -07:00
|
|
|
// Draw a border segment in the table collapsing border model without
|
|
|
|
// beveling corners
|
2011-04-07 18:04:40 -07:00
|
|
|
static void DrawTableBorderSegment(nsRenderingContext& aContext,
|
2008-09-25 09:03:15 -07:00
|
|
|
PRUint8 aBorderStyle,
|
|
|
|
nscolor aBorderColor,
|
2007-03-22 10:30:00 -07:00
|
|
|
const nsStyleBackground* aBGColor,
|
2008-09-25 09:03:15 -07:00
|
|
|
const nsRect& aBorderRect,
|
|
|
|
PRInt32 aAppUnitsPerCSSPixel,
|
|
|
|
PRUint8 aStartBevelSide = 0,
|
|
|
|
nscoord aStartBevelOffset = 0,
|
|
|
|
PRUint8 aEndBevelSide = 0,
|
|
|
|
nscoord aEndBevelOffset = 0);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2007-08-06 01:15:00 -07:00
|
|
|
/**
|
|
|
|
* Function for painting the decoration lines for the text.
|
2008-02-16 07:31:27 -08:00
|
|
|
* NOTE: aPt, aLineSize, aAscent and aOffset are non-rounded device pixels,
|
|
|
|
* not app units.
|
2007-08-06 01:15:00 -07:00
|
|
|
* input:
|
|
|
|
* @param aGfxContext
|
2011-09-16 12:23:29 -07:00
|
|
|
* @param aDirtyRect no need to paint outside this rect
|
2007-08-06 01:15:00 -07:00
|
|
|
* @param aColor the color of the decoration line
|
|
|
|
* @param aPt the top/left edge of the text
|
|
|
|
* @param aLineSize the width and the height of the decoration
|
|
|
|
* line
|
|
|
|
* @param aAscent the ascent of the text
|
|
|
|
* @param aOffset the offset of the decoration line from
|
|
|
|
* the baseline of the text (if the value is
|
|
|
|
* positive, the line is lifted up)
|
|
|
|
* @param aDecoration which line will be painted. The value can be
|
2011-04-22 22:16:41 -07:00
|
|
|
* NS_STYLE_TEXT_DECORATION_LINE_UNDERLINE or
|
|
|
|
* NS_STYLE_TEXT_DECORATION_LINE_OVERLINE or
|
|
|
|
* NS_STYLE_TEXT_DECORATION_LINE_LINE_THROUGH.
|
2011-03-31 05:26:49 -07:00
|
|
|
* @param aStyle the style of the decoration line such as
|
|
|
|
* NS_STYLE_TEXT_DECORATION_STYLE_*.
|
2009-04-05 22:53:00 -07:00
|
|
|
* @param aDescentLimit If aDescentLimit is zero or larger and the
|
|
|
|
* underline overflows from the descent space,
|
|
|
|
* the underline should be lifted up as far as
|
|
|
|
* possible. Note that this does not mean the
|
|
|
|
* underline never overflows from this
|
|
|
|
* limitation. Because if the underline is
|
|
|
|
* positioned to the baseline or upper, it causes
|
|
|
|
* unreadability. Note that if this is zero
|
|
|
|
* or larger, the underline rect may be shrunken
|
|
|
|
* if it's possible. Therefore, this value is
|
|
|
|
* used for strikeout line and overline too.
|
2007-08-06 01:15:00 -07:00
|
|
|
*/
|
|
|
|
static void PaintDecorationLine(gfxContext* aGfxContext,
|
2011-09-16 12:23:29 -07:00
|
|
|
const gfxRect& aDirtyRect,
|
2007-08-06 01:15:00 -07:00
|
|
|
const nscolor aColor,
|
|
|
|
const gfxPoint& aPt,
|
|
|
|
const gfxSize& aLineSize,
|
|
|
|
const gfxFloat aAscent,
|
|
|
|
const gfxFloat aOffset,
|
|
|
|
const PRUint8 aDecoration,
|
2009-04-05 22:53:00 -07:00
|
|
|
const PRUint8 aStyle,
|
|
|
|
const gfxFloat aDescentLimit = -1.0);
|
2007-08-06 01:15:00 -07:00
|
|
|
|
2008-02-16 07:31:27 -08:00
|
|
|
/**
|
|
|
|
* Function for getting the decoration line rect for the text.
|
|
|
|
* NOTE: aLineSize, aAscent and aOffset are non-rounded device pixels,
|
|
|
|
* not app units.
|
|
|
|
* input:
|
|
|
|
* @param aPresContext
|
|
|
|
* @param aLineSize the width and the height of the decoration
|
|
|
|
* line
|
|
|
|
* @param aAscent the ascent of the text
|
|
|
|
* @param aOffset the offset of the decoration line from
|
|
|
|
* the baseline of the text (if the value is
|
|
|
|
* positive, the line is lifted up)
|
|
|
|
* @param aDecoration which line will be painted. The value can be
|
2011-04-22 22:16:41 -07:00
|
|
|
* NS_STYLE_TEXT_DECORATION_LINE_UNDERLINE or
|
|
|
|
* NS_STYLE_TEXT_DECORATION_LINE_OVERLINE or
|
|
|
|
* NS_STYLE_TEXT_DECORATION_LINE_LINE_THROUGH.
|
2011-03-31 05:26:49 -07:00
|
|
|
* @param aStyle the style of the decoration line such as
|
|
|
|
* NS_STYLE_TEXT_DECORATION_STYLE_*.
|
2009-04-05 22:53:00 -07:00
|
|
|
* @param aDescentLimit If aDescentLimit is zero or larger and the
|
|
|
|
* underline overflows from the descent space,
|
|
|
|
* the underline should be lifted up as far as
|
|
|
|
* possible. Note that this does not mean the
|
|
|
|
* underline never overflows from this
|
|
|
|
* limitation. Because if the underline is
|
|
|
|
* positioned to the baseline or upper, it causes
|
|
|
|
* unreadability. Note that if this is zero
|
|
|
|
* or larger, the underline rect may be shrunken
|
|
|
|
* if it's possible. Therefore, this value is
|
|
|
|
* used for strikeout line and overline too.
|
2008-02-16 07:31:27 -08:00
|
|
|
* output:
|
|
|
|
* @return the decoration line rect for the input,
|
|
|
|
* the each values are app units.
|
|
|
|
*/
|
|
|
|
static nsRect GetTextDecorationRect(nsPresContext* aPresContext,
|
|
|
|
const gfxSize& aLineSize,
|
|
|
|
const gfxFloat aAscent,
|
|
|
|
const gfxFloat aOffset,
|
|
|
|
const PRUint8 aDecoration,
|
2009-04-05 22:53:00 -07:00
|
|
|
const PRUint8 aStyle,
|
|
|
|
const gfxFloat aDescentLimit = -1.0);
|
2009-04-03 00:26:28 -07:00
|
|
|
|
|
|
|
protected:
|
|
|
|
static gfxRect GetTextDecorationRectInternal(const gfxPoint& aPt,
|
|
|
|
const gfxSize& aLineSize,
|
|
|
|
const gfxFloat aAscent,
|
|
|
|
const gfxFloat aOffset,
|
|
|
|
const PRUint8 aDecoration,
|
2009-04-05 22:53:00 -07:00
|
|
|
const PRUint8 aStyle,
|
|
|
|
const gfxFloat aDscentLimit);
|
2007-03-22 10:30:00 -07:00
|
|
|
};
|
|
|
|
|
2008-06-12 15:02:32 -07:00
|
|
|
/*
|
|
|
|
* nsContextBoxBlur
|
|
|
|
* Creates an 8-bit alpha channel context for callers to draw in, blurs the
|
|
|
|
* contents of that context and applies it as a 1-color mask on a
|
2008-09-25 12:53:09 -07:00
|
|
|
* different existing context. Uses gfxAlphaBoxBlur as its back end.
|
2008-06-12 15:02:32 -07:00
|
|
|
*
|
2008-09-25 09:03:15 -07:00
|
|
|
* You must call Init() first to create a suitable temporary surface to draw
|
|
|
|
* on. You must then draw any desired content onto the given context, then
|
|
|
|
* call DoPaint() to apply the blurred content as a single-color mask. You
|
|
|
|
* can only call Init() once, so objects cannot be reused.
|
2008-06-12 15:02:32 -07:00
|
|
|
*
|
|
|
|
* This is very useful for creating drop shadows or silhouettes.
|
|
|
|
*/
|
|
|
|
class nsContextBoxBlur {
|
|
|
|
public:
|
2010-08-23 02:30:07 -07:00
|
|
|
enum {
|
|
|
|
FORCE_MASK = 0x01
|
|
|
|
};
|
2008-06-12 15:02:32 -07:00
|
|
|
/**
|
2008-09-25 09:03:15 -07:00
|
|
|
* Prepares a gfxContext to draw on. Do not call this twice; if you want
|
|
|
|
* to get the gfxContext again use GetContext().
|
2008-06-12 15:02:32 -07:00
|
|
|
*
|
|
|
|
* @param aRect The coordinates of the surface to create.
|
|
|
|
* All coordinates must be in app units.
|
2008-09-25 09:03:15 -07:00
|
|
|
* This must not include the blur radius, pass
|
|
|
|
* it as the second parameter and everything
|
|
|
|
* is taken care of.
|
2008-06-12 15:02:32 -07:00
|
|
|
*
|
|
|
|
* @param aBlurRadius The blur radius in app units.
|
|
|
|
*
|
2008-09-25 09:03:15 -07:00
|
|
|
* @param aAppUnitsPerDevPixel The number of app units in a device pixel,
|
|
|
|
* for conversion. Most of the time you'll
|
|
|
|
* pass this from the current PresContext if
|
|
|
|
* available.
|
2008-06-12 15:02:32 -07:00
|
|
|
*
|
2008-09-25 09:03:15 -07:00
|
|
|
* @param aDestinationCtx The graphics context to apply the blurred
|
|
|
|
* mask to when you call DoPaint(). Make sure
|
|
|
|
* it is not destroyed before you call
|
|
|
|
* DoPaint(). To set the color of the
|
|
|
|
* resulting blurred graphic mask, you must
|
|
|
|
* set the color on this context before
|
|
|
|
* calling Init().
|
2008-06-12 15:02:32 -07:00
|
|
|
*
|
2008-12-02 13:16:22 -08:00
|
|
|
* @param aDirtyRect The absolute dirty rect in app units. Used to
|
|
|
|
* optimize the temporary surface size and speed up blur.
|
|
|
|
*
|
2010-04-17 20:13:10 -07:00
|
|
|
* @param aSkipRect An area in device pixels (NOT app units!) to avoid
|
|
|
|
* blurring over, to prevent unnecessary work.
|
2010-08-23 02:30:07 -07:00
|
|
|
*
|
|
|
|
* @param aFlags FORCE_MASK to ensure that the content drawn to the
|
|
|
|
* returned gfxContext is used as a mask, and not
|
|
|
|
* drawn directly to aDestinationCtx.
|
2010-04-17 20:13:10 -07:00
|
|
|
*
|
2008-09-25 09:03:15 -07:00
|
|
|
* @return A blank 8-bit alpha-channel-only graphics context to
|
|
|
|
* draw on, or null on error. Must not be freed. The
|
|
|
|
* context has a device offset applied to it given by
|
|
|
|
* aRect. This means you can use coordinates as if it
|
|
|
|
* were at the desired position at aRect and you don't
|
|
|
|
* need to worry about translating any coordinates to
|
|
|
|
* draw on this temporary surface.
|
2008-06-12 15:02:32 -07:00
|
|
|
*
|
2008-09-25 09:03:15 -07:00
|
|
|
* If aBlurRadius is 0, the returned context is aDestinationCtx and
|
|
|
|
* DoPaint() does nothing, because no blurring is required. Therefore, you
|
|
|
|
* should prepare the destination context as if you were going to draw
|
|
|
|
* directly on it instead of any temporary surface created in this class.
|
2008-06-12 15:02:32 -07:00
|
|
|
*/
|
2010-08-23 02:30:07 -07:00
|
|
|
gfxContext* Init(const nsRect& aRect, nscoord aSpreadRadius,
|
|
|
|
nscoord aBlurRadius,
|
2008-12-02 13:16:22 -08:00
|
|
|
PRInt32 aAppUnitsPerDevPixel, gfxContext* aDestinationCtx,
|
2010-08-23 02:30:07 -07:00
|
|
|
const nsRect& aDirtyRect, const gfxRect* aSkipRect,
|
|
|
|
PRUint32 aFlags = 0);
|
2008-06-12 15:02:32 -07:00
|
|
|
|
2010-08-23 02:30:07 -07:00
|
|
|
/**
|
|
|
|
* Does the actual blurring/spreading. Users of this object *must*
|
|
|
|
* have called Init() first, then have drawn whatever they want to be
|
|
|
|
* blurred onto the internal gfxContext before calling this.
|
|
|
|
*/
|
|
|
|
void DoEffects();
|
|
|
|
|
2008-06-12 15:02:32 -07:00
|
|
|
/**
|
|
|
|
* Does the actual blurring and mask applying. Users of this object *must*
|
|
|
|
* have called Init() first, then have drawn whatever they want to be
|
|
|
|
* blurred onto the internal gfxContext before calling this.
|
|
|
|
*/
|
|
|
|
void DoPaint();
|
|
|
|
|
|
|
|
/**
|
2008-09-25 09:03:15 -07:00
|
|
|
* Gets the internal gfxContext at any time. Must not be freed. Avoid
|
|
|
|
* calling this before calling Init() since the context would not be
|
|
|
|
* constructed at that point.
|
2008-06-12 15:02:32 -07:00
|
|
|
*/
|
|
|
|
gfxContext* GetContext();
|
|
|
|
|
2010-09-11 09:27:12 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the margin associated with the given blur radius, i.e., the
|
|
|
|
* additional area that might be painted as a result of it. (The
|
|
|
|
* margin for a spread radius is itself, on all sides.)
|
|
|
|
*/
|
|
|
|
static nsMargin GetBlurRadiusMargin(nscoord aBlurRadius,
|
|
|
|
PRInt32 aAppUnitsPerDevPixel);
|
|
|
|
|
2008-06-12 15:02:32 -07:00
|
|
|
protected:
|
2008-09-25 12:53:09 -07:00
|
|
|
gfxAlphaBoxBlur blur;
|
2008-06-12 15:02:32 -07:00
|
|
|
nsRefPtr<gfxContext> mContext;
|
|
|
|
gfxContext* mDestinationCtx;
|
2012-05-16 13:45:25 -07:00
|
|
|
|
|
|
|
/* This is true if the blur already has it's content transformed
|
|
|
|
* by mDestinationCtx's transform */
|
|
|
|
bool mPreTransformed;
|
|
|
|
|
2008-06-12 15:02:32 -07:00
|
|
|
};
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
#endif /* nsCSSRendering_h___ */
|