view src/video/cocoa/SDL_cocoashape.m @ 4829:7160e833c4ac

Implemented shaped windows for Cocoa. Still need to see if they actually work.
author egottlieb
date Tue, 03 Aug 2010 21:56:05 -0400
parents 5660aac926e9
children 3dfe779900f5
line wrap: on
line source

/*
    SDL - Simple DirectMedia Layer
    Copyright (C) 2010 Eli Gottlieb

    This library is free software; you can redistribute it and/or
    modify it under the terms of the GNU Lesser General Public
    License as published by the Free Software Foundation; either
    version 2.1 of the License, or (at your option) any later version.

    This library is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
    Lesser General Public License for more details.

    You should have received a copy of the GNU Lesser General Public
    License along with this library; if not, write to the Free Software
    Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA

    Eli Gottlieb
    eligottlieb@gmail.com
*/

#include "SDL_stdinc.h"
#include "SDL_cocoavideo.h"
#include "SDL_shape.h"
#include "SDL_cocoashape.h"
#include "SDL_sysvideo.h"

SDL_WindowShaper* Cocoa_CreateShaper(SDL_Window* window) {
	SDL_WindowData* data = (SDL_WindowData*)window->driverdata;
	[data->nswindow setAlpha:1.0];
	[data->nswindow setOpaque:YES];
	[data->nswindow setStyleMask:NSBorderlessWindowMask];
	SDL_WindowShaper* result = SDL_malloc(sizeof(SDL_WindowShaper));
	result->window = window;
	result->mode.mode = ShapeModeDefault;
	result->mode.parameters.binarizationCutoff = 1;
	result->usershownflag = 0;
	window->shaper = result;
	
	SDL_ShapeData* shape_data = SDL_malloc(sizeof(SDL_ShapeData));
	result->driverdata = shape_data;
	shape_data->context = [data->nswindow graphicsContext];
	shape_data->saved = SDL_FALSE;
	shape_data->shape = NULL;
	
	int resized_properly = Cocoa_ResizeWindowShape(window);
	assert(resized_properly == 0);
	return result;
}

typedef struct {
	NSBezierPath* clipPath;
	SDL_Window* window;
} SDL_PathConglomeration;

NSRect convert_rect(SDL_Rect rect,SDL_Window* window) {
	NSRect nsrect = NSMakeRect(rect.x,window->h-(rect.y+rect.h),rect.w,rect.h);
	return [[((SDL_WindowData*)window->driverdata)->nswindow contentView] convertRectFromBase:nsrect];
}

void ConglomerateShapeTree(SDL_ShapeTree* tree,SDL_PathConglomeration cong) {
	if(tree->kind == OpaqueShape) {
		NSRect rect = convert_rect(tree->data.shape,cong->window);
		[cong->clipPath appendBezierPathWithRect:rect];
	}
}

int Cocoa_SetWindowShape(SDL_WindowShaper *shaper,SDL_Surface *shape,SDL_WindowShapeMode *shapeMode) {
	SDL_ShapeData* data = (SDL_ShapeData*)shaper->driverdata;
	if(data->saved == SDL_TRUE) {
		[data->context restoreGraphicsState];
		data->saved = SDL_FALSE;
	}
		
	[data->context saveGraphicsState];
	data->saved = SDL_TRUE;
	
	//[[NSColor clearColor] set];
	//NSRectFill([[((SDL_WindowData*)shaper->window->driverdata)->nswindow contentView] frame]);
	/* TODO: It looks like Cocoa can set a clipping path based on a list of rectangles.  That's what we get from the
           Windoze shape-calculation code: a list of rectangles.  This will work... I think. */
	NSBezierPath* clipPath = [NSBezierPath bezierPath];
	
	SDL_PathConglomeration cong = {clipPath,shaper->window};
	
	SDL_TraverseShapeTree(data->shape,&ConglomerateShapeTree,cong);
	
	[clipPath addClip];
}

int Cocoa_ResizeWindowShape(SDL_Window *window) {
	SDL_ShapeData* data = window->shaper->driverdata;
	assert(data != NULL);
	
	if(data->shape != NULL)
		SDL_FreeShapeTree(&data->shape);
	
	return 0;
}