blob: c33a7fc0fcd6c73778c4447107fe677fb56e59cd [file] [log] [blame]
reed@android.com8a1c16f2008-12-17 15:59:43 +00001#include "SkCanvas.h"
2#include "SkDevice.h"
reed@google.comac10a2d2010-12-22 21:39:39 +00003#include "SkGpuCanvas.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +00004#include "SkGraphics.h"
reed@android.comb08eb2b2009-01-06 20:16:26 +00005#include "SkImageEncoder.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +00006#include "SkPaint.h"
7#include "SkPicture.h"
8#include "SkStream.h"
reed@android.com44177402009-11-23 21:07:51 +00009#include "SkTime.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000010#include "SkWindow.h"
11
12#include "SampleCode.h"
reed@google.comac10a2d2010-12-22 21:39:39 +000013#include "GrContext.h"
reed@android.comf2b98d62010-12-20 18:26:13 +000014
15//#define DEFAULT_TO_GPU
16
reed@android.come191b162009-12-18 21:33:39 +000017extern SkView* create_overview(int, const SkViewFactory[]);
reed@android.com34245c72009-11-03 04:00:48 +000018
reed@android.comcb342352010-07-22 18:27:53 +000019#define SK_SUPPORT_GL
reed@android.comf2b98d62010-12-20 18:26:13 +000020//#define SK_SUPPORT_D3D9
reed@android.com8a1c16f2008-12-17 15:59:43 +000021
22#define ANIMATING_EVENTTYPE "nextSample"
23#define ANIMATING_DELAY 750
24
reed@android.comf2b98d62010-12-20 18:26:13 +000025#if !defined(SK_BUILD_FOR_WIN32)
26//#define USE_OFFSCREEN
27#endif
reed@android.com8a1c16f2008-12-17 15:59:43 +000028
reed@google.comac10a2d2010-12-22 21:39:39 +000029#ifdef SK_SUPPORT_GL
30 #include "GrGLConfig.h"
31#elif defined(SK_SUPPORT_D3D9)
32 #include <d3d9.h>
33#endif
34
reed@android.com8a1c16f2008-12-17 15:59:43 +000035SkViewRegister* SkViewRegister::gHead;
36SkViewRegister::SkViewRegister(SkViewFactory fact) : fFact(fact) {
37 static bool gOnce;
38 if (!gOnce) {
39 gHead = NULL;
40 gOnce = true;
41 }
42
43 fChain = gHead;
44 gHead = this;
45}
46
reed@android.comf2b98d62010-12-20 18:26:13 +000047#if defined(SK_SUPPORT_GL) && defined(SK_SUPPORT_D3D9)
48 #error "choose either GL or D3D9"
reed@android.com8a1c16f2008-12-17 15:59:43 +000049#endif
50
reed@android.comf2b98d62010-12-20 18:26:13 +000051#if defined(SK_SUPPORT_GL)
52 #define SK_USE_SHADERS
53#endif
54
55static GrContext* get_global_grctx(SkOSWindow* oswin) {
reed@android.comf2b98d62010-12-20 18:26:13 +000056 // should be pthread-local at least
57 static GrContext* ctx;
58 if (NULL == ctx) {
59#if defined(SK_SUPPORT_GL)
60 #if defined(SK_USE_SHADERS)
61 ctx = GrContext::Create(GrGpu::kOpenGL_Shaders_Engine, NULL);
62 #else
63 ctx = GrContext::Create(GrGpu::kOpenGL_Fixed_Engine, NULL);
64 #endif
65#elif defined(SK_SUPPORT_D3D9)
66 if (oswin->d3d9Device()) {
67 ctx = GrContext::Create(GrGpu::kDirect3D9_Engine,
68 (IDirect3DDevice9*) oswin->d3d9Device());
69 }
70#endif
reed@android.com8a1c16f2008-12-17 15:59:43 +000071 }
reed@android.comf2b98d62010-12-20 18:26:13 +000072 return ctx;
reed@android.comf2b98d62010-12-20 18:26:13 +000073}
reed@android.com8a1c16f2008-12-17 15:59:43 +000074
75//////////////////////////////////////////////////////////////////////////////
76
reed@android.comf2b98d62010-12-20 18:26:13 +000077static const char gCharEvtName[] = "SampleCode_Char_Event";
78static const char gKeyEvtName[] = "SampleCode_Key_Event";
reed@android.com8a1c16f2008-12-17 15:59:43 +000079static const char gTitleEvtName[] = "SampleCode_Title_Event";
80static const char gPrefSizeEvtName[] = "SampleCode_PrefSize_Event";
reed@android.comf2b98d62010-12-20 18:26:13 +000081static const char gFastTextEvtName[] = "SampleCode_FastText_Event";
82
83bool SampleCode::CharQ(const SkEvent& evt, SkUnichar* outUni) {
84 if (evt.isType(gCharEvtName, sizeof(gCharEvtName) - 1)) {
85 if (outUni) {
86 *outUni = evt.getFast32();
87 }
88 return true;
89 }
90 return false;
91}
92
93bool SampleCode::KeyQ(const SkEvent& evt, SkKey* outKey) {
94 if (evt.isType(gKeyEvtName, sizeof(gKeyEvtName) - 1)) {
95 if (outKey) {
96 *outKey = (SkKey)evt.getFast32();
97 }
98 return true;
99 }
100 return false;
101}
reed@android.com8a1c16f2008-12-17 15:59:43 +0000102
103bool SampleCode::TitleQ(const SkEvent& evt) {
104 return evt.isType(gTitleEvtName, sizeof(gTitleEvtName) - 1);
105}
106
107void SampleCode::TitleR(SkEvent* evt, const char title[]) {
108 SkASSERT(evt && TitleQ(*evt));
109 evt->setString(gTitleEvtName, title);
110}
111
112bool SampleCode::PrefSizeQ(const SkEvent& evt) {
113 return evt.isType(gPrefSizeEvtName, sizeof(gPrefSizeEvtName) - 1);
114}
115
116void SampleCode::PrefSizeR(SkEvent* evt, SkScalar width, SkScalar height) {
117 SkASSERT(evt && PrefSizeQ(*evt));
118 SkScalar size[2];
119 size[0] = width;
120 size[1] = height;
121 evt->setScalars(gPrefSizeEvtName, 2, size);
122}
123
reed@android.comf2b98d62010-12-20 18:26:13 +0000124bool SampleCode::FastTextQ(const SkEvent& evt) {
125 return evt.isType(gFastTextEvtName, sizeof(gFastTextEvtName) - 1);
126}
127
128///////////////////////////////////////////////////////////////////////////////
129
reed@android.com44177402009-11-23 21:07:51 +0000130static SkMSec gAnimTime;
reed@android.comf2b98d62010-12-20 18:26:13 +0000131static SkMSec gAnimTimePrev;
132
reed@android.com44177402009-11-23 21:07:51 +0000133SkMSec SampleCode::GetAnimTime() { return gAnimTime; }
reed@android.comf2b98d62010-12-20 18:26:13 +0000134SkMSec SampleCode::GetAnimTimeDelta() { return gAnimTime - gAnimTimePrev; }
135SkScalar SampleCode::GetAnimSecondsDelta() {
136 return SkDoubleToScalar(GetAnimTimeDelta() / 1000.0);
137}
reed@android.com44177402009-11-23 21:07:51 +0000138
139SkScalar SampleCode::GetAnimScalar(SkScalar speed, SkScalar period) {
reed@android.comf2b98d62010-12-20 18:26:13 +0000140 // since gAnimTime can be up to 32 bits, we can't convert it to a float
141 // or we'll lose the low bits. Hence we use doubles for the intermediate
142 // calculations
143 double seconds = (double)gAnimTime / 1000.0;
144 double value = SkScalarToDouble(speed) * seconds;
reed@android.com44177402009-11-23 21:07:51 +0000145 if (period) {
reed@android.comf2b98d62010-12-20 18:26:13 +0000146 value = ::fmod(value, SkScalarToDouble(period));
reed@android.com44177402009-11-23 21:07:51 +0000147 }
reed@android.comf2b98d62010-12-20 18:26:13 +0000148 return SkDoubleToScalar(value);
reed@android.com44177402009-11-23 21:07:51 +0000149}
150
reed@android.com8a1c16f2008-12-17 15:59:43 +0000151//////////////////////////////////////////////////////////////////////////////
152
reed@android.comf2b98d62010-12-20 18:26:13 +0000153static SkView* curr_view(SkWindow* wind) {
154 SkView::F2BIter iter(wind);
155 return iter.next();
156}
157
reed@android.com8a1c16f2008-12-17 15:59:43 +0000158class SampleWindow : public SkOSWindow {
reed@android.com34245c72009-11-03 04:00:48 +0000159 SkTDArray<SkViewFactory> fSamples;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000160public:
161 SampleWindow(void* hwnd);
162 virtual ~SampleWindow();
163
reed@android.come522ca52009-11-23 20:10:41 +0000164 virtual void draw(SkCanvas* canvas);
165
reed@android.com8a1c16f2008-12-17 15:59:43 +0000166protected:
167 virtual void onDraw(SkCanvas* canvas);
168 virtual bool onHandleKey(SkKey key);
169 virtual bool onHandleChar(SkUnichar);
170 virtual void onSizeChange();
171
172 virtual SkCanvas* beforeChildren(SkCanvas*);
173 virtual void afterChildren(SkCanvas*);
reed@android.com6c5f6f22009-08-14 16:08:38 +0000174 virtual void beforeChild(SkView* child, SkCanvas* canvas);
175 virtual void afterChild(SkView* child, SkCanvas* canvas);
176
reed@android.com8a1c16f2008-12-17 15:59:43 +0000177 virtual bool onEvent(const SkEvent& evt);
reed@android.comf2b98d62010-12-20 18:26:13 +0000178 virtual bool onQuery(SkEvent* evt);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000179
180#if 0
181 virtual bool handleChar(SkUnichar uni);
182 virtual bool handleEvent(const SkEvent& evt);
183 virtual bool handleKey(SkKey key);
184 virtual bool handleKeyUp(SkKey key);
185
186 virtual bool onClick(Click* click);
187 virtual Click* onFindClickHandler(SkScalar x, SkScalar y);
188 virtual bool onHandleKeyUp(SkKey key);
189#endif
190private:
reed@android.com34245c72009-11-03 04:00:48 +0000191 int fCurrIndex;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000192
193 SkPicture* fPicture;
reed@android.comf2b98d62010-12-20 18:26:13 +0000194 SkGpuCanvas* fGpuCanvas;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000195 SkPath fClipPath;
196
197 enum CanvasType {
198 kRaster_CanvasType,
199 kPicture_CanvasType,
reed@android.comf2b98d62010-12-20 18:26:13 +0000200 kGPU_CanvasType
reed@android.com8a1c16f2008-12-17 15:59:43 +0000201 };
202 CanvasType fCanvasType;
203
204 bool fUseClip;
reed@android.come522ca52009-11-23 20:10:41 +0000205 bool fNClip;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000206 bool fRepeatDrawing;
207 bool fAnimating;
reed@android.com6c5f6f22009-08-14 16:08:38 +0000208 bool fRotate;
209 bool fScale;
reed@android.comf2b98d62010-12-20 18:26:13 +0000210 bool fRequestGrabImage;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000211
212 int fScrollTestX, fScrollTestY;
213
214 void loadView(SkView*);
215 void updateTitle();
216 bool nextSample();
217
218 void postAnimatingEvent() {
219 if (fAnimating) {
220 SkEvent* evt = new SkEvent(ANIMATING_EVENTTYPE);
221 evt->post(this->getSinkID(), ANIMATING_DELAY);
222 }
223 }
224
225
226 static CanvasType cycle_canvastype(CanvasType);
227
228 typedef SkOSWindow INHERITED;
229};
230
231SampleWindow::CanvasType SampleWindow::cycle_canvastype(CanvasType ct) {
232 static const CanvasType gCT[] = {
233 kPicture_CanvasType,
reed@android.comf2b98d62010-12-20 18:26:13 +0000234 kGPU_CanvasType,
reed@android.com8a1c16f2008-12-17 15:59:43 +0000235 kRaster_CanvasType
236 };
237 return gCT[ct];
238}
239
240SampleWindow::SampleWindow(void* hwnd) : INHERITED(hwnd) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000241 fPicture = NULL;
reed@android.comf2b98d62010-12-20 18:26:13 +0000242 fGpuCanvas = NULL;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000243
reed@android.comf2b98d62010-12-20 18:26:13 +0000244#ifdef DEFAULT_TO_GPU
245 fCanvasType = kGPU_CanvasType;
246#else
reed@android.com8a1c16f2008-12-17 15:59:43 +0000247 fCanvasType = kRaster_CanvasType;
reed@android.comf2b98d62010-12-20 18:26:13 +0000248#endif
reed@android.com8a1c16f2008-12-17 15:59:43 +0000249 fUseClip = false;
reed@android.come522ca52009-11-23 20:10:41 +0000250 fNClip = false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000251 fRepeatDrawing = false;
252 fAnimating = false;
reed@android.com6c5f6f22009-08-14 16:08:38 +0000253 fRotate = false;
254 fScale = false;
reed@android.comf2b98d62010-12-20 18:26:13 +0000255 fRequestGrabImage = false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000256
257 fScrollTestX = fScrollTestY = 0;
258
259// this->setConfig(SkBitmap::kRGB_565_Config);
260 this->setConfig(SkBitmap::kARGB_8888_Config);
261 this->setVisibleP(true);
reed@android.comf2b98d62010-12-20 18:26:13 +0000262 this->setClipToBounds(false);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000263
reed@android.com34245c72009-11-03 04:00:48 +0000264 {
265 const SkViewRegister* reg = SkViewRegister::Head();
266 while (reg) {
267 *fSamples.append() = reg->factory();
268 reg = reg->next();
269 }
270 }
271 fCurrIndex = 0;
reed@android.come0f13ee2009-11-04 19:40:25 +0000272 this->loadView(fSamples[fCurrIndex]());
reed@android.com8a1c16f2008-12-17 15:59:43 +0000273}
274
275SampleWindow::~SampleWindow() {
276 delete fPicture;
reed@android.comf2b98d62010-12-20 18:26:13 +0000277 delete fGpuCanvas;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000278}
279
reed@android.com55e76b22009-11-23 21:46:47 +0000280static SkBitmap capture_bitmap(SkCanvas* canvas) {
281 SkBitmap bm;
282 const SkBitmap& src = canvas->getDevice()->accessBitmap(false);
283 src.copyTo(&bm, src.config());
284 return bm;
285}
286
287static bool bitmap_diff(SkCanvas* canvas, const SkBitmap& orig,
288 SkBitmap* diff) {
289 const SkBitmap& src = canvas->getDevice()->accessBitmap(false);
290
291 SkAutoLockPixels alp0(src);
292 SkAutoLockPixels alp1(orig);
293 for (int y = 0; y < src.height(); y++) {
294 const void* srcP = src.getAddr(0, y);
295 const void* origP = orig.getAddr(0, y);
296 size_t bytes = src.width() * src.bytesPerPixel();
297 if (memcmp(srcP, origP, bytes)) {
298 SkDebugf("---------- difference on line %d\n", y);
299 return true;
300 }
301 }
302 return false;
303}
304
reed@android.com44177402009-11-23 21:07:51 +0000305#define XCLIP_N 8
306#define YCLIP_N 8
reed@android.come522ca52009-11-23 20:10:41 +0000307
308void SampleWindow::draw(SkCanvas* canvas) {
reed@android.com44177402009-11-23 21:07:51 +0000309 // update the animation time
reed@android.comf2b98d62010-12-20 18:26:13 +0000310 gAnimTimePrev = gAnimTime;
reed@android.com44177402009-11-23 21:07:51 +0000311 gAnimTime = SkTime::GetMSecs();
312
reed@android.come522ca52009-11-23 20:10:41 +0000313 if (fNClip) {
reed@android.com55e76b22009-11-23 21:46:47 +0000314 this->INHERITED::draw(canvas);
315 SkBitmap orig = capture_bitmap(canvas);
reed@android.come522ca52009-11-23 20:10:41 +0000316
317 const SkScalar w = this->width();
318 const SkScalar h = this->height();
319 const SkScalar cw = w / XCLIP_N;
320 const SkScalar ch = h / YCLIP_N;
321 for (int y = 0; y < YCLIP_N; y++) {
reed@android.com55e76b22009-11-23 21:46:47 +0000322 SkRect r;
323 r.fTop = y * ch;
324 r.fBottom = (y + 1) * ch;
325 if (y == YCLIP_N - 1) {
326 r.fBottom = h;
327 }
reed@android.come522ca52009-11-23 20:10:41 +0000328 for (int x = 0; x < XCLIP_N; x++) {
329 SkAutoCanvasRestore acr(canvas, true);
reed@android.com55e76b22009-11-23 21:46:47 +0000330 r.fLeft = x * cw;
331 r.fRight = (x + 1) * cw;
332 if (x == XCLIP_N - 1) {
333 r.fRight = w;
334 }
reed@android.come522ca52009-11-23 20:10:41 +0000335 canvas->clipRect(r);
336 this->INHERITED::draw(canvas);
337 }
338 }
reed@android.com55e76b22009-11-23 21:46:47 +0000339
340 SkBitmap diff;
341 if (bitmap_diff(canvas, orig, &diff)) {
342 }
reed@android.come522ca52009-11-23 20:10:41 +0000343 } else {
344 this->INHERITED::draw(canvas);
345 }
346}
347
reed@android.com8a1c16f2008-12-17 15:59:43 +0000348void SampleWindow::onDraw(SkCanvas* canvas) {
349 if (fRepeatDrawing) {
350 this->inval(NULL);
351 }
352}
353
354#include "SkColorPriv.h"
355
356static void reverseRedAndBlue(const SkBitmap& bm) {
357 SkASSERT(bm.config() == SkBitmap::kARGB_8888_Config);
358 uint8_t* p = (uint8_t*)bm.getPixels();
359 uint8_t* stop = p + bm.getSize();
360 while (p < stop) {
361 // swap red/blue (to go from ARGB(int) to RGBA(memory) and premultiply
362 unsigned scale = SkAlpha255To256(p[3]);
363 unsigned r = p[2];
364 unsigned b = p[0];
365 p[0] = SkAlphaMul(r, scale);
366 p[1] = SkAlphaMul(p[1], scale);
367 p[2] = SkAlphaMul(b, scale);
368 p += 4;
369 }
370}
371
372SkCanvas* SampleWindow::beforeChildren(SkCanvas* canvas) {
reed@android.comf2b98d62010-12-20 18:26:13 +0000373 SkIPoint viewport;
374 bool alreadyGPU = canvas->getViewport(&viewport);
375
376 if (kGPU_CanvasType != fCanvasType) {
reed@android.com6efdc472008-12-19 18:24:35 +0000377#ifdef SK_SUPPORT_GL
reed@android.comf2b98d62010-12-20 18:26:13 +0000378 detachGL();
379#elif defined(SK_SUPPORT_D3D9)
380 detachD3D9();
381#endif
382 }
383
reed@android.com8a1c16f2008-12-17 15:59:43 +0000384 switch (fCanvasType) {
385 case kRaster_CanvasType:
386 canvas = this->INHERITED::beforeChildren(canvas);
387 break;
388 case kPicture_CanvasType:
389 fPicture = new SkPicture;
390 canvas = fPicture->beginRecording(9999, 9999);
391 break;
reed@google.comac10a2d2010-12-22 21:39:39 +0000392 case kGPU_CanvasType: {
reed@android.comf2b98d62010-12-20 18:26:13 +0000393 if (!alreadyGPU) {
394 SkDevice* device = canvas->getDevice();
395 const SkBitmap& bitmap = device->accessBitmap(true);
396#ifdef SK_SUPPORT_GL
397 #ifdef USE_OFFSCREEN
398 // first clear the raster bitmap, so we don't see any leftover bits
399 bitmap.eraseColor(0);
400 // now setup our glcanvas
401 attachGL(&bitmap);
402 #else
403 attachGL(NULL);
404 #endif
405 glClear(GL_COLOR_BUFFER_BIT | GL_STENCIL_BUFFER_BIT);
406#elif defined(SK_SUPPORT_D3D9)
407 // now setup our canvas
408 attachD3D9();
409#endif
reed@android.comf2b98d62010-12-20 18:26:13 +0000410 fGpuCanvas = new SkGpuCanvas(get_global_grctx(this));
reed@google.comac10a2d2010-12-22 21:39:39 +0000411 device = fGpuCanvas->createDevice(SkBitmap::kARGB_8888_Config,
412 bitmap.width(), bitmap.height(),
413 false, false);
414 fGpuCanvas->setDevice(device)->unref();
reed@android.comf2b98d62010-12-20 18:26:13 +0000415 canvas = fGpuCanvas;
416
417 } else {
418 canvas = this->INHERITED::beforeChildren(canvas);
419 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000420 break;
reed@google.comac10a2d2010-12-22 21:39:39 +0000421 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000422 }
423
424 if (fUseClip) {
425 canvas->drawColor(0xFFFF88FF);
426 canvas->clipPath(fClipPath);
427 }
428
429 return canvas;
430}
431
432static void paint_rgn(const SkBitmap& bm, const SkIRect& r,
433 const SkRegion& rgn) {
434 SkCanvas canvas(bm);
435 SkRegion inval(rgn);
436
437 inval.translate(r.fLeft, r.fTop);
438 canvas.clipRegion(inval);
439 canvas.drawColor(0xFFFF8080);
440}
441
442void SampleWindow::afterChildren(SkCanvas* orig) {
reed@android.comf2b98d62010-12-20 18:26:13 +0000443 if (fRequestGrabImage) {
444 fRequestGrabImage = false;
445
446 SkCanvas* canvas = fGpuCanvas ? fGpuCanvas : orig;
447 SkDevice* device = canvas->getDevice();
448 SkBitmap bitmap;
449 SkIRect bounds = { 0, 0, this->width(), this->height() };
450 if (device->readPixels(bounds, &bitmap)) {
451 static int gSampleGrabCounter;
452 SkString name;
453 name.printf("sample_grab_%d", gSampleGrabCounter++);
454 SkImageEncoder::EncodeFile(name.c_str(), bitmap,
455 SkImageEncoder::kPNG_Type, 100);
456 }
457 }
458
reed@android.com8a1c16f2008-12-17 15:59:43 +0000459 switch (fCanvasType) {
460 case kRaster_CanvasType:
461 break;
462 case kPicture_CanvasType:
reed@android.comaefd2bc2009-03-30 21:02:14 +0000463 if (true) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000464 SkPicture* pict = new SkPicture(*fPicture);
465 fPicture->unref();
466 orig->drawPicture(*pict);
467 pict->unref();
reed@android.comaefd2bc2009-03-30 21:02:14 +0000468 } else if (true) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000469 SkDynamicMemoryWStream ostream;
470 fPicture->serialize(&ostream);
471 fPicture->unref();
472
473 SkMemoryStream istream(ostream.getStream(), ostream.getOffset());
474 SkPicture pict(&istream);
475 orig->drawPicture(pict);
476 } else {
477 fPicture->draw(orig);
478 fPicture->unref();
479 }
480 fPicture = NULL;
481 break;
reed@android.com6efdc472008-12-19 18:24:35 +0000482#ifdef SK_SUPPORT_GL
reed@android.comf2b98d62010-12-20 18:26:13 +0000483 case kGPU_CanvasType:
484 delete fGpuCanvas;
485 fGpuCanvas = NULL;
486 presentGL();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000487#ifdef USE_OFFSCREEN
488 reverseRedAndBlue(orig->getDevice()->accessBitmap(true));
489#endif
490 break;
reed@android.comf2b98d62010-12-20 18:26:13 +0000491#elif defined(SK_SUPPORT_D3D9)
492 case kGPU_CanvasType: {
493 delete fGpuCanvas;
494 fGpuCanvas = NULL;
495 presentD3D9();
496 break;
497 }
reed@android.com6efdc472008-12-19 18:24:35 +0000498#endif
reed@android.comf2b98d62010-12-20 18:26:13 +0000499
reed@android.com8a1c16f2008-12-17 15:59:43 +0000500 }
501
502// if ((fScrollTestX | fScrollTestY) != 0)
reed@android.comf2b98d62010-12-20 18:26:13 +0000503 if (false) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000504 const SkBitmap& bm = orig->getDevice()->accessBitmap(true);
505 int dx = fScrollTestX * 7;
506 int dy = fScrollTestY * 7;
507 SkIRect r;
508 SkRegion inval;
509
510 r.set(50, 50, 50+100, 50+100);
511 bm.scrollRect(&r, dx, dy, &inval);
512 paint_rgn(bm, r, inval);
513 }
514}
515
reed@android.com6c5f6f22009-08-14 16:08:38 +0000516void SampleWindow::beforeChild(SkView* child, SkCanvas* canvas) {
517 if (fScale) {
518 SkScalar scale = SK_Scalar1 * 7 / 10;
519 SkScalar cx = this->width() / 2;
520 SkScalar cy = this->height() / 2;
521 canvas->translate(cx, cy);
522 canvas->scale(scale, scale);
523 canvas->translate(-cx, -cy);
524 }
525 if (fRotate) {
526 SkScalar cx = this->width() / 2;
527 SkScalar cy = this->height() / 2;
528 canvas->translate(cx, cy);
529 canvas->rotate(SkIntToScalar(30));
530 canvas->translate(-cx, -cy);
531 }
532}
533
534void SampleWindow::afterChild(SkView* child, SkCanvas* canvas) {
535}
536
reed@android.com8a1c16f2008-12-17 15:59:43 +0000537static SkBitmap::Config gConfigCycle[] = {
538 SkBitmap::kNo_Config, // none -> none
539 SkBitmap::kNo_Config, // a1 -> none
540 SkBitmap::kNo_Config, // a8 -> none
541 SkBitmap::kNo_Config, // index8 -> none
542 SkBitmap::kARGB_4444_Config, // 565 -> 4444
543 SkBitmap::kARGB_8888_Config, // 4444 -> 8888
544 SkBitmap::kRGB_565_Config // 8888 -> 565
545};
546
547static SkBitmap::Config cycle_configs(SkBitmap::Config c) {
548 return gConfigCycle[c];
549}
550
551bool SampleWindow::nextSample() {
reed@android.com34245c72009-11-03 04:00:48 +0000552 fCurrIndex = (fCurrIndex + 1) % fSamples.count();
553 this->loadView(fSamples[fCurrIndex]());
554 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000555}
556
557bool SampleWindow::onEvent(const SkEvent& evt) {
558 if (evt.isType(ANIMATING_EVENTTYPE)) {
559 if (fAnimating) {
560 this->nextSample();
561 this->postAnimatingEvent();
562 }
563 return true;
564 }
reed@android.com34245c72009-11-03 04:00:48 +0000565 if (evt.isType("set-curr-index")) {
566 fCurrIndex = evt.getFast32() % fSamples.count();
567 this->loadView(fSamples[fCurrIndex]());
568 return true;
569 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000570 return this->INHERITED::onEvent(evt);
571}
572
reed@android.comf2b98d62010-12-20 18:26:13 +0000573bool SampleWindow::onQuery(SkEvent* query) {
574 if (query->isType("get-slide-count")) {
575 query->setFast32(fSamples.count());
576 return true;
577 }
578 if (query->isType("get-slide-title")) {
579 SkView* view = fSamples[query->getFast32()]();
580 SkEvent evt(gTitleEvtName);
581 if (view->doQuery(&evt)) {
582 query->setString("title", evt.findString(gTitleEvtName));
583 }
584 SkSafeUnref(view);
585 return true;
586 }
587 if (query->isType("use-fast-text")) {
588 SkEvent evt(gFastTextEvtName);
589 return curr_view(this)->doQuery(&evt);
590 }
591 return this->INHERITED::onQuery(query);
592}
593
reed@android.com0ae6b242008-12-23 16:49:54 +0000594static void cleanup_for_filename(SkString* name) {
595 char* str = name->writable_str();
reed@android.come191b162009-12-18 21:33:39 +0000596 for (size_t i = 0; i < name->size(); i++) {
reed@android.com0ae6b242008-12-23 16:49:54 +0000597 switch (str[i]) {
598 case ':': str[i] = '-'; break;
599 case '/': str[i] = '-'; break;
600 case ' ': str[i] = '_'; break;
601 default: break;
602 }
603 }
604}
reed@android.com8a1c16f2008-12-17 15:59:43 +0000605
606bool SampleWindow::onHandleChar(SkUnichar uni) {
reed@android.comf2b98d62010-12-20 18:26:13 +0000607 {
608 SkView* view = curr_view(this);
609 if (view) {
610 SkEvent evt(gCharEvtName);
611 evt.setFast32(uni);
612 if (view->doQuery(&evt)) {
613 return true;
614 }
615 }
616 }
617
reed@android.com8a1c16f2008-12-17 15:59:43 +0000618 int dx = 0xFF;
619 int dy = 0xFF;
620
621 switch (uni) {
622 case '5': dx = 0; dy = 0; break;
623 case '8': dx = 0; dy = -1; break;
624 case '6': dx = 1; dy = 0; break;
625 case '2': dx = 0; dy = 1; break;
626 case '4': dx = -1; dy = 0; break;
627 case '7': dx = -1; dy = -1; break;
628 case '9': dx = 1; dy = -1; break;
629 case '3': dx = 1; dy = 1; break;
630 case '1': dx = -1; dy = 1; break;
631
632 default:
633 break;
634 }
635
636 if (0xFF != dx && 0xFF != dy) {
637 if ((dx | dy) == 0) {
638 fScrollTestX = fScrollTestY = 0;
639 } else {
640 fScrollTestX += dx;
641 fScrollTestY += dy;
642 }
643 this->inval(NULL);
644 return true;
645 }
646
reed@android.com0ae6b242008-12-23 16:49:54 +0000647 switch (uni) {
648 case 'a':
649 fAnimating = !fAnimating;
650 this->postAnimatingEvent();
651 this->updateTitle();
652 return true;
653 case 'f': {
654 const char* title = this->getTitle();
655 if (title[0] == 0) {
656 title = "sampleapp";
657 }
658 SkString name(title);
659 cleanup_for_filename(&name);
660 name.append(".png");
661 if (SkImageEncoder::EncodeFile(name.c_str(), this->getBitmap(),
reed@android.comb08eb2b2009-01-06 20:16:26 +0000662 SkImageEncoder::kPNG_Type, 100)) {
reed@android.com0ae6b242008-12-23 16:49:54 +0000663 SkDebugf("Created %s\n", name.c_str());
664 }
665 return true;
666 }
reed@android.com6c5f6f22009-08-14 16:08:38 +0000667 case 'r':
668 fRotate = !fRotate;
669 this->inval(NULL);
670 this->updateTitle();
671 return true;
672 case 's':
673 fScale = !fScale;
674 this->inval(NULL);
675 this->updateTitle();
676 return true;
reed@android.comf2b98d62010-12-20 18:26:13 +0000677 case 'd':
678 SkGraphics::SetFontCacheUsed(0);
679 return true;
680 case 'g':
681 fRequestGrabImage = true;
682 this->inval(NULL);
683 break;
reed@android.com0ae6b242008-12-23 16:49:54 +0000684 default:
685 break;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000686 }
687
688 return this->INHERITED::onHandleChar(uni);
689}
690
691#include "SkDumpCanvas.h"
692
693bool SampleWindow::onHandleKey(SkKey key) {
reed@android.comf2b98d62010-12-20 18:26:13 +0000694 {
695 SkView* view = curr_view(this);
696 if (view) {
697 SkEvent evt(gKeyEvtName);
698 evt.setFast32(key);
699 if (view->doQuery(&evt)) {
700 return true;
701 }
702 }
703 }
704
reed@android.com8a1c16f2008-12-17 15:59:43 +0000705 switch (key) {
706 case kRight_SkKey:
707 if (this->nextSample()) {
708 return true;
709 }
710 break;
711 case kLeft_SkKey:
712 fCanvasType = cycle_canvastype(fCanvasType);
713 this->updateTitle();
714 this->inval(NULL);
715 return true;
716 case kUp_SkKey:
reed@android.come522ca52009-11-23 20:10:41 +0000717 fNClip = !fNClip;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000718 this->updateTitle();
719 this->inval(NULL);
720 return true;
721 case kDown_SkKey:
722 this->setConfig(cycle_configs(this->getBitmap().config()));
723 this->updateTitle();
724 return true;
725 case kOK_SkKey:
reed@android.comf2b98d62010-12-20 18:26:13 +0000726 if (false) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000727 SkDebugfDumper dumper;
728 SkDumpCanvas dc(&dumper);
729 this->draw(&dc);
730 } else {
731 fRepeatDrawing = !fRepeatDrawing;
732 if (fRepeatDrawing) {
733 this->inval(NULL);
734 }
735 }
736 return true;
reed@android.com34245c72009-11-03 04:00:48 +0000737 case kBack_SkKey:
738 this->loadView(NULL);
739 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000740 default:
741 break;
742 }
743 return this->INHERITED::onHandleKey(key);
744}
745
746void SampleWindow::loadView(SkView* view) {
747 SkView::F2BIter iter(this);
748 SkView* prev = iter.next();
749 if (prev) {
750 prev->detachFromParent();
751 }
reed@android.com34245c72009-11-03 04:00:48 +0000752
753 if (NULL == view) {
754 view = create_overview(fSamples.count(), fSamples.begin());
755 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000756 view->setVisibleP(true);
reed@android.comf2b98d62010-12-20 18:26:13 +0000757 view->setClipToBounds(false);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000758 this->attachChildToFront(view)->unref();
759 view->setSize(this->width(), this->height());
760
761 this->updateTitle();
762}
763
764static const char* gConfigNames[] = {
765 "unknown config",
766 "A1",
767 "A8",
768 "Index8",
769 "565",
770 "4444",
771 "8888"
772};
773
774static const char* configToString(SkBitmap::Config c) {
775 return gConfigNames[c];
776}
777
778static const char* gCanvasTypePrefix[] = {
779 "raster: ",
780 "picture: ",
781 "opengl: "
782};
783
784void SampleWindow::updateTitle() {
785 SkString title;
786
787 SkView::F2BIter iter(this);
788 SkView* view = iter.next();
789 SkEvent evt(gTitleEvtName);
790 if (view->doQuery(&evt)) {
791 title.set(evt.findString(gTitleEvtName));
792 }
793 if (title.size() == 0) {
794 title.set("<unknown>");
795 }
796
797 title.prepend(gCanvasTypePrefix[fCanvasType]);
798
799 title.prepend(" ");
800 title.prepend(configToString(this->getBitmap().config()));
801
802 if (fAnimating) {
803 title.prepend("<A> ");
804 }
reed@android.com6c5f6f22009-08-14 16:08:38 +0000805 if (fScale) {
806 title.prepend("<S> ");
807 }
808 if (fRotate) {
809 title.prepend("<R> ");
810 }
reed@android.come522ca52009-11-23 20:10:41 +0000811 if (fNClip) {
812 title.prepend("<C> ");
813 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000814 this->setTitle(title.c_str());
815}
816
817void SampleWindow::onSizeChange() {
818 this->INHERITED::onSizeChange();
819
820 SkView::F2BIter iter(this);
821 SkView* view = iter.next();
822 view->setSize(this->width(), this->height());
823
824 // rebuild our clippath
825 {
826 const SkScalar W = this->width();
827 const SkScalar H = this->height();
828
829 fClipPath.reset();
830#if 0
831 for (SkScalar y = SK_Scalar1; y < H; y += SkIntToScalar(32)) {
832 SkRect r;
833 r.set(SK_Scalar1, y, SkIntToScalar(30), y + SkIntToScalar(30));
834 for (; r.fLeft < W; r.offset(SkIntToScalar(32), 0))
835 fClipPath.addRect(r);
836 }
837#else
838 SkRect r;
839 r.set(0, 0, W, H);
840 fClipPath.addRect(r, SkPath::kCCW_Direction);
841 r.set(W/4, H/4, W*3/4, H*3/4);
842 fClipPath.addRect(r, SkPath::kCW_Direction);
843#endif
844 }
845
846 this->updateTitle(); // to refresh our config
847}
848
849///////////////////////////////////////////////////////////////////////////////
850
reed@android.comf2b98d62010-12-20 18:26:13 +0000851template <typename T> void SkTBSort(T array[], int count) {
852 for (int i = 1; i < count - 1; i++) {
853 bool didSwap = false;
854 for (int j = count - 1; j > i; --j) {
855 if (array[j] < array[j-1]) {
856 T tmp(array[j-1]);
857 array[j-1] = array[j];
858 array[j] = tmp;
859 didSwap = true;
860 }
861 }
862 if (!didSwap) {
863 break;
864 }
865 }
866
867 for (int k = 0; k < count - 1; k++) {
868 SkASSERT(!(array[k+1] < array[k]));
869 }
870}
871
872#include "SkRandom.h"
873
874static void rand_rect(SkIRect* rect, SkRandom& rand) {
875 int bits = 8;
876 int shift = 32 - bits;
877 rect->set(rand.nextU() >> shift, rand.nextU() >> shift,
878 rand.nextU() >> shift, rand.nextU() >> shift);
879 rect->sort();
880}
881
882static void dumpRect(const SkIRect& r) {
883 SkDebugf(" { %d, %d, %d, %d },\n",
884 r.fLeft, r.fTop,
885 r.fRight, r.fBottom);
886}
887
888static void test_rects(const SkIRect rect[], int count) {
889 SkRegion rgn0, rgn1;
890
891 for (int i = 0; i < count; i++) {
892 rgn0.op(rect[i], SkRegion::kUnion_Op);
893 // dumpRect(rect[i]);
894 }
895 rgn1.setRects(rect, count);
896
897 if (rgn0 != rgn1) {
898 SkDebugf("\n");
899 for (int i = 0; i < count; i++) {
900 dumpRect(rect[i]);
901 }
902 SkDebugf("\n");
903 }
904}
905
906static void test() {
907 size_t i;
908
909 const SkIRect r0[] = {
910 { 0, 0, 1, 1 },
911 { 2, 2, 3, 3 },
912 };
913 const SkIRect r1[] = {
914 { 0, 0, 1, 3 },
915 { 1, 1, 2, 2 },
916 { 2, 0, 3, 3 },
917 };
918 const SkIRect r2[] = {
919 { 0, 0, 1, 2 },
920 { 2, 1, 3, 3 },
921 { 4, 0, 5, 1 },
922 { 6, 0, 7, 4 },
923 };
924
925 static const struct {
926 const SkIRect* fRects;
927 int fCount;
928 } gRecs[] = {
929 { r0, SK_ARRAY_COUNT(r0) },
930 { r1, SK_ARRAY_COUNT(r1) },
931 { r2, SK_ARRAY_COUNT(r2) },
932 };
933
934 for (i = 0; i < SK_ARRAY_COUNT(gRecs); i++) {
935 test_rects(gRecs[i].fRects, gRecs[i].fCount);
936 }
937
938 SkRandom rand;
939 for (i = 0; i < 10000; i++) {
940 SkRegion rgn0, rgn1;
941
942 const int N = 8;
943 SkIRect rect[N];
944 for (int j = 0; j < N; j++) {
945 rand_rect(&rect[j], rand);
946 }
947 test_rects(rect, N);
948 }
949}
950
reed@android.com8a1c16f2008-12-17 15:59:43 +0000951SkOSWindow* create_sk_window(void* hwnd) {
reed@android.comf2b98d62010-12-20 18:26:13 +0000952// test();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000953 return new SampleWindow(hwnd);
954}
955
956void get_preferred_size(int* x, int* y, int* width, int* height) {
957 *x = 10;
958 *y = 50;
959 *width = 640;
960 *height = 480;
961}
962
963void application_init() {
964// setenv("ANDROID_ROOT", "../../../data", 0);
reed@android.come191b162009-12-18 21:33:39 +0000965#ifdef SK_BUILD_FOR_MAC
reed@android.com8a1c16f2008-12-17 15:59:43 +0000966 setenv("ANDROID_ROOT", "/android/device/data", 0);
reed@android.come191b162009-12-18 21:33:39 +0000967#endif
reed@android.com5e5adfd2009-03-07 03:39:23 +0000968 SkGraphics::Init();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000969 SkEvent::Init();
970}
971
972void application_term() {
973 SkEvent::Term();
974 SkGraphics::Term();
975}