384 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			384 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* 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 <martinn@src.gnome.org>
 | |
|  *
 | |
|  * 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 <https://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| #include "config.h"
 | |
| 
 | |
| #include <gegl.h>
 | |
| #include <gtk/gtk.h>
 | |
| 
 | |
| #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 <windows.h>
 | |
| #endif /* G_OS_WIN32 */
 | |
| 
 | |
| #ifdef GDK_WINDOWING_QUARTZ
 | |
| // only to get keycode definitions from HIToolbox/Events.h
 | |
| #include <Carbon/Carbon.h>
 | |
| #include <Cocoa/Cocoa.h>
 | |
| #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;
 | |
| }
 | |
| 
 |