/*
* 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 "SkOSWindow_SDL.h"
#include "SkCanvas.h"
#include "SkColorPriv.h"
#include "SkGLCanvas.h"
#include "SkOSMenu.h"
#include "SkTime.h"
static void post_SkEvent_event() {
SDL_Event evt;
evt.type = SDL_USEREVENT;
evt.user.type = SDL_USEREVENT;
evt.user.code = 0;
evt.user.data1 = NULL;
evt.user.data2 = NULL;
SDL_PushEvent(&evt);
}
static bool skia_setBitmapFromSurface(SkBitmap* dst, SDL_Surface* src) {
SkBitmap::Config config;
switch (src->format->BytesPerPixel) {
case 2:
config = SkBitmap::kRGB_565_Config;
break;
case 4:
config = SkBitmap::kARGB_8888_Config;
break;
default:
return false;
}
dst->setConfig(config, src->w, src->h, src->pitch);
dst->setPixels(src->pixels);
return true;
}
SkOSWindow::SkOSWindow(void* screen) {
fScreen = reinterpret_cast<SDL_Surface*>(screen);
this->resize(fScreen->w, fScreen->h);
uint32_t rmask = SK_R32_MASK << SK_R32_SHIFT;
uint32_t gmask = SK_G32_MASK << SK_G32_SHIFT;
uint32_t bmask = SK_B32_MASK << SK_B32_SHIFT;
uint32_t amask = SK_A32_MASK << SK_A32_SHIFT;
if (fScreen->flags & SDL_OPENGL) {
fSurface = NULL;
fGLCanvas = new SkGLCanvas;
fGLCanvas->setViewport(fScreen->w, fScreen->h);
} else {
fGLCanvas = NULL;
fSurface = SDL_CreateRGBSurface(SDL_SWSURFACE, fScreen->w, fScreen->h,
32, rmask, gmask, bmask, amask);
}
}
SkOSWindow::~SkOSWindow() {
delete fGLCanvas;
if (fSurface) {
SDL_FreeSurface(fSurface);
}
}
#include <OpenGL/gl.h>
void SkOSWindow::doDraw() {
if (fGLCanvas) {
glEnable(GL_BLEND);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
glHint(GL_LINE_SMOOTH_HINT, GL_DONT_CARE);
glEnable(GL_TEXTURE_2D);
glClearColor(0, 0, 0, 0);
glClear(GL_COLOR_BUFFER_BIT);
int count = fGLCanvas->save();
this->draw(fGLCanvas);
fGLCanvas->restoreToCount(count);
SDL_GL_SwapBuffers( );
} else {
if ( SDL_MUSTLOCK(fSurface) ) {
if ( SDL_LockSurface(fSurface) < 0 ) {
return;
}
}
SkBitmap bitmap;
if (skia_setBitmapFromSurface(&bitmap, fSurface)) {
SkCanvas canvas(bitmap);
this->draw(&canvas);
}
if ( SDL_MUSTLOCK(fSurface) ) {
SDL_UnlockSurface(fSurface);
}
int result = SDL_BlitSurface(fSurface, NULL, fScreen, NULL);
if (result) {
SkDebugf("------- SDL_BlitSurface returned %d\n", result);
}
SDL_UpdateRect(fScreen, 0, 0, fScreen->w, fScreen->h);
}
}
static SkKey find_skkey(SDLKey src) {
// this array must match the enum order in SkKey.h
static const SDLKey gKeys[] = {
SDLK_UNKNOWN,
SDLK_UNKNOWN, // left softkey
SDLK_UNKNOWN, // right softkey
SDLK_UNKNOWN, // home
SDLK_UNKNOWN, // back
SDLK_UNKNOWN, // send
SDLK_UNKNOWN, // end
SDLK_0,
SDLK_1,
SDLK_2,
SDLK_3,
SDLK_4,
SDLK_5,
SDLK_6,
SDLK_7,
SDLK_8,
SDLK_9,
SDLK_ASTERISK,
SDLK_HASH,
SDLK_UP,
SDLK_DOWN,
SDLK_LEFT,
SDLK_RIGHT,
SDLK_RETURN, // OK
SDLK_UNKNOWN, // volume up
SDLK_UNKNOWN, // volume down
SDLK_UNKNOWN, // power
SDLK_UNKNOWN, // camera
};
const SDLKey* array = gKeys;
for (size_t i = 0; i < SK_ARRAY_COUNT(gKeys); i++) {
if (array[i] == src) {
return static_cast<SkKey>(i);
}
}
return kNONE_SkKey;
}
void SkOSWindow::handleSDLEvent(const SDL_Event& event) {
switch (event.type) {
case SDL_VIDEORESIZE:
this->resize(event.resize.w, event.resize.h);
break;
case SDL_VIDEOEXPOSE:
this->doDraw();
break;
case SDL_MOUSEMOTION:
if (event.motion.state == SDL_PRESSED) {
this->handleClick(event.motion.x, event.motion.y,
SkView::Click::kMoved_State);
}
break;
case SDL_MOUSEBUTTONDOWN:
case SDL_MOUSEBUTTONUP:
this->handleClick(event.button.x, event.button.y,
event.button.state == SDL_PRESSED ?
SkView::Click::kDown_State :
SkView::Click::kUp_State);
break;
case SDL_KEYDOWN: {
SkKey sk = find_skkey(event.key.keysym.sym);
if (kNONE_SkKey != sk) {
if (event.key.state == SDL_PRESSED) {
this->handleKey(sk);
} else {
this->handleKeyUp(sk);
}
}
break;
}
case SDL_USEREVENT:
if (SkEvent::ProcessEvent()) {
post_SkEvent_event();
}
break;
}
}
void SkOSWindow::onHandleInval(const SkIRect& r) {
SDL_Event evt;
evt.type = SDL_VIDEOEXPOSE;
evt.expose.type = SDL_VIDEOEXPOSE;
SDL_PushEvent(&evt);
}
void SkOSWindow::onSetTitle(const char title[]) {
SDL_WM_SetCaption(title, NULL);
}
void SkOSWindow::onAddMenu(const SkOSMenu* sk_menu) {}
///////////////////////////////////////////////////////////////////////////////////////
void SkEvent::SignalNonEmptyQueue() {
SkDebugf("-------- signal nonempty\n");
post_SkEvent_event();
}
static Uint32 timer_callback(Uint32 interval) {
// SkDebugf("-------- timercallback %d\n", interval);
SkEvent::ServiceQueueTimer();
return 0;
}
void SkEvent::SignalQueueTimer(SkMSec delay)
{
SDL_SetTimer(0, NULL);
if (delay) {
SDL_SetTimer(delay, timer_callback);
}
}