summaryrefslogtreecommitdiffstats
path: root/share/extensions/tests/test_inkex_gui.py
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 18:24:48 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 18:24:48 +0000
commitcca66b9ec4e494c1d919bff0f71a820d8afab1fa (patch)
tree146f39ded1c938019e1ed42d30923c2ac9e86789 /share/extensions/tests/test_inkex_gui.py
parentInitial commit. (diff)
downloadinkscape-cca66b9ec4e494c1d919bff0f71a820d8afab1fa.tar.xz
inkscape-cca66b9ec4e494c1d919bff0f71a820d8afab1fa.zip
Adding upstream version 1.2.2.upstream/1.2.2upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'share/extensions/tests/test_inkex_gui.py')
-rw-r--r--share/extensions/tests/test_inkex_gui.py174
1 files changed, 174 insertions, 0 deletions
diff --git a/share/extensions/tests/test_inkex_gui.py b/share/extensions/tests/test_inkex_gui.py
new file mode 100644
index 0000000..cd4093c
--- /dev/null
+++ b/share/extensions/tests/test_inkex_gui.py
@@ -0,0 +1,174 @@
+# coding=utf-8
+#
+# Copyright 2022 Martin Owens <doctormo@geek-2.com>
+#
+# 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 <http://www.gnu.org/licenses/>
+#
+"""
+Test GUI App, running main loops and threading from and to them.
+
+Do NOT copy this test if you are looking for examples of how to write
+unit tests for your GUI enabled inkscape extensions. This code contains
+a lot of weird and confusing checking of loops and process running.
+"""
+
+import os
+import sys
+import time
+import pytest
+
+from inkex.tester import TestCase
+from inkex.utils import DependencyError
+
+try:
+ from inkex.gui.tester import MainLoopProtection
+ from inkex.gui.listview import label
+ from inkex.gui import GtkApp, Window, asyncme
+ from gi.repository import Gtk, GLib
+
+except DependencyError:
+ Window = object
+ GtkApp = None
+
+ class asyncme:
+ run_or_none = classmethod(lambda a, f: f)
+ run_or_wait = classmethod(lambda a, f: f)
+ mainloop_only = classmethod(lambda a, f: f)
+
+
+class LoopyWindow(Window):
+ """A testing window with various loops"""
+
+ def realized(self, widget=None):
+ """Called by the window's realise signal, see glade ui xml"""
+ # Spawn a thread to do something without locking up the UI
+ self.thread_action()
+ # Add a function to the GLib main loop manually
+ GLib.idle_add(self.loop_action)
+
+ def loop_action(self):
+ """Called while we are in the idle loop"""
+ # Test Mainloop to mainloop doesn't lock up.
+ assert "pancakes" == self.get_pancakes()
+ time.sleep(0.2)
+ # Run some useful action we might want to test
+ self.in_loop_action()
+ # We actually want to quit to make the test faster
+ if self.gapp.exit() is None:
+ return True # Run again
+ return False
+
+ def in_loop_action(self):
+ """Defined by the test loader if needed"""
+ pass
+
+ @asyncme.run_or_none
+ def thread_action(self):
+ """Run in a different thread from Gtk"""
+ self.in_thread_action()
+
+ @asyncme.run_or_wait
+ def in_thread_action(self):
+ """Defined by the test loader if needed"""
+ pass
+
+ @asyncme.mainloop_only
+ def get_pancakes(self):
+ """
+ A call which always runs in the mainloop,
+ even if not called from one.
+ """
+ return "pancakes"
+
+ def test_signal(self, widget=None):
+ """A signal from Gtk widgets defined in the glade xml"""
+ if widget:
+ pass
+
+
+@pytest.mark.skipif(GtkApp is None, reason="PyGObject is required")
+class GtkAppTest(TestCase):
+ """Tests for GUI App"""
+
+ def construct_app(self, windows=(), ui_file="app-test"):
+ """Create a gtk app based on some inputs"""
+ return type(
+ "_GtkApp",
+ (GtkApp,),
+ {
+ "app_name": "application-test",
+ "windows": list(windows),
+ "ui_dir": self.datadir(),
+ "prefix": "ui",
+ "ui_file": ui_file,
+ },
+ )
+
+ def construct_window(self, name, **kwargs):
+ return type("_GtkWindow", (LoopyWindow,), {"name": name, **kwargs})
+
+ def test_app_errors(self):
+ """Various consruction errors"""
+ self.assertRaises(NotImplementedError, GtkApp)
+ bad_win = self.construct_window("bad-window")
+ bad_app = self.construct_app([bad_win], ui_file="bad-file")
+ self.assertRaises(FileNotFoundError, bad_app)
+ self.assertRaises(KeyError, self.construct_app())
+ good_win = self.construct_window("basic_app")
+ good_app = self.construct_app([good_win])()
+ self.assertRaises(KeyError, good_app.load_window, "no-window")
+
+ def test_args(self):
+ """Test app arguments"""
+ self.assertEqual(label(4), "int")
+ self.assertEqual(label((4, 5)), "int or int")
+ GtkApp(
+ app_name="inline-app",
+ ui_dir=self.datadir(),
+ prefix="ui",
+ ui_file="app-test",
+ windows=[self.construct_window("basic_app")],
+ )
+
+ def test_app_inline_run(self):
+ """Test a basic gui loop"""
+ cls = self.construct_app([self.construct_window("basic_app")])
+ with MainLoopProtection():
+ cls(start_loop=True)
+
+ def test_app_outside_run(self):
+ """Test various threading in and out of Gtk"""
+ other_win = self.construct_window("basic_app")
+
+ def _thread(self):
+ self.looped = self.widget("button1").get_label()
+
+ def _loop(self, widget=None):
+ self.gapp.remove_window(other_win(self.gapp))
+ self.gapp.remove_window(self)
+
+ app = self.construct_app(
+ [
+ self.construct_window(
+ "basic_app",
+ in_loop_action=_loop,
+ in_thread_action=asyncme.mainloop_only(_thread),
+ )
+ ]
+ )()
+ with MainLoopProtection():
+ app.run()
+
+ # Threading doesn't work in the CI builder, disabled for now.
+ # self.assertEqual(app._primary.looped, 'gtk-apply')