blob: b20267a6dcf3e9a74cb686f7230a18c318226b0a [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
2/*
3 * Copyright 2011 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
reed@android.com8a1c16f2008-12-17 15:59:43 +00008#include "SkWindow.h"
9#include "SkCanvas.h"
reed@android.comf2b98d62010-12-20 18:26:13 +000010#include "SkDevice.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000011#include "SkOSMenu.h"
12#include "SkSystemEventTypes.h"
13#include "SkTime.h"
14
15#define SK_EventDelayInval "\xd" "n" "\xa" "l"
16
17#define TEST_BOUNDERx
18
19#include "SkBounder.h"
20class test_bounder : public SkBounder {
21public:
rmistry@google.comd6176b02012-08-23 18:14:13 +000022 test_bounder(const SkBitmap& bm) : fCanvas(bm) {}
reed@android.com8a1c16f2008-12-17 15:59:43 +000023protected:
rmistry@google.comd6176b02012-08-23 18:14:13 +000024 virtual bool onIRect(const SkIRect& r)
25 {
26 SkRect rr;
reed@android.com8a1c16f2008-12-17 15:59:43 +000027
rmistry@google.comd6176b02012-08-23 18:14:13 +000028 rr.set(SkIntToScalar(r.fLeft), SkIntToScalar(r.fTop),
29 SkIntToScalar(r.fRight), SkIntToScalar(r.fBottom));
reed@android.com8a1c16f2008-12-17 15:59:43 +000030
rmistry@google.comd6176b02012-08-23 18:14:13 +000031 SkPaint p;
reed@android.com8a1c16f2008-12-17 15:59:43 +000032
rmistry@google.comd6176b02012-08-23 18:14:13 +000033 p.setStyle(SkPaint::kStroke_Style);
34 p.setColor(SK_ColorYELLOW);
reed@android.com8a1c16f2008-12-17 15:59:43 +000035
36#if 0
rmistry@google.comd6176b02012-08-23 18:14:13 +000037 rr.inset(SK_ScalarHalf, SK_ScalarHalf);
reed@android.com8a1c16f2008-12-17 15:59:43 +000038#else
rmistry@google.comd6176b02012-08-23 18:14:13 +000039 rr.inset(-SK_ScalarHalf, -SK_ScalarHalf);
reed@android.com8a1c16f2008-12-17 15:59:43 +000040#endif
41
rmistry@google.comd6176b02012-08-23 18:14:13 +000042 fCanvas.drawRect(rr, p);
43 return true;
44 }
reed@android.com8a1c16f2008-12-17 15:59:43 +000045private:
rmistry@google.comd6176b02012-08-23 18:14:13 +000046 SkCanvas fCanvas;
reed@android.com8a1c16f2008-12-17 15:59:43 +000047};
48
49SkWindow::SkWindow() : fFocusView(NULL)
50{
Scroggod3aed392011-06-22 13:26:56 +000051 fClicks.reset();
52 fWaitingOnInval = false;
reed@android.com8a1c16f2008-12-17 15:59:43 +000053
54#ifdef SK_BUILD_FOR_WINCE
Scroggod3aed392011-06-22 13:26:56 +000055 fConfig = SkBitmap::kRGB_565_Config;
reed@android.com8a1c16f2008-12-17 15:59:43 +000056#else
Scroggod3aed392011-06-22 13:26:56 +000057 fConfig = SkBitmap::kARGB_8888_Config;
reed@android.com8a1c16f2008-12-17 15:59:43 +000058#endif
reed@android.comf2b98d62010-12-20 18:26:13 +000059
60 fMatrix.reset();
reed@android.com8a1c16f2008-12-17 15:59:43 +000061}
62
63SkWindow::~SkWindow()
64{
Scroggod3aed392011-06-22 13:26:56 +000065 fClicks.deleteAll();
66 fMenus.deleteAll();
reed@android.com8a1c16f2008-12-17 15:59:43 +000067}
68
reed@google.com5957f472012-10-01 20:31:56 +000069SkCanvas* SkWindow::createCanvas() {
70 return new SkCanvas(this->getBitmap());
71}
72
reed@android.comf2b98d62010-12-20 18:26:13 +000073void SkWindow::setMatrix(const SkMatrix& matrix) {
74 if (fMatrix != matrix) {
75 fMatrix = matrix;
76 this->inval(NULL);
77 }
78}
79
80void SkWindow::preConcat(const SkMatrix& matrix) {
81 SkMatrix m;
82 m.setConcat(fMatrix, matrix);
83 this->setMatrix(m);
84}
85
86void SkWindow::postConcat(const SkMatrix& matrix) {
87 SkMatrix m;
88 m.setConcat(matrix, fMatrix);
89 this->setMatrix(m);
90}
91
reed@android.com8a1c16f2008-12-17 15:59:43 +000092void SkWindow::setConfig(SkBitmap::Config config)
93{
rmistry@google.comd6176b02012-08-23 18:14:13 +000094 this->resize(fBitmap.width(), fBitmap.height(), config);
reed@android.com8a1c16f2008-12-17 15:59:43 +000095}
96
reed@android.com8a1c16f2008-12-17 15:59:43 +000097void SkWindow::resize(int width, int height, SkBitmap::Config config)
98{
rmistry@google.comd6176b02012-08-23 18:14:13 +000099 if (config == SkBitmap::kNo_Config)
100 config = fConfig;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000101
rmistry@google.comd6176b02012-08-23 18:14:13 +0000102 if (width != fBitmap.width() || height != fBitmap.height() || config != fConfig)
103 {
104 fConfig = config;
105 fBitmap.setConfig(config, width, height);
106 fBitmap.allocPixels();
reed@android.comf2b98d62010-12-20 18:26:13 +0000107 fBitmap.setIsOpaque(true);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000108
rmistry@google.comd6176b02012-08-23 18:14:13 +0000109 this->setSize(SkIntToScalar(width), SkIntToScalar(height));
110 this->inval(NULL);
111 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000112}
113
114void SkWindow::eraseARGB(U8CPU a, U8CPU r, U8CPU g, U8CPU b)
115{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000116 fBitmap.eraseARGB(a, r, g, b);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000117}
118
119void SkWindow::eraseRGB(U8CPU r, U8CPU g, U8CPU b)
120{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000121 fBitmap.eraseRGB(r, g, b);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000122}
123
reed@android.comf2b98d62010-12-20 18:26:13 +0000124bool SkWindow::handleInval(const SkRect* localR)
reed@android.com8a1c16f2008-12-17 15:59:43 +0000125{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000126 SkIRect ir;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000127
reed@android.comf2b98d62010-12-20 18:26:13 +0000128 if (localR) {
129 SkRect devR;
130 SkMatrix inverse;
131 if (!fMatrix.invert(&inverse)) {
132 return false;
133 }
134 fMatrix.mapRect(&devR, *localR);
135 devR.round(&ir);
136 } else {
reed@google.comf9bb7a82011-03-01 15:15:13 +0000137 ir.set(0, 0,
rmistry@google.comd6176b02012-08-23 18:14:13 +0000138 SkScalarRound(this->width()),
139 SkScalarRound(this->height()));
reed@android.comf2b98d62010-12-20 18:26:13 +0000140 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000141 fDirtyRgn.op(ir, SkRegion::kUnion_Op);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000142
rmistry@google.comd6176b02012-08-23 18:14:13 +0000143 this->onHandleInval(ir);
144 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000145}
146
reed@android.comf2b98d62010-12-20 18:26:13 +0000147void SkWindow::forceInvalAll() {
reed@google.com261b8e22011-04-14 17:53:24 +0000148 fDirtyRgn.setRect(0, 0,
149 SkScalarCeil(this->width()),
150 SkScalarCeil(this->height()));
reed@android.comf2b98d62010-12-20 18:26:13 +0000151}
152
reed@android.com8a1c16f2008-12-17 15:59:43 +0000153#if defined(SK_BUILD_FOR_WINCE) && defined(USE_GX_SCREEN)
rmistry@google.comd6176b02012-08-23 18:14:13 +0000154 #include <windows.h>
155 #include <gx.h>
156 extern GXDisplayProperties gDisplayProps;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000157#endif
158
159#ifdef SK_SIMULATE_FAILED_MALLOC
160extern bool gEnableControlledThrow;
161#endif
162
reed@google.com511aa0e2012-09-27 21:09:30 +0000163bool SkWindow::update(SkIRect* updateArea)
reed@android.com8a1c16f2008-12-17 15:59:43 +0000164{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000165 if (!fDirtyRgn.isEmpty())
166 {
167 SkBitmap bm = this->getBitmap();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000168
169#if defined(SK_BUILD_FOR_WINCE) && defined(USE_GX_SCREEN)
rmistry@google.comd6176b02012-08-23 18:14:13 +0000170 char* buffer = (char*)GXBeginDraw();
171 SkASSERT(buffer);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000172
rmistry@google.comd6176b02012-08-23 18:14:13 +0000173 RECT rect;
174 GetWindowRect((HWND)((SkOSWindow*)this)->getHWND(), &rect);
175 buffer += rect.top * gDisplayProps.cbyPitch + rect.left * gDisplayProps.cbxPitch;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000176
rmistry@google.comd6176b02012-08-23 18:14:13 +0000177 bm.setPixels(buffer);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000178#endif
reed@google.com4c9e2aa2012-09-27 20:58:01 +0000179 SkCanvas rasterCanvas(bm);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000180
reed@google.com4c9e2aa2012-09-27 20:58:01 +0000181 rasterCanvas.clipRegion(fDirtyRgn);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000182 if (updateArea)
183 *updateArea = fDirtyRgn.getBounds();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000184
reed@google.com4c9e2aa2012-09-27 20:58:01 +0000185 SkAutoCanvasRestore acr(&rasterCanvas, true);
186 rasterCanvas.concat(fMatrix);
reed@android.comf2b98d62010-12-20 18:26:13 +0000187
rmistry@google.comd6176b02012-08-23 18:14:13 +0000188 // empty this now, so we can correctly record any inval calls that
189 // might be made during the draw call.
190 fDirtyRgn.setEmpty();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000191
192#ifdef TEST_BOUNDER
rmistry@google.comd6176b02012-08-23 18:14:13 +0000193 test_bounder b(bm);
reed@google.com4c9e2aa2012-09-27 20:58:01 +0000194 rasterCanvas.setBounder(&b);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000195#endif
196#ifdef SK_SIMULATE_FAILED_MALLOC
rmistry@google.comd6176b02012-08-23 18:14:13 +0000197 gEnableControlledThrow = true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000198#endif
199#ifdef SK_BUILD_FOR_WIN32
rmistry@google.comd6176b02012-08-23 18:14:13 +0000200 //try {
reed@google.com200e53f2012-09-27 21:22:56 +0000201 this->draw(&rasterCanvas);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000202 //}
203 //catch (...) {
204 //}
reed@android.com8a1c16f2008-12-17 15:59:43 +0000205#else
reed@google.com4c9e2aa2012-09-27 20:58:01 +0000206 this->draw(&rasterCanvas);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000207#endif
208#ifdef SK_SIMULATE_FAILED_MALLOC
rmistry@google.comd6176b02012-08-23 18:14:13 +0000209 gEnableControlledThrow = false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000210#endif
211#ifdef TEST_BOUNDER
reed@google.com4c9e2aa2012-09-27 20:58:01 +0000212 rasterCanvas.setBounder(NULL);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000213#endif
214
215#if defined(SK_BUILD_FOR_WINCE) && defined(USE_GX_SCREEN)
rmistry@google.comd6176b02012-08-23 18:14:13 +0000216 GXEndDraw();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000217#endif
218
rmistry@google.comd6176b02012-08-23 18:14:13 +0000219 return true;
220 }
221 return false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000222}
223
224bool SkWindow::handleChar(SkUnichar uni)
225{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000226 if (this->onHandleChar(uni))
227 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000228
rmistry@google.comd6176b02012-08-23 18:14:13 +0000229 SkView* focus = this->getFocusView();
230 if (focus == NULL)
231 focus = this;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000232
rmistry@google.comd6176b02012-08-23 18:14:13 +0000233 SkEvent evt(SK_EventType_Unichar);
234 evt.setFast32(uni);
235 return focus->doEvent(evt);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000236}
237
238bool SkWindow::handleKey(SkKey key)
239{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000240 if (key == kNONE_SkKey)
241 return false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000242
rmistry@google.comd6176b02012-08-23 18:14:13 +0000243 if (this->onHandleKey(key))
244 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000245
rmistry@google.comd6176b02012-08-23 18:14:13 +0000246 // send an event to the focus-view
247 {
248 SkView* focus = this->getFocusView();
249 if (focus == NULL)
250 focus = this;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000251
rmistry@google.comd6176b02012-08-23 18:14:13 +0000252 SkEvent evt(SK_EventType_Key);
253 evt.setFast32(key);
254 if (focus->doEvent(evt))
255 return true;
256 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000257
rmistry@google.comd6176b02012-08-23 18:14:13 +0000258 if (key == kUp_SkKey || key == kDown_SkKey)
259 {
260 if (this->moveFocus(key == kUp_SkKey ? kPrev_FocusDirection : kNext_FocusDirection) == NULL)
261 this->onSetFocusView(NULL);
262 return true;
263 }
264 return false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000265}
266
267bool SkWindow::handleKeyUp(SkKey key)
268{
269 if (key == kNONE_SkKey)
270 return false;
rmistry@google.comd6176b02012-08-23 18:14:13 +0000271
reed@android.com8a1c16f2008-12-17 15:59:43 +0000272 if (this->onHandleKeyUp(key))
273 return true;
rmistry@google.comd6176b02012-08-23 18:14:13 +0000274
reed@android.com8a1c16f2008-12-17 15:59:43 +0000275 //send an event to the focus-view
276 {
277 SkView* focus = this->getFocusView();
278 if (focus == NULL)
279 focus = this;
rmistry@google.comd6176b02012-08-23 18:14:13 +0000280
reed@android.com8a1c16f2008-12-17 15:59:43 +0000281 //should this one be the same?
282 SkEvent evt(SK_EventType_KeyUp);
283 evt.setFast32(key);
284 if (focus->doEvent(evt))
285 return true;
286 }
287 return false;
288}
289
yangsu@google.com654d72f2011-08-01 17:27:33 +0000290void SkWindow::addMenu(SkOSMenu* menu) {
rmistry@google.comd6176b02012-08-23 18:14:13 +0000291 *fMenus.append() = menu;
292 this->onAddMenu(menu);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000293}
294
reed@android.com0ae6b242008-12-23 16:49:54 +0000295void SkWindow::setTitle(const char title[]) {
296 if (NULL == title) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000297 title = "";
reed@android.com0ae6b242008-12-23 16:49:54 +0000298 }
299 fTitle.set(title);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000300 this->onSetTitle(title);
301}
302
reed@android.com8a1c16f2008-12-17 15:59:43 +0000303//////////////////////////////////////////////////////////////////////
304
305bool SkWindow::onEvent(const SkEvent& evt)
306{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000307 if (evt.isType(SK_EventDelayInval))
308 {
309 SkRegion::Iterator iter(fDirtyRgn);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000310
rmistry@google.comd6176b02012-08-23 18:14:13 +0000311 for (; !iter.done(); iter.next())
312 this->onHandleInval(iter.rect());
313 fWaitingOnInval = false;
314 return true;
315 }
316 return this->INHERITED::onEvent(evt);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000317}
318
319bool SkWindow::onGetFocusView(SkView** focus) const
320{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000321 if (focus)
322 *focus = fFocusView;
323 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000324}
325
326bool SkWindow::onSetFocusView(SkView* focus)
327{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000328 if (fFocusView != focus)
329 {
330 if (fFocusView)
331 fFocusView->onFocusChange(false);
332 fFocusView = focus;
333 if (focus)
334 focus->onFocusChange(true);
335 }
336 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000337}
338
339//////////////////////////////////////////////////////////////////////
340
341void SkWindow::onHandleInval(const SkIRect&)
342{
343}
344
345bool SkWindow::onHandleChar(SkUnichar)
346{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000347 return false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000348}
349
350bool SkWindow::onHandleKey(SkKey key)
351{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000352 return false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000353}
354
355bool SkWindow::onHandleKeyUp(SkKey key)
356{
357 return false;
358}
359
Scroggod3aed392011-06-22 13:26:56 +0000360bool SkWindow::handleClick(int x, int y, Click::State state, void *owner) {
361 return this->onDispatchClick(x, y, state, owner);
mike@reedtribe.orgdd0cd342011-03-21 00:53:39 +0000362}
363
Scroggod3aed392011-06-22 13:26:56 +0000364bool SkWindow::onDispatchClick(int x, int y, Click::State state,
365 void* owner) {
rmistry@google.comd6176b02012-08-23 18:14:13 +0000366 bool handled = false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000367
Scroggod3aed392011-06-22 13:26:56 +0000368 // First, attempt to find an existing click with this owner.
369 int index = -1;
370 for (int i = 0; i < fClicks.count(); i++) {
371 if (owner == fClicks[i]->fOwner) {
372 index = i;
373 break;
374 }
375 }
376
rmistry@google.comd6176b02012-08-23 18:14:13 +0000377 switch (state) {
Scroggod3aed392011-06-22 13:26:56 +0000378 case Click::kDown_State: {
379 if (index != -1) {
380 delete fClicks[index];
381 fClicks.remove(index);
382 }
383 Click* click = this->findClickHandler(SkIntToScalar(x),
384 SkIntToScalar(y));
385
386 if (click) {
387 click->fOwner = owner;
388 *fClicks.append() = click;
389 SkView::DoClickDown(click, x, y);
390 handled = true;
391 }
392 break;
393 }
394 case Click::kMoved_State:
395 if (index != -1) {
396 SkView::DoClickMoved(fClicks[index], x, y);
397 handled = true;
398 }
399 break;
400 case Click::kUp_State:
401 if (index != -1) {
402 SkView::DoClickUp(fClicks[index], x, y);
403 delete fClicks[index];
404 fClicks.remove(index);
405 handled = true;
406 }
407 break;
408 default:
409 // Do nothing
410 break;
rmistry@google.comd6176b02012-08-23 18:14:13 +0000411 }
412 return handled;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000413}
414