blob: 271f8f4001edce730d8fbcf12c80234c227d3cf1 [file] [log] [blame]
scroggo19b91532016-10-24 09:03:26 -07001/*
2 * Copyright 2016 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8#include "gm.h"
Matt Sarett68b8e3d2017-04-28 11:15:22 -04009#include "sk_tool_utils.h"
scroggo19b91532016-10-24 09:03:26 -070010#include "SkAnimTimer.h"
11#include "SkCanvas.h"
12#include "SkCodec.h"
13#include "SkColor.h"
14#include "SkCommandLineFlags.h"
Mike Reed17c574a2018-12-12 18:10:38 -050015#include "SkFont.h"
scroggo19b91532016-10-24 09:03:26 -070016#include "SkPaint.h"
17#include "SkString.h"
18#include "Resources.h"
19
20#include <vector>
21
Hal Canaryc465d132017-12-08 10:21:31 -050022DEFINE_string(animatedGif, "images/test640x479.gif", "Animated gif in resources folder");
scroggo19b91532016-10-24 09:03:26 -070023
24namespace {
25 void error(SkCanvas* canvas, const SkString& errorText) {
26 constexpr SkScalar kOffset = 5.0f;
27 canvas->drawColor(SK_ColorRED);
28 SkPaint paint;
Mike Reed17c574a2018-12-12 18:10:38 -050029 SkFont font;
scroggo19b91532016-10-24 09:03:26 -070030 SkRect bounds;
Mike Reed17c574a2018-12-12 18:10:38 -050031 font.measureText(errorText.c_str(), errorText.size(), kUTF8_SkTextEncoding, &bounds);
Hal Canary89a644b2019-01-07 09:36:09 -050032 canvas->drawString(errorText, kOffset, bounds.height() + kOffset, font, paint);
scroggo19b91532016-10-24 09:03:26 -070033 }
34}
35
36class AnimatedGifGM : public skiagm::GM {
37private:
38 std::unique_ptr<SkCodec> fCodec;
Leon Scroggins III249b8e32017-04-17 12:46:33 -040039 int fFrame;
scroggo19b91532016-10-24 09:03:26 -070040 double fNextUpdate;
Leon Scroggins III249b8e32017-04-17 12:46:33 -040041 int fTotalFrames;
scroggo19b91532016-10-24 09:03:26 -070042 std::vector<SkCodec::FrameInfo> fFrameInfos;
43 std::vector<SkBitmap> fFrames;
44
45 void drawFrame(SkCanvas* canvas, int frameIndex) {
46 // FIXME: Create from an Image/ImageGenerator?
47 if (frameIndex >= (int) fFrames.size()) {
48 fFrames.resize(frameIndex + 1);
49 }
50 SkBitmap& bm = fFrames[frameIndex];
51 if (!bm.getPixels()) {
52 const SkImageInfo info = fCodec->getInfo().makeColorType(kN32_SkColorType);
53 bm.allocPixels(info);
54
55 SkCodec::Options opts;
56 opts.fFrameIndex = frameIndex;
Leon Scroggins III249b8e32017-04-17 12:46:33 -040057 const int requiredFrame = fFrameInfos[frameIndex].fRequiredFrame;
Nigel Tao66bc5242018-08-22 10:56:03 +100058 if (requiredFrame != SkCodec::kNoFrame) {
Leon Scroggins III249b8e32017-04-17 12:46:33 -040059 SkASSERT(requiredFrame >= 0
60 && static_cast<size_t>(requiredFrame) < fFrames.size());
scroggo19b91532016-10-24 09:03:26 -070061 SkBitmap& requiredBitmap = fFrames[requiredFrame];
62 // For simplicity, do not try to cache old frames
Matt Sarett68b8e3d2017-04-28 11:15:22 -040063 if (requiredBitmap.getPixels() &&
64 sk_tool_utils::copy_to(&bm, requiredBitmap.colorType(), requiredBitmap)) {
Leon Scroggins III33deb7e2017-06-07 12:31:51 -040065 opts.fPriorFrame = requiredFrame;
scroggo19b91532016-10-24 09:03:26 -070066 }
67 }
68
69 if (SkCodec::kSuccess != fCodec->getPixels(info, bm.getPixels(),
Leon Scroggins571b30f2017-07-11 17:35:31 +000070 bm.rowBytes(), &opts)) {
scroggo19b91532016-10-24 09:03:26 -070071 SkDebugf("Could not getPixels for frame %i: %s", frameIndex, FLAGS_animatedGif[0]);
72 return;
73 }
74 }
75
76 canvas->drawBitmap(bm, 0, 0);
77 }
78
79public:
80 AnimatedGifGM()
81 : fFrame(0)
82 , fNextUpdate (-1)
83 , fTotalFrames (-1) {}
84
85private:
86 SkString onShortName() override {
87 return SkString("animatedGif");
88 }
89
90 SkISize onISize() override {
91 if (this->initCodec()) {
92 SkISize dim = fCodec->getInfo().dimensions();
93 // Wide enough to display all the frames.
94 dim.fWidth *= fTotalFrames;
95 // Tall enough to show the row of frames plus an animating version.
96 dim.fHeight *= 2;
97 return dim;
98 }
99 return SkISize::Make(640, 480);
100 }
101
102 void onDrawBackground(SkCanvas* canvas) override {
scroggocf280a42016-10-24 09:59:53 -0700103 canvas->clear(SK_ColorWHITE);
scroggo19b91532016-10-24 09:03:26 -0700104 if (this->initCodec()) {
105 SkAutoCanvasRestore acr(canvas, true);
Leon Scroggins III249b8e32017-04-17 12:46:33 -0400106 for (int frameIndex = 0; frameIndex < fTotalFrames; frameIndex++) {
scroggo19b91532016-10-24 09:03:26 -0700107 this->drawFrame(canvas, frameIndex);
Jim Van Verth3cfdf6c2016-10-26 09:45:23 -0400108 canvas->translate(SkIntToScalar(fCodec->getInfo().width()), 0);
scroggo19b91532016-10-24 09:03:26 -0700109 }
110 }
111 }
112
113 bool initCodec() {
114 if (fCodec) {
115 return true;
116 }
117 if (FLAGS_animatedGif.isEmpty()) {
118 SkDebugf("Nothing specified for --animatedGif!");
119 return false;
120 }
121
122 std::unique_ptr<SkStream> stream(GetResourceAsStream(FLAGS_animatedGif[0]));
123 if (!stream) {
124 return false;
125 }
126
Mike Reedede7bac2017-07-23 15:30:02 -0400127 fCodec = SkCodec::MakeFromStream(std::move(stream));
scroggo19b91532016-10-24 09:03:26 -0700128 if (!fCodec) {
129 SkDebugf("Could create codec from %s", FLAGS_animatedGif[0]);
130 return false;
131 }
132
133 fFrame = 0;
134 fFrameInfos = fCodec->getFrameInfo();
135 fTotalFrames = fFrameInfos.size();
136 return true;
137 }
138
139 void onDraw(SkCanvas* canvas) override {
140 if (!fCodec) {
141 SkString errorText = SkStringPrintf("Nothing to draw; %s", FLAGS_animatedGif[0]);
142 error(canvas, errorText);
143 return;
144 }
145
146 SkAutoCanvasRestore acr(canvas, true);
Jim Van Verth3cfdf6c2016-10-26 09:45:23 -0400147 canvas->translate(0, SkIntToScalar(fCodec->getInfo().height()));
scroggo19b91532016-10-24 09:03:26 -0700148 this->drawFrame(canvas, fFrame);
149 }
150
151 bool onAnimate(const SkAnimTimer& timer) override {
152 if (!fCodec || fTotalFrames == 1) {
153 return false;
154 }
155
156 double secs = timer.msec() * .1;
157 if (fNextUpdate < double(0)) {
158 // This is a sentinel that we have not done any updates yet.
159 // I'm assuming this gets called *after* onOnceBeforeDraw, so our first frame should
160 // already have been retrieved.
161 SkASSERT(fFrame == 0);
162 fNextUpdate = secs + fFrameInfos[fFrame].fDuration;
163
164 return true;
165 }
166
167 if (secs < fNextUpdate) {
168 return true;
169 }
170
171 while (secs >= fNextUpdate) {
172 // Retrieve the next frame.
173 fFrame++;
174 if (fFrame == fTotalFrames) {
175 fFrame = 0;
176 }
177
178 // Note that we loop here. This is not safe if we need to draw the intermediate frame
179 // in order to draw correctly.
180 fNextUpdate += fFrameInfos[fFrame].fDuration;
181 }
182
183 return true;
184 }
185};
scroggo19b91532016-10-24 09:03:26 -0700186DEF_GM(return new AnimatedGifGM);
187
Mike Reed964aedb2018-10-03 10:16:55 -0400188
189#include "SkAnimCodecPlayer.h"
190#include "SkOSFile.h"
191#include "SkMakeUnique.h"
192
193static std::unique_ptr<SkCodec> load_codec(const char filename[]) {
194 return SkCodec::MakeFromData(SkData::MakeFromFileName(filename));
195}
196
197class AnimCodecPlayerGM : public skiagm::GM {
198private:
199 std::vector<std::unique_ptr<SkAnimCodecPlayer> > fPlayers;
200 uint32_t fBaseMSec = 0;
201
202public:
203 AnimCodecPlayerGM() {
204 const char* root = "/skia/anim/";
205 SkOSFile::Iter iter(root);
206 SkString path;
207 while (iter.next(&path)) {
208 SkString completepath;
209 completepath.printf("%s%s", root, path.c_str());
210 auto codec = load_codec(completepath.c_str());
211 if (codec) {
212 fPlayers.push_back(skstd::make_unique<SkAnimCodecPlayer>(std::move(codec)));
213 }
214 }
215 }
216
217private:
218 SkString onShortName() override {
219 return SkString("AnimCodecPlayer");
220 }
221
222 SkISize onISize() override {
223 return { 1024, 768 };
224 }
225
226 void onDraw(SkCanvas* canvas) override {
227 canvas->scale(0.25f, 0.25f);
228 for (auto& p : fPlayers) {
229 canvas->drawImage(p->getFrame(), 0, 0, nullptr);
230 canvas->translate(p->dimensions().width(), 0);
231 }
232 }
233
234 bool onAnimate(const SkAnimTimer& timer) override {
235 if (fBaseMSec == 0) {
236 fBaseMSec = timer.msec();
237 }
238 for (auto& p : fPlayers) {
239 (void)p->seek(timer.msec() - fBaseMSec);
240 }
241 return true;
242 }
243};
244DEF_GM(return new AnimCodecPlayerGM);