Skip to content

Commit 0088eac

Browse files
committed
fix plan feature digitize tool and promoted widget references after refactoring
- fix PlanDigitizeMapTool to reinitialize itself after layer changes - implement similar config file fetching for regulation group libraries as for feature templates - reorganize library files in resources folder - deactivate selections in NewFeatureDock when PlanFeatureDigitizeMapTool finishes/deactivates
1 parent d5c5aef commit 0088eac

18 files changed

+123
-70
lines changed

arho_feature_template/core/models.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@
2121

2222
logger = logging.getLogger(__name__)
2323

24-
DEFAULT_PLAN_REGULATIONS_CONFIG_PATH = Path(os.path.join(resources_path(), "kaavamaaraykset.yaml"))
24+
DEFAULT_PLAN_REGULATIONS_CONFIG_PATH = Path(os.path.join(resources_path(), "libraries", "kaavamaaraykset.yaml"))
2525

2626

2727
class ValueType(Enum):

arho_feature_template/core/plan_manager.py

+44-32
Original file line numberDiff line numberDiff line change
@@ -4,11 +4,7 @@
44
import logging
55
from typing import TYPE_CHECKING
66

7-
from qgis.core import (
8-
QgsExpressionContextUtils,
9-
QgsProject,
10-
QgsVectorLayer,
11-
)
7+
from qgis.core import QgsExpressionContextUtils, QgsProject, QgsVectorLayer, QgsWkbTypes
128
from qgis.gui import QgsMapToolDigitizeFeature
139
from qgis.PyQt.QtWidgets import QDialog, QMessageBox
1410

@@ -40,7 +36,8 @@
4036
RegulationGroupLayer,
4137
plan_layers,
4238
)
43-
from arho_feature_template.resources.template_libraries import library_config_files
39+
from arho_feature_template.resources.libraries.feature_templates import feature_template_library_config_files
40+
from arho_feature_template.resources.libraries.regulation_groups import regulation_group_library_config_files
4441
from arho_feature_template.utils.db_utils import get_existing_database_connection_names
4542
from arho_feature_template.utils.misc_utils import (
4643
check_layer_changes,
@@ -68,41 +65,61 @@
6865
class PlanDigitizeMapTool(QgsMapToolDigitizeFeature): ...
6966

7067

71-
class PlanFeatureDigitizeMapTool(QgsMapToolDigitizeFeature): ...
68+
class PlanFeatureDigitizeMapTool(QgsMapToolDigitizeFeature):
69+
def __init__(self, mode: QgsMapToolDigitizeFeature.CaptureMode):
70+
super().__init__(iface.mapCanvas(), iface.cadDockWidget(), mode)
7271

7372

7473
class PlanManager:
7574
def __init__(self):
7675
self.json_plan_path = None
7776
self.json_plan_outline_path = None
7877

79-
self.digitize_map_tool = PlanDigitizeMapTool(iface.mapCanvas(), iface.cadDockWidget())
80-
self.digitize_map_tool.digitizingCompleted.connect(self._plan_geom_digitized)
81-
82-
# Initialize new feature dock
78+
# Initialize libraries
8379
self.feature_template_libraries = [
84-
FeatureTemplateLibrary.from_config_file(config_file) for config_file in library_config_files()
80+
FeatureTemplateLibrary.from_config_file(file) for file in feature_template_library_config_files()
8581
]
86-
# self.feature_template_libraries = []
87-
88-
# TODO: change
89-
import os
90-
from pathlib import Path
91-
92-
from arho_feature_template.qgis_plugin_tools.tools.resources import resources_path
93-
94-
katja_asemakaava_path = Path(os.path.join(resources_path(), "katja_asemakaava.yaml"))
9582
self.regulation_group_libraries = [
96-
RegulationGroupLibrary.from_config_file(katja_asemakaava_path),
97-
regulation_group_library_from_active_plan(),
83+
RegulationGroupLibrary.from_config_file(file) for file in regulation_group_library_config_files()
9884
]
85+
self.regulation_group_libraries.append(regulation_group_library_from_active_plan())
9986

87+
# Initialize new feature dock
10088
self.new_feature_dock = NewFeatureDock(self.feature_template_libraries)
101-
self.new_feature_dock.start_digitize_btn.clicked.connect(self.add_new_plan_feature)
89+
self.new_feature_dock.tool_activated.connect(self.add_new_plan_feature)
10290
self.new_feature_dock.hide()
10391

104-
self.feature_digitize_map_tool = PlanFeatureDigitizeMapTool(iface.mapCanvas(), iface.cadDockWidget())
92+
# Initialize digitize tools
93+
self.digitize_map_tool = PlanDigitizeMapTool(iface.mapCanvas(), iface.cadDockWidget())
94+
self.digitize_map_tool.digitizingCompleted.connect(self._plan_geom_digitized)
95+
96+
self.feature_digitize_map_tool = None
97+
self.initialize_feature_digitize_map_tool()
98+
99+
def initialize_feature_digitize_map_tool(self, layer: QgsVectorLayer | None = None):
100+
# Get matcing capture mode for given layer
101+
if layer is None:
102+
mode = PlanFeatureDigitizeMapTool.CaptureMode.CaptureNone
103+
elif layer.geometryType() == QgsWkbTypes.PointGeometry:
104+
mode = PlanFeatureDigitizeMapTool.CaptureMode.CapturePoint
105+
elif layer.geometryType() == QgsWkbTypes.LineGeometry:
106+
mode = PlanFeatureDigitizeMapTool.CaptureMode.CaptureLine
107+
elif layer.geometryType() == QgsWkbTypes.PolygonGeometry:
108+
mode = PlanFeatureDigitizeMapTool.CaptureMode.CapturePolygon
109+
110+
# Disconnect signals first to not trigger them unwantedly
111+
if self.feature_digitize_map_tool:
112+
self.feature_digitize_map_tool.digitizingCompleted.disconnect()
113+
self.feature_digitize_map_tool.digitizingFinished.disconnect()
114+
115+
# Reinitialize and reconnect signals
116+
self.feature_digitize_map_tool = PlanFeatureDigitizeMapTool(mode)
105117
self.feature_digitize_map_tool.digitizingCompleted.connect(self._plan_feature_geom_digitized)
118+
self.feature_digitize_map_tool.digitizingFinished.connect(self.new_feature_dock.deactivate_and_clear_selections)
119+
120+
# Set layer if given
121+
if layer:
122+
self.feature_digitize_map_tool.setLayer(layer)
106123

107124
def add_new_plan(self):
108125
"""Initiate the process to add a new plan to the Kaava layer."""
@@ -125,8 +142,6 @@ def add_new_plan(self):
125142
iface.mapCanvas().setMapTool(self.digitize_map_tool)
126143

127144
def add_new_plan_feature(self):
128-
self.previous_map_tool = iface.mapCanvas().mapTool()
129-
130145
if not handle_unsaved_changes():
131146
return
132147

@@ -141,10 +156,9 @@ def add_new_plan_feature(self):
141156
msg = f"Could not find plan feature layer class for layer name {layer_name}"
142157
raise ValueError(msg)
143158
layer = layer_class.get_from_project()
144-
145159
layer.startEditing()
146-
self.feature_digitize_map_tool.clean() # Is this needed? add_new_plan does not call this
147-
self.feature_digitize_map_tool.setLayer(layer)
160+
161+
self.initialize_feature_digitize_map_tool(layer)
148162
iface.mapCanvas().setMapTool(self.feature_digitize_map_tool)
149163

150164
def _plan_geom_digitized(self, feature: QgsFeature):
@@ -187,8 +201,6 @@ def _plan_feature_geom_digitized(self, feature: QgsFeature):
187201
if attribute_form.exec_():
188202
save_plan_feature(attribute_form.model)
189203

190-
iface.mapCanvas().setMapTool(self.previous_map_tool)
191-
192204
def set_active_plan(self, plan_id: str | None):
193205
"""Update the project layers based on the selected land use plan.
194206

arho_feature_template/gui/components/plan_regulation_group_widget.ui

+1-1
Original file line numberDiff line numberDiff line change
@@ -166,7 +166,7 @@
166166
<customwidget>
167167
<class>CodeComboBox</class>
168168
<extends>QComboBox</extends>
169-
<header>arho_feature_template.gui.code_combobox</header>
169+
<header>arho_feature_template.gui.components.code_combobox</header>
170170
</customwidget>
171171
</customwidgets>
172172
<resources/>

arho_feature_template/gui/dialogs/new_plan_regulation_group_form.ui

+1-1
Original file line numberDiff line numberDiff line change
@@ -269,7 +269,7 @@ p, li { white-space: pre-wrap; }
269269
<customwidget>
270270
<class>CodeComboBox</class>
271271
<extends>QComboBox</extends>
272-
<header>arho_feature_template.gui.code_combobox</header>
272+
<header>arho_feature_template.gui.components.code_combobox</header>
273273
</customwidget>
274274
</customwidgets>
275275
<resources/>

arho_feature_template/gui/dialogs/plan_attribute_form.ui

+2-2
Original file line numberDiff line numberDiff line change
@@ -377,12 +377,12 @@
377377
<customwidget>
378378
<class>CodeComboBox</class>
379379
<extends>QComboBox</extends>
380-
<header>arho_feature_template.gui.code_combobox</header>
380+
<header>arho_feature_template.gui.components.code_combobox</header>
381381
</customwidget>
382382
<customwidget>
383383
<class>HierarchicalCodeComboBox</class>
384384
<extends>QComboBox</extends>
385-
<header>arho_feature_template.gui.code_combobox</header>
385+
<header>arho_feature_template.gui.components.code_combobox</header>
386386
</customwidget>
387387
</customwidgets>
388388
<tabstops>

arho_feature_template/gui/dialogs/plan_feature_form.ui

+1-1
Original file line numberDiff line numberDiff line change
@@ -197,7 +197,7 @@
197197
<customwidget>
198198
<class>CodeComboBox</class>
199199
<extends>QComboBox</extends>
200-
<header>arho_feature_template.gui.code_combobox</header>
200+
<header>arho_feature_template.gui.components.code_combobox</header>
201201
</customwidget>
202202
</customwidgets>
203203
<resources/>

arho_feature_template/gui/dialogs/plugin_settings.ui

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
<x>0</x>
88
<y>0</y>
99
<width>300</width>
10-
<height>150</height>
10+
<height>205</height>
1111
</rect>
1212
</property>
1313
<property name="windowTitle">

arho_feature_template/gui/docks/new_feature_dock.py

+18-9
Original file line numberDiff line numberDiff line change
@@ -3,14 +3,14 @@
33

44
from qgis.gui import QgsDockWidget
55
from qgis.PyQt import uic
6-
from qgis.PyQt.QtCore import Qt
6+
from qgis.PyQt.QtCore import Qt, pyqtSignal
77
from qgis.PyQt.QtWidgets import QListWidget, QListWidgetItem
88

99
from arho_feature_template.gui.components.new_feature_grid_widget import NewFeatureGridWidget
1010

1111
if TYPE_CHECKING:
1212
from qgis.gui import QgsFilterLineEdit
13-
from qgis.PyQt.QtWidgets import QComboBox, QLabel, QPushButton, QWidget
13+
from qgis.PyQt.QtWidgets import QComboBox, QLabel, QWidget
1414

1515
from arho_feature_template.core.models import FeatureTemplateLibrary
1616

@@ -19,6 +19,8 @@
1919

2020

2121
class NewFeatureDock(QgsDockWidget, DockClass): # type: ignore
22+
tool_activated = pyqtSignal()
23+
2224
def __init__(self, feature_template_libraries: list):
2325
super().__init__()
2426
self.setupUi(self)
@@ -29,7 +31,6 @@ def __init__(self, feature_template_libraries: list):
2931
self.template_list: QListWidget
3032
self.txt_tip: QLabel
3133
self.dockWidgetContents: QWidget
32-
self.start_digitize_btn: QPushButton
3334

3435
# INIT
3536
# 1. New feature grid
@@ -47,11 +48,8 @@ def __init__(self, feature_template_libraries: list):
4748
self.set_active_feature_template_library(0)
4849

4950
self.template_list.setSelectionMode(self.template_list.SingleSelection)
51+
self.search_box.valueChanged.connect(self.filter_plan_feature_templates)
5052

51-
self.search_box.setShowSearchIcon(True)
52-
self.search_box.valueChanged.connect(lambda _: self.filter_plan_feature_templates)
53-
54-
# Activate map tool when template is selected
5553
# NOTE: If user moves selection with arrow keys, this is not registered currently
5654
self.template_list.clicked.connect(self.on_template_item_clicked)
5755
self.new_feature_grid.active_feature_type_changed.connect(self.on_active_feature_type_changed)
@@ -60,6 +58,8 @@ def on_active_feature_type_changed(self, feature_name: str, layer_name: str):
6058
self.active_feature_type = feature_name if feature_name else None
6159
self.active_feature_layer = layer_name if layer_name else None
6260
self.filter_plan_feature_templates()
61+
if self.active_feature_type:
62+
self.tool_activated.emit()
6363

6464
def filter_plan_feature_templates(self):
6565
# Consider both search text and active plan feature type
@@ -70,19 +70,28 @@ def filter_plan_feature_templates(self):
7070
item_text = item.text().lower()
7171
plan_feature = item.data(Qt.UserRole)
7272
text_matches = search_text in item_text
73-
feature_type_matches = plan_feature.layer_name == self.active_feature_layer
73+
feature_type_matches = (
74+
plan_feature.layer_name == self.active_feature_layer if self.active_feature_layer else True
75+
)
7476
item.setHidden(not (text_matches and feature_type_matches))
7577

7678
def on_template_item_clicked(self, index: int):
7779
item = self.template_list.itemFromIndex(index)
7880
template = item.data(Qt.UserRole)
81+
# Clicked new list item => activate new template
7982
if template != self.active_template:
8083
self.active_template = template
8184
else:
82-
# If user clicks an already selected list item, clear selection
85+
# Clicked selected list item => clear selection
8386
self.active_template = None
8487
self.template_list.clearSelection()
8588

89+
def deactivate_and_clear_selections(self):
90+
self.on_active_feature_type_changed("", "")
91+
self.new_feature_grid.clear_selections()
92+
self.active_template = None
93+
self.template_list.clearSelection()
94+
8695
def set_active_feature_template_library(self, index: int) -> None:
8796
self.template_list.clear()
8897
library = self.feature_template_libraries[index]

arho_feature_template/gui/docks/new_feature_dock.ui

+32-9
Original file line numberDiff line numberDiff line change
@@ -34,23 +34,42 @@
3434
<number>0</number>
3535
</property>
3636
<item>
37-
<widget class="QPushButton" name="start_digitize_btn">
37+
<widget class="QLabel" name="label">
38+
<property name="font">
39+
<font>
40+
<pointsize>13</pointsize>
41+
</font>
42+
</property>
3843
<property name="text">
39-
<string>Piirrä</string>
44+
<string>Kaavakohteen tyyppi</string>
4045
</property>
4146
</widget>
4247
</item>
4348
<item>
44-
<widget class="QLabel" name="label">
45-
<property name="text">
46-
<string>Kohteen tyyppi</string>
49+
<spacer name="verticalSpacer">
50+
<property name="orientation">
51+
<enum>Qt::Vertical</enum>
4752
</property>
48-
</widget>
53+
<property name="sizeType">
54+
<enum>QSizePolicy::Fixed</enum>
55+
</property>
56+
<property name="sizeHint" stdset="0">
57+
<size>
58+
<width>20</width>
59+
<height>20</height>
60+
</size>
61+
</property>
62+
</spacer>
4963
</item>
5064
<item>
5165
<widget class="QLabel" name="label_2">
66+
<property name="font">
67+
<font>
68+
<pointsize>13</pointsize>
69+
</font>
70+
</property>
5271
<property name="text">
53-
<string>Alusta kaavamääräykset templaatista</string>
72+
<string>Alusta kaavakohde templaatista</string>
5473
</property>
5574
</widget>
5675
</item>
@@ -65,7 +84,7 @@
6584
</sizepolicy>
6685
</property>
6786
<property name="text">
68-
<string>Kirjasto</string>
87+
<string>Kaavakohdekirjasto</string>
6988
</property>
7089
</widget>
7190
</item>
@@ -75,7 +94,11 @@
7594
</layout>
7695
</item>
7796
<item>
78-
<widget class="QgsFilterLineEdit" name="search_box"/>
97+
<widget class="QgsFilterLineEdit" name="search_box">
98+
<property name="showSearchIcon">
99+
<bool>true</bool>
100+
</property>
101+
</widget>
79102
</item>
80103
<item>
81104
<widget class="QListWidget" name="template_list"/>

arho_feature_template/gui/docks/validation_dock.ui

+1-1
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,7 @@
4545
<customwidget>
4646
<class>ValidationTreeView</class>
4747
<extends>QTreeView</extends>
48-
<header>arho_feature_template.gui.validation_tree_view</header>
48+
<header>arho_feature_template.gui.docks.validation_tree_view</header>
4949
</customwidget>
5050
</customwidgets>
5151
<resources/>

arho_feature_template/plugin.py

+1-6
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,6 @@
88
from qgis.PyQt.QtWidgets import QAction, QWidget
99
from qgis.utils import iface
1010

11-
from arho_feature_template.core.feature_template_library import TemplateGeometryDigitizeMapTool
1211
from arho_feature_template.core.plan_manager import PlanManager
1312
from arho_feature_template.gui.dialogs.plugin_settings import PluginSettings
1413
from arho_feature_template.gui.docks.validation_dock import ValidationDock
@@ -17,7 +16,7 @@
1716
from arho_feature_template.qgis_plugin_tools.tools.resources import plugin_name
1817

1918
if TYPE_CHECKING:
20-
from qgis.gui import QgisInterface, QgsMapTool
19+
from qgis.gui import QgisInterface
2120

2221
iface: QgisInterface = cast("QgisInterface", iface) # type: ignore[no-redef]
2322

@@ -210,10 +209,6 @@ def initGui(self) -> None: # noqa N802
210209
status_tip="Muokkaa pluginin asetuksia",
211210
)
212211

213-
def on_map_tool_changed(self, new_tool: QgsMapTool, old_tool: QgsMapTool) -> None: # noqa: ARG002
214-
if not isinstance(new_tool, TemplateGeometryDigitizeMapTool):
215-
self.new_feature_dock_action.setChecked(False)
216-
217212
def add_new_plan(self):
218213
self.plan_manager.add_new_plan()
219214

arho_feature_template/resources/template_libraries/__init__.py arho_feature_template/resources/libraries/feature_templates/__init__.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@
88
from pathlib import Path
99

1010

11-
def library_config_files() -> Iterator[Path]:
11+
def feature_template_library_config_files() -> Iterator[Path]:
1212
"""Get all template library configuration files."""
1313

1414
for resource in resources.files(__package__).iterdir():

0 commit comments

Comments
 (0)