Compare commits
11 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| faed780dbf | |||
|
|
6a7e2c28f6 | ||
|
|
0be47794b0 | ||
| 2e2799930d | |||
| 01cbb76dcd | |||
| 1c3de100e4 | |||
| 34d8253919 | |||
| 178487f22e | |||
|
|
cc757452bc | ||
|
|
bf479bdb64 | ||
|
|
f69ee89a3c |
90
functions/verfahrensgebiet_alkis.py
Normal file
90
functions/verfahrensgebiet_alkis.py
Normal file
@@ -0,0 +1,90 @@
|
||||
from enum import Enum
|
||||
from qgis.core import (
|
||||
QgsVectorLayer, QgsProject, QgsFeature, QgsField, QgsGeometry
|
||||
)
|
||||
from qgis.PyQt.QtWidgets import QMessageBox
|
||||
from qgis.PyQt.QtCore import QVariant
|
||||
from sn_basis import get_variable
|
||||
|
||||
alkis_NAS_url = "https://geodienste.sachsen.de/aaa/public_alkis/nas/wfs"
|
||||
typename = "adv:AX_BauRaumOderBodenordnungsrecht"
|
||||
|
||||
class LoadStatus(Enum):
|
||||
NONE = "none" # nichts geladen
|
||||
FIRST = "first" # erstmalig geladen
|
||||
RELOAD = "reload" # neu geladen
|
||||
KEEP = "keep" # vorhandener Layer behalten
|
||||
|
||||
|
||||
def _check_existing_layer(tab_widget):
|
||||
project = QgsProject.instance()
|
||||
existing_layers = project.mapLayersByName("Verfahrensgebiet")
|
||||
if not existing_layers:
|
||||
return None, LoadStatus.FIRST
|
||||
|
||||
reply = QMessageBox.question(
|
||||
tab_widget,
|
||||
"Layer bereits vorhanden",
|
||||
"Der Layer 'Verfahrensgebiet' existiert bereits.\n"
|
||||
"Möchten Sie ihn löschen und neu laden?",
|
||||
QMessageBox.StandardButton.Yes | QMessageBox.StandardButton.No,
|
||||
QMessageBox.StandardButton.No
|
||||
)
|
||||
if reply == QMessageBox.StandardButton.Yes:
|
||||
for lyr in existing_layers:
|
||||
project.removeMapLayer(lyr.id())
|
||||
return None, LoadStatus.RELOAD
|
||||
else:
|
||||
return existing_layers[0], LoadStatus.KEEP
|
||||
|
||||
|
||||
def verfahrensgebiet_alkis(tab_widget):
|
||||
verfahrensnummer = get_variable("verfahrensnummer")
|
||||
if not verfahrensnummer:
|
||||
QMessageBox.critical(tab_widget, "Fehler",
|
||||
"In den Projekteigenschaften ist noch keine Verfahrensnummer eingetragen!")
|
||||
return None, LoadStatus.NONE
|
||||
|
||||
# Vorhandenen Layer prüfen
|
||||
existing, status = _check_existing_layer(tab_widget)
|
||||
if status == LoadStatus.KEEP:
|
||||
return existing, status
|
||||
|
||||
# WFS mit Filter laden
|
||||
wfs_uri_find = (
|
||||
f"url='{alkis_NAS_url}' typename='{typename}' srsname='EPSG:25833' "
|
||||
f"sql=SELECT * FROM AX_BauRaumOderBodenordnungsrecht "
|
||||
f"WHERE bezeichnung LIKE '{verfahrensnummer}%'"
|
||||
)
|
||||
temp_layer = QgsVectorLayer(wfs_uri_find, "TempVerfahrensgebiet", "WFS")
|
||||
|
||||
if not temp_layer.isValid():
|
||||
QMessageBox.critical(tab_widget, "Fehler", "Layer konnte nicht geladen werden.")
|
||||
return None, LoadStatus.NONE
|
||||
|
||||
features = list(temp_layer.getFeatures())
|
||||
if not features:
|
||||
QMessageBox.critical(tab_widget, "Fehler",
|
||||
f"Verfahrenskennzeichen {verfahrensnummer} nicht im WFS gefunden.")
|
||||
return None, LoadStatus.NONE
|
||||
|
||||
union_geom = QgsGeometry.unaryUnion([f.geometry() for f in features])
|
||||
if union_geom is None or union_geom.isEmpty():
|
||||
QMessageBox.critical(tab_widget, "Fehler", "Keine Geometrien zum Verschmelzen gefunden.")
|
||||
return None, LoadStatus.NONE
|
||||
|
||||
# Memory-Layer erzeugen
|
||||
crs = temp_layer.crs().authid()
|
||||
memory_layer = QgsVectorLayer(f"Polygon?crs={crs}", "Verfahrensgebiet", "memory")
|
||||
pr = memory_layer.dataProvider()
|
||||
pr.addAttributes([QgsField("VKZ", QVariant.String)])
|
||||
memory_layer.updateFields()
|
||||
|
||||
feat_union = QgsFeature()
|
||||
feat_union.setGeometry(union_geom)
|
||||
feat_union.setAttributes([verfahrensnummer])
|
||||
pr.addFeatures([feat_union])
|
||||
memory_layer.updateExtents()
|
||||
|
||||
# Status: FIRST oder RELOAD
|
||||
return memory_layer, status
|
||||
20
main.py
20
main.py
@@ -2,7 +2,6 @@ from qgis.utils import plugins
|
||||
from sn_basis.ui.dockmanager import DockManager
|
||||
from .ui.dockwidget import DockWidget
|
||||
|
||||
|
||||
class Verfahrensgebiet:
|
||||
def __init__(self, iface):
|
||||
self.iface = iface
|
||||
@@ -13,8 +12,11 @@ class Verfahrensgebiet:
|
||||
self.plugin_name = self.__class__.__name__
|
||||
self.dock_name = f"sn_dock_{self.plugin_name.lower()}"
|
||||
|
||||
def _basis(self):
|
||||
return plugins.get("sn_basis")
|
||||
|
||||
def initGui(self):
|
||||
basis = plugins.get("sn_basis")
|
||||
basis = self._basis()
|
||||
if basis and basis.ui:
|
||||
self.action = basis.ui.add_action(
|
||||
self.plugin_name,
|
||||
@@ -31,12 +33,22 @@ class Verfahrensgebiet:
|
||||
self.dockwidget = None
|
||||
|
||||
if self.action:
|
||||
basis = plugins.get("sn_basis")
|
||||
basis = self._basis()
|
||||
if basis and basis.ui:
|
||||
# Action aus Menü und Toolbar entfernen
|
||||
basis.ui.remove_action(self.action)
|
||||
self.action = None
|
||||
|
||||
# def run(self):
|
||||
# if not self.dockwidget:
|
||||
# self.dockwidget = DockWidget(self.iface.mainWindow(), subtitle=self.plugin_name)
|
||||
# self.dockwidget.setObjectName(self.dock_name)
|
||||
# self.dockwidget.action = self.action
|
||||
|
||||
# DockManager.show(self.dockwidget)
|
||||
|
||||
# basis = self._basis()
|
||||
# if basis and basis.ui:
|
||||
# basis.ui.set_active_plugin(self.action)
|
||||
def run(self):
|
||||
self.dockwidget = DockWidget(self.iface.mainWindow(), subtitle=self.plugin_name)
|
||||
self.dockwidget.setObjectName(self.dock_name)
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
name=LNO Sachsen | Verfahrensgebiet
|
||||
qgisMinimumVersion=3.0
|
||||
description=Plugin zum Erzeugen eines Objektes "Verfahrensgebiet"
|
||||
version=25.11.3
|
||||
version=25.11.4
|
||||
author=Michael Otto
|
||||
email=michael.otto@landkreis-mittelsachsen.de
|
||||
about=Plugin zum Erzeugen eines Objektes "Verfahrensgebiet"
|
||||
|
||||
@@ -1,8 +1,7 @@
|
||||
from sn_basis.ui.tabs.settings_tab import SettingsTab
|
||||
from sn_verfahrensgebiet.ui.tabs.tab_a import TabA
|
||||
from sn_verfahrensgebiet.ui.tabs.tab_b import TabB
|
||||
from sn_verfahrensgebiet.ui.tabs.working_tab import WorkingTab
|
||||
from sn_basis.ui.base_dockwidget import BaseDockWidget
|
||||
|
||||
|
||||
class DockWidget(BaseDockWidget):
|
||||
tabs = [TabA, TabB, SettingsTab]
|
||||
tabs = [WorkingTab, SettingsTab]
|
||||
|
||||
@@ -1,12 +0,0 @@
|
||||
from qgis.PyQt.QtWidgets import QWidget, QVBoxLayout, QLabel, QLineEdit
|
||||
|
||||
class TabA(QWidget):
|
||||
tab_title = "Tab A"
|
||||
|
||||
def __init__(self, parent=None):
|
||||
super().__init__(parent)
|
||||
layout = QVBoxLayout()
|
||||
layout.addWidget(QLabel("Plugin1 – Tab A"))
|
||||
layout.addWidget(QLineEdit("Feld A1"))
|
||||
layout.addWidget(QLineEdit("Feld A2"))
|
||||
self.setLayout(layout)
|
||||
@@ -1,11 +0,0 @@
|
||||
from qgis.PyQt.QtWidgets import QWidget, QVBoxLayout, QLabel, QTextEdit
|
||||
|
||||
class TabB(QWidget):
|
||||
tab_title = "Tab B"
|
||||
|
||||
def __init__(self, parent=None):
|
||||
super().__init__(parent)
|
||||
layout = QVBoxLayout()
|
||||
layout.addWidget(QLabel("Plugin1 – Tab B"))
|
||||
layout.addWidget(QTextEdit("Mehrzeiliger Text für Plugin1"))
|
||||
self.setLayout(layout)
|
||||
112
ui/tabs/working_tab.py
Normal file
112
ui/tabs/working_tab.py
Normal file
@@ -0,0 +1,112 @@
|
||||
from qgis.PyQt.QtWidgets import QWidget, QVBoxLayout, QHBoxLayout, QLabel, QPushButton, QMessageBox
|
||||
from qgis.PyQt.QtCore import Qt
|
||||
from qgis.core import Qgis, QgsProject, QgsMessageLog
|
||||
from qgis.utils import iface
|
||||
|
||||
from sn_verfahrensgebiet.functions.verfahrensgebiet_alkis import verfahrensgebiet_alkis, LoadStatus
|
||||
from sn_basis.functions.messages import success, info, warning, error
|
||||
from sn_basis.functions.styles import apply_style
|
||||
|
||||
class WorkingTab(QWidget):
|
||||
tab_title = "Bearbeitung"
|
||||
|
||||
def __init__(self, parent=None):
|
||||
super().__init__(parent)
|
||||
|
||||
layout = QVBoxLayout()
|
||||
layout.setAlignment(Qt.AlignmentFlag.AlignTop)
|
||||
|
||||
# Abschnitt 1: Verfahrensgebiet
|
||||
row_verf_header = QHBoxLayout()
|
||||
lbl_verf = QLabel("<b>Verfahrensgebiet</b>")
|
||||
self.haken_verf = self._haken_label()
|
||||
row_verf_header.addWidget(lbl_verf)
|
||||
row_verf_header.addWidget(self.haken_verf)
|
||||
row_verf_header.addStretch()
|
||||
layout.addLayout(row_verf_header)
|
||||
|
||||
row_verf_buttons = QHBoxLayout()
|
||||
self.btn_verf_alkis = QPushButton("Aus ALKIS laden")
|
||||
self.btn_verf_shape = QPushButton("Aus Shape laden")
|
||||
row_verf_buttons.addWidget(self.btn_verf_alkis)
|
||||
row_verf_buttons.addWidget(self.btn_verf_shape)
|
||||
layout.addLayout(row_verf_buttons)
|
||||
|
||||
# Abschnitt 2: Flurstücke (Platzhalter für spätere Implementierung)
|
||||
row_flurst_header = QHBoxLayout()
|
||||
lbl_flurst = QLabel("<b>Flurstücke</b>")
|
||||
self.haken_flurst = self._haken_label()
|
||||
row_flurst_header.addWidget(lbl_flurst)
|
||||
row_flurst_header.addWidget(self.haken_flurst)
|
||||
row_flurst_header.addStretch()
|
||||
layout.addLayout(row_flurst_header)
|
||||
|
||||
row_flurst_buttons = QHBoxLayout()
|
||||
self.btn_flurst_alkis = QPushButton("Aus ALKIS laden")
|
||||
self.btn_flurst_shape = QPushButton("Aus Shape laden")
|
||||
row_flurst_buttons.addWidget(self.btn_flurst_alkis)
|
||||
row_flurst_buttons.addWidget(self.btn_flurst_shape)
|
||||
layout.addLayout(row_flurst_buttons)
|
||||
|
||||
layout.addStretch()
|
||||
self.setLayout(layout)
|
||||
|
||||
# Signale verbinden
|
||||
self.btn_verf_alkis.clicked.connect(self.handle_verf_alkis)
|
||||
# self.btn_verf_shape.clicked.connect(self.handle_verf_shape)
|
||||
# self.btn_flurst_alkis.clicked.connect(self.handle_flurst_alkis)
|
||||
# self.btn_flurst_shape.clicked.connect(self.handle_flurst_shape)
|
||||
|
||||
def handle_verf_alkis(self):
|
||||
self._set_busy(self.btn_verf_alkis, True)
|
||||
try:
|
||||
layer, status = verfahrensgebiet_alkis(self)
|
||||
if not layer or not layer.isValid():
|
||||
return
|
||||
|
||||
if status in (LoadStatus.FIRST, LoadStatus.RELOAD):
|
||||
# Gemeinsame Logik für erstmaliges und erneutes Laden
|
||||
QgsProject.instance().addMapLayer(layer)
|
||||
iface.mapCanvas().setExtent(layer.extent())
|
||||
iface.mapCanvas().refresh()
|
||||
|
||||
apply_style(layer, "verfahrensgebiet.qml")
|
||||
|
||||
self.setze_haken(self.haken_verf, True)
|
||||
|
||||
# Unterschied nur in der Meldung
|
||||
msg = "erstmalig geladen" if status == LoadStatus.FIRST else "neu geladen"
|
||||
success("Verfahrensgebiet", f"Layer wurde {msg}.", duration=5)
|
||||
|
||||
elif status == LoadStatus.KEEP:
|
||||
info("Verfahrensgebiet", "Vorhandener Layer wurde behalten.", duration=4)
|
||||
|
||||
elif status == LoadStatus.NONE:
|
||||
warning("Verfahrensgebiet", "Layer konnte nicht geladen werden.", duration=None)
|
||||
|
||||
finally:
|
||||
self._set_busy(self.btn_verf_alkis, False)
|
||||
|
||||
def _set_busy(self, button: QPushButton, busy: bool):
|
||||
button.setEnabled(not busy)
|
||||
if busy:
|
||||
button.setText("Lade ...")
|
||||
else:
|
||||
button.setText("Aus ALKIS laden")
|
||||
|
||||
def _haken_label(self) -> QLabel:
|
||||
label = QLabel()
|
||||
label.setFixedSize(20, 20)
|
||||
label.setAlignment(Qt.AlignmentFlag.AlignCenter)
|
||||
label.setText("") # Start ohne Haken
|
||||
return label
|
||||
|
||||
def setze_haken(self, label: QLabel, aktiv: bool):
|
||||
"""Zeigt oder entfernt den grünen Haken (Unicode), ohne Theme-Icons."""
|
||||
if aktiv:
|
||||
label.setText("✓")
|
||||
label.setStyleSheet("color: #2ea043; font-weight: bold;")
|
||||
else:
|
||||
label.setText("")
|
||||
label.setStyleSheet("")
|
||||
|
||||
Reference in New Issue
Block a user