/* PIKA - Photo and Image Kooker Application * a rebranding of The GNU Image Manipulation Program (created with heckimp) * A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio * * Original copyright, applying to most contents (license remains unchanged): * Copyright (C) 2009 Martin Nordholts * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "config.h" #include #include #include "display/display-types.h" #include "display/pikadisplay.h" #include "display/pikadisplayshell.h" #include "display/pikaimagewindow.h" #include "menus/menus.h" #include "widgets/pikauimanager.h" #include "widgets/pikadialogfactory.h" #include "core/pika.h" #include "core/pikaimage.h" #include "core/pikalayer.h" #include "core/pikalayer-new.h" #include "tests.h" #include "pika-app-test-utils.h" #ifdef G_OS_WIN32 /* SendInput() requirement is Windows 2000 pro or over. * We may need to set WINVER to make sure the compiler does not try to * compile for on older version of win32, thus breaking the build. * See * http://msdn.microsoft.com/en-us/library/aa383745%28v=vs.85%29.aspx#setting_winver_or__win32_winnt */ #define WINVER 0x0500 #include #endif /* G_OS_WIN32 */ #ifdef GDK_WINDOWING_QUARTZ // only to get keycode definitions from HIToolbox/Events.h #include #include #endif /* GDK_WINDOWING_QUARTZ */ void pika_test_utils_set_env_to_subdir (const gchar *root_env_var, const gchar *subdir, const gchar *target_env_var) { const gchar *root_dir = NULL; gchar *target_dir = NULL; /* Get root dir */ root_dir = g_getenv (root_env_var); if (! root_dir) g_printerr ("*\n" "* The env var %s is not set, you are probably running\n" "* in a debugger. Set it manually, e.g.:\n" "*\n" "* set env %s=%s/source/pika\n" "*\n", root_env_var, root_env_var, g_get_home_dir ()); /* Construct path and setup target env var */ target_dir = g_build_filename (root_dir, subdir, NULL); g_setenv (target_env_var, target_dir, TRUE); g_free (target_dir); } void pika_test_utils_set_env_to_subpath (const gchar *root_env_var1, const gchar *root_env_var2, const gchar *subdir, const gchar *target_env_var) { const gchar *root_dir1 = NULL; const gchar *root_dir2 = NULL; gchar *target_dir1 = NULL; gchar *target_dir2 = NULL; gchar *target_path = NULL; /* Get root dir */ root_dir1 = g_getenv (root_env_var1); if (! root_dir1) g_printerr ("*\n" "* The env var %s is not set, you are probably running\n" "* in a debugger. Set it manually, e.g.:\n" "*\n" "* set env %s=%s/source/pika\n" "*\n", root_env_var1, root_env_var1, g_get_home_dir ()); root_dir2 = g_getenv (root_env_var2); if (! root_dir2) g_printerr ("*\n" "* The env var %s is not set, you are probably running\n" "* in a debugger. Set it manually, e.g.:\n" "*\n" "* set env %s=%s/source/pika\n" "*\n", root_env_var2, root_env_var2, g_get_home_dir ()); /* Construct path and setup target env var */ target_dir1 = g_build_filename (root_dir1, subdir, NULL); target_dir2 = g_build_filename (root_dir2, subdir, NULL); target_path = g_strconcat (target_dir1, G_SEARCHPATH_SEPARATOR_S, target_dir2, NULL); g_free (target_dir1); g_free (target_dir2); g_setenv (target_env_var, target_path, TRUE); g_free (target_path); } /** * pika_test_utils_set_pika3_directory: * @root_env_var: Either "PIKA_TESTING_ABS_TOP_SRCDIR" or * "PIKA_TESTING_ABS_TOP_BUILDDIR" * @subdir: Subdir, may be %NULL * * Sets PIKA3_DIRECTORY to the source dir @root_env_var/@subdir. The * environment variables is set up by the test runner, see Makefile.am **/ void pika_test_utils_set_pika3_directory (const gchar *root_env_var, const gchar *subdir) { pika_test_utils_set_env_to_subdir (root_env_var, subdir, "PIKA3_DIRECTORY" /*target_env_var*/); } /** * pika_test_utils_setup_menus_path: * * Sets PIKA_TESTING_MENUS_PATH to "$top_srcdir/menus:$top_builddir/menus". **/ void pika_test_utils_setup_menus_path (void) { /* PIKA_TESTING_ABS_TOP_SRCDIR is set by the automake test runner, * see Makefile.am */ pika_test_utils_set_env_to_subpath ("PIKA_TESTING_ABS_TOP_SRCDIR", "PIKA_TESTING_ABS_TOP_BUILDDIR", "menus", "PIKA_TESTING_MENUS_PATH"); } /** * pika_test_utils_create_image: * @pika: A #Pika instance. * @width: Width of image (and layer) * @height: Height of image (and layer) * * Creates a new image of a given size with one layer of same size and * a display. * * Returns: The new #PikaImage. **/ void pika_test_utils_create_image (Pika *pika, gint width, gint height) { PikaImage *image; PikaLayer *layer; image = pika_image_new (pika, width, height, PIKA_RGB, PIKA_PRECISION_U8_NON_LINEAR); layer = pika_layer_new (image, width, height, pika_image_get_layer_format (image, TRUE), "layer1", 1.0, PIKA_LAYER_MODE_NORMAL); pika_image_add_layer (image, layer, NULL /*parent*/, 0 /*position*/, FALSE /*push_undo*/); pika_create_display (pika, image, PIKA_UNIT_PIXEL, 1.0 /*scale*/, NULL); } /** * pika_test_utils_synthesize_key_event: * @widget: Widget to target. * @keyval: Keyval, e.g. GDK_Return * * Simulates a keypress and release with gdk_test_simulate_key(). **/ void pika_test_utils_synthesize_key_event (GtkWidget *widget, guint keyval) { #if defined(GDK_WINDOWING_QUARTZ) GdkKeymapKey *keys = NULL; gint n_keys = 0; gint i; CGEventRef keyUp, keyDown; if (gdk_keymap_get_entries_for_keyval (gdk_keymap_get_for_display (gdk_display_get_default ()), keyval, &keys, &n_keys)) { /* XXX not in use yet */ CGEventRef commandDown = CGEventCreateKeyboardEvent (NULL, (CGKeyCode)kVK_Command, true); CGEventRef commandUp = CGEventCreateKeyboardEvent (NULL, (CGKeyCode)kVK_Command, false); CGEventRef shiftDown = CGEventCreateKeyboardEvent (NULL, (CGKeyCode)kVK_Shift, true); CGEventRef shiftUp = CGEventCreateKeyboardEvent (NULL, (CGKeyCode)kVK_Shift, false); CGEventRef optionDown = CGEventCreateKeyboardEvent (NULL, (CGKeyCode)kVK_Option, true); CGEventRef optionUp = CGEventCreateKeyboardEvent (NULL, (CGKeyCode)kVK_Option, false); for (i = 0; i < n_keys; i++) { /* Option press. */ if (keys[i].group) { CGEventPost (kCGHIDEventTap, optionDown); } /* Shift press. */ if (keys[i].level) { CGEventPost(kCGHIDEventTap, shiftDown); } keyDown = CGEventCreateKeyboardEvent (NULL, (CGKeyCode)keys[i].keycode, true); keyUp = CGEventCreateKeyboardEvent (NULL, (CGKeyCode)keys[i].keycode, false); /* Key pressed. */ CGEventPost (kCGHIDEventTap, keyDown); CFRelease (keyDown); usleep (100); /* key released */ CGEventPost (kCGHIDEventTap, keyUp); CFRelease (keyUp); /* Shift release. */ if (keys[i].level) { CGEventPost (kCGHIDEventTap, shiftDown); } /* Option release. */ if (keys[i].group) { CGEventPost (kCGHIDEventTap, optionUp); } /* No need to loop for alternative keycodes. We want only one * key generated. */ break; } CFRelease (commandDown); CFRelease (commandUp); CFRelease (shiftDown); CFRelease (shiftUp); CFRelease (optionDown); CFRelease (optionUp); g_free (keys); } else { g_warning ("%s: no macOS key mapping found for keyval %d.", G_STRFUNC, keyval); } #else /* ! GDK_WINDOWING_QUARTZ */ gdk_test_simulate_key (gtk_widget_get_window (widget), -1, -1, /*x, y*/ keyval, 0 /*modifiers*/, GDK_KEY_PRESS); gdk_test_simulate_key (gtk_widget_get_window (widget), -1, -1, /*x, y*/ keyval, 0 /*modifiers*/, GDK_KEY_RELEASE); #endif /* ! GDK_WINDOWING_QUARTZ */ } /** * pika_test_utils_get_ui_manager: * @pika: The #Pika instance. * * Returns the "best" #PikaUIManager to use when performing * actions. It gives the ui manager of the empty display if it exists, * otherwise it gives it the ui manager of the first display. * * Returns: The #PikaUIManager. **/ PikaUIManager * pika_test_utils_get_ui_manager (Pika *pika) { return menus_get_image_manager_singleton (pika); } /** * pika_test_utils_create_image_from_dalog: * @pika: * * Creates a new image using the "New image" dialog, and then returns * the #PikaImage created. * * Returns: The created #PikaImage. **/ PikaImage * pika_test_utils_create_image_from_dialog (Pika *pika) { PikaImage *image = NULL; GtkWidget *new_image_dialog = NULL; guint n_initial_images = g_list_length (pika_get_image_iter (pika)); guint n_images = -1; gint tries_left = 100; PikaUIManager *ui_manager = pika_test_utils_get_ui_manager (pika); /* Bring up the new image dialog */ pika_ui_manager_activate_action (ui_manager, "image", "image-new"); pika_test_run_mainloop_until_idle (); /* Get the GtkWindow of the dialog */ new_image_dialog = pika_dialog_factory_dialog_raise (pika_dialog_factory_get_singleton (), gdk_display_get_monitor (gdk_display_get_default (), 0), NULL, "pika-image-new-dialog", -1 /*view_size*/); /* Press the OK button. It will take a while for the image to be * created so loop for a while */ gtk_dialog_response (GTK_DIALOG (new_image_dialog), GTK_RESPONSE_OK); do { g_usleep (20 * 1000); pika_test_run_mainloop_until_idle (); n_images = g_list_length (pika_get_image_iter (pika)); } while (tries_left-- && n_images != n_initial_images + 1); /* Make sure there now is one image more than initially */ g_assert_cmpint (n_images, ==, n_initial_images + 1); image = PIKA_IMAGE (pika_get_image_iter (pika)->data); return image; }