12120
|
1 /*
|
|
2 * vo_quartz.c
|
|
3 *
|
|
4 * Copyright (c) Nicolas Plourde - January 2004
|
|
5 *
|
|
6 * MPlayer Mac OSX Quartz video out module.
|
|
7 *
|
|
8 * TODO: -Fullscreen
|
|
9 * -Better event handling
|
|
10 *
|
|
11 * Note on performance:
|
|
12 * Right now i can play fullsize dvd video with -framedrop on my
|
|
13 * iBook G4 800mhz. YUV to RGB converstion will speed up thing alot.
|
|
14 * Another thing is the slow fps when you maximize the window, I was
|
|
15 * not expecting that. I will fix this a.s.a.p. Im new to Mac
|
|
16 * programming so help is welcome.
|
|
17 */
|
|
18
|
|
19 //SYS
|
|
20 #include <stdio.h>
|
|
21 #include <stdlib.h>
|
|
22 #include <string.h>
|
|
23 #include <math.h>
|
|
24 #include <errno.h>
|
|
25
|
|
26 //OSX
|
|
27 #include <Carbon/Carbon.h>
|
|
28 #include <QuickTime/QuickTime.h>
|
|
29
|
|
30 //MPLAYER
|
|
31 #include "config.h"
|
|
32 #include "video_out.h"
|
|
33 #include "video_out_internal.h"
|
|
34 #include "aspect.h"
|
|
35
|
|
36 #include "../input/input.h"
|
|
37 #include "../input/mouse.h"
|
|
38
|
|
39 #include "vo_quartz.h"
|
|
40
|
|
41 static vo_info_t info = {
|
|
42 "MacOS X (Quartz)",
|
|
43 "quartz",
|
|
44 "Nicolas Plourde <nicolasplourde@hotmail.com>",
|
|
45 ""
|
|
46 };
|
|
47
|
|
48 LIBVO_EXTERN (quartz)
|
|
49 static unsigned char *ImageData = NULL;
|
|
50
|
|
51 static uint32_t image_width;
|
|
52 static uint32_t image_height;
|
|
53 static uint32_t image_depth;
|
|
54 static uint32_t image_bytes;
|
|
55 static uint32_t image_format;
|
|
56
|
|
57 static int int_pause = 0;
|
|
58
|
|
59 int screen_width, screen_height;
|
|
60
|
|
61 WindowRef theWindow;
|
|
62 CGContextRef context;
|
|
63 CGRect bounds;
|
|
64 CGRect winBounds;
|
|
65 Rect contentRect;
|
|
66 CGImageRef image;
|
|
67 CGDataProviderRef dataProviderRef;
|
|
68 Ptr oldscreenstate;
|
|
69 RGBColor black = { 0, 0, 0 };
|
|
70 float winAlpha = 1;
|
|
71
|
|
72 #include "../osdep/keycodes.h"
|
|
73 extern void mplayer_put_key (int code);
|
|
74
|
|
75 //PROTOTYPE/////////////////////////////////////////////////////////////////
|
|
76 void resize_window (uint32_t width, uint32_t height);
|
|
77 static OSStatus MainWindowEventHandler (EventHandlerCallRef nextHandler,
|
|
78 EventRef event, void *userData);
|
|
79 static OSStatus MainKeyEventHandler (EventHandlerCallRef nextHandler,
|
|
80 EventRef event, void *userData);
|
|
81
|
|
82
|
|
83 //default window event handler
|
|
84 static OSStatus
|
|
85 MainWindowEventHandler (EventHandlerCallRef nextHandler, EventRef event,
|
|
86 void *userData)
|
|
87 {
|
|
88 OSStatus err = noErr;
|
|
89 WindowRef window;
|
|
90 Rect rectPort = { 0, 0, 0, 0 };
|
|
91 OSStatus result = eventNotHandledErr;
|
|
92 UInt32 class = GetEventClass (event);
|
|
93 UInt32 kind = GetEventKind (event);
|
|
94
|
|
95 GetEventParameter (event, kEventParamDirectObject, typeWindowRef, NULL,
|
|
96 sizeof (WindowRef), NULL, &window);
|
|
97 if (window)
|
|
98 {
|
|
99 GetWindowPortBounds (window, &rectPort);
|
|
100 }
|
|
101
|
|
102 switch (kind)
|
|
103 {
|
|
104 case kEventWindowActivated:
|
|
105
|
|
106 case kEventWindowDrawContent:
|
|
107 break;
|
|
108
|
|
109 case kEventWindowClosed:
|
|
110 HideWindow (window);
|
|
111 mplayer_put_key (KEY_ESC);
|
|
112 break;
|
|
113
|
|
114 case kEventWindowShown:
|
|
115 InvalWindowRect (window, &rectPort);
|
|
116 break;
|
|
117
|
|
118 case kEventWindowBoundsChanged:
|
|
119 resize_window (rectPort.right, rectPort.bottom);
|
|
120 break;
|
|
121
|
|
122 case kEventWindowZoomed:
|
|
123 resize_window (rectPort.right, rectPort.bottom);
|
|
124 break;
|
|
125
|
|
126 default:
|
|
127 err = eventNotHandledErr;
|
|
128 break;
|
|
129 }
|
|
130
|
|
131 return err;
|
|
132 }
|
|
133
|
|
134 //keyboard event handler
|
|
135 static OSStatus
|
|
136 MainKeyEventHandler (EventHandlerCallRef nextHandler, EventRef event,
|
|
137 void *userData)
|
|
138 {
|
|
139 OSStatus err = noErr;
|
|
140 UInt32 macKeyCode;
|
|
141
|
|
142 GetEventParameter (event, kEventParamKeyCode, typeUInt32, NULL,
|
|
143 sizeof (macKeyCode), NULL, &macKeyCode);
|
|
144
|
|
145 switch (GetEventKind (event))
|
|
146 {
|
|
147 case kEventRawKeyDown:
|
|
148 {
|
|
149 switch (macKeyCode)
|
|
150 {
|
|
151 case QZ_RETURN:
|
|
152 mplayer_put_key (KEY_ENTER);
|
|
153 break;
|
|
154 case QZ_ESCAPE:
|
|
155 EndFullScreen (oldscreenstate, 0);
|
|
156 QuitApplicationEventLoop ();
|
|
157 mplayer_put_key (KEY_ESC);
|
|
158 break;
|
|
159 case QZ_q:
|
|
160 mplayer_put_key ('q');
|
|
161 break;
|
|
162 case QZ_F1:
|
|
163 mplayer_put_key (KEY_F + 1);
|
|
164 break;
|
|
165 case QZ_F2:
|
|
166 mplayer_put_key (KEY_F + 2);
|
|
167 break;
|
|
168 case QZ_F3:
|
|
169 mplayer_put_key (KEY_F + 3);
|
|
170 break;
|
|
171 case QZ_F4:
|
|
172 mplayer_put_key (KEY_F + 4);
|
|
173 break;
|
|
174 case QZ_F5:
|
|
175 mplayer_put_key (KEY_F + 5);
|
|
176 break;
|
|
177 case QZ_F6:
|
|
178 mplayer_put_key (KEY_F + 6);
|
|
179 break;
|
|
180 case QZ_F7:
|
|
181 mplayer_put_key (KEY_F + 7);
|
|
182 break;
|
|
183 case QZ_F8:
|
|
184 mplayer_put_key (KEY_F + 8);
|
|
185 break;
|
|
186 case QZ_F9:
|
|
187 mplayer_put_key (KEY_F + 9);
|
|
188 break;
|
|
189 case QZ_F10:
|
|
190 mplayer_put_key (KEY_F + 10);
|
|
191 break;
|
|
192 case QZ_F11:
|
|
193 mplayer_put_key (KEY_F + 11);
|
|
194 break;
|
|
195 case QZ_F12:
|
|
196 mplayer_put_key (KEY_F + 12);
|
|
197 break;
|
|
198 case QZ_o:
|
|
199 mplayer_put_key ('o');
|
|
200 break;
|
|
201 case QZ_SPACE:
|
|
202 mplayer_put_key (' ');
|
|
203 break;
|
|
204 case QZ_p:
|
|
205 mplayer_put_key ('p');
|
|
206 break;
|
|
207 //case QZ_7: mplayer_put_key(shift_key?'/':'7');
|
|
208 //case QZ_PLUS: mplayer_put_key(shift_key?'*':'+');
|
|
209 case QZ_KP_PLUS:
|
|
210 mplayer_put_key ('+');
|
|
211 break;
|
|
212 case QZ_MINUS:
|
|
213 case QZ_KP_MINUS:
|
|
214 mplayer_put_key ('-');
|
|
215 break;
|
|
216 case QZ_TAB:
|
|
217 mplayer_put_key ('\t');
|
|
218 break;
|
|
219 case QZ_PAGEUP:
|
|
220 mplayer_put_key (KEY_PAGE_UP);
|
|
221 break;
|
|
222 case QZ_PAGEDOWN:
|
|
223 mplayer_put_key (KEY_PAGE_DOWN);
|
|
224 break;
|
|
225 case QZ_UP:
|
|
226 mplayer_put_key (KEY_UP);
|
|
227 break;
|
|
228 case QZ_DOWN:
|
|
229 mplayer_put_key (KEY_DOWN);
|
|
230 break;
|
|
231 case QZ_LEFT:
|
|
232 mplayer_put_key (KEY_LEFT);
|
|
233 break;
|
|
234 case QZ_RIGHT:
|
|
235 mplayer_put_key (KEY_RIGHT);
|
|
236 break;
|
|
237 //case QZ_LESS: mplayer_put_key(shift_key?'>':'<'); break;
|
|
238 //case QZ_GREATER: mplayer_put_key('>'); break;
|
|
239 //case QZ_ASTERISK:
|
|
240 case QZ_KP_MULTIPLY:
|
|
241 mplayer_put_key ('*');
|
|
242 break;
|
|
243 case QZ_SLASH:
|
|
244 case QZ_KP_DIVIDE:
|
|
245 mplayer_put_key ('/');
|
|
246 break;
|
|
247 case QZ_KP0:
|
|
248 mplayer_put_key (KEY_KP0);
|
|
249 break;
|
|
250 case QZ_KP1:
|
|
251 mplayer_put_key (KEY_KP1);
|
|
252 break;
|
|
253 case QZ_KP2:
|
|
254 mplayer_put_key (KEY_KP2);
|
|
255 break;
|
|
256 case QZ_KP3:
|
|
257 mplayer_put_key (KEY_KP3);
|
|
258 break;
|
|
259 case QZ_KP4:
|
|
260 mplayer_put_key (KEY_KP4);
|
|
261 break;
|
|
262 case QZ_KP5:
|
|
263 mplayer_put_key (KEY_KP5);
|
|
264 break;
|
|
265 case QZ_KP6:
|
|
266 mplayer_put_key (KEY_KP6);
|
|
267 break;
|
|
268 case QZ_KP7:
|
|
269 mplayer_put_key (KEY_KP7);
|
|
270 break;
|
|
271 case QZ_KP8:
|
|
272 mplayer_put_key (KEY_KP8);
|
|
273 break;
|
|
274 case QZ_KP9:
|
|
275 mplayer_put_key (KEY_KP9);
|
|
276 break;
|
|
277 case QZ_KP_PERIOD:
|
|
278 mplayer_put_key (KEY_KPDEC);
|
|
279 break;
|
|
280 case QZ_KP_ENTER:
|
|
281 mplayer_put_key (KEY_KPENTER);
|
|
282 break;
|
|
283 case QZ_LEFTBRACKET:
|
|
284 SetWindowAlpha (theWindow, winAlpha -= 0.05);
|
|
285 break;
|
|
286 case QZ_RIGHTBRACKET:
|
|
287 SetWindowAlpha (theWindow, winAlpha += 0.05);
|
|
288 break;
|
|
289 case QZ_f:
|
|
290 break;
|
|
291 default:
|
|
292 break;
|
|
293 }
|
|
294 }
|
|
295 break;
|
|
296 default:
|
|
297 err = eventNotHandledErr;
|
|
298 break;
|
|
299 }
|
|
300
|
|
301 return err;
|
|
302 }
|
|
303
|
|
304 static uint32_t
|
|
305 config (uint32_t width, uint32_t height, uint32_t d_width, uint32_t d_height,
|
|
306 uint32_t flags, char *title, uint32_t format)
|
|
307 {
|
|
308 WindowAttributes windowAttrs;
|
|
309 CFStringRef titleKey;
|
|
310 CFStringRef windowTitle;
|
|
311 OSStatus result;
|
|
312 GDHandle deviceHdl;
|
|
313 Rect deviceRect;
|
|
314
|
|
315 //Get Main device info///////////////////////////////////////////////////
|
|
316 deviceHdl = GetMainDevice ();
|
|
317 deviceRect = (*deviceHdl)->gdRect;
|
|
318
|
|
319 screen_width = deviceRect.right;
|
|
320 screen_height = deviceRect.bottom;
|
|
321
|
|
322 //misc mplayer setup/////////////////////////////////////////////////////
|
|
323 image_width = width;
|
|
324 image_height = height;
|
|
325 image_depth = IMGFMT_RGB_DEPTH (format);
|
|
326 image_bytes = (IMGFMT_RGB_DEPTH (format) + 7) / 8;
|
|
327
|
|
328 aspect_save_orig (width, height);
|
|
329 aspect_save_prescale (d_width, d_height);
|
|
330 aspect_save_screenres (screen_width, screen_height);
|
|
331
|
|
332 aspect (&d_width, &d_height, A_NOZOOM);
|
|
333
|
|
334 if (ImageData)
|
|
335 free (ImageData);
|
|
336 ImageData = malloc (image_width * image_height * image_bytes);
|
|
337
|
|
338 //Create player window//////////////////////////////////////////////////
|
|
339 windowAttrs = kWindowStandardDocumentAttributes
|
|
340 | kWindowMetalAttribute
|
|
341 | kWindowStandardHandlerAttribute
|
|
342 | kWindowInWindowMenuAttribute
|
|
343 | kWindowLiveResizeAttribute | kWindowCompositingAttribute;
|
|
344
|
|
345 SetRect (&contentRect, 0, 0, d_width, d_height);
|
|
346 CreateNewWindow (kDocumentWindowClass, windowAttrs, &contentRect,
|
|
347 &theWindow);
|
|
348
|
|
349 titleKey = CFSTR ("MPlayer");
|
|
350 windowTitle = CFCopyLocalizedString (titleKey, NULL);
|
|
351 result = SetWindowTitleWithCFString (theWindow, windowTitle);
|
|
352 CFRelease (titleKey);
|
|
353 CFRelease (windowTitle);
|
|
354
|
|
355 const EventTypeSpec winEvents[] = {
|
|
356 {kEventClassWindow, kEventWindowActivated},
|
|
357 {kEventClassWindow, kEventWindowDrawContent},
|
|
358 {kEventClassWindow, kEventWindowClosed},
|
|
359 {kEventClassWindow, kEventWindowShown},
|
|
360 {kEventClassWindow, kEventWindowBoundsChanged},
|
|
361 {kEventClassWindow, kEventWindowZoomed}
|
|
362 };
|
|
363
|
|
364 const EventTypeSpec keyEvents[] =
|
|
365 { {kEventClassKeyboard, kEventRawKeyDown} };
|
|
366 InstallWindowEventHandler (theWindow,
|
|
367 NewEventHandlerUPP (MainWindowEventHandler),
|
|
368 GetEventTypeCount (winEvents), winEvents,
|
|
369 theWindow, NULL);
|
|
370 InstallWindowEventHandler (theWindow,
|
|
371 NewEventHandlerUPP (MainKeyEventHandler),
|
|
372 GetEventTypeCount (keyEvents), keyEvents,
|
|
373 theWindow, NULL);
|
|
374
|
|
375 RepositionWindow (theWindow, NULL, kWindowCascadeOnMainScreen);
|
|
376 ShowWindow (theWindow);
|
|
377
|
|
378 //Setup Quartz context
|
|
379 CreateCGContextForPort (GetWindowPort (theWindow), &context);
|
|
380
|
|
381 //set size and aspect for current window
|
|
382 resize_window (d_width, d_height);
|
|
383
|
|
384 return 0;
|
|
385 }
|
|
386
|
|
387 //resize drawing context to fit window
|
|
388 void
|
|
389 resize_window (uint32_t width, uint32_t height)
|
|
390 {
|
|
391 //this is a "wow it work". Need some improvement.
|
|
392 uint32_t d_width;
|
|
393 uint32_t d_height;
|
|
394 uint32_t size;
|
|
395 Rect tmpRect;
|
|
396
|
|
397 float aspectX;
|
|
398 float aspectY;
|
|
399
|
|
400 aspect (&d_width, &d_height, A_NOZOOM);
|
|
401
|
|
402 aspectX = (float) ((float) d_width * (width / (float) d_width));
|
|
403 aspectY = (float) ((float) d_height * (width / (float) d_width));
|
|
404
|
|
405 if (aspectY > height)
|
|
406 {
|
|
407 aspectX = (float) ((float) d_width * (height / (float) d_height));
|
|
408 aspectY = (float) ((float) d_height * (height / (float) d_height));
|
|
409
|
|
410 bounds = CGRectMake ((width - aspectX) / 2, 0, aspectX, aspectY);
|
|
411 }
|
|
412 else
|
|
413 {
|
|
414 bounds = CGRectMake (0, (height - aspectY) / 2, aspectX, aspectY);
|
|
415 }
|
|
416
|
|
417 //create a graphic context for the window
|
|
418 GetWindowPortBounds (theWindow, &tmpRect);
|
|
419 SetPortBounds (GetWindowPort (theWindow), &tmpRect);
|
|
420 CreateCGContextForPort (GetWindowPort (theWindow), &context);
|
|
421
|
|
422 //fill background with black
|
|
423 winBounds =
|
|
424 CGRectMake (tmpRect.top, tmpRect.left, tmpRect.right, tmpRect.bottom);
|
|
425 CGContextSetRGBFillColor (context, 0.0, 0.0, 0.0, 1.0);
|
|
426 CGContextFillRect (context, winBounds);
|
|
427 }
|
|
428
|
|
429 static void
|
|
430 check_events (void)
|
|
431 {
|
|
432 EventRef theEvent;
|
|
433 EventTargetRef theTarget;
|
|
434
|
|
435 theTarget = GetEventDispatcherTarget ();
|
|
436
|
|
437 ReceiveNextEvent (0, NULL, kEventDurationNoWait, true, &theEvent);
|
|
438 SendEventToEventTarget (theEvent, theTarget);
|
|
439 ReleaseEvent (theEvent);
|
|
440
|
|
441 //if(VO_EVENT_RESIZE) resize_window(vo_dwidth,vo_dheight);
|
|
442 if (VO_EVENT_EXPOSE && int_pause)
|
|
443 flip_page ();
|
|
444 }
|
|
445
|
|
446 static void
|
|
447 draw_osd (void)
|
|
448 {
|
|
449 }
|
|
450
|
|
451 static void
|
|
452 flip_page (void)
|
|
453 {
|
|
454 CGContextFlush (context);
|
|
455 }
|
|
456
|
|
457 static uint32_t
|
|
458 draw_slice (uint8_t * src[], int stride[], int w, int h, int x, int y)
|
|
459 {
|
|
460 return -1;
|
|
461 }
|
|
462
|
|
463 static uint32_t
|
|
464 draw_frame (uint8_t * src[])
|
|
465 {
|
|
466 //this is very slow. I have to find another way.
|
|
467 CGImageAlphaInfo alphaInfo;
|
|
468
|
|
469 dataProviderRef =
|
|
470 CGDataProviderCreateWithData (0, src[0],
|
|
471 image_width * image_height * image_bytes,
|
|
472 0);
|
|
473
|
|
474 if (image_format == IMGFMT_RGB24)
|
|
475 alphaInfo = kCGImageAlphaNone;
|
|
476 else if (image_format == IMGFMT_RGB32)
|
|
477 alphaInfo = kCGImageAlphaNoneSkipFirst;
|
|
478
|
|
479 image = CGImageCreate (image_width,
|
|
480 image_height,
|
|
481 8,
|
|
482 image_depth,
|
|
483 ((image_width * image_depth) + 7) / 8,
|
|
484 CGColorSpaceCreateDeviceRGB (),
|
|
485 alphaInfo,
|
|
486 dataProviderRef, 0, 0, kCGRenderingIntentDefault);
|
|
487
|
|
488 CGContextDrawImage (context, bounds, image);
|
|
489
|
|
490 return 0;
|
|
491 }
|
|
492
|
|
493 static uint32_t
|
|
494 query_format (uint32_t format)
|
|
495 {
|
|
496 image_format = format;
|
|
497
|
|
498 //Curently supporting only rgb format.
|
|
499 if ((format == IMGFMT_RGB24) || (format == IMGFMT_RGB32))
|
|
500 return VFCAP_CSP_SUPPORTED | VFCAP_CSP_SUPPORTED_BY_HW;
|
|
501 return 0;
|
|
502 }
|
|
503
|
|
504
|
|
505 static void
|
|
506 uninit (void)
|
|
507 {
|
|
508 }
|
|
509
|
|
510 static uint32_t
|
|
511 preinit (const char *arg)
|
|
512 {
|
|
513 return 0;
|
|
514 }
|
|
515
|
|
516 static uint32_t
|
|
517 control (uint32_t request, void *data, ...)
|
|
518 {
|
|
519 switch (request)
|
|
520 {
|
|
521 case VOCTRL_PAUSE:
|
|
522 return (int_pause = 1);
|
|
523 case VOCTRL_RESUME:
|
|
524 return (int_pause = 0);
|
|
525 case VOCTRL_QUERY_FORMAT:
|
|
526 return query_format (*((uint32_t *) data));
|
|
527 }
|
|
528 return VO_NOTIMPL;
|
|
529 }
|