Bug 1855375: Basic implementation for Yelp Suggestions r=fluent-reviewers,flod,adw
[gecko.git] / image / Downscaler.h
blobe5fcf0387e1efb4b6886be0cea987eae473cef0f
1 /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
3 * This Source Code Form is subject to the terms of the Mozilla Public
4 * License, v. 2.0. If a copy of the MPL was not distributed with this
5 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
7 /**
8 * Downscaler is a high-quality, streaming image downscaler based upon Skia's
9 * scaling implementation.
12 #ifndef mozilla_image_Downscaler_h
13 #define mozilla_image_Downscaler_h
15 #include "mozilla/Maybe.h"
16 #include "mozilla/UniquePtr.h"
17 #include "gfxPoint.h"
18 #include "nsRect.h"
19 #include "mozilla/gfx/ConvolutionFilter.h"
20 #include "mozilla/gfx/Matrix.h"
22 namespace mozilla {
23 namespace image {
25 /**
26 * DownscalerInvalidRect wraps two invalidation rects: one in terms of the
27 * original image size, and one in terms of the target size.
29 struct DownscalerInvalidRect {
30 nsIntRect mOriginalSizeRect;
31 nsIntRect mTargetSizeRect;
34 /**
35 * Downscaler is a high-quality, streaming image downscaler based upon Skia's
36 * scaling implementation.
38 * Decoders can construct a Downscaler once they know their target size, then
39 * call BeginFrame() for each frame they decode. They should write a decoded row
40 * into the buffer returned by RowBuffer(), and then call CommitRow() to signal
41 * that they have finished.
44 * Because invalidations need to be computed in terms of the scaled version of
45 * the image, Downscaler also tracks them. Decoders can call HasInvalidation()
46 * and TakeInvalidRect() instead of tracking invalidations themselves.
48 class Downscaler {
49 public:
50 /// Constructs a new Downscaler which to scale to size @aTargetSize.
51 explicit Downscaler(const nsIntSize& aTargetSize);
52 ~Downscaler();
54 const nsIntSize& OriginalSize() const { return mOriginalSize; }
55 const nsIntSize& TargetSize() const { return mTargetSize; }
56 const nsIntSize FrameSize() const {
57 return nsIntSize(mFrameRect.Width(), mFrameRect.Height());
59 const gfx::MatrixScalesDouble& Scale() const { return mScale; }
61 /**
62 * Begins a new frame and reinitializes the Downscaler.
64 * @param aOriginalSize The original size of this frame, before scaling.
65 * @param aFrameRect The region of the original image which has data.
66 * Every pixel outside @aFrameRect is considered blank and
67 * has zero alpha.
68 * @param aOutputBuffer The buffer to which the Downscaler should write its
69 * output; this is the same buffer where the Decoder
70 * would write its output when not downscaling during
71 * decode.
72 * @param aHasAlpha Whether or not this frame has an alpha channel.
73 * Performance is a little better if it doesn't have one.
74 * @param aFlipVertically If true, output rows will be written to the output
75 * buffer in reverse order vertically, which matches
76 * the way they are stored in some image formats.
78 nsresult BeginFrame(const nsIntSize& aOriginalSize,
79 const Maybe<nsIntRect>& aFrameRect,
80 uint8_t* aOutputBuffer, bool aHasAlpha,
81 bool aFlipVertically = false);
83 bool IsFrameComplete() const {
84 return mCurrentInLine >= mOriginalSize.height;
87 /// Retrieves the buffer into which the Decoder should write each row.
88 uint8_t* RowBuffer() {
89 return mRowBuffer.get() + mFrameRect.X() * sizeof(uint32_t);
92 /// Clears the current row buffer.
93 void ClearRow() { ClearRestOfRow(0); }
95 /// Clears the current row buffer starting at @aStartingAtCol.
96 void ClearRestOfRow(uint32_t aStartingAtCol);
98 /// Signals that the decoder has finished writing a row into the row buffer.
99 void CommitRow();
101 /// Returns true if there is a non-empty invalid rect available.
102 bool HasInvalidation() const;
104 /// Takes the Downscaler's current invalid rect and resets it.
105 DownscalerInvalidRect TakeInvalidRect();
108 * Resets the Downscaler's position in the image, for a new progressive pass
109 * over the same frame. Because the same data structures can be reused, this
110 * is more efficient than calling BeginFrame.
112 void ResetForNextProgressivePass();
114 private:
115 void DownscaleInputLine();
116 void ReleaseWindow();
117 void SkipToRow(int32_t aRow);
119 nsIntSize mOriginalSize;
120 nsIntSize mTargetSize;
121 nsIntRect mFrameRect;
122 gfx::MatrixScalesDouble mScale;
124 uint8_t* mOutputBuffer;
126 UniquePtr<uint8_t[]> mRowBuffer;
127 UniquePtr<uint8_t*[]> mWindow;
129 gfx::ConvolutionFilter mXFilter;
130 gfx::ConvolutionFilter mYFilter;
132 int32_t mWindowCapacity;
134 int32_t mLinesInBuffer;
135 int32_t mPrevInvalidatedLine;
136 int32_t mCurrentOutLine;
137 int32_t mCurrentInLine;
139 bool mHasAlpha : 1;
140 bool mFlipVertically : 1;
143 } // namespace image
144 } // namespace mozilla
146 #endif // mozilla_image_Downscaler_h