| /* |
| * Copyright 2015 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "SampleCode.h" |
| #include "Resources.h" |
| #include "SkAnimTimer.h" |
| #include "SkView.h" |
| #include "SkCanvas.h" |
| #include "SkRSXform.h" |
| #include "SkSurface.h" |
| #include "Timer.h" |
| |
| #include <stdio.h> |
| |
| static const int kGrid = 100; |
| static const int kWidth = 960; |
| static const int kHeight = 640; |
| |
| typedef void (*DrawAtlasProc)(SkCanvas*, SkImage*, const SkRSXform[], const SkRect[], |
| const SkColor[], int, const SkRect*, const SkPaint*); |
| |
| static void draw_atlas(SkCanvas* canvas, SkImage* atlas, const SkRSXform xform[], |
| const SkRect tex[], const SkColor colors[], int count, const SkRect* cull, |
| const SkPaint* paint) { |
| canvas->drawAtlas(atlas, xform, tex, colors, count, SkXfermode::kModulate_Mode, cull, paint); |
| } |
| |
| static void draw_atlas_sim(SkCanvas* canvas, SkImage* atlas, const SkRSXform xform[], |
| const SkRect tex[], const SkColor colors[], int count, const SkRect* cull, |
| const SkPaint* paint) { |
| for (int i = 0; i < count; ++i) { |
| SkMatrix matrix; |
| matrix.setRSXform(xform[i]); |
| |
| canvas->save(); |
| canvas->concat(matrix); |
| canvas->drawImageRect(atlas, tex[i], tex[i].makeOffset(-tex[i].x(), -tex[i].y()), paint, |
| SkCanvas::kFast_SrcRectConstraint); |
| canvas->restore(); |
| } |
| } |
| |
| |
| class DrawShipView : public SampleView { |
| public: |
| DrawShipView(const char name[], DrawAtlasProc proc) : fName(name), fProc(proc) { |
| fAtlas.reset(GetResourceAsImage("ship.png")); |
| if (!fAtlas) { |
| SkDebugf("\nCould not decode file ship.png. Falling back to penguin mode.\n"); |
| fAtlas.reset(GetResourceAsImage("baby_tux.png")); |
| if (!fAtlas) { |
| SkDebugf("\nCould not decode file baby_tux.png. Did you forget" |
| " to set the resourcePath?\n"); |
| return; |
| } |
| } |
| |
| SkScalar anchorX = fAtlas->width()*0.5f; |
| SkScalar anchorY = fAtlas->height()*0.5f; |
| int currIndex = 0; |
| for (int x = 0; x < kGrid; x++) { |
| for (int y = 0; y < kGrid; y++) { |
| float xPos = (x / (kGrid - 1.0f)) * kWidth; |
| float yPos = (y / (kGrid - 1.0f)) * kWidth; |
| |
| fTex[currIndex] = SkRect::MakeLTRB(0.0f, 0.0f, |
| SkIntToScalar(fAtlas->width()), |
| SkIntToScalar(fAtlas->height())); |
| fXform[currIndex] = SkRSXform::MakeFromRadians(2.0f, SK_ScalarPI*0.5f, |
| xPos, yPos, anchorX, anchorY); |
| currIndex++; |
| } |
| } |
| fTex[currIndex] = SkRect::MakeLTRB(0.0f, 0.0f, |
| SkIntToScalar(fAtlas->width()), |
| SkIntToScalar(fAtlas->height())); |
| fXform[currIndex] = SkRSXform::MakeFromRadians(2.0f, SK_ScalarPI*0.5f, |
| kWidth*0.5f, kHeight*0.5f, anchorX, anchorY); |
| |
| fCurrentTime = 0; |
| fTimer.start(); |
| } |
| |
| ~DrawShipView() override {} |
| |
| protected: |
| // overrides from SkEventSink |
| bool onQuery(SkEvent* evt) override { |
| if (SampleCode::TitleQ(*evt)) { |
| SampleCode::TitleR(evt, fName); |
| return true; |
| } |
| return this->INHERITED::onQuery(evt); |
| } |
| |
| void onDrawContent(SkCanvas* canvas) override { |
| const float kCosDiff = 0.99984769515f; |
| const float kSinDiff = 0.01745240643f; |
| |
| if (!fAtlas) { |
| return; |
| } |
| |
| SkPaint paint; |
| paint.setFilterQuality(kLow_SkFilterQuality); |
| paint.setColor(SK_ColorWHITE); |
| paint.setTextSize(15.0f); |
| |
| fTimer.end(); |
| |
| fTimes[fCurrentTime] = (float)(fTimer.fWall); |
| fCurrentTime = (fCurrentTime + 1) & 0x1f; |
| |
| float meanTime = 0.0f; |
| for (int i = 0; i < 32; ++i) { |
| meanTime += fTimes[i]; |
| } |
| meanTime /= 32.f; |
| SkString outString("fps: "); |
| SkScalar fps = 1000.f/meanTime; |
| outString.appendScalar(fps); |
| outString.append(" ms: "); |
| outString.appendScalar(meanTime); |
| |
| fTimer.start(); |
| |
| SkScalar anchorX = fAtlas->width()*0.5f; |
| SkScalar anchorY = fAtlas->height()*0.5f; |
| for (int i = 0; i < kGrid*kGrid+1; ++i) { |
| SkScalar c = fXform[i].fSCos; |
| SkScalar s = fXform[i].fSSin; |
| |
| SkScalar dx = c*anchorX - s*anchorY; |
| SkScalar dy = s*anchorX + c*anchorY; |
| |
| fXform[i].fSCos = kCosDiff*c - kSinDiff*s; |
| fXform[i].fSSin = kSinDiff*c + kCosDiff*s; |
| |
| dx -= fXform[i].fSCos*anchorX - fXform[i].fSSin*anchorY; |
| dy -= fXform[i].fSSin*anchorX + fXform[i].fSCos*anchorY; |
| fXform[i].fTx += dx; |
| fXform[i].fTy += dy; |
| } |
| |
| fProc(canvas, fAtlas, fXform, fTex, nullptr, kGrid*kGrid+1, nullptr, &paint); |
| canvas->drawText(outString.c_str(), outString.size(), 100.f, 100.f, paint); |
| |
| this->inval(nullptr); |
| } |
| |
| #if 0 |
| // TODO: switch over to use this for our animation |
| bool onAnimate(const SkAnimTimer& timer) override { |
| SkScalar angle = SkDoubleToScalar(fmod(timer.secs() * 360 / 24, 360)); |
| fAnimatingDrawable->setSweep(angle); |
| return true; |
| } |
| #endif |
| |
| private: |
| const char* fName; |
| DrawAtlasProc fProc; |
| |
| SkAutoTUnref<SkImage> fAtlas; |
| SkRSXform fXform[kGrid*kGrid+1]; |
| SkRect fTex[kGrid*kGrid+1]; |
| WallTimer fTimer; |
| float fTimes[32]; |
| int fCurrentTime; |
| |
| |
| typedef SampleView INHERITED; |
| }; |
| |
| ////////////////////////////////////////////////////////////////////////////// |
| |
| DEF_SAMPLE( return new DrawShipView("DrawShip", draw_atlas); ) |
| DEF_SAMPLE( return new DrawShipView("DrawShipSim", draw_atlas_sim); ) |