src/video/uikit/SDL_uikitvideo.m
author Ryan C. Gordon
Sun, 27 Mar 2011 01:35:19 -0400
changeset 5518 f84dd424d514
parent 5262 b530ef003506
child 5535 96594ac5fd1a
permissions -rw-r--r--
iOS: Report both landscape and portrait orientation as display modes.
slouken@2765
     1
/*
slouken@2765
     2
    SDL - Simple DirectMedia Layer
slouken@5262
     3
    Copyright (C) 1997-2011 Sam Lantinga
slouken@2765
     4
slouken@2765
     5
    This library is free software; you can redistribute it and/or
slouken@2765
     6
    modify it under the terms of the GNU Lesser General Public
slouken@2765
     7
    License as published by the Free Software Foundation; either
slouken@2765
     8
    version 2.1 of the License, or (at your option) any later version.
slouken@2765
     9
slouken@2765
    10
    This library is distributed in the hope that it will be useful,
slouken@2765
    11
    but WITHOUT ANY WARRANTY; without even the implied warranty of
slouken@2765
    12
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
slouken@2765
    13
    Lesser General Public License for more details.
slouken@2765
    14
slouken@2765
    15
    You should have received a copy of the GNU Lesser General Public
slouken@2765
    16
    License along with this library; if not, write to the Free Software
slouken@2765
    17
    Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
slouken@2765
    18
slouken@2765
    19
    Sam Lantinga
slouken@2765
    20
    slouken@libsdl.org
slouken@2765
    21
*/
icculus@4443
    22
icculus@4443
    23
#import <UIKit/UIKit.h>
icculus@4443
    24
slouken@2765
    25
#include "SDL_config.h"
slouken@2765
    26
slouken@2765
    27
#include "SDL_video.h"
slouken@2765
    28
#include "SDL_mouse.h"
slouken@2765
    29
#include "../SDL_sysvideo.h"
slouken@2765
    30
#include "../SDL_pixels_c.h"
slouken@2765
    31
#include "../../events/SDL_events_c.h"
slouken@2765
    32
slouken@2765
    33
#include "SDL_uikitvideo.h"
slouken@2765
    34
#include "SDL_uikitevents.h"
slouken@2765
    35
#include "SDL_uikitwindow.h"
slouken@2765
    36
#include "SDL_uikitopengles.h"
slouken@2765
    37
icculus@4443
    38
#include "SDL_assert.h"
icculus@4443
    39
slouken@2765
    40
#define UIKITVID_DRIVER_NAME "uikit"
slouken@2765
    41
slouken@2765
    42
/* Initialization/Query functions */
slouken@2765
    43
static int UIKit_VideoInit(_THIS);
icculus@4443
    44
static void UIKit_GetDisplayModes(_THIS, SDL_VideoDisplay * sdl_display);
slouken@3643
    45
static int UIKit_SetDisplayMode(_THIS, SDL_VideoDisplay * display,
slouken@3643
    46
                                SDL_DisplayMode * mode);
slouken@2765
    47
static void UIKit_VideoQuit(_THIS);
slouken@2765
    48
icculus@4446
    49
BOOL SDL_UIKit_supports_multiple_displays = NO;
icculus@4443
    50
slouken@2765
    51
/* DUMMY driver bootstrap functions */
slouken@2765
    52
slouken@2765
    53
static int
slouken@2765
    54
UIKit_Available(void)
slouken@2765
    55
{
slouken@5131
    56
    return (1);
slouken@2765
    57
}
slouken@2765
    58
slouken@2765
    59
static void UIKit_DeleteDevice(SDL_VideoDevice * device)
slouken@2765
    60
{
slouken@2765
    61
    SDL_free(device);
slouken@2765
    62
}
slouken@2765
    63
slouken@2765
    64
static SDL_VideoDevice *
slouken@2765
    65
UIKit_CreateDevice(int devindex)
slouken@2765
    66
{
slouken@2765
    67
    SDL_VideoDevice *device;
slouken@2765
    68
slouken@2765
    69
    /* Initialize all variables that we clean on shutdown */
slouken@2765
    70
    device = (SDL_VideoDevice *) SDL_calloc(1, sizeof(SDL_VideoDevice));
slouken@2765
    71
    if (!device) {
slouken@2765
    72
        SDL_OutOfMemory();
slouken@2765
    73
        if (device) {
slouken@2765
    74
            SDL_free(device);
slouken@2765
    75
        }
slouken@2765
    76
        return (0);
slouken@2765
    77
    }
slouken@2765
    78
slouken@2765
    79
    /* Set the function pointers */
slouken@2765
    80
    device->VideoInit = UIKit_VideoInit;
slouken@2765
    81
    device->VideoQuit = UIKit_VideoQuit;
icculus@4443
    82
    device->GetDisplayModes = UIKit_GetDisplayModes;
slouken@2765
    83
    device->SetDisplayMode = UIKit_SetDisplayMode;
slouken@2765
    84
    device->PumpEvents = UIKit_PumpEvents;
slouken@5131
    85
    device->CreateWindow = UIKit_CreateWindow;
slouken@5131
    86
    device->DestroyWindow = UIKit_DestroyWindow;
slouken@5056
    87
    device->GetWindowWMInfo = UIKit_GetWindowWMInfo;
slouken@5131
    88
    
slouken@5131
    89
    
slouken@5131
    90
    /* OpenGL (ES) functions */
slouken@5131
    91
    device->GL_MakeCurrent        = UIKit_GL_MakeCurrent;
slouken@5131
    92
    device->GL_SwapWindow        = UIKit_GL_SwapWindow;
slouken@5131
    93
    device->GL_CreateContext    = UIKit_GL_CreateContext;
slouken@5131
    94
    device->GL_DeleteContext    = UIKit_GL_DeleteContext;
slouken@5131
    95
    device->GL_GetProcAddress   = UIKit_GL_GetProcAddress;
slouken@5131
    96
    device->GL_LoadLibrary        = UIKit_GL_LoadLibrary;
slouken@5131
    97
    device->free = UIKit_DeleteDevice;
slouken@2765
    98
slouken@5131
    99
    device->gl_config.accelerated = 1;
slouken@5131
   100
slouken@2765
   101
    return device;
slouken@2765
   102
}
slouken@2765
   103
slouken@2765
   104
VideoBootStrap UIKIT_bootstrap = {
slouken@2765
   105
    UIKITVID_DRIVER_NAME, "SDL UIKit video driver",
slouken@2765
   106
    UIKit_Available, UIKit_CreateDevice
slouken@2765
   107
};
slouken@2765
   108
slouken@2765
   109
icculus@4443
   110
/*
icculus@4443
   111
!!! FIXME:
icculus@4443
   112
icculus@4443
   113
The main screen should list a AxB mode for portrait orientation, and then
icculus@4443
   114
 also list BxA for landscape mode. When setting a given resolution, we should
icculus@4443
   115
 rotate the view's transform appropriately (extra credit if you check the
icculus@4443
   116
 accelerometer and rotate the display so it's never upside down).
icculus@4443
   117
icculus@4443
   118
  http://iphonedevelopment.blogspot.com/2008/10/starting-in-landscape-mode-without.html
icculus@4443
   119
icculus@4443
   120
*/
icculus@4443
   121
icculus@5518
   122
static CGSize
icculus@5518
   123
UIKit_ForcePortrait(const CGSize size)
icculus@5518
   124
{
icculus@5518
   125
    CGSize retval;
icculus@5518
   126
    if (size.width < size.height) { // portrait
icculus@5518
   127
        retval = size;
icculus@5518
   128
    } else {  // landscape
icculus@5518
   129
        retval.width = size.height;
icculus@5518
   130
        retval.height = size.width;
icculus@5518
   131
    }
icculus@5518
   132
    return retval;
icculus@5518
   133
}
icculus@5518
   134
icculus@5518
   135
static CGSize
icculus@5518
   136
UIKit_ForceLandscape(const CGSize size)
icculus@5518
   137
{
icculus@5518
   138
    CGSize retval;
icculus@5518
   139
    if (size.width > size.height) { // landscape
icculus@5518
   140
        retval = size;
icculus@5518
   141
    } else {  // portrait
icculus@5518
   142
        retval.width = size.height;
icculus@5518
   143
        retval.height = size.width;
icculus@5518
   144
    }
icculus@5518
   145
    return retval;
icculus@5518
   146
}
icculus@5518
   147
icculus@4443
   148
static void
icculus@4443
   149
UIKit_GetDisplayModes(_THIS, SDL_VideoDisplay * display)
icculus@4443
   150
{
icculus@4446
   151
    UIScreen *uiscreen = (UIScreen *) display->driverdata;
icculus@4443
   152
    SDL_DisplayMode mode;
icculus@4443
   153
    SDL_zero(mode);
icculus@4443
   154
icculus@4443
   155
    // availableModes showed up in 3.2 (the iPad and later). We should only
icculus@4443
   156
    //  land here for at least that version of the OS.
icculus@4446
   157
    if (!SDL_UIKit_supports_multiple_displays) {
icculus@4446
   158
        const CGRect rect = [uiscreen bounds];
icculus@4443
   159
        mode.format = SDL_PIXELFORMAT_ABGR8888;
icculus@4443
   160
        mode.w = (int) rect.size.width;
icculus@4443
   161
        mode.h = (int) rect.size.height;
icculus@4443
   162
        mode.refresh_rate = 0;
icculus@4443
   163
        mode.driverdata = NULL;
icculus@4443
   164
        SDL_AddDisplayMode(display, &mode);
icculus@5518
   165
        mode.w = (int) rect.size.height;  // swap the orientation, add again.
icculus@5518
   166
        mode.h = (int) rect.size.width;
icculus@5518
   167
        SDL_AddDisplayMode(display, &mode);
icculus@4443
   168
        return;
icculus@4443
   169
    }
icculus@4443
   170
icculus@5518
   171
    const int ismain = (uiscreen == [UIScreen mainScreen]);
icculus@4446
   172
    const NSArray *modes = [uiscreen availableModes];
icculus@4443
   173
    const NSUInteger mode_count = [modes count];
icculus@4443
   174
    NSUInteger i;
icculus@4443
   175
    for (i = 0; i < mode_count; i++) {
icculus@4443
   176
        UIScreenMode *uimode = (UIScreenMode *) [modes objectAtIndex:i];
icculus@5518
   177
        CGSize size = [uimode size];
icculus@4443
   178
        mode.format = SDL_PIXELFORMAT_ABGR8888;
icculus@5518
   179
        mode.refresh_rate = 0;
icculus@5518
   180
        mode.driverdata = uimode;
icculus@4443
   181
        mode.w = (int) size.width;
icculus@4443
   182
        mode.h = (int) size.height;
icculus@5518
   183
        if (SDL_AddDisplayMode(display, &mode))
icculus@5518
   184
            [uimode retain];
icculus@5518
   185
icculus@5518
   186
        if (ismain) {
icculus@5518
   187
            // Add the mode twice, flipped to portrait and landscape.
icculus@5518
   188
            //  SDL_AddDisplayMode() will ignore duplicates.
icculus@5518
   189
            size = UIKit_ForcePortrait([uimode size]);
icculus@5518
   190
            mode.w = (int) size.width;
icculus@5518
   191
            mode.h = (int) size.height;
icculus@5518
   192
            if (SDL_AddDisplayMode(display, &mode))
icculus@5518
   193
                [uimode retain];
icculus@5518
   194
icculus@5518
   195
            size = UIKit_ForceLandscape(size);
icculus@5518
   196
            mode.w = (int) size.width;
icculus@5518
   197
            mode.h = (int) size.height;
icculus@5518
   198
            if (SDL_AddDisplayMode(display, &mode))
icculus@5518
   199
                [uimode retain];
icculus@5518
   200
        }
icculus@4443
   201
    }
icculus@4443
   202
}
icculus@4443
   203
icculus@4443
   204
icculus@4443
   205
static void
icculus@4446
   206
UIKit_AddDisplay(UIScreen *uiscreen, int w, int h)
icculus@4443
   207
{
icculus@4443
   208
    SDL_VideoDisplay display;
icculus@4443
   209
    SDL_DisplayMode mode;
icculus@4443
   210
    SDL_zero(mode);
icculus@4443
   211
    mode.format = SDL_PIXELFORMAT_ABGR8888;
icculus@4443
   212
    mode.w = w;
icculus@4443
   213
    mode.h = h;
icculus@4443
   214
    mode.refresh_rate = 0;
icculus@4443
   215
icculus@4443
   216
    SDL_zero(display);
icculus@4443
   217
    display.desktop_mode = mode;
icculus@4443
   218
    display.current_mode = mode;
icculus@4446
   219
icculus@4446
   220
    [uiscreen retain];
icculus@4446
   221
    display.driverdata = uiscreen;
icculus@4443
   222
    SDL_AddVideoDisplay(&display);
icculus@4443
   223
}
icculus@4443
   224
icculus@4443
   225
slouken@2765
   226
int
slouken@2765
   227
UIKit_VideoInit(_THIS)
slouken@2765
   228
{
icculus@4443
   229
    _this->gl_config.driver_loaded = 1;
slouken@2765
   230
icculus@4445
   231
    NSString *reqSysVer = @"3.2";
icculus@4445
   232
    NSString *currSysVer = [[UIDevice currentDevice] systemVersion];
icculus@4445
   233
    if ([currSysVer compare:reqSysVer options:NSNumericSearch] != NSOrderedAscending)
icculus@4446
   234
        SDL_UIKit_supports_multiple_displays = YES;
slouken@2765
   235
icculus@4443
   236
    // If this is iPhoneOS < 3.2, all devices are one screen, 320x480 pixels.
icculus@4443
   237
    //  The iPad added both a larger main screen and the ability to use
icculus@4443
   238
    //  external displays.
icculus@4446
   239
    if (!SDL_UIKit_supports_multiple_displays) {
icculus@4443
   240
        // Just give 'em the whole main screen.
icculus@4446
   241
        UIScreen *uiscreen = [UIScreen mainScreen];
icculus@4446
   242
        const CGRect rect = [uiscreen bounds];
icculus@4446
   243
        UIKit_AddDisplay(uiscreen, (int)rect.size.width, (int)rect.size.height);
icculus@4443
   244
    } else {
icculus@4443
   245
        const NSArray *screens = [UIScreen screens];
icculus@4443
   246
        const NSUInteger screen_count = [screens count];
icculus@4443
   247
        NSUInteger i;
icculus@4443
   248
        for (i = 0; i < screen_count; i++) {
icculus@4443
   249
            // the main screen is the first element in the array.
icculus@4446
   250
            UIScreen *uiscreen = (UIScreen *) [screens objectAtIndex:i];
icculus@4446
   251
            const CGSize size = [[uiscreen currentMode] size];
icculus@4446
   252
            UIKit_AddDisplay(uiscreen, (int) size.width, (int) size.height);
icculus@4443
   253
        }
icculus@4443
   254
    }
slouken@2765
   255
slouken@2765
   256
    /* We're done! */
slouken@2765
   257
    return 0;
slouken@2765
   258
}
slouken@2765
   259
slouken@2765
   260
static int
slouken@3643
   261
UIKit_SetDisplayMode(_THIS, SDL_VideoDisplay * display, SDL_DisplayMode * mode)
slouken@2765
   262
{
icculus@4446
   263
    UIScreen *uiscreen = (UIScreen *) display->driverdata;
icculus@4446
   264
    if (!SDL_UIKit_supports_multiple_displays) {
icculus@4443
   265
        // Not on at least iPhoneOS 3.2 (versions prior to iPad).
icculus@4443
   266
        SDL_assert(mode->driverdata == NULL);
icculus@4443
   267
    } else {
icculus@4443
   268
        UIScreenMode *uimode = (UIScreenMode *) mode->driverdata;
icculus@4446
   269
        [uiscreen setCurrentMode:uimode];
icculus@4443
   270
    }
icculus@4443
   271
slouken@2765
   272
    return 0;
slouken@2765
   273
}
slouken@2765
   274
slouken@2765
   275
void
slouken@2765
   276
UIKit_VideoQuit(_THIS)
slouken@2765
   277
{
icculus@4443
   278
    // Release Objective-C objects, so higher level doesn't free() them.
icculus@4443
   279
    int i, j;
icculus@4443
   280
    for (i = 0; i < _this->num_displays; i++) {
icculus@4443
   281
        SDL_VideoDisplay *display = &_this->displays[i];
icculus@4446
   282
        UIScreen *uiscreen = (UIScreen *) display->driverdata;
icculus@4446
   283
        [uiscreen release];
icculus@4443
   284
        display->driverdata = NULL;
icculus@4443
   285
        for (j = 0; j < display->num_display_modes; j++) {
icculus@4443
   286
            SDL_DisplayMode *mode = &display->display_modes[j];
icculus@4443
   287
            UIScreenMode *uimode = (UIScreenMode *) mode->driverdata;
icculus@4443
   288
            if (uimode) {
icculus@4443
   289
                [uimode release];
icculus@4443
   290
                mode->driverdata = NULL;
icculus@4443
   291
            }
icculus@4443
   292
        }
icculus@4443
   293
    }
slouken@2765
   294
}
slouken@2765
   295
slouken@2765
   296
/* vi: set ts=4 sw=4 expandtab: */