diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml
index 58eeaa7..f74f9f9 100644
--- a/.github/workflows/ci.yml
+++ b/.github/workflows/ci.yml
@@ -28,7 +28,8 @@ jobs:
fail-fast: false
matrix:
qgis_version_tag:
- - release-3_16
+ - release-3_26
+ - release-3_30
- release-3_34
- release-3_36
os: [ubuntu-22.04]
diff --git a/admin.py b/admin.py
index ba25dc1..0c6f33e 100644
--- a/admin.py
+++ b/admin.py
@@ -4,6 +4,7 @@
"""
import os
+import sys
import configparser
import datetime as dt
@@ -61,6 +62,20 @@ def main(context: typer.Context, verbose: bool = False, qgis_profile: str = "def
}
+def _qgis_profile_path() -> str:
+ """Returns the path segment to QGIS profiles folder based on the platform.
+
+ :returns: Correct path segment corresponding to the current platform.
+ :rtype: str
+ """
+ if sys.platform == "win32":
+ app_data_dir = "AppData/Roaming"
+ else:
+ app_data_dir = ".local/share"
+
+ return f"{app_data_dir}/QGIS/QGIS3/profiles/"
+
+
@app.command()
def install(context: typer.Context, build_src: bool = True):
"""Deploys plugin to QGIS plugins directory
@@ -80,8 +95,7 @@ def install(context: typer.Context, build_src: bool = True):
)
root_directory = (
- Path.home() / f".local/share/QGIS/QGIS3/profiles/"
- f"{context.obj['qgis_profile']}"
+ Path.home() / f"{_qgis_profile_path()}{context.obj['qgis_profile']}"
)
base_target_directory = root_directory / "python/plugins" / SRC_NAME
@@ -104,8 +118,7 @@ def symlink(context: typer.Context):
build_path = LOCAL_ROOT_DIR / "build" / SRC_NAME
root_directory = (
- Path.home() / f".local/share/QGIS/QGIS3/profiles/"
- f"{context.obj['qgis_profile']}"
+ Path.home() / f"{_qgis_profile_path()}{context.obj['qgis_profile']}"
)
destination_path = root_directory / "python/plugins" / SRC_NAME
@@ -124,8 +137,7 @@ def uninstall(context: typer.Context):
:type context: typer.Context
"""
root_directory = (
- Path.home() / f".local/share/QGIS/QGIS3/profiles/"
- f"{context.obj['qgis_profile']}"
+ Path.home() / f"{_qgis_profile_path()}{context.obj['qgis_profile']}"
)
base_target_directory = root_directory / "python/plugins" / SRC_NAME
shutil.rmtree(str(base_target_directory), ignore_errors=True)
@@ -136,6 +148,7 @@ def uninstall(context: typer.Context):
def generate_zip(
context: typer.Context,
version: str = None,
+ file_name: str = None,
output_directory: typing.Optional[Path] = LOCAL_ROOT_DIR / "dist",
):
"""Generates plugin zip folder, that can be used to installed the
@@ -147,6 +160,9 @@ def generate_zip(
:param version: Plugin version
:type version: str
+ :param file_name: Plugin zip file name
+ :type file_name: str
+
:param output_directory: Directory where the zip folder will be saved.
:type context: Path
"""
@@ -154,7 +170,10 @@ def generate_zip(
metadata = _get_metadata()
plugin_version = metadata["version"] if version is None else version
output_directory.mkdir(parents=True, exist_ok=True)
- zip_path = output_directory / f"{SRC_NAME}.{plugin_version}.zip"
+ zip_file_name = (
+ f"{SRC_NAME}.{plugin_version}.zip" if file_name is None else file_name
+ )
+ zip_path = output_directory / f"{zip_file_name}"
with zipfile.ZipFile(zip_path, "w") as fh:
_add_to_zip(build_dir, fh, arc_path_base=build_dir.parent)
typer.echo(
@@ -190,7 +209,7 @@ def build(
:returns: Build directory path.
:rtype: Path
"""
- if clean:
+ if clean and output_directory.exists():
shutil.rmtree(str(output_directory), ignore_errors=True)
output_directory.mkdir(parents=True, exist_ok=True)
copy_source_files(output_directory, tests=tests)
@@ -201,7 +220,6 @@ def build(
generate_metadata(context, output_directory)
return output_directory
-
@app.command()
def copy_icon(
output_directory: typing.Optional[Path] = LOCAL_ROOT_DIR / "build/temp",
@@ -247,14 +265,26 @@ def copy_source_files(
for child in (LOCAL_ROOT_DIR / "src" / SRC_NAME).iterdir():
if child.name != "__pycache__":
target_path = output_directory / child.name
- handler = shutil.copytree if child.is_dir() else shutil.copy
- handler(str(child.resolve()), str(target_path))
+ if child.is_dir():
+ shutil.copytree(
+ str(child.resolve()),
+ str(target_path),
+ ignore=shutil.ignore_patterns("*.pyc", "__pycache*"),
+ )
+ else:
+ shutil.copy(str(child.resolve()), str(target_path))
if tests:
for child in LOCAL_ROOT_DIR.iterdir():
if child.name in TEST_FILES:
target_path = output_directory / child.name
- handler = shutil.copytree if child.is_dir() else shutil.copy
- handler(str(child.resolve()), str(target_path))
+ if child.is_dir():
+ shutil.copytree(
+ str(child.resolve()),
+ str(target_path),
+ ignore=shutil.ignore_patterns("*.pyc", "__pycache*"),
+ )
+ else:
+ shutil.copy(str(child.resolve()), str(target_path))
@app.command()
@@ -273,6 +303,12 @@ def compile_resources(
resources_path = LOCAL_ROOT_DIR / "resources" / "resources.qrc"
target_path = output_directory / "resources.py"
target_path.parent.mkdir(parents=True, exist_ok=True)
+
+ # Windows handling of paths for shlex.split function
+ if sys.platform == "win32":
+ target_path = target_path.as_posix()
+ resources_path = resources_path.as_posix()
+
_log(f"compile_resources target_path: {target_path}", context=context)
subprocess.run(shlex.split(f"pyrcc5 -o {target_path} {resources_path}"))
diff --git a/config.json b/config.json
index 721721b..d5885a0 100644
--- a/config.json
+++ b/config.json
@@ -16,7 +16,7 @@
"author": "Kartoza",
"email": "info@kartoza.com",
"description": "View, browse and navigate through imagery.",
- "version": "0.0.1",
+ "version": "0.0.11dev",
"changelog": ""
}
}
\ No newline at end of file
diff --git a/run-tests.sh b/run-tests.sh
index d376a50..c589215 100755
--- a/run-tests.sh
+++ b/run-tests.sh
@@ -5,10 +5,12 @@ QGIS_IMAGE=qgis/qgis
QGIS_IMAGE_latest=latest
QGIS_IMAGE_V_3_26=release-3_26
-QGIS_VERSION_TAGS=($QGIS_IMAGE_latest $QGIS_IMAGE_V_3_26)
+QGIS_VERSION_TAGS=($QGIS_IMAGE_V_3_26)
export IMAGE=$QGIS_IMAGE
+python admin.py build --tests
+
for TAG in "${QGIS_VERSION_TAGS[@]}"
do
echo "Running tests for QGIS $TAG"
diff --git a/scripts/docker/qgis-gea-plugin-test-pre-scripts.sh b/scripts/docker/qgis-gea-plugin-test-pre-scripts.sh
index 47cc2bc..25a87e9 100755
--- a/scripts/docker/qgis-gea-plugin-test-pre-scripts.sh
+++ b/scripts/docker/qgis-gea-plugin-test-pre-scripts.sh
@@ -2,7 +2,7 @@
qgis_setup.sh
-# FIX default installation because the sources must be in "qgis-gea-plugin" parent folder
-rm -rf /root/.local/share/QGIS/QGIS3/profiles/default/python/plugins/qgis-gea-plugin
-ln -sf /tests_directory /root/.local/share/QGIS/QGIS3/profiles/default/python/plugins/qgis-gea-plugin
-ln -sf /tests_directory /usr/share/qgis/python/plugins/qgis-gea-plugin
+# FIX default installation because the sources must be in "qgis_gea_plugin" parent folder
+rm -rf /root/.local/share/QGIS/QGIS3/profiles/default/python/plugins/qgis_gea_plugin
+ln -sf /tests_directory /root/.local/share/QGIS/QGIS3/profiles/default/python/plugins/qgis_gea_plugin
+ln -sf /tests_directory /usr/share/qgis/python/plugins/qgis_gea_plugin
diff --git a/src/qgis_gea_plugin/conf.py b/src/qgis_gea_plugin/conf.py
index e69de29..632ca66 100644
--- a/src/qgis_gea_plugin/conf.py
+++ b/src/qgis_gea_plugin/conf.py
@@ -0,0 +1,128 @@
+# -*- coding: utf-8 -*-
+"""
+ Handles storage and retrieval of the plugin QgsSettings.
+"""
+
+import contextlib
+import dataclasses
+import datetime
+import enum
+import json
+import os.path
+import uuid
+from pathlib import Path
+
+from qgis.PyQt import QtCore
+from qgis.core import QgsSettings
+
+from .utils import log
+
+
+@contextlib.contextmanager
+def qgis_settings(group_root: str, settings=None):
+ """Context manager to help defining groups when creating QgsSettings.
+
+ :param group_root: Name of the root group for the settings
+ :type group_root: str
+
+ :param settings: QGIS settings to use
+ :type settings: QgsSettings
+
+ :yields: Instance of the created settings
+ :ytype: QgsSettings
+ """
+ if settings is None:
+ settings = QgsSettings()
+ settings.beginGroup(group_root)
+ try:
+ yield settings
+ finally:
+ settings.endGroup()
+
+
+class Settings(enum.Enum):
+ """Plugin settings names"""
+
+ HISTORICAL_VIEW = "historical"
+ NICFI_VIEW = "nicfi"
+
+
+class SettingsManager(QtCore.QObject):
+ """Manages saving/loading settings for the plugin in QgsSettings."""
+
+ BASE_GROUP_NAME: str = "qgis_gea_plugin"
+
+ settings = QgsSettings()
+
+ scenarios_settings_updated = QtCore.pyqtSignal()
+ priority_layers_changed = QtCore.pyqtSignal()
+ settings_updated = QtCore.pyqtSignal([str, object], [Settings, object])
+
+ def set_value(self, name: str, value):
+ """Adds a new setting key and value on the plugin specific settings.
+
+ :param name: Name of setting key
+ :type name: str
+
+ :param value: Value of the setting
+ :type value: Any
+ """
+ self.settings.setValue(f"{self.BASE_GROUP_NAME}/{name}", value)
+ if isinstance(name, Settings):
+ name = name.value
+
+ self.settings_updated.emit(name, value)
+
+ def get_value(self, name: str, default=None, setting_type=None):
+ """Gets value of the setting with the passed name.
+
+ :param name: Name of setting key
+ :type name: str
+
+ :param default: Default value returned when the setting key does not exist
+ :type default: Any
+
+ :param setting_type: Type of the store setting
+ :type setting_type: Any
+
+ :returns: Value of the setting
+ :rtype: Any
+ """
+ if setting_type:
+ return self.settings.value(
+ f"{self.BASE_GROUP_NAME}/{name}", default, setting_type
+ )
+ return self.settings.value(f"{self.BASE_GROUP_NAME}/{name}", default)
+
+ def find_settings(self, name):
+ """Returns the plugin setting keys from the
+ plugin root group that matches the passed name
+
+ :param name: Setting name to search for
+ :type name: str
+
+ :returns result: List of the matching settings names
+ :rtype result: list
+ """
+
+ result = []
+ with qgis_settings(f"{self.BASE_GROUP_NAME}") as settings:
+ for settings_name in settings.childKeys():
+ if name in settings_name:
+ result.append(settings_name)
+ return result
+
+ def remove(self, name):
+ """Remove the setting with the specified name.
+
+ :param name: Name of the setting key
+ :type name: str
+ """
+ self.settings.remove(f"{self.BASE_GROUP_NAME}/{name}")
+
+ def delete_settings(self):
+ """Deletes the all the plugin settings."""
+ self.settings.remove(f"{self.BASE_GROUP_NAME}")
+
+
+settings_manager = SettingsManager()
diff --git a/src/qgis_gea_plugin/definitions/defaults.py b/src/qgis_gea_plugin/definitions/defaults.py
new file mode 100644
index 0000000..c6193d9
--- /dev/null
+++ b/src/qgis_gea_plugin/definitions/defaults.py
@@ -0,0 +1,9 @@
+# -*- coding: utf-8 -*-
+"""
+ Definitions for all defaults settings
+"""
+
+PLUGIN_ICON = ":/plugins/qgis_gea_plugin/icon.png"
+
+ANIMATION_PLAY_ICON = ":/images/themes/default/mActionPlay.svg"
+ANIMATION_PAUSE_ICON = ":/images/themes/default/temporal_navigation/pause.svg"
\ No newline at end of file
diff --git a/src/qgis_gea_plugin/gui/qgis_gea.py b/src/qgis_gea_plugin/gui/qgis_gea.py
new file mode 100644
index 0000000..027af9e
--- /dev/null
+++ b/src/qgis_gea_plugin/gui/qgis_gea.py
@@ -0,0 +1,216 @@
+# -*- coding: utf-8 -*-
+
+"""
+The plugin main window class file.
+"""
+
+import os
+
+from qgis.PyQt import (
+ QtCore,
+ QtGui,
+ QtWidgets,
+ QtNetwork,
+)
+from qgis.PyQt.uic import loadUiType
+
+from qgis.core import QgsProject, QgsInterval, QgsUnitTypes, QgsTemporalNavigationObject
+from qgis.gui import QgsLayerTreeView
+
+from ..resources import *
+from ..models.base import IMAGERY
+from ..definitions.defaults import ANIMATION_PLAY_ICON, ANIMATION_PAUSE_ICON, PLUGIN_ICON
+from ..conf import settings_manager, Settings
+from ..utils import animation_state_change, log, tr
+
+WidgetUi, _ = loadUiType(
+ os.path.join(os.path.dirname(__file__), "../ui/main_dockwidget.ui")
+)
+
+
+class QgisGeaPlugin(QtWidgets.QDockWidget, WidgetUi):
+ """
+ Main plugin UI class for QGIS GEA Plugin.
+
+ This class represents the main dock widget for the plugin, providing
+ functionality for temporal navigation, layer management and plugin settings.
+
+ """
+
+ def __init__(self, iface, parent=None):
+ """
+ Initialize the QGIS Gea Plugin dock widget.
+
+ :param iface: Reference to the QGIS interface.
+ :type iface: QgsInterface
+
+ :param parent: Parent widget. Defaults to None.
+ :type parent: QWidget
+
+ """
+ super().__init__(parent)
+ self.setupUi(self)
+ self.iface = iface
+
+ self.navigation_object = QgsTemporalNavigationObject(self)
+ self.navigation_object.setFrameDuration(
+ QgsInterval(1, QgsUnitTypes.TemporalIrregularStep)
+ )
+
+ self.current_imagery_type = IMAGERY.HISTORICAL
+
+ icon_pixmap = QtGui.QPixmap(PLUGIN_ICON)
+ self.icon_la.setPixmap(icon_pixmap)
+
+ self.play_btn.setIcon(QtGui.QIcon(ANIMATION_PLAY_ICON))
+
+ self.time_values = []
+
+ self.historical_imagery.setChecked(
+ settings_manager.get_value(
+ Settings.HISTORICAL_VIEW,
+ setting_type=bool,
+ default=True)
+ )
+
+ self.nicfi_imagery.setChecked(
+ settings_manager.get_value(
+ Settings.NICFI_VIEW,
+ setting_type=bool,
+ default=False)
+ )
+ self.prepare_time_slider()
+
+ self.historical_imagery.toggled.connect(self.prepare_time_slider)
+ self.nicfi_imagery.toggled.connect(self.prepare_time_slider)
+
+ self.play_btn.clicked.connect(self.animate_layers)
+ self.navigation_object.updateTemporalRange.connect(
+ self.temporal_range_changed
+ )
+ self.time_slider.valueChanged.connect(
+ self.slider_value_changed
+ )
+
+ self.iface.projectRead.connect(self.prepare_time_slider)
+
+ def slider_value_changed(self, value):
+ """
+ Slot function for handling time slider value change.
+
+ :param value: New value of the slider.
+ :type value: int
+ """
+ self.navigation_object.setCurrentFrameNumber(value)
+
+ def animate_layers(self):
+ """
+ Toggle animation of layers based on the current animation state.
+ This function is called when user press the play button.
+ """
+ if self.navigation_object.animationState() == \
+ QgsTemporalNavigationObject.AnimationState.Idle:
+ self.play_btn.setIcon(QtGui.QIcon(ANIMATION_PAUSE_ICON))
+ self.play_btn.setToolTip(tr("Pause animation"))
+ self.navigation_object.playForward()
+ else:
+ self.navigation_object.pause()
+ self.play_btn.setToolTip(tr("Click to play animation"))
+ self.play_btn.setIcon(QtGui.QIcon(ANIMATION_PLAY_ICON))
+
+ def temporal_range_changed(self, temporal_range):
+ """
+ Update temporal range and UI elements when temporal range changes.
+
+ :param temporal_range: New temporal range.
+ :type temporal_range: QgsDateTimeRange
+ """
+ self.iface.mapCanvas().setTemporalRange(temporal_range)
+ self.temporal_range_la.setText(
+ tr(
+ f'Current time range: '
+ f'{temporal_range.begin().toString("yyyy-MM-ddTHH:mm:ss")} to '
+ f'{temporal_range.end().toString("yyyy-MM-ddTHH:mm:ss")} '
+ ))
+ self.time_slider.setValue(
+ self.navigation_object.currentFrameNumber()
+ )
+
+ # On the last animation frame
+ if self.navigation_object.currentFrameNumber() == \
+ len(self.navigation_object.availableTemporalRanges()) - 1:
+
+ self.play_btn.setIcon(QtGui.QIcon(ANIMATION_PLAY_ICON))
+
+ def prepare_time_slider(self):
+ """
+ Prepare the time slider based on current selected imagery type.
+ """
+ values = []
+ set_layer = None
+ active_layer = None
+
+ closed_imagery = None
+
+ if self.historical_imagery.isChecked():
+ settings_manager.set_value(Settings.HISTORICAL_VIEW, True)
+ settings_manager.set_value(Settings.NICFI_VIEW, False)
+
+ self.current_imagery_type = IMAGERY.HISTORICAL
+ closed_imagery = IMAGERY.NICFI
+ else:
+ settings_manager.set_value(Settings.NICFI_VIEW, True)
+ settings_manager.set_value(Settings.HISTORICAL_VIEW, False)
+
+ self.current_imagery_type = IMAGERY.NICFI
+ closed_imagery = IMAGERY.HISTORICAL
+
+ layers = QgsProject.instance().mapLayers()
+ for path, layer in layers.items():
+ if layer.metadata().contains(
+ self.current_imagery_type.value.lower()
+ ):
+ values.append(
+ layer.temporalProperties().fixedTemporalRange()
+ )
+ active_layer = layer
+ elif layer.metadata().contains(
+ closed_imagery.value.lower()
+ ):
+ set_layer = layer
+
+ self.update_layer_group(set_layer)
+ self.update_layer_group(active_layer, True)
+
+ self.time_slider.setRange(0, len(values) - 1)
+ self.navigation_object.setAvailableTemporalRanges(values)
+
+ temporal_range = values[0] if len(values) > 0 else None
+
+ if temporal_range:
+ self.iface.mapCanvas().setTemporalRange(temporal_range)
+ self.temporal_range_la.setText(
+ tr(
+ f'Current time range: '
+ f'{temporal_range.begin().toString("yyyy-MM-ddTHH:mm:ss")} to '
+ f'{temporal_range.end().toString("yyyy-MM-ddTHH:mm:ss")} '
+ ))
+
+ def update_layer_group(self, layer, show=False):
+ """
+ Update visibility of provided layer parent group.
+
+ :param layer: Layer to update.
+ :type layer: QgsMapLayer
+
+ :param show: Group visibility state. Defaults to False.
+ :type show: bool
+ """
+ if layer is not None:
+ root = QgsProject.instance().layerTreeRoot()
+ layer_tree = root.findLayer(layer.id())
+
+ if layer_tree is not None:
+ group_tree = layer_tree.parent()
+ if group_tree is not None:
+ group_tree.setItemVisibilityCheckedRecursive(show)
diff --git a/src/qgis_gea_plugin/gui/qgis_gea_plugin.py b/src/qgis_gea_plugin/gui/qgis_gea_plugin.py
deleted file mode 100644
index dd8ee32..0000000
--- a/src/qgis_gea_plugin/gui/qgis_gea_plugin.py
+++ /dev/null
@@ -1,43 +0,0 @@
-# -*- coding: utf-8 -*-
-
-"""
- The plugin main window class file
-"""
-
-import os
-
-from qgis.PyQt import (
- QtCore,
- QtGui,
- QtWidgets,
- QtNetwork,
-)
-from qgis.PyQt.uic import loadUiType
-
-from ..resources import *
-
-
-WidgetUi, _ = loadUiType(
- os.path.join(os.path.dirname(__file__), "../ui/main_dockwidget.ui")
-)
-
-
-class QgisGeaPlugin(QtWidgets.QDockWidget, WidgetUi):
- """Main plugin UI"""
-
- def __init__(
- self,
- iface,
- parent=None,
- ):
- super().__init__(parent)
- self.setupUi(self)
- self.iface = iface
-
- icon_pixmap = QtGui.QPixmap(":/plugins/qgis_gea_plugin/icon.png")
- self.icon_la.setPixmap(icon_pixmap)
-
- self.play_btn.setIcon(
- QtGui.QIcon(":/images/themes/default/mActionPlay.svg")
- )
-
diff --git a/src/qgis_gea_plugin/main.py b/src/qgis_gea_plugin/main.py
index 117494c..287ac01 100644
--- a/src/qgis_gea_plugin/main.py
+++ b/src/qgis_gea_plugin/main.py
@@ -19,7 +19,7 @@
# Initialize Qt resources from file resources.py
from .resources import *
-from .gui.qgis_gea_plugin import QgisGeaPlugin
+from .gui.qgis_gea import QgisGeaPlugin
class QgisGea:
diff --git a/src/qgis_gea_plugin/models/base.py b/src/qgis_gea_plugin/models/base.py
new file mode 100644
index 0000000..cb486b2
--- /dev/null
+++ b/src/qgis_gea_plugin/models/base.py
@@ -0,0 +1,14 @@
+# -*- coding: utf-8 -*-
+
+""" Plugin models.
+"""
+
+import dataclasses
+import datetime
+from enum import Enum, IntEnum
+
+class IMAGERY(Enum):
+ """Project imagery types"""
+
+ HISTORICAL = "Historical"
+ NICFI = "Nicfi"
\ No newline at end of file
diff --git a/src/qgis_gea_plugin/ui/main_dockwidget.ui b/src/qgis_gea_plugin/ui/main_dockwidget.ui
index 0f0f65d..9a8fe3a 100644
--- a/src/qgis_gea_plugin/ui/main_dockwidget.ui
+++ b/src/qgis_gea_plugin/ui/main_dockwidget.ui
@@ -7,7 +7,7 @@
0
0
725
- 588
+ 610
@@ -118,7 +118,7 @@
-
-
+
10
@@ -126,31 +126,50 @@
Qt::Horizontal
- QSlider::TicksBothSides
+ QSlider::TicksBelow
-
-
+
Current Imagery (NICIFI)
+
+ buttonGroup
+
-
-
+
Historical Imagery (Landsat)
+
+ true
+
+
+ buttonGroup
+
-
+
+ Play animation
+
...
+ -
+
+
+
+
+
+
@@ -291,4 +310,7 @@
+
+
+
diff --git a/src/qgis_gea_plugin/utils.py b/src/qgis_gea_plugin/utils.py
index e69de29..2bd23f4 100644
--- a/src/qgis_gea_plugin/utils.py
+++ b/src/qgis_gea_plugin/utils.py
@@ -0,0 +1,80 @@
+# -*- coding: utf-8 -*-
+"""
+ Plugin utilities
+"""
+
+import hashlib
+import json
+import os
+import uuid
+import datetime
+from pathlib import Path
+
+from qgis.PyQt import QtCore, QtGui
+from qgis.core import (
+ Qgis,
+ QgsCoordinateReferenceSystem,
+ QgsCoordinateTransform,
+ QgsCoordinateTransformContext,
+ QgsDistanceArea,
+ QgsMessageLog,
+ QgsProcessingFeedback,
+ QgsProject,
+ QgsProcessing,
+ QgsRasterLayer,
+ QgsRectangle,
+ QgsUnitTypes,
+)
+
+
+
+
+def log(
+ message: str,
+ name: str = "qgis_gea",
+ info: bool = True,
+ notify: bool = True,
+):
+ """Logs the message into QGIS logs using qgis_cplus as the default
+ log instance.
+ If notify_user is True, user will be notified about the log.
+
+ :param message: The log message
+ :type message: str
+
+ :param name: Name of te log instance, qgis_cplus is the default
+ :type message: str
+
+ :param info: Whether the message is about info or a
+ warning
+ :type info: bool
+
+ :param notify: Whether to notify user about the log
+ :type notify: bool
+ """
+ level = Qgis.Info if info else Qgis.Warning
+ QgsMessageLog.logMessage(
+ message,
+ name,
+ level=level,
+ notifyUser=notify,
+ )
+
+def tr(message):
+ """Get the translation for a string using Qt translation API.
+ We implement this ourselves since we do not inherit QObject.
+
+ :param message: String for translation.
+ :type message: str, QString
+
+ :returns: Translated version of message.
+ :rtype: QString
+ """
+ # noinspection PyTypeChecker,PyArgumentList,PyCallByClass
+ return QtCore.QCoreApplication.translate("QgisGea", message)
+
+
+def animation_state_change(value):
+ log(f"{value}")
+ pass
+
diff --git a/test/test_main_dock.py b/test/test_main_dock.py
new file mode 100644
index 0000000..689a1f9
--- /dev/null
+++ b/test/test_main_dock.py
@@ -0,0 +1,114 @@
+import unittest
+from qgis.PyQt.QtCore import Qt, QDateTime
+from qgis.PyQt.QtTest import QTest
+
+from qgis.core import QgsDateTimeRange, QgsInterval, QgsUnitTypes, QgsTemporalNavigationObject
+from qgis.gui import QgsMapCanvas
+
+from qgis.utils import plugins, iface
+
+from qgis_gea_plugin.gui.qgis_gea import QgisGeaPlugin
+
+from qgis_gea_plugin.models.base import IMAGERY
+
+from utilities_for_testing import get_qgis_app
+
+QGIS_APP, CANVAS, IFACE, PARENT = get_qgis_app()
+
+class TestQgisGeaPlugin(unittest.TestCase):
+ """
+ Unit tests for QgisGeaPlugin class.
+ """
+
+ @classmethod
+ def setUpClass(cls):
+ """
+ Set up the QGIS GEA Plugin dock instance for testing.
+ """
+ cls.plugin_dock = QgisGeaPlugin(iface)
+ iface.mainWindow().addDockWidget(
+ Qt.RightDockWidgetArea,
+ cls.plugin_dock
+ )
+
+ def test_slider_value_changed(self):
+ """
+ Test slider value change functionality.
+ """
+
+ ranges = [
+ QgsDateTimeRange(
+ QDateTime.fromString("2020-01-01T00:00:00", Qt.ISODate),
+ QDateTime.fromString("2020-01-01T00:00:00", Qt.ISODate)),
+ QgsDateTimeRange(
+ QDateTime.fromString("2020-01-01T00:00:00", Qt.ISODate),
+ QDateTime.fromString("2020-01-01T00:00:00", Qt.ISODate))
+ ]
+ self.plugin_dock.navigation_object.setFrameDuration(
+ QgsInterval(1, QgsUnitTypes.TemporalIrregularStep)
+ )
+ self.plugin_dock.navigation_object.setAvailableTemporalRanges(ranges)
+
+ self.assertEqual(self.plugin_dock.navigation_object.currentFrameNumber(), 0)
+
+ # Simulate setting slider value
+ slider_value = 1
+ self.plugin_dock.slider_value_changed(slider_value)
+
+ # Check if navigation object's current frame number is set correctly
+ self.assertEqual(
+ self.plugin_dock.navigation_object.currentFrameNumber(),
+ slider_value
+ )
+
+ def test_animate_layers(self):
+ """
+ Test the main animation function.
+ """
+ self.assertEqual(
+ self.plugin_dock.navigation_object.animationState(),
+ QgsTemporalNavigationObject.AnimationState.Idle
+ )
+ # Simulate clicking on the play button
+ QTest.mouseClick(self.plugin_dock.play_btn, Qt.LeftButton)
+
+ self.assertEqual(
+ self.plugin_dock.navigation_object.animationState(),
+ QgsTemporalNavigationObject.AnimationState.Forward
+ )
+
+ # Simulate clicking on the play button again
+ QTest.mouseClick(self.plugin_dock.play_btn, Qt.LeftButton)
+
+ self.assertEqual(
+ self.plugin_dock.navigation_object.animationState(),
+ QgsTemporalNavigationObject.AnimationState.Idle
+ )
+
+ def test_prepare_time_slider(self):
+ """
+ Test preparation of time slider UI components.
+ """
+ self.plugin_dock.historical_imagery.setChecked(True)
+ self.plugin_dock.prepare_time_slider()
+
+ # Check if the current imagery type is set correctly
+ self.assertEqual(self.plugin_dock.current_imagery_type, IMAGERY.HISTORICAL)
+
+ self.plugin_dock.nicfi_imagery.setChecked(True)
+ self.plugin_dock.prepare_time_slider()
+
+ # Check if the current imagery type is set correctly
+ self.assertEqual(self.plugin_dock.current_imagery_type, IMAGERY.NICFI)
+
+ @classmethod
+ def tearDownClass(cls):
+ """
+ Clean up after tests.
+ """
+ # Remove the dock widget after tests
+ iface.mainWindow().removeDockWidget(cls.plugin_dock)
+
+
+if __name__ == "__main__":
+ unittest.main()
diff --git a/test_suite.py b/test_suite.py
index a2b7876..25f8a71 100644
--- a/test_suite.py
+++ b/test_suite.py
@@ -11,7 +11,7 @@
try:
import coverage
except ImportError:
- pipmain(['install', 'coverage'])
+ pipmain(["install", "coverage"])
import coverage
import tempfile
from osgeo import gdal
@@ -27,17 +27,17 @@ def _run_tests(test_suite, package_name, with_coverage=False):
version = str(Qgis.QGIS_VERSION_INT)
version = int(version)
- print('########')
- print('%s tests has been discovered in %s' % (count, package_name))
- print('QGIS : %s' % version)
- print('Python GDAL : %s' % gdal.VersionInfo('VERSION_NUM'))
- print('QT : %s' % Qt.QT_VERSION_STR)
- print('Run slow tests : %s' % (not os.environ.get('ON_TRAVIS', False)))
- print('########')
+ print("########")
+ print("%s tests has been discovered in %s" % (count, package_name))
+ print("QGIS : %s" % version)
+ print("Python GDAL : %s" % gdal.VersionInfo("VERSION_NUM"))
+ print("QT : %s" % Qt.QT_VERSION_STR)
+ print("Run slow tests : %s" % (not os.environ.get("ON_TRAVIS", False)))
+ print("########")
if with_coverage:
cov = coverage.Coverage(
- source=['./'],
- omit=['*/test/*', './definitions/*'],
+ source=["./"],
+ omit=["*/test/*", "./definitions/*"],
)
cov.start()
@@ -48,14 +48,14 @@ def _run_tests(test_suite, package_name, with_coverage=False):
cov.save()
report = tempfile.NamedTemporaryFile(delete=False)
cov.report(file=report)
- # Produce HTML reports in the `htmlcov` folder and open index.html
+ # Produce HTML report_templates in the `htmlcov` folder and open index.html
# cov.html_report()
report.close()
- with open(report.name, 'r') as fin:
+ with open(report.name, "r") as fin:
print(fin.read())
-def test_package(package='test'):
+def test_package(package="test"):
"""Test package.
This function is called by Github actions or travis without arguments.
:param package: The package to test.