2 * Copyright 2022 Olaf Wintermann
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
11 * The above copyright notice and this permission notice shall be included in
12 * all copies or substantial portions of the Software.
14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
19 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
20 * DEALINGS IN THE SOFTWARE.
33 static MainWindow *main_window;
35 static void WindowCreateMenu(MainWindow *win, Widget parent, Arg *args, int nargs);
37 static void FileOpenCB(Widget w, void *udata, void *cdata);
38 static void ViewFullscreenCB(Widget w, void *udata, void *cdata);
40 static void WindowRealized(MainWindow *win);
42 static int blank_cursor_init = 0;
43 static Pixmap blank_cursor_pixmap;
44 static Cursor blank_cursor;
46 static void init_blank_cursor(Widget w) {
51 blank_cursor_pixmap = XCreateBitmapFromData(XtDisplay(w), XtWindow(w), &data, 1, 1);
52 if(!blank_cursor_pixmap) return;
54 blank_cursor = XCreatePixmapCursor(XtDisplay(w), blank_cursor_pixmap, blank_cursor_pixmap, &c, &c, 0, 0);
56 XFreePixmap(XtDisplay(w), blank_cursor_pixmap);
57 blank_cursor_init = 1;
60 static void window_close_handler(Widget window, void *udata, void *cdata) {
61 WindowClosePlayer(main_window);
65 static unsigned int keycodeF;
67 static void windowKeyEH(Widget widget, XtPointer data, XEvent *event, Boolean *dispatch) {
68 MainWindow *win = data;
69 if(win->fullscreen && event->xkey.keycode == keycodeF) {
70 WindowFullscreen(main_window, FALSE);
75 static int main_window_is_realized = 0;
77 static void resizeEH(Widget widget, XtPointer data, XEvent *event, Boolean *dispatch) {
78 if(!main_window_is_realized) {
79 if(XtIsRealized(widget)) {
80 main_window_is_realized = 1;
84 WindowAdjustAspectRatio(data);
87 static void WindowRealized(MainWindow *win) {
88 char *open_file = GetOpenFileArg();
90 size_t len = strlen(open_file);
91 char *file = XtMalloc(len+1);
92 memcpy(file, open_file, len);
94 WindowSetFile(win, file);
99 if(!blank_cursor_init) {
100 init_blank_cursor(win->player_widget);
104 static void playerWidgetInputCB(Widget widget, XtPointer u, XtPointer c) {
106 XmDrawingAreaCallbackStruct *cb = c;
108 if(win->player && win->player->isactive) {
109 PlayerHandleInput(win, win->player, cb);
113 static void windowGrabButton(MainWindow *win) {
120 ButtonPressMask | ButtonReleaseMask | PointerMotionMask | FocusChangeMask | EnterWindowMask | LeaveWindowMask,
125 win->buttongrab = True;
128 static void playerEH(Widget widget, XtPointer data, XEvent *event, Boolean *dispatch) {
129 MainWindow *win = data;
130 int etype = event->type;
133 if(etype == EnterNotify) {
135 windowGrabButton(win);
138 if(etype == LeaveNotify) {
140 //XtUngrabButton(win->player_widget, AnyButton, AnyModifier);
141 //win->buttongrab = False;
146 if(etype == ButtonPress || etype == ButtonRelease || etype == KeyPress || etype == KeyRelease) {
147 //printf("button press\n");
151 if(!win->player || win->player->window == 0) return;
153 WindowPlayerWidgetEvent(win, event);
156 // redirect key events to the player window
157 //printf("redirect\n");
158 event->xkey.window = win->player->window;
160 XtDisplay(win->player_widget),
168 void WindowPlayerWidgetEvent(MainWindow *win, XEvent *event) {
169 int etype = event->type;
171 if(etype == MotionNotify) {
173 } else if(etype == ButtonPress) {
175 } else if(etype == ButtonRelease) {
180 MainWindow* WindowCreate(Display *display) {
184 MainWindow *window = malloc(sizeof(MainWindow));
185 memset(window, 0, sizeof(MainWindow));
186 main_window = window;
190 XtSetArg(args[n], XmNtitle, APP_NAME); n++;
191 window->window = XtAppCreateShell(
194 applicationShellWidgetClass,
195 //vendorShellWidgetClass,
201 Atom wm_delete_window;
202 wm_delete_window = XmInternAtom(
206 XmAddWMProtocolCallback(
209 window_close_handler,
213 XtAddEventHandler(window->window, StructureNotifyMask, False, resizeEH, window);
216 XtSetArg(args[n], XmNwidth, 360); n++;
217 XtSetArg(args[n], XmNheight, 220); n++;
218 XtSetArg(args[n], XmNshadowThickness, 0); n++;
219 Widget container = XmCreateForm(window->window, "form", args, n);
220 XtManageChild(container);
221 XtAddEventHandler(container, KeyPressMask, FALSE, windowKeyEH, window);
224 XtSetArg(args[n], XmNleftAttachment, XmATTACH_FORM); n++;
225 XtSetArg(args[n], XmNrightAttachment, XmATTACH_FORM); n++;
226 XtSetArg(args[n], XmNtopAttachment, XmATTACH_FORM); n++;
227 WindowCreateMenu(window, container, args, n);
230 XtSetArg(args[n], XmNleftAttachment, XmATTACH_FORM); n++;
231 XtSetArg(args[n], XmNrightAttachment, XmATTACH_FORM); n++;
232 XtSetArg(args[n], XmNbottomAttachment, XmATTACH_FORM); n++;
233 XtSetArg(args[n], XmNtopAttachment, XmATTACH_WIDGET); n++;
234 XtSetArg(args[n], XmNtopWidget, window->menubar); n++;
235 XtSetArg(args[n], XmNbackground, BlackPixelOfScreen(XtScreen(window->window))); n++;
236 window->player_widget = XmCreateDrawingArea(container, "player", args, n);
237 XtManageChild(window->player_widget);
238 XtAddCallback(window->player_widget, XmNinputCallback, playerWidgetInputCB, window);
239 XmProcessTraversal(window->player_widget, XmTRAVERSE_CURRENT);
240 XtAddEventHandler(window->player_widget, PointerMotionMask | ButtonPressMask | ButtonReleaseMask | FocusChangeMask |
241 EnterWindowMask | KeyPressMask | KeyReleaseMask |
242 LeaveWindowMask, FALSE, playerEH, window);
246 keycodeF = XKeysymToKeycode(XtDisplay(window->window), XStringToKeysym("F"));
251 MainWindow* GetMainWindow(void) {
255 void WindowShow(MainWindow *win) {
256 XtRealizeWidget(win->window);
260 * Creates a XmPushButton menu item
262 static Widget createMenuItem(
267 const char *accelerator,
268 char *accelerator_text,
269 XtCallbackProc callback,
275 XmString s1 = XmStringCreateSimple(label);
276 XtSetArg(args[n], XmNlabelString, s1); n++;
277 XtSetArg(args[n], XmNmnemonic, mnemonic); n++;
279 if(accelerator && accelerator_text) {
280 at = XmStringCreateSimple(accelerator_text);
281 XtSetArg(args[n], XmNaccelerator, accelerator); n++;
282 XtSetArg(args[n], XmNacceleratorText, at); n++;
285 Widget menuItem = XmCreatePushButtonGadget(menu, name, args, n);
286 XtManageChild(menuItem);
288 if(at) XmStringFree(at);
291 XtAddCallback(menuItem, XmNactivateCallback, (XtCallbackProc)callback, cbData);
297 static void WindowCreateMenu(MainWindow *win, Widget parent, Arg *mbargs, int nmbargs) {
298 Widget menubar = XmCreateMenuBar(parent, "menubar", mbargs, nmbargs);
299 XtManageChild(menubar);
300 win->menubar = menubar;
306 XmString s = XmStringCreateSimple("File");
307 Widget fileMenuItem = XtVaCreateManagedWidget(
309 xmCascadeButtonWidgetClass,
314 Widget fileMenu = XmVaCreateSimplePulldownMenu(menubar, "menu", 0, NULL, NULL);
316 s = XmStringCreateSimple("Playback");
317 Widget playMenuItem = XtVaCreateManagedWidget(
319 xmCascadeButtonWidgetClass,
324 Widget playMenu = XmVaCreateSimplePulldownMenu(menubar, "menu", 1, NULL, NULL);
326 s = XmStringCreateSimple("View");
327 Widget viewMenuItem = XtVaCreateManagedWidget(
329 xmCascadeButtonWidgetClass,
334 Widget viewMenu = XmVaCreateSimplePulldownMenu(menubar, "menu", 2, NULL, NULL);
337 createMenuItem(fileMenu, "fileOpen", "Open...", 'O', "Ctrl<Key>O", "Ctrl+O", FileOpenCB, NULL);
340 createMenuItem(viewMenu, "viewFullscreen", "Fullscreen", 'F', "<Key>F", "F", ViewFullscreenCB, NULL);
343 void go_fullscreen(Display *dsp, Window win)
346 Atom wm_state = XInternAtom(dsp, "_NET_WM_STATE", False);
347 Atom fullscreen = XInternAtom(dsp, "_NET_WM_STATE_FULLSCREEN", False);
348 memset(&xev, 0, sizeof(xev));
349 xev.type = ClientMessage;
350 xev.xclient.window = win;
351 xev.xclient.message_type = wm_state;
352 xev.xclient.format = 32;
353 xev.xclient.data.l[0] = 1; // _NET_WM_STATE_ADD
354 xev.xclient.data.l[1] = fullscreen;
355 xev.xclient.data.l[2] = 0;
356 XSendEvent(dsp, DefaultRootWindow(dsp), False,
357 SubstructureNotifyMask, &xev);
360 static Atom net_wm_state;
361 static Atom net_wm_state_fullscreen;
362 static int net_wm_atoms_initialized = 0;
364 void WindowFullscreen(MainWindow *win, bool enableFullscreen) {
365 Display *dpy = XtDisplay(win->window);
367 // init net_wm_state atoms
368 if(!net_wm_atoms_initialized) {
369 net_wm_state = XInternAtom(dpy, "_NET_WM_STATE", False);
370 net_wm_state_fullscreen = XInternAtom(dpy, "_NET_WM_STATE_FULLSCREEN", False);
371 net_wm_atoms_initialized = 1;
374 WindowMenubarSetVisible(win, !enableFullscreen);
375 if(enableFullscreen && !win->fullscreen) {
376 XtUnmanageChild(main_window->menubar);
377 main_window->fullscreen = TRUE;
378 } else if(!enableFullscreen && win->fullscreen) {
379 XtManageChild(main_window->menubar);
380 main_window->fullscreen = FALSE;
384 memset(&ev, 0, sizeof(XEvent));
385 ev.type = ClientMessage;
386 ev.xclient.window = XtWindow(win->window);
387 ev.xclient.message_type = net_wm_state;
388 ev.xclient.format = 32;
389 ev.xclient.data.l[0] = enableFullscreen ? 1 : 0;
390 ev.xclient.data.l[1] = net_wm_state_fullscreen;
391 ev.xclient.data.l[2] = 0;
392 XSendEvent(dpy, DefaultRootWindow(dpy), False, SubstructureNotifyMask | SubstructureRedirectMask, &ev);
395 void WindowMenubarSetVisible(MainWindow *win, bool visible) {
397 XtManageChild(main_window->menubar);
398 XtVaSetValues(main_window->player_widget, XmNtopAttachment, XmATTACH_WIDGET, XmNtopWidget, main_window->menubar, NULL);
400 XtUnmanageChild(main_window->menubar);
401 XtVaSetValues(main_window->player_widget, XmNtopAttachment, XmATTACH_FORM, NULL);
405 void WindowSetFile(MainWindow *win, char *file) {
412 static void filedialog_end(
415 XmFileSelectionBoxCallbackStruct *selection)
417 XtUnmanageChild(widget);
418 XtDestroyWidget(widget);
421 static void filedialog_select(
424 XmFileSelectionBoxCallbackStruct *selection)
427 if(selection->value) {
428 XmStringGetLtoR(selection->value, XmSTRING_DEFAULT_CHARSET, &value);
430 WindowSetFile(data, value);
431 // no need to free the value, because it is stored in MainWindow
433 PlayerOpenFile(data);
436 filedialog_end(widget, data, NULL);
442 static void FileOpenCB(Widget w, void *udata, void *cdata) {
443 MainWindow *win = main_window;
448 XtSetArg(args[n], XnNshowViewMenu, 1); n++;
449 Widget dialog = XnCreateFileSelectionDialog(win->window, "dialog", args, n);
450 XtAddCallback(dialog, XmNokCallback, (XtCallbackProc)filedialog_select, win);
451 XtAddCallback(dialog, XmNcancelCallback, (XtCallbackProc)filedialog_end, win);
453 Widget dirUp = XnFileSelectionBoxGetChild(dialog, XnFSB_DIR_UP_BUTTON);
454 XtUnmanageChild(dirUp);
456 XtManageChild(dialog);
459 static void ViewFullscreenCB(Widget w, void *udata, void *cdata) {
460 if(main_window->fullscreen) {
461 WindowFullscreen(main_window, FALSE);
463 WindowFullscreen(main_window, TRUE);
468 void WindowAdjustAspectRatio(MainWindow *win) {
469 if(!win->player) return;
470 if(!win->player->isactive || win->player->width <= 0 || win->player->height <= 0) return;
472 // we have a running player width video
473 // adjust window aspect ratio (the window aspect ratio is different from
474 // the video, because of window decoration, menubar and other extra controls)
476 Dimension win_width, win_height;
477 XtVaGetValues(win->window, XmNwidth, &win_width, XmNheight, &win_height, NULL);
478 Dimension player_width, player_height;
479 XtVaGetValues(win->player_widget, XmNwidth, &player_width, XmNheight, &player_height, NULL);
481 double r = (double)win->player->width / (double)win->player->height;
482 double p_width = player_width;
483 double p_height = p_width / r;
485 Dimension new_width = p_width + win_width - player_width;
486 Dimension new_height = p_height + win_height - player_height;
489 hints.flags = PAspect;
490 hints.min_aspect.x = new_width;
491 hints.min_aspect.y = new_height;
492 hints.max_aspect.x = new_width;
493 hints.max_aspect.y = new_height;
494 XSetWMNormalHints(XtDisplay(win->window), XtWindow(win->window), &hints);
497 void WindowClosePlayer(MainWindow *win) {
499 PlayerDestroy(win->player);
504 void WindowHidePlayerCursor(MainWindow *win) {
505 if(!win->cursorhidden && win->player && win->player->window != 0) {
506 //XDefineCursor(XtDisplay(win->player_widget), XtWindow(win->player_widget), blank_cursor);
507 win->cursorhidden = True;
508 XFlush(XtDisplay(win->player_widget));
512 void WindowShowPlayerCursor(MainWindow *win) {
513 if(win->cursorhidden && win->player && win->player->window != 0) {
514 XDefineCursor(XtDisplay(win->player_widget), XtWindow(win->player_widget), None);
515 win->cursorhidden = False;
516 XFlush(XtDisplay(win->player_widget));