Hi Robert,

attached you'll find some small bugfixes and enhancements for the
ios-implementation of GraphicsWindow.

* add a RootViewController to the UIWindow, if applicable
* fixed a crash when adding the view to another view and not to a window
(reported by Mike Wozniewski)
* sometimes, the osg-side did not get notified, when the underlying view
got resized, hopefully fixed.

cheers,

Stephan

#include <iostream>
#include <osgViewer/api/IOS/GraphicsWindowIOS>

#import <QuartzCore/QuartzCore.h>
#import <UIKit/UIKit.h>

#if OSG_GLES1_FEATURES
    #import <OpenGLES/ES1/glext.h>
#else
    #import <OpenGLES/ES2/glext.h>
    // in GLES2, the OES suffix if dropped from function names (from rti) 
    #define glGenFramebuffersOES glGenFramebuffers 
    #define glGenRenderbuffersOES glGenRenderbuffers 
    #define glBindFramebufferOES glBindFramebuffer 
    #define glBindRenderbufferOES glBindRenderbuffer 
    #define glFramebufferRenderbufferOES glFramebufferRenderbuffer 
    #define glGetRenderbufferParameterivOES glGetRenderbufferParameteriv 
    #define glRenderbufferStorageOES glRenderbufferStorage 
    #define glDeleteRenderbuffersOES glDeleteRenderbuffers 
    #define glDeleteFramebuffersOES glDeleteFramebuffers 
    #define glCheckFramebufferStatusOES glCheckFramebufferStatus 

    #define GL_FRAMEBUFFER_OES GL_FRAMEBUFFER 
    #define GL_RENDERBUFFER_OES GL_RENDERBUFFER 
    #define GL_RENDERBUFFER_WIDTH_OES GL_RENDERBUFFER_WIDTH 
    #define GL_RENDERBUFFER_HEIGHT_OES GL_RENDERBUFFER_HEIGHT 
    #define GL_COLOR_ATTACHMENT0_OES GL_COLOR_ATTACHMENT0 
    #define GL_DEPTH_ATTACHMENT_OES GL_DEPTH_ATTACHMENT 
    #define GL_DEPTH_COMPONENT16_OES GL_DEPTH_COMPONENT16 
    #define GL_STENCIL_INDEX8_OES GL_STENCIL_INDEX8 
    #define GL_FRAMEBUFFER_COMPLETE_OES GL_FRAMEBUFFER_COMPLETE 
    #define GL_STENCIL_ATTACHMENT_OES GL_STENCIL_ATTACHMENT 

    #define GL_RGB5_A1_OES GL_RGB5_A1
#endif 

#include "IOSUtils.h"




#pragma mark GraphicsWindowIOSWindow

// 
----------------------------------------------------------------------------------------------------------
// GraphicsWindowIOSWindow, implements canBecomeKeyWindow + canBecomeMainWindow
// 
----------------------------------------------------------------------------------------------------------

@interface GraphicsWindowIOSWindow : UIWindow
{
}

- (BOOL) canBecomeKeyWindow;
- (BOOL) canBecomeMainWindow;

@end

@implementation GraphicsWindowIOSWindow

//
//Implement dealloc 
//
- (void) dealloc
{
    [super dealloc];
}

- (BOOL) canBecomeKeyWindow
{
    return YES;
}

- (BOOL) canBecomeMainWindow
{
    return YES;
}

@end

#pragma mark GraphicsWindowIOSGLView

// 
----------------------------------------------------------------------------------------------------------
// GraphicsWindowIOSGLView
// custom UIView-class handling creation and display of frame/render buffers 
plus receives touch input
// 
----------------------------------------------------------------------------------------------------------

typedef std::map<void*, unsigned int> TouchPointsIdMapping;

@interface GraphicsWindowIOSGLView : UIView
{
    @private
        osgViewer::GraphicsWindowIOS* _win;
        EAGLContext* _context;
    
        /* The pixel dimensions of the backbuffer */
        GLint _backingWidth;
        GLint _backingHeight;
    
        //the pixel buffers for the video
        /* OpenGL names for the renderbuffer and framebuffers used to render to 
this view */
        GLuint _viewRenderbuffer, _viewFramebuffer;
        
        /* OpenGL name for the depth buffer that is attached to 
viewFramebuffer, if it exists (0 if it does not exist) */
        GLuint _depthRenderbuffer;
    
        /* OpenGL name for the stencil buffer that is attached to 
viewFramebuffer, if it exists (0 if it does not exist) */
        GLuint _stencilBuffer;
        
        // for multisampled antialiased rendering
        GLuint _msaaFramebuffer, _msaaRenderBuffer, _msaaDepthBuffer;
        
        TouchPointsIdMapping* _touchPointsIdMapping;
        unsigned int _lastTouchPointId;


    
}

- (void)setGraphicsWindow: (osgViewer::GraphicsWindowIOS*) win;
- (osgViewer::GraphicsWindowIOS*) getGraphicsWindow;
- (void)setOpenGLContext: (EAGLContext*) context;
- (void)updateDimensions;
- (BOOL)createFramebuffer;
- (void)destroyFramebuffer;
- (void)swapBuffers;
- (void)bindFrameBuffer;

- (BOOL)acceptsFirstResponder;
- (BOOL)becomeFirstResponder;
- (BOOL)resignFirstResponder;

- (osgGA::GUIEventAdapter::TouchPhase) convertTouchPhase: (UITouchPhase) phase;
- (osg::Vec2) convertPointToPixel: (osg::Vec2) point;
- (void) dealloc;
@end

@implementation GraphicsWindowIOSGLView 

- (osgGA::GUIEventAdapter::TouchPhase) convertTouchPhase: (UITouchPhase) phase 
{
    switch(phase) {
    
        case UITouchPhaseBegan:
            return osgGA::GUIEventAdapter::TOUCH_BEGAN;
            break;
        case UITouchPhaseMoved:
            return osgGA::GUIEventAdapter::TOUCH_MOVED;
            break;

        case UITouchPhaseStationary:
            return osgGA::GUIEventAdapter::TOUCH_STATIONERY;
            break;

        case UITouchPhaseEnded:
        case UITouchPhaseCancelled:
            return osgGA::GUIEventAdapter::TOUCH_ENDED;
            break;
    }
    
    return osgGA::GUIEventAdapter::TOUCH_ENDED;

}


- (unsigned int)computeTouchId: (UITouch*) touch 
{
    unsigned int result(0);
    
    if (!_touchPointsIdMapping) {
        _lastTouchPointId = 0;
        _touchPointsIdMapping = new TouchPointsIdMapping();
    }
    
    switch([touch phase])
    {
    
        case UITouchPhaseBegan:
            {
                TouchPointsIdMapping::iterator itr = 
_touchPointsIdMapping->find(touch);
                // std::cout << "new: " << touch << " num: " << 
_touchPointsIdMapping->size() << " found: " << (itr != 
_touchPointsIdMapping->end()) << std::endl;
                 
                if (itr == _touchPointsIdMapping->end()) 
                {
                    (*_touchPointsIdMapping)[touch] = result = 
_lastTouchPointId;
                    _lastTouchPointId++;
                    break;
                }
               
            }
            // missing "break" by intention!
        
        case UITouchPhaseMoved:
        case UITouchPhaseStationary:
            {
                result = (*_touchPointsIdMapping)[touch];
            }
            break;
       
        case UITouchPhaseEnded:
        case UITouchPhaseCancelled:
            {
                TouchPointsIdMapping::iterator itr = 
_touchPointsIdMapping->find(touch);
                // std::cout<< "remove: " << touch << " num: " << 
_touchPointsIdMapping->size() << " found: " << (itr != 
_touchPointsIdMapping->end()) << std::endl;
                
                if (itr != _touchPointsIdMapping->end()) {
                    result = itr->second;
                    _touchPointsIdMapping->erase(itr);
                }
                if(_touchPointsIdMapping->size() == 0) {
                    _lastTouchPointId = 0;
                }
                // std::cout<< "remove: " << touch << " num: " << 
_touchPointsIdMapping->size() << std::endl;
            }
            break;
            
        default:
            break;
    }
        
    return result;
}


- (osg::Vec2) convertPointToPixel: (osg::Vec2) point
{
    //get the views contentscale factor and multiply the point by it
    float scale = 1.0f;
    
#if defined(__IPHONE_4_0) && (__IPHONE_OS_VERSION_MIN_REQUIRED >= __IPHONE_4_0)
    scale = self.contentScaleFactor;
#endif
    return osg::Vec2(point.x()*scale, point.y()*scale);
    
}

-(void) setGraphicsWindow: (osgViewer::GraphicsWindowIOS*) win
{
    _win = win;
    _touchPointsIdMapping = new TouchPointsIdMapping();
    _lastTouchPointId = 0;
}

- (osgViewer::GraphicsWindowIOS*) getGraphicsWindow {
    return _win;
}

-(void) setOpenGLContext: (EAGLContext*) context
{
    _context = context;
}


// You must implement this method
+ (Class)layerClass {
    return [CAEAGLLayer class];
}

//
//Called when the view is created using a frame for dimensions
//
- (id)initWithFrame:(CGRect)frame : (osgViewer::GraphicsWindowIOS*)win{
    
    _win = win;

    if ((self = [super initWithFrame:frame])) {
        // Get the layer
        CAEAGLLayer *eaglLayer = (CAEAGLLayer *)self.layer;
        
        osgViewer::GraphicsWindowIOS::WindowData* win_data(NULL);
        if (_win->getTraits()->inheritedWindowData.valid())
            win_data = 
dynamic_cast<osgViewer::GraphicsWindowIOS::WindowData*>(_win->getTraits()->inheritedWindowData.get());
        
        eaglLayer.opaque = win_data ? !win_data->getCreateTransparentView() : 
YES ;
        if(_win->getTraits()->alpha > 0)
        {
            //create layer with alpha channel RGBA8
            eaglLayer.drawableProperties = [NSDictionary 
dictionaryWithObjectsAndKeys:
                                            [NSNumber numberWithBool:NO], 
kEAGLDrawablePropertyRetainedBacking, kEAGLColorFormatRGBA8, 
kEAGLDrawablePropertyColorFormat, nil];
        }else{
            //else no alpha, IOS uses RBG565
            eaglLayer.drawableProperties = [NSDictionary 
dictionaryWithObjectsAndKeys:
                                            [NSNumber numberWithBool:NO], 
kEAGLDrawablePropertyRetainedBacking, kEAGLColorFormatRGB565, 
kEAGLDrawablePropertyColorFormat, nil];

        }
    }
    self.multipleTouchEnabled = YES;
    
    return self;
}

//
//Implement dealloc to destory our frame buffer
//
- (void) dealloc
{
    OSG_INFO << "GraphicsWindowIOSGLView::dealloc" << std::endl;
    if(_touchPointsIdMapping) 
        delete _touchPointsIdMapping;
    _touchPointsIdMapping = NULL;
    [super dealloc];
}

- (void)layoutSubviews {
    [super layoutSubviews];
    [self updateDimensions];
}


- (void) setFrame:(CGRect)frame
{
    [super setFrame:frame];
    [self updateDimensions];
}


- (void) updateDimensions
{
    if (_win)
    {
        CGRect frame = self.bounds;
        osg::Vec2 pointOrigin = osg::Vec2(frame.origin.x,frame.origin.y);
        osg::Vec2 pointSize = osg::Vec2(frame.size.width,frame.size.height);
        osg::Vec2 pixelOrigin = [(GraphicsWindowIOSGLView*)(self) 
convertPointToPixel:pointOrigin];
        osg::Vec2 pixelSize = [(GraphicsWindowIOSGLView*)(self) 
convertPointToPixel:pointSize];
        
        OSG_INFO << "updateDimensions, resize to "
            <<  pixelOrigin.x() << " " << pixelOrigin.y() << " " 
            << pixelSize.x() << " " << pixelSize.y() 
            << std::endl;
        _win->resized(pixelOrigin.x(), pixelOrigin.y(), pixelSize.x(), 
pixelSize.y());
    }
}

- (BOOL)createFramebuffer {

    _msaaFramebuffer = _msaaRenderBuffer = 0;
    
    glGenFramebuffersOES(1, &_viewFramebuffer);
    glGenRenderbuffersOES(1, &_viewRenderbuffer);
    
    // set the default id for osg to switch back after using fbos.
    _win->setDefaultFboId(_viewFramebuffer);
    
    glBindFramebufferOES(GL_FRAMEBUFFER_OES, _viewFramebuffer);
    glBindRenderbufferOES(GL_RENDERBUFFER_OES, _viewRenderbuffer);
    [_context renderbufferStorage:GL_RENDERBUFFER_OES 
fromDrawable:(CAEAGLLayer*)self.layer];
    glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES, GL_COLOR_ATTACHMENT0_OES, 
GL_RENDERBUFFER_OES, _viewRenderbuffer);
    
    glGetRenderbufferParameterivOES(GL_RENDERBUFFER_OES, 
GL_RENDERBUFFER_WIDTH_OES, &_backingWidth);
    glGetRenderbufferParameterivOES(GL_RENDERBUFFER_OES, 
GL_RENDERBUFFER_HEIGHT_OES, &_backingHeight);
    
    osg::notify(osg::DEBUG_INFO) << "GraphicsWindowIOS::createFramebuffer INFO: 
Created GL RenderBuffer of size " << _backingWidth << ", " << _backingHeight << 
" ." << std::endl;

#if __IPHONE_OS_VERSION_MIN_REQUIRED >= 50000
    //on ios 5 we have to use a packed depth stencil buffer if we want stencil
    if(_win->getTraits()->depth > 0) {
        //add stencil if requested
        if(_win->getTraits()->stencil > 0) {
            // Create a packed depth stencil buffer.
            glGenRenderbuffersOES(1, &_depthRenderbuffer);
            glBindRenderbufferOES(GL_RENDERBUFFER_OES, _depthRenderbuffer);
            
            glRenderbufferStorageOES(GL_RENDERBUFFER_OES, 
GL_DEPTH24_STENCIL8_OES, _backingWidth, _backingHeight);
            
            glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES, 
GL_DEPTH_ATTACHMENT_OES,
                                         GL_RENDERBUFFER_OES, 
_depthRenderbuffer);
            glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES, 
GL_STENCIL_ATTACHMENT_OES,
                                         GL_RENDERBUFFER_OES, 
_depthRenderbuffer);
        }else{
            //normal depth buffer
            glGenRenderbuffersOES(1, &_depthRenderbuffer);
            glBindRenderbufferOES(GL_RENDERBUFFER_OES, _depthRenderbuffer);
            if(_win->getTraits()->depth == 16)
            {
                glRenderbufferStorageOES(GL_RENDERBUFFER_OES, 
GL_DEPTH_COMPONENT16_OES, _backingWidth, _backingHeight);
            }else if(_win->getTraits()->depth == 24){
                glRenderbufferStorageOES(GL_RENDERBUFFER_OES, 
GL_DEPTH_COMPONENT24_OES, _backingWidth, _backingHeight);
            }

            glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES, 
GL_DEPTH_ATTACHMENT_OES, GL_RENDERBUFFER_OES, _depthRenderbuffer);
        }
    }
    
#else
    //add depth if requested
    if(_win->getTraits()->depth > 0) {
        glGenRenderbuffersOES(1, &_depthRenderbuffer);
        glBindRenderbufferOES(GL_RENDERBUFFER_OES, _depthRenderbuffer);
        if(_win->getTraits()->depth == 16)
        {
            glRenderbufferStorageOES(GL_RENDERBUFFER_OES, 
GL_DEPTH_COMPONENT16_OES, _backingWidth, _backingHeight);
        }else if(_win->getTraits()->depth == 24){
            glRenderbufferStorageOES(GL_RENDERBUFFER_OES, 
GL_DEPTH_COMPONENT24_OES, _backingWidth, _backingHeight);
        }

#if defined(GL_DEPTH_COMPONENT32_OES)
        else if(_win->getTraits()->depth == 32){
            glRenderbufferStorageOES(GL_RENDERBUFFER_OES, 
GL_DEPTH_COMPONENT32_OES, _backingWidth, _backingHeight);
        }
#endif

        glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES, 
GL_DEPTH_ATTACHMENT_OES, GL_RENDERBUFFER_OES, _depthRenderbuffer);
    }
    
    //add stencil if requested
    if(_win->getTraits()->stencil > 0) {
        glGenRenderbuffersOES(1, &_stencilBuffer);
        glBindRenderbufferOES(GL_RENDERBUFFER_OES, _stencilBuffer);
        glRenderbufferStorageOES(GL_RENDERBUFFER_OES, GL_STENCIL_INDEX8_OES, 
_backingWidth, _backingHeight);
        glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES, 
GL_STENCIL_ATTACHMENT_OES, GL_RENDERBUFFER_OES, _stencilBuffer);
    }  
#endif
    
    //MSAA only available for >= 4.0 sdk
    
#if defined(__IPHONE_4_0) && (__IPHONE_OS_VERSION_MIN_REQUIRED >= __IPHONE_4_0)
    
    if(_win->getTraits()->sampleBuffers > 0) 
    {
        glGenFramebuffersOES(1, &_msaaFramebuffer); 
        glGenRenderbuffersOES(1, &_msaaRenderBuffer);
        
        _win->setDefaultFboId(_msaaFramebuffer);
        
        glBindFramebufferOES(GL_FRAMEBUFFER_OES, _msaaFramebuffer); 
        glBindRenderbufferOES(GL_RENDERBUFFER_OES, _msaaRenderBuffer);
        
        // Samples is the amount of pixels the MSAA buffer uses to make one 
pixel on the render // buffer. Use a small number like 2 for the 3G and below 
and 4 or more for newer models
        
        glRenderbufferStorageMultisampleAPPLE(GL_RENDERBUFFER_OES, 
_win->getTraits()->samples, GL_RGB5_A1_OES, _backingWidth, _backingHeight);
        
        glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES, 
GL_COLOR_ATTACHMENT0_OES, GL_RENDERBUFFER_OES, _msaaRenderBuffer);
        glGenRenderbuffersOES(1, &_msaaDepthBuffer); 
        glBindRenderbufferOES(GL_RENDERBUFFER_OES, _msaaDepthBuffer);
        
        glRenderbufferStorageMultisampleAPPLE(GL_RENDERBUFFER_OES, 
_win->getTraits()->samples, ( _win->getTraits()->depth == 16) ? 
GL_DEPTH_COMPONENT16_OES : GL_DEPTH_COMPONENT24_OES, _backingWidth , 
_backingHeight);
        glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES, 
GL_DEPTH_ATTACHMENT_OES, GL_RENDERBUFFER_OES, _msaaDepthBuffer);
    
    }
#endif
    
    if(glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES) != 
GL_FRAMEBUFFER_COMPLETE_OES) {
        OSG_FATAL << "GraphicsWindowIOS::createFramebuffer ERROR: Failed to 
create a GL RenderBuffer, glCheckFramebufferStatusOES returned '" 
                  << glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES) << "'." << 
std::endl;
        return NO;
    }
    
    return YES;
}


- (void)destroyFramebuffer {
    
    if(_viewFramebuffer)
    {
        glDeleteFramebuffersOES(1, &_viewFramebuffer);
        _viewFramebuffer = 0;
    }
    if(_viewRenderbuffer)
    {
        glDeleteRenderbuffersOES(1, &_viewRenderbuffer);
        _viewRenderbuffer = 0;
    }
    
    if(_depthRenderbuffer) {
        glDeleteRenderbuffersOES(1, &_depthRenderbuffer);
        _depthRenderbuffer = 0;
    }
    
    if(_stencilBuffer) {
        glDeleteFramebuffersOES(1, &_stencilBuffer);
        _stencilBuffer = 0;
    }
    
    if(_msaaRenderBuffer) {
        glDeleteFramebuffersOES(1, &_msaaRenderBuffer);
        _msaaRenderBuffer = 0;
    }
    
    if(_msaaDepthBuffer) {
        glDeleteFramebuffersOES(1, &_msaaDepthBuffer);
        _msaaDepthBuffer = 0;
    }

    if(_msaaFramebuffer) {
        glDeleteFramebuffersOES(1, &_msaaFramebuffer);
        _msaaFramebuffer = 0;
    }
}

//
//Swap the view and render buffers
//
- (void)swapBuffers {


#if defined(__IPHONE_4_0) && (__IPHONE_OS_VERSION_MIN_REQUIRED >= __IPHONE_4_0) 
   
    if(_msaaFramebuffer) 
    {
        glBindFramebufferOES(GL_FRAMEBUFFER_OES, _msaaFramebuffer);
        
        glBindFramebufferOES(GL_READ_FRAMEBUFFER_APPLE, _msaaFramebuffer); 
        glBindFramebufferOES(GL_DRAW_FRAMEBUFFER_APPLE, _viewFramebuffer);
        
        glResolveMultisampleFramebufferAPPLE();
        
        GLenum attachments[] = {GL_DEPTH_ATTACHMENT_OES, 
GL_COLOR_ATTACHMENT0_OES}; 
        glDiscardFramebufferEXT(GL_READ_FRAMEBUFFER_APPLE, 2, attachments);
    }
#endif


      //swap buffers (sort of i think?)
    glBindRenderbufferOES(GL_RENDERBUFFER_OES, _viewRenderbuffer);
    
    //display render in context
    [_context presentRenderbuffer:GL_RENDERBUFFER_OES];
    
    //re bind the frame buffer for next frames renders
    glBindFramebufferOES(GL_FRAMEBUFFER_OES, _viewFramebuffer);
    
#if defined(__IPHONE_4_0) && (__IPHONE_OS_VERSION_MIN_REQUIRED >= __IPHONE_4_0)
    if (_msaaFramebuffer)
        glBindFramebufferOES(GL_FRAMEBUFFER_OES, _msaaFramebuffer);;
#endif
}

//
//bind view buffer as current for new render pass
//
- (void)bindFrameBuffer {

    //bind the frame buffer
    glBindFramebufferOES(GL_FRAMEBUFFER_OES, _viewFramebuffer);
    
#if defined(__IPHONE_4_0) && (__IPHONE_OS_VERSION_MIN_REQUIRED >= __IPHONE_4_0)
    if (_msaaFramebuffer)
        glBindFramebufferOES(GL_READ_FRAMEBUFFER_APPLE, _msaaFramebuffer);
#endif
}


- (BOOL)acceptsFirstResponder
{
  return YES;
}

- (BOOL)becomeFirstResponder
{
  return YES;
}

- (BOOL)resignFirstResponder
{
  return YES;
}

//
//Touch input callbacks
//
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
    
    NSSet *allTouches = [event allTouches];
    
    osg::ref_ptr<osgGA::GUIEventAdapter> osg_event(NULL);

    for(int i=0; i<[allTouches count]; i++)
    {
        
        UITouch *touch = [[allTouches allObjects] objectAtIndex:i];
        CGPoint pos = [touch locationInView:touch.view];
        osg::Vec2 pixelPos = [self convertPointToPixel: osg::Vec2(pos.x,pos.y)];
        unsigned int touch_id = [self computeTouchId: touch];
        
        if (!osg_event) {
            osg_event = _win->getEventQueue()->touchBegan(touch_id, [self 
convertTouchPhase: [touch phase]], pixelPos.x(), pixelPos.y());
        } else {
            osg_event->addTouchPoint(touch_id, [self convertTouchPhase: [touch 
phase]], pixelPos.x(), pixelPos.y());
        }
    }
    
}

- (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event {
    
    NSSet *allTouches = [event allTouches];
    
    osg::ref_ptr<osgGA::GUIEventAdapter> osg_event(NULL);

    for(int i=0; i<[allTouches count]; i++)
    {
        UITouch *touch = [[allTouches allObjects] objectAtIndex:i];
        CGPoint pos = [touch locationInView:touch.view];
        osg::Vec2 pixelPos = [self convertPointToPixel: osg::Vec2(pos.x,pos.y)];
        unsigned int touch_id = [self computeTouchId: touch];

        if (!osg_event) {
            osg_event = _win->getEventQueue()->touchMoved(touch_id, [self 
convertTouchPhase: [touch phase]], pixelPos.x(), pixelPos.y());
        } else {
            osg_event->addTouchPoint(touch_id, [self convertTouchPhase: [touch 
phase]], pixelPos.x(), pixelPos.y());
        }


    }
}

- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event

{    
    NSSet *allTouches = [event allTouches];
    
    osg::ref_ptr<osgGA::GUIEventAdapter> osg_event(NULL);
    
    for(int i=0; i<[allTouches count]; i++)
    {
        UITouch *touch = [[allTouches allObjects] objectAtIndex:i];
        CGPoint pos = [touch locationInView:touch.view];
        osg::Vec2 pixelPos = [self convertPointToPixel: osg::Vec2(pos.x,pos.y)];
        unsigned int touch_id = [self computeTouchId: touch];
        if (!osg_event) {
            osg_event = _win->getEventQueue()->touchEnded(touch_id, [self 
convertTouchPhase: [touch phase]], pixelPos.x(), pixelPos.y(), [touch 
tapCount]);
        } else {
            osg_event->addTouchPoint(touch_id, [self convertTouchPhase: [touch 
phase]], pixelPos.x(), pixelPos.y(), [touch tapCount]);
        }

    }
}

-(void) touchesCancelled:(NSSet *)touches withEvent:(UIEvent *)event 
{
    [self touchesEnded: touches withEvent:event];
}

@end



@interface GraphicsWindowIOSGLViewController : UIViewController
{

}
- 
(BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation;
- 
(void)willAnimateRotationToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
 duration:(NSTimeInterval)duration;

@end

@implementation GraphicsWindowIOSGLViewController


- 
(BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
{
    osgViewer::GraphicsWindowIOS* win = [(GraphicsWindowIOSGLView*)(self.view) 
getGraphicsWindow];
    if(!win){return  NO;}
    
    osgViewer::GraphicsWindowIOS::WindowData::DeviceOrientationFlags flags = 
win->getDeviceOrientationFlags();
    
 
    BOOL result(NO);
    
    switch (interfaceOrientation) {
        case UIDeviceOrientationPortrait:
            if(flags & 
osgViewer::GraphicsWindowIOS::WindowData::PORTRAIT_ORIENTATION){
                result = YES;
            }
            break;
        case UIDeviceOrientationPortraitUpsideDown:
            if(flags & 
osgViewer::GraphicsWindowIOS::WindowData::PORTRAIT_UPSIDEDOWN_ORIENTATION){
                result = YES;
            }
            break;
        case UIInterfaceOrientationLandscapeLeft:
            if(win->getTraits()->supportsResize && flags & 
osgViewer::GraphicsWindowIOS::WindowData::LANDSCAPE_LEFT_ORIENTATION){
                result = YES;
            }
            break;
        case UIInterfaceOrientationLandscapeRight:
            if(win->getTraits()->supportsResize && flags & 
osgViewer::GraphicsWindowIOS::WindowData::LANDSCAPE_RIGHT_ORIENTATION){
                result = YES;
            }
            break;
        default:
            break;
    }
    OSG_INFO << "shouldAutorotateToInterfaceOrientation for " << 
interfaceOrientation << ": " << ((result==YES) ? "YES" : "NO") << std::endl;
    return result;
}


- 
(void)willAnimateRotationToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
 duration:(NSTimeInterval)duration 
{
    [(GraphicsWindowIOSGLView*)(self.view) updateDimensions];
}



@end



using namespace osgIOS; 
namespace osgViewer {

    
    
#pragma mark GraphicsWindowIOS



// 
----------------------------------------------------------------------------------------------------------
// init
// 
----------------------------------------------------------------------------------------------------------

void GraphicsWindowIOS::init()
{
    if (_initialized) return;

    _ownsWindow = false;
    _context = NULL;
    _window = NULL;
    _view = NULL;
    _viewController = NULL;
    
    _updateContext = true;
    //if -1.0 we use the screens scale factor
    _viewContentScaleFactor = -1.0f;
    _valid = _initialized = true;
}


// 
----------------------------------------------------------------------------------------------------------
// realizeImplementation, creates the window + context
// 
----------------------------------------------------------------------------------------------------------

bool GraphicsWindowIOS::realizeImplementation()
{
    NSAutoreleasePool* pool = [[NSAutoreleasePool alloc] init];
    
    BOOL bar_hidden = (_traits->windowDecoration) ? NO: YES;
    #ifdef __IPHONE_OS_VERSION_MIN_REQUIRED
    #if __IPHONE_OS_VERSION_MIN_REQUIRED > 30100
        [[UIApplication sharedApplication] setStatusBarHidden: bar_hidden 
withAnimation:UIStatusBarAnimationNone];
    #else
        [[UIApplication sharedApplication] setStatusBarHidden: bar_hidden 
animated:NO];
    #endif
    #endif
    
    //Get info about the requested screen
    IOSWindowingSystemInterface* wsi = 
dynamic_cast<IOSWindowingSystemInterface*>(osg::GraphicsContext::getWindowingSystemInterface());
    osg::Vec2 screenSizePoints;
    osg::Vec2 screenSizePixels;
    float screenScaleFactor = 1.0f;
    UIScreen* screen = nil;
    osg::GraphicsContext::ScreenSettings screenSettings;
    if (wsi) {
        wsi->getScreenContentScaleFactor((*_traits), screenScaleFactor);
        wsi->getScreenSizeInPoints((*_traits), screenSizePoints); 
        screenSizePixels = osg::Vec2(screenSettings.width, 
screenSettings.height);
        wsi->getScreenSettings((*_traits), screenSettings);
        screen = wsi->getUIScreen((*_traits));
    }else{
        OSG_FATAL << "GraphicsWindowIOS::realizeImplementation: ERROR: Failed 
to create IOS windowing system, OSG will be unable to create a vaild gl context 
and will not be able to render." << std::endl;
        return false;
    }
    
    _ownsWindow = true;
    
    // see if an existing inherited window was passed in
    WindowData* windowData = _traits->inheritedWindowData ? 
dynamic_cast<WindowData*>(_traits->inheritedWindowData.get()) : NULL;
    if (windowData) 
    {
        if (windowData->getWindowOrParentView())
        {
            _ownsWindow = false;        
            _window = windowData->getWindowOrParentView();
        }
        
        _deviceOrientationFlags = windowData->_deviceOrientationFlags;
        _viewContentScaleFactor = windowData->_viewContentScaleFactor;
    } 
    
    //if the user hasn't specified a viewScaleFactor we will use the screens 
scale factor
    //so we get a full res buffer
    if(_viewContentScaleFactor < 0.0f)
    {_viewContentScaleFactor = screenScaleFactor;}
    

    OSG_DEBUG << "GraphicsWindowIOS::realizeImplementation / ownsWindow: " << 
_ownsWindow << std::endl;

    
    //Here's the confusing bit, the default traits use the screen res which is 
in pixels and the user will want to use pixels also
    //but we need to create our views and windows in points. By default we 
create a full res buffer across all devices. This
    //means that for backward compatibility you need to set the windowData 
_viewContentScaleFactor to 1.0f and set the screen res to the
    //res of the older gen device.
    CGRect window_bounds;
    osg::Vec2 pointsOrigin = this->pixelToPoint(osg::Vec2(_traits->x, 
_traits->y));
    osg::Vec2 pointsSize = this->pixelToPoint(osg::Vec2(_traits->width, 
_traits->height));

    window_bounds.origin.x = pointsOrigin.x(); 
    window_bounds.origin.y = pointsOrigin.y();
    window_bounds.size.width = pointsSize.x(); 
    window_bounds.size.height = pointsSize.y();
    
    
    //if we own the window we need to create one
    if (_ownsWindow) 
    {
        //create the IOS window object using the viewbounds (in points) 
required for our context size
        _window = [[GraphicsWindowIOSWindow alloc] initWithFrame: 
window_bounds];// styleMask: style backing: NSBackingStoreBuffered defer: NO];
        
        if (!_window) {
            OSG_WARN << "GraphicsWindowIOS::realizeImplementation: ERROR: 
Failed to create GraphicsWindowIOSWindow can not display gl view" << std::endl;
            return false;
        }
        
        OSG_DEBUG << "GraphicsWindowIOS::realizeImplementation: INFO: Created 
UIWindow with bounds '" << window_bounds.size.width << ", " << 
window_bounds.size.height << "' (points)." << std::endl;
        
        //if the user has requested a differnet screenNum from default 0 get 
the UIScreen object and
        //apply to our window (this is for IPad external screens, I don't have 
one, so I've no idea if it works)
        //I'm also not sure if we should apply this to external windows also?
        if(_traits->screenNum > 0 && screen != nil)
        {
            _window.screen = screen;
        }
    } 
            
    //create the desired OpenGLES context type
#if OSG_GLES1_FEATURES
    _context = [[EAGLContext alloc] initWithAPI:kEAGLRenderingAPIOpenGLES1];
#elif OSG_GLES2_FEATURES
    _context = [[EAGLContext alloc] initWithAPI:kEAGLRenderingAPIOpenGLES2];
#endif
    
    if (!_context || ![EAGLContext setCurrentContext:_context]) {
        
        #if OSG_GLES1_FEATURES
        OSG_FATAL << "GraphicsWindowIOS::realizeImplementation: ERROR: Failed 
to create a valid OpenGLES1 context" << std::endl;
        #elif OSG_GLES2_FEATURES
        OSG_FATAL << "GraphicsWindowIOS::realizeImplementation: ERROR: Failed 
to create a valid OpenGLES2 context" << std::endl;
        #endif
        return false;
    }

    //create the view to display our context in our window
    CGRect gl_view_bounds = (_ownsWindow) ? [_window frame] : window_bounds;
    GraphicsWindowIOSGLView* theView = [[ GraphicsWindowIOSGLView alloc ] 
initWithFrame: gl_view_bounds : this ];
    if(!theView)
    {
        OSG_FATAL << "GraphicsWindowIOS::realizeImplementation: ERROR: Failed 
to create GraphicsWindowIOSGLView, can not create frame buffers." << std::endl;
        return false;
    }
    
    [theView setAutoresizingMask:  ( UIViewAutoresizingFlexibleWidth |  
UIViewAutoresizingFlexibleHeight) ];
    
    //Apply our content scale factor to our view, this is what converts the 
views points
    //size to our desired context size.
#if defined(__IPHONE_4_0) && (__IPHONE_OS_VERSION_MIN_REQUIRED >= __IPHONE_4_0)
    theView.contentScaleFactor = _viewContentScaleFactor;
#endif    
    [theView setGraphicsWindow: this];
    [theView setOpenGLContext:_context];
    _view = theView;
    
    OSG_DEBUG << "GraphicsWindowIOS::realizeImplementation / view: " << theView 
<< std::endl;

    if (getDeviceOrientationFlags() != WindowData::IGNORE_ORIENTATION) 
    {
        _viewController = [[GraphicsWindowIOSGLViewController alloc] init];
        _viewController.view = _view;
    }
    
    // Attach view to window
    [_window addSubview: _view];
    if ([_window isKindOfClass:[UIWindow class]])
        _window.rootViewController = _viewController;
    [theView release];
    
    //if we own the window also make it visible
    if (_ownsWindow) 
    {
        
        //show window
        [_window makeKeyAndVisible];
    }

    [pool release];
    
    // IOSs origin is top/left:
    
getEventQueue()->getCurrentEventState()->setMouseYOrientation(osgGA::GUIEventAdapter::Y_INCREASING_DOWNWARDS);
    
    _valid = _initialized = _realized = true;
    return _valid;
}




// 
----------------------------------------------------------------------------------------------------------
// closeImplementation
// 
----------------------------------------------------------------------------------------------------------
void GraphicsWindowIOS::closeImplementation()
{
    OSG_INFO << "close IOS window" << std::endl;
    _valid = false;
    _realized = false;
   
    
    if (_view) 
    {
        [_view setOpenGLContext: NULL];
        [_context release];
        [_view removeFromSuperview];
        [_view setGraphicsWindow: NULL];
    }
    
    if (_viewController) 
    {
        [_viewController release];
        _viewController = NULL;
    }
        
    if (_window && _ownsWindow) 
    {  
        [_window release];
        //[glView release];
    }

    
    _window = NULL;
    _view = NULL;  
    _context = NULL;  
}


// 
----------------------------------------------------------------------------------------------------------
// makeCurrentImplementation
// 
----------------------------------------------------------------------------------------------------------

bool GraphicsWindowIOS:: makeCurrentImplementation()
{
    
    
    //bind the context
    [EAGLContext setCurrentContext:_context];
    
    if (_updateContext)
    {
        [_view destroyFramebuffer];
        [_view createFramebuffer];

        _updateContext = false; 
    }
    //i think we also want to bind the frame buffer here
    //[_view bindFrameBuffer];

    return true;
}


// 
----------------------------------------------------------------------------------------------------------
// releaseContextImplementation
// 
----------------------------------------------------------------------------------------------------------

bool GraphicsWindowIOS::releaseContextImplementation()
{
    if ([EAGLContext currentContext] == _context) {
        [EAGLContext setCurrentContext:nil];
    }
    return true;
}


// 
----------------------------------------------------------------------------------------------------------
// swapBuffersImplementation
// 
----------------------------------------------------------------------------------------------------------

void GraphicsWindowIOS::swapBuffersImplementation()
{
    //[_context flushBuffer];
    [_view swapBuffers];
}



// 
----------------------------------------------------------------------------------------------------------
// setWindowDecorationImplementation
//
// We will use this to toggle the status bar on IPhone, nearest thing to window 
decoration
// 
----------------------------------------------------------------------------------------------------------

bool GraphicsWindowIOS::setWindowDecorationImplementation(bool flag)
{
    if (!_realized || !_ownsWindow) return false;

    BOOL bar_hidden = (flag) ? NO: YES;
    #ifdef __IPHONE_OS_VERSION_MIN_REQUIRED
    #if __IPHONE_OS_VERSION_MIN_REQUIRED > 30100
        [[UIApplication sharedApplication] setStatusBarHidden: bar_hidden 
withAnimation:UIStatusBarAnimationNone];
    #else
        [[UIApplication sharedApplication] setStatusBarHidden: bar_hidden 
animated:NO];
    #endif
    #endif
    
    return true;
}


// 
----------------------------------------------------------------------------------------------------------
// grabFocus
// 
----------------------------------------------------------------------------------------------------------
void GraphicsWindowIOS::grabFocus()
{
    //i think make key is the equivalent of focus on iphone 
    [_window makeKeyWindow];
}


// 
----------------------------------------------------------------------------------------------------------
// grabFocusIfPointerInWindow
// 
----------------------------------------------------------------------------------------------------------
void GraphicsWindowIOS::grabFocusIfPointerInWindow()
{
    OSG_INFO << "GraphicsWindowIOS :: grabFocusIfPointerInWindow not 
implemented yet " << std::endl;
}

// 
----------------------------------------------------------------------------------------------------------
// raiseWindow
// Raise the window to the top.
// 
----------------------------------------------------------------------------------------------------------
void GraphicsWindowIOS::raiseWindow()
{
    [_window bringSubviewToFront:_view];
}

// 
----------------------------------------------------------------------------------------------------------
// resizedImplementation
// 
----------------------------------------------------------------------------------------------------------

void GraphicsWindowIOS::resizedImplementation(int x, int y, int width, int 
height)
{
    GraphicsContext::resizedImplementation(x, y, width, height);
    
    _updateContext = true;
    
    getEventQueue()->windowResize(x,y,width, height, 
getEventQueue()->getTime());
}




// 
----------------------------------------------------------------------------------------------------------
// setWindowRectangleImplementation
// 
----------------------------------------------------------------------------------------------------------
bool GraphicsWindowIOS::setWindowRectangleImplementation(int x, int y, int 
width, int height)
{
    OSG_INFO << "GraphicsWindowIOS :: setWindowRectangleImplementation not 
implemented yet " << std::endl;
    if (!_ownsWindow)
        return false;
            
    return true;
}

    
void GraphicsWindowIOS::checkEvents()
{
    
    
}



// 
----------------------------------------------------------------------------------------------------------
// setWindowName
// 
----------------------------------------------------------------------------------------------------------

void GraphicsWindowIOS::setWindowName (const std::string & name)
{
    OSG_INFO << "GraphicsWindowIOS :: setWindowName not implemented yet " << 
std::endl;
}


// 
----------------------------------------------------------------------------------------------------------
// useCursor, no cursor on IOS
// 
----------------------------------------------------------------------------------------------------------

void GraphicsWindowIOS::useCursor(bool cursorOn)
{
    OSG_INFO << "GraphicsWindowIOS :: useCursor not implemented yet " << 
std::endl;
}


// 
----------------------------------------------------------------------------------------------------------
// setCursor, no cursor on IOS
// 
----------------------------------------------------------------------------------------------------------

void GraphicsWindowIOS::setCursor(MouseCursor mouseCursor)
{
    OSG_INFO << "GraphicsWindowIOS :: setCursor not implemented yet " << 
std::endl;
}


// 
----------------------------------------------------------------------------------------------------------
// setVSync, no vsync on IOS
// 
----------------------------------------------------------------------------------------------------------

void GraphicsWindowIOS::setVSync(bool f) 
{
    OSG_INFO << "GraphicsWindowIOS :: setVSync not implemented yet " << 
std::endl;
}
    
    
// 
----------------------------------------------------------------------------------------------------------
// helper funcs for converting points to pixels taking into account the views 
contents scale factor
// 
----------------------------------------------------------------------------------------------------------

osg::Vec2 GraphicsWindowIOS::pointToPixel(const osg::Vec2& point)
{
    return point * _viewContentScaleFactor;
}
    
osg::Vec2 GraphicsWindowIOS::pixelToPoint(const osg::Vec2& pixel)
{
    float scaler = 1.0f / _viewContentScaleFactor;
    return pixel * scaler;
}


// 
----------------------------------------------------------------------------------------------------------
// d'tor
// 
----------------------------------------------------------------------------------------------------------

GraphicsWindowIOS::~GraphicsWindowIOS() 
{
    close();
}



class ConcreteIOSWindowingSystemInterface : public  IOSWindowingSystemInterface 
{
public:
    ConcreteIOSWindowingSystemInterface()
    :    IOSWindowingSystemInterface()
    {
    }
    
    virtual osg::GraphicsContext* 
createGraphicsContext(osg::GraphicsContext::Traits* traits) 
    {
        if (traits->pbuffer)
        {
            // pbuffers not supported on iOS
            return 0;
        }
        else
        {
            osg::ref_ptr<GraphicsWindowIOS> window = new 
GraphicsWindowIOS(traits);
            if (window->valid()) return window.release();
            else return 0;
        }
    }
};

}//end namspace


RegisterWindowingSystemInterfaceProxy<osgViewer::ConcreteIOSWindowingSystemInterface>
 createWindowingSystemInterfaceProxy;


// declare C entry point for static compilation.
extern "C" void graphicswindow_IOS(void)
{
    osg::GraphicsContext::setWindowingSystemInterface(new 
osgViewer::ConcreteIOSWindowingSystemInterface());
}
_______________________________________________
osg-submissions mailing list
[email protected]
http://lists.openscenegraph.org/listinfo.cgi/osg-submissions-openscenegraph.org

Reply via email to