| /* |
| * Copyright 2011 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "SkWindow.h" |
| #include "SkCanvas.h" |
| #include "SkOSMenu.h" |
| #include "SkSurface.h" |
| #include "SkSystemEventTypes.h" |
| #include "SkTime.h" |
| |
| #define SK_EventDelayInval "\xd" "n" "\xa" "l" |
| |
| SkWindow::SkWindow() |
| : fSurfaceProps(SkSurfaceProps::kLegacyFontHost_InitType) |
| , fFocusView(nullptr) |
| { |
| fClicks.reset(); |
| fWaitingOnInval = false; |
| fMatrix.reset(); |
| |
| fBitmap.allocN32Pixels(0, 0); |
| } |
| |
| SkWindow::~SkWindow() { |
| fClicks.deleteAll(); |
| fMenus.deleteAll(); |
| } |
| |
| sk_sp<SkSurface> SkWindow::makeSurface() { |
| const SkBitmap& bm = this->getBitmap(); |
| return SkSurface::MakeRasterDirect(bm.info(), bm.getPixels(), bm.rowBytes(), &fSurfaceProps); |
| } |
| |
| void SkWindow::setMatrix(const SkMatrix& matrix) { |
| if (fMatrix != matrix) { |
| fMatrix = matrix; |
| this->inval(nullptr); |
| } |
| } |
| |
| void SkWindow::preConcat(const SkMatrix& matrix) { |
| SkMatrix m; |
| m.setConcat(fMatrix, matrix); |
| this->setMatrix(m); |
| } |
| |
| void SkWindow::postConcat(const SkMatrix& matrix) { |
| SkMatrix m; |
| m.setConcat(matrix, fMatrix); |
| this->setMatrix(m); |
| } |
| |
| void SkWindow::resize(const SkImageInfo& info) { |
| if (fBitmap.info() != info) { |
| fBitmap.allocPixels(info); |
| this->inval(nullptr); |
| } |
| this->setSize(SkIntToScalar(fBitmap.width()), SkIntToScalar(fBitmap.height())); |
| } |
| |
| void SkWindow::resize(int width, int height) { |
| this->resize(fBitmap.info().makeWH(width, height)); |
| } |
| |
| void SkWindow::setColorType(SkColorType ct, sk_sp<SkColorSpace> cs) { |
| const SkImageInfo& info = fBitmap.info(); |
| this->resize(SkImageInfo::Make(info.width(), info.height(), ct, kPremul_SkAlphaType, cs)); |
| } |
| |
| bool SkWindow::handleInval(const SkRect* localR) { |
| SkIRect ir; |
| |
| if (localR) { |
| SkRect devR; |
| SkMatrix inverse; |
| if (!fMatrix.invert(&inverse)) { |
| return false; |
| } |
| fMatrix.mapRect(&devR, *localR); |
| devR.round(&ir); |
| } else { |
| ir.set(0, 0, |
| SkScalarRoundToInt(this->width()), |
| SkScalarRoundToInt(this->height())); |
| } |
| fDirtyRgn.op(ir, SkRegion::kUnion_Op); |
| |
| this->onHandleInval(ir); |
| return true; |
| } |
| |
| void SkWindow::forceInvalAll() { |
| fDirtyRgn.setRect(0, 0, |
| SkScalarCeilToInt(this->width()), |
| SkScalarCeilToInt(this->height())); |
| } |
| |
| #ifdef SK_SIMULATE_FAILED_MALLOC |
| extern bool gEnableControlledThrow; |
| #endif |
| |
| bool SkWindow::update(SkIRect* updateArea) { |
| if (!fDirtyRgn.isEmpty()) { |
| sk_sp<SkSurface> surface(this->makeSurface()); |
| SkCanvas* canvas = surface->getCanvas(); |
| |
| canvas->clipRegion(fDirtyRgn); |
| if (updateArea) { |
| *updateArea = fDirtyRgn.getBounds(); |
| } |
| |
| SkAutoCanvasRestore acr(canvas, true); |
| canvas->concat(fMatrix); |
| |
| // empty this now, so we can correctly record any inval calls that |
| // might be made during the draw call. |
| fDirtyRgn.setEmpty(); |
| |
| #ifdef SK_SIMULATE_FAILED_MALLOC |
| gEnableControlledThrow = true; |
| #endif |
| #ifdef SK_BUILD_FOR_WIN32 |
| //try { |
| this->draw(canvas); |
| //} |
| //catch (...) { |
| //} |
| #else |
| this->draw(canvas); |
| #endif |
| #ifdef SK_SIMULATE_FAILED_MALLOC |
| gEnableControlledThrow = false; |
| #endif |
| |
| return true; |
| } |
| return false; |
| } |
| |
| bool SkWindow::handleChar(SkUnichar uni) { |
| if (this->onHandleChar(uni)) |
| return true; |
| |
| SkView* focus = this->getFocusView(); |
| if (focus == nullptr) |
| focus = this; |
| |
| SkEvent evt(SK_EventType_Unichar); |
| evt.setFast32(uni); |
| return focus->doEvent(evt); |
| } |
| |
| bool SkWindow::handleKey(SkKey key) { |
| if (key == kNONE_SkKey) |
| return false; |
| |
| if (this->onHandleKey(key)) |
| return true; |
| |
| // send an event to the focus-view |
| { |
| SkView* focus = this->getFocusView(); |
| if (focus == nullptr) |
| focus = this; |
| |
| SkEvent evt(SK_EventType_Key); |
| evt.setFast32(key); |
| if (focus->doEvent(evt)) |
| return true; |
| } |
| |
| if (key == kUp_SkKey || key == kDown_SkKey) { |
| if (this->moveFocus(key == kUp_SkKey ? kPrev_FocusDirection : kNext_FocusDirection) == nullptr) |
| this->onSetFocusView(nullptr); |
| return true; |
| } |
| return false; |
| } |
| |
| bool SkWindow::handleKeyUp(SkKey key) { |
| if (key == kNONE_SkKey) |
| return false; |
| |
| if (this->onHandleKeyUp(key)) |
| return true; |
| |
| //send an event to the focus-view |
| { |
| SkView* focus = this->getFocusView(); |
| if (focus == nullptr) |
| focus = this; |
| |
| //should this one be the same? |
| SkEvent evt(SK_EventType_KeyUp); |
| evt.setFast32(key); |
| if (focus->doEvent(evt)) |
| return true; |
| } |
| return false; |
| } |
| |
| void SkWindow::addMenu(SkOSMenu* menu) { |
| *fMenus.append() = menu; |
| this->onAddMenu(menu); |
| } |
| |
| void SkWindow::setTitle(const char title[]) { |
| if (nullptr == title) { |
| title = ""; |
| } |
| fTitle.set(title); |
| this->onSetTitle(title); |
| } |
| |
| bool SkWindow::onEvent(const SkEvent& evt) { |
| if (evt.isType(SK_EventDelayInval)) { |
| for (SkRegion::Iterator iter(fDirtyRgn); !iter.done(); iter.next()) |
| this->onHandleInval(iter.rect()); |
| fWaitingOnInval = false; |
| return true; |
| } |
| return this->INHERITED::onEvent(evt); |
| } |
| |
| bool SkWindow::onGetFocusView(SkView** focus) const { |
| if (focus) |
| *focus = fFocusView; |
| return true; |
| } |
| |
| bool SkWindow::onSetFocusView(SkView* focus) { |
| if (fFocusView != focus) { |
| if (fFocusView) |
| fFocusView->onFocusChange(false); |
| fFocusView = focus; |
| if (focus) |
| focus->onFocusChange(true); |
| } |
| return true; |
| } |
| |
| void SkWindow::onHandleInval(const SkIRect&) { |
| } |
| |
| bool SkWindow::onHandleChar(SkUnichar) { |
| return false; |
| } |
| |
| bool SkWindow::onHandleKey(SkKey) { |
| return false; |
| } |
| |
| bool SkWindow::onHandleKeyUp(SkKey) { |
| return false; |
| } |
| |
| bool SkWindow::handleClick(int x, int y, Click::State state, void *owner, |
| unsigned modifierKeys) { |
| return this->onDispatchClick(x, y, state, owner, modifierKeys); |
| } |
| |
| bool SkWindow::onDispatchClick(int x, int y, Click::State state, |
| void* owner, unsigned modifierKeys) { |
| bool handled = false; |
| |
| // First, attempt to find an existing click with this owner. |
| int index = -1; |
| for (int i = 0; i < fClicks.count(); i++) { |
| if (owner == fClicks[i]->fOwner) { |
| index = i; |
| break; |
| } |
| } |
| |
| switch (state) { |
| case Click::kDown_State: { |
| if (index != -1) { |
| delete fClicks[index]; |
| fClicks.remove(index); |
| } |
| Click* click = this->findClickHandler(SkIntToScalar(x), |
| SkIntToScalar(y), modifierKeys); |
| |
| if (click) { |
| click->fOwner = owner; |
| *fClicks.append() = click; |
| SkView::DoClickDown(click, x, y, modifierKeys); |
| handled = true; |
| } |
| break; |
| } |
| case Click::kMoved_State: |
| if (index != -1) { |
| SkView::DoClickMoved(fClicks[index], x, y, modifierKeys); |
| handled = true; |
| } |
| break; |
| case Click::kUp_State: |
| if (index != -1) { |
| SkView::DoClickUp(fClicks[index], x, y, modifierKeys); |
| delete fClicks[index]; |
| fClicks.remove(index); |
| handled = true; |
| } |
| break; |
| default: |
| // Do nothing |
| break; |
| } |
| return handled; |
| } |
| |
| #if SK_SUPPORT_GPU |
| |
| #include "GrContext.h" |
| #include "gl/GrGLInterface.h" |
| #include "gl/GrGLUtil.h" |
| #include "SkGr.h" |
| |
| sk_sp<SkSurface> SkWindow::makeGpuBackedSurface(const AttachmentInfo& attachmentInfo, |
| const GrGLInterface* interface, |
| GrContext* grContext) { |
| GrBackendRenderTargetDesc desc; |
| desc.fWidth = SkScalarRoundToInt(this->width()); |
| desc.fHeight = SkScalarRoundToInt(this->height()); |
| if (0 == desc.fWidth || 0 == desc.fHeight) { |
| return nullptr; |
| } |
| |
| // TODO: Query the actual framebuffer for sRGB capable. However, to |
| // preserve old (fake-linear) behavior, we don't do this. Instead, rely |
| // on the flag (currently driven via 'C' mode in SampleApp). |
| // |
| // Also, we may not have real sRGB support (ANGLE, in particular), so check for |
| // that, and fall back to L32: |
| // |
| // ... and, if we're using a 10-bit/channel FB0, it doesn't do sRGB conversion on write, |
| // so pretend that it's non-sRGB 8888: |
| desc.fConfig = |
| grContext->caps()->srgbSupport() && |
| info().colorSpace() && |
| (attachmentInfo.fColorBits != 30) |
| ? kSRGBA_8888_GrPixelConfig : kRGBA_8888_GrPixelConfig; |
| desc.fOrigin = kBottomLeft_GrSurfaceOrigin; |
| desc.fSampleCnt = attachmentInfo.fSampleCount; |
| desc.fStencilBits = attachmentInfo.fStencilBits; |
| GrGLint buffer; |
| GR_GL_GetIntegerv(interface, GR_GL_FRAMEBUFFER_BINDING, &buffer); |
| desc.fRenderTargetHandle = buffer; |
| |
| sk_sp<SkColorSpace> colorSpace = |
| grContext->caps()->srgbSupport() && info().colorSpace() |
| ? SkColorSpace::MakeNamed(SkColorSpace::kSRGB_Named) : nullptr; |
| return SkSurface::MakeFromBackendRenderTarget(grContext, desc, colorSpace, &fSurfaceProps); |
| } |
| |
| #endif |