blob: f25eb12379de90e2a3971b26b432cd1eeb89bcf9 [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
179
robertphillips@google.comce9dce02012-10-02 12:32:22 +0000180 SkAutoTUnref<SkCanvas> canvas(this->createCanvas());
181
182 canvas->clipRegion(fDirtyRgn);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000183 if (updateArea)
184 *updateArea = fDirtyRgn.getBounds();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000185
robertphillips@google.comce9dce02012-10-02 12:32:22 +0000186 SkAutoCanvasRestore acr(canvas, true);
187 canvas->concat(fMatrix);
reed@android.comf2b98d62010-12-20 18:26:13 +0000188
rmistry@google.comd6176b02012-08-23 18:14:13 +0000189 // empty this now, so we can correctly record any inval calls that
190 // might be made during the draw call.
191 fDirtyRgn.setEmpty();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000192
193#ifdef TEST_BOUNDER
rmistry@google.comd6176b02012-08-23 18:14:13 +0000194 test_bounder b(bm);
robertphillips@google.comce9dce02012-10-02 12:32:22 +0000195 canvas->setBounder(&b);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000196#endif
197#ifdef SK_SIMULATE_FAILED_MALLOC
rmistry@google.comd6176b02012-08-23 18:14:13 +0000198 gEnableControlledThrow = true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000199#endif
200#ifdef SK_BUILD_FOR_WIN32
rmistry@google.comd6176b02012-08-23 18:14:13 +0000201 //try {
robertphillips@google.comce9dce02012-10-02 12:32:22 +0000202 this->draw(canvas);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000203 //}
204 //catch (...) {
205 //}
reed@android.com8a1c16f2008-12-17 15:59:43 +0000206#else
robertphillips@google.comce9dce02012-10-02 12:32:22 +0000207 this->draw(canvas);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000208#endif
209#ifdef SK_SIMULATE_FAILED_MALLOC
rmistry@google.comd6176b02012-08-23 18:14:13 +0000210 gEnableControlledThrow = false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000211#endif
212#ifdef TEST_BOUNDER
robertphillips@google.comce9dce02012-10-02 12:32:22 +0000213 canvas->setBounder(NULL);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000214#endif
215
216#if defined(SK_BUILD_FOR_WINCE) && defined(USE_GX_SCREEN)
rmistry@google.comd6176b02012-08-23 18:14:13 +0000217 GXEndDraw();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000218#endif
219
rmistry@google.comd6176b02012-08-23 18:14:13 +0000220 return true;
221 }
222 return false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000223}
224
225bool SkWindow::handleChar(SkUnichar uni)
226{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000227 if (this->onHandleChar(uni))
228 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000229
rmistry@google.comd6176b02012-08-23 18:14:13 +0000230 SkView* focus = this->getFocusView();
231 if (focus == NULL)
232 focus = this;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000233
rmistry@google.comd6176b02012-08-23 18:14:13 +0000234 SkEvent evt(SK_EventType_Unichar);
235 evt.setFast32(uni);
236 return focus->doEvent(evt);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000237}
238
239bool SkWindow::handleKey(SkKey key)
240{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000241 if (key == kNONE_SkKey)
242 return false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000243
rmistry@google.comd6176b02012-08-23 18:14:13 +0000244 if (this->onHandleKey(key))
245 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000246
rmistry@google.comd6176b02012-08-23 18:14:13 +0000247 // send an event to the focus-view
248 {
249 SkView* focus = this->getFocusView();
250 if (focus == NULL)
251 focus = this;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000252
rmistry@google.comd6176b02012-08-23 18:14:13 +0000253 SkEvent evt(SK_EventType_Key);
254 evt.setFast32(key);
255 if (focus->doEvent(evt))
256 return true;
257 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000258
rmistry@google.comd6176b02012-08-23 18:14:13 +0000259 if (key == kUp_SkKey || key == kDown_SkKey)
260 {
261 if (this->moveFocus(key == kUp_SkKey ? kPrev_FocusDirection : kNext_FocusDirection) == NULL)
262 this->onSetFocusView(NULL);
263 return true;
264 }
265 return false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000266}
267
268bool SkWindow::handleKeyUp(SkKey key)
269{
270 if (key == kNONE_SkKey)
271 return false;
rmistry@google.comd6176b02012-08-23 18:14:13 +0000272
reed@android.com8a1c16f2008-12-17 15:59:43 +0000273 if (this->onHandleKeyUp(key))
274 return true;
rmistry@google.comd6176b02012-08-23 18:14:13 +0000275
reed@android.com8a1c16f2008-12-17 15:59:43 +0000276 //send an event to the focus-view
277 {
278 SkView* focus = this->getFocusView();
279 if (focus == NULL)
280 focus = this;
rmistry@google.comd6176b02012-08-23 18:14:13 +0000281
reed@android.com8a1c16f2008-12-17 15:59:43 +0000282 //should this one be the same?
283 SkEvent evt(SK_EventType_KeyUp);
284 evt.setFast32(key);
285 if (focus->doEvent(evt))
286 return true;
287 }
288 return false;
289}
290
yangsu@google.com654d72f2011-08-01 17:27:33 +0000291void SkWindow::addMenu(SkOSMenu* menu) {
rmistry@google.comd6176b02012-08-23 18:14:13 +0000292 *fMenus.append() = menu;
293 this->onAddMenu(menu);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000294}
295
reed@android.com0ae6b242008-12-23 16:49:54 +0000296void SkWindow::setTitle(const char title[]) {
297 if (NULL == title) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000298 title = "";
reed@android.com0ae6b242008-12-23 16:49:54 +0000299 }
300 fTitle.set(title);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000301 this->onSetTitle(title);
302}
303
reed@android.com8a1c16f2008-12-17 15:59:43 +0000304//////////////////////////////////////////////////////////////////////
305
306bool SkWindow::onEvent(const SkEvent& evt)
307{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000308 if (evt.isType(SK_EventDelayInval))
309 {
310 SkRegion::Iterator iter(fDirtyRgn);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000311
rmistry@google.comd6176b02012-08-23 18:14:13 +0000312 for (; !iter.done(); iter.next())
313 this->onHandleInval(iter.rect());
314 fWaitingOnInval = false;
315 return true;
316 }
317 return this->INHERITED::onEvent(evt);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000318}
319
320bool SkWindow::onGetFocusView(SkView** focus) const
321{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000322 if (focus)
323 *focus = fFocusView;
324 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000325}
326
327bool SkWindow::onSetFocusView(SkView* focus)
328{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000329 if (fFocusView != focus)
330 {
331 if (fFocusView)
332 fFocusView->onFocusChange(false);
333 fFocusView = focus;
334 if (focus)
335 focus->onFocusChange(true);
336 }
337 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000338}
339
340//////////////////////////////////////////////////////////////////////
341
342void SkWindow::onHandleInval(const SkIRect&)
343{
344}
345
346bool SkWindow::onHandleChar(SkUnichar)
347{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000348 return false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000349}
350
sugoi@google.com93c7ee32013-03-12 14:36:57 +0000351bool SkWindow::onHandleKey(SkKey)
reed@android.com8a1c16f2008-12-17 15:59:43 +0000352{
rmistry@google.comd6176b02012-08-23 18:14:13 +0000353 return false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000354}
355
sugoi@google.com93c7ee32013-03-12 14:36:57 +0000356bool SkWindow::onHandleKeyUp(SkKey)
reed@android.com8a1c16f2008-12-17 15:59:43 +0000357{
358 return false;
359}
360
reed@google.com4d5c26d2013-01-08 16:17:50 +0000361bool SkWindow::handleClick(int x, int y, Click::State state, void *owner,
362 unsigned modifierKeys) {
363 return this->onDispatchClick(x, y, state, owner, modifierKeys);
mike@reedtribe.orgdd0cd342011-03-21 00:53:39 +0000364}
365
Scroggod3aed392011-06-22 13:26:56 +0000366bool SkWindow::onDispatchClick(int x, int y, Click::State state,
reed@google.com4d5c26d2013-01-08 16:17:50 +0000367 void* owner, unsigned modifierKeys) {
rmistry@google.comd6176b02012-08-23 18:14:13 +0000368 bool handled = false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000369
Scroggod3aed392011-06-22 13:26:56 +0000370 // First, attempt to find an existing click with this owner.
371 int index = -1;
372 for (int i = 0; i < fClicks.count(); i++) {
373 if (owner == fClicks[i]->fOwner) {
374 index = i;
375 break;
376 }
377 }
378
rmistry@google.comd6176b02012-08-23 18:14:13 +0000379 switch (state) {
Scroggod3aed392011-06-22 13:26:56 +0000380 case Click::kDown_State: {
381 if (index != -1) {
382 delete fClicks[index];
383 fClicks.remove(index);
384 }
385 Click* click = this->findClickHandler(SkIntToScalar(x),
reed@google.com4d5c26d2013-01-08 16:17:50 +0000386 SkIntToScalar(y), modifierKeys);
Scroggod3aed392011-06-22 13:26:56 +0000387
388 if (click) {
389 click->fOwner = owner;
390 *fClicks.append() = click;
reed@google.com4d5c26d2013-01-08 16:17:50 +0000391 SkView::DoClickDown(click, x, y, modifierKeys);
Scroggod3aed392011-06-22 13:26:56 +0000392 handled = true;
393 }
394 break;
395 }
396 case Click::kMoved_State:
397 if (index != -1) {
reed@google.com4d5c26d2013-01-08 16:17:50 +0000398 SkView::DoClickMoved(fClicks[index], x, y, modifierKeys);
Scroggod3aed392011-06-22 13:26:56 +0000399 handled = true;
400 }
401 break;
402 case Click::kUp_State:
403 if (index != -1) {
reed@google.com4d5c26d2013-01-08 16:17:50 +0000404 SkView::DoClickUp(fClicks[index], x, y, modifierKeys);
Scroggod3aed392011-06-22 13:26:56 +0000405 delete fClicks[index];
406 fClicks.remove(index);
407 handled = true;
408 }
409 break;
410 default:
411 // Do nothing
412 break;
rmistry@google.comd6176b02012-08-23 18:14:13 +0000413 }
414 return handled;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000415}