QGIS-Plugin VLN Karten: Verfahrens-Layer der Karten-API laden und hochladen
Lädt Verfahrensgebiet, Plan 41, Karte alter Stand (KAS) und Wertermittlung (WE) je VKZ vollständig aus KARTE_OBJEKT (Listen-Endpunkt mit Paging) und schreibt sie per PUT zurück. Einmaliger Login (mail/password -> userauth), API-Key persistiert in QSettings; Verfahrens-Auswahl in der Toolbar wird je QGIS-Projekt gemerkt. Gemischte Geometrietypen werden beim Laden in Punkte-/Linien-/Flächen-Layer gesplittet und beim Hochladen wieder vereint. Qt5/Qt6-kompatibel (QGIS 3.22+ und QGIS 4). Co-Authored-By: Claude Fable 5 <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,3 @@
|
||||
__pycache__/
|
||||
*.pyc
|
||||
.DS_Store
|
||||
@@ -1,2 +1,145 @@
|
||||
# API_Karte_QGISDemo
|
||||
# VLN Karten — QGIS-Plugin für die Karten-API des VLN Managers
|
||||
|
||||
QGIS-Plugin, das die Karten-Layer des [VLN Managers](https://www.vlnsachsen.de)
|
||||
direkt in QGIS bearbeitbar macht. Es liest und schreibt dieselben
|
||||
Datenbank-Daten (`KARTE_OBJEKT`), die auch das Web-GIS
|
||||
(karte.flurneuordnung-sachsen.de) über die `/maps/*`-Endpunkte nutzt.
|
||||
|
||||
## Funktionen
|
||||
|
||||
- **Einmaliger Login** mit dem VLN-Manager-Konto (E-Mail/Passwort).
|
||||
Der zurückgelieferte API-Key wird in den QGIS-Einstellungen gespeichert
|
||||
und bei jedem QGIS-Start automatisch wiederverwendet; erst bei
|
||||
HTTP 401/403 erscheint der Anmeldedialog erneut.
|
||||
- **Verfahrens-Auswahl in der Toolbar** — alle Teilnehmergemeinschaften
|
||||
aus `GET /tgen` als „VKZ — Name“. Die Auswahl wird in der Projektdatei
|
||||
gespeichert und beim Öffnen des Projekts wiederhergestellt.
|
||||
- **Vier Layer je Verfahren laden:**
|
||||
|
||||
| Button | API-Layer | DB-Art |
|
||||
|---|---|---|
|
||||
| Verfahrensgebiet | `umringe` | `UMRING` |
|
||||
| Plan 41 (Wege- und Gewässerplan) | `p41` | `P41` |
|
||||
| Karte alter Stand | `kas` | `KAS` |
|
||||
| Wertermittlung | `we` | `WE` |
|
||||
|
||||
Geladen wird **vollständig** über den Listen-Endpunkt mit
|
||||
limit/offset-Paging — auch bei mehr als 2000 Objekten.
|
||||
- **Gemischte Geometrietypen:** Die API liefert Punkte, Linien und
|
||||
Flächen in einer FeatureCollection. Beim Laden wird nach
|
||||
Geometrie-Familie in bis zu drei Memory-Layer gesplittet
|
||||
(Single-Typen werden zu Multi-Typen befördert); bei nur einem Typ
|
||||
entsteht ein einzelner Layer.
|
||||
- **Hochladen:** „Aktiven Layer hochladen“ vereint alle Teil-Layer
|
||||
desselben Datensatzes (Punkte + Linien + Flächen derselben VKZ) wieder
|
||||
zu einer FeatureCollection und schreibt sie per `PUT` zurück.
|
||||
Vorher: Commit offener Bearbeitungssitzungen und Sicherheitsabfrage
|
||||
mit Auflistung der beteiligten Layer.
|
||||
- **Schutz vor Duplikaten:** Erneutes Laden eines bereits geladenen
|
||||
Datensatzes ersetzt die vorhandenen Layer nach Rückfrage, statt sie zu
|
||||
stapeln. Liefert der Server 0 Objekte, wird das deutlich gemeldet und
|
||||
ein leerer Layer zum Digitalisieren angelegt.
|
||||
|
||||
## Voraussetzungen
|
||||
|
||||
- QGIS **3.22 oder neuer**, einschließlich **QGIS 4** (Qt6/PyQt6 —
|
||||
der Code verwendet durchgehend scoped Enums und `exec()`).
|
||||
- Ein VLN-Manager-Konto mit Zugriff auf die Karten-API.
|
||||
|
||||
## Installation
|
||||
|
||||
Repository klonen und den Plugin-Ordner in das QGIS-Profil verlinken
|
||||
(alternativ kopieren), dann QGIS neu starten:
|
||||
|
||||
```bash
|
||||
git clone https://entwicklung.flurneuordnung-sachsen.de/VLN_SN/API_Karte_QGISDemo.git
|
||||
cd API_Karte_QGISDemo
|
||||
|
||||
# macOS, QGIS 4 (bei QGIS 3: "QGIS4" durch "QGIS3" ersetzen):
|
||||
ln -s "$PWD/vln_karten" \
|
||||
~/Library/Application\ Support/QGIS/QGIS4/profiles/default/python/plugins/vln_karten
|
||||
|
||||
# Linux:
|
||||
# ~/.local/share/QGIS/QGIS4/profiles/default/python/plugins/
|
||||
# Windows:
|
||||
# %APPDATA%\QGIS\QGIS4\profiles\default\python\plugins\
|
||||
```
|
||||
|
||||
Anschließend in QGIS unter *Erweiterungen → Erweiterungen verwalten*
|
||||
„VLN Karten“ aktivieren (experimentelle Erweiterungen zulassen).
|
||||
|
||||
## Bedienung
|
||||
|
||||
1. **Anmelden …** in der Toolbar „VLN Karten“ (auch unter
|
||||
*Web → VLN Karten*): E-Mail und Passwort des VLN-Manager-Kontos.
|
||||
Das Passwort wird nicht gespeichert, nur der API-Key.
|
||||
2. **Verfahren wählen** in der Auswahlliste.
|
||||
3. **Layer laden**, in QGIS normal editieren (Memory-Layer,
|
||||
EPSG:25833), **hochladen**.
|
||||
|
||||
> ⚠️ Der `PUT` der API ersetzt den kompletten Layer-Bestand der
|
||||
> jeweiligen VKZ (versionierter Snapshot in `KARTE_SPEICHERSTAND`).
|
||||
> Deshalb lädt das Plugin immer alle Teil-Layer gemeinsam hoch — nur
|
||||
> einen Teil zu senden, würde die übrigen Geometrietypen serverseitig
|
||||
> löschen.
|
||||
|
||||
## API-Vertrag
|
||||
|
||||
Basis-URL: `https://api.flurneuordnung-sachsen.de/v2`
|
||||
(fest hinterlegt als `DEFAULT_BASE_URL` in
|
||||
[vln_karten/api_client.py](vln_karten/api_client.py))
|
||||
|
||||
```
|
||||
POST /person/login
|
||||
Body: {"mail": "...", "password": "..."}
|
||||
Antwort: {"data": {"userauth": "<url_token>", "id": ...}}
|
||||
userauth = API-Key, danach Header "X-API-Key: <userauth>"
|
||||
|
||||
GET /tgen Verfahren/TGs (Auth nötig)
|
||||
|
||||
GET /maps/<layer>?vkz=&limit=&offset= Layer lesen (kas/we: Auth nötig)
|
||||
Antwort: GeoJSON FeatureCollection in EPSG:25833 (ETRS89/UTM33)
|
||||
|
||||
PUT /maps/<layer>/{vkz} Layer schreiben (Auth nötig)
|
||||
Body: GeoJSON (ersetzt den Layer dieser VKZ, neuer Snapshot)
|
||||
Antwort: {"data": {"vkz","layer","art","speicher_id"}, "status":"ok"}
|
||||
|
||||
Fehlerformat: RFC 7807 (application/problem+json)
|
||||
```
|
||||
|
||||
Ein weiterer Layer der API (`st` = Servicetermin) wäre ein zusätzlicher
|
||||
Eintrag im `DATASETS`-Dict in [vln_karten/plugin.py](vln_karten/plugin.py).
|
||||
|
||||
## Aufbau
|
||||
|
||||
| Datei | Zweck |
|
||||
|---|---|
|
||||
| [vln_karten/plugin.py](vln_karten/plugin.py) | Toolbar, Aktionen, Verfahrens-Auswahl, `DATASETS`-Registry |
|
||||
| [vln_karten/api_client.py](vln_karten/api_client.py) | HTTP-Client (Login, Paging-Loader, PUT) über `QgsBlockingNetworkRequest` |
|
||||
| [vln_karten/layer_manager.py](vln_karten/layer_manager.py) | GeoJSON ↔ Memory-Layer, Geometrie-Splitting, Layer-Zusammenführung |
|
||||
| [vln_karten/login_dialog.py](vln_karten/login_dialog.py) | Anmeldedialog |
|
||||
| [vln_karten/metadata.txt](vln_karten/metadata.txt) | QGIS-Plugin-Metadaten |
|
||||
|
||||
## Technische Hinweise
|
||||
|
||||
- **CRS:** Die API liefert und erwartet Koordinaten in **EPSG:25833**
|
||||
(abweichend von RFC 7946). Import/Export laufen ohne
|
||||
WGS84-Transformation (`GEOJSON_CRS` in `layer_manager.py`).
|
||||
- **Geometrien** gehen beim Hochladen als Multi-Typen an die API
|
||||
(`ST_GeomFromGeoJSON` akzeptiert beides). Features ohne Geometrie
|
||||
landen in einem Tabellen-Layer „ohne Geometrie“.
|
||||
- **API-Key-Ablage:** unverschlüsselt in den QGIS-Einstellungen
|
||||
(QSettings, Gruppe `vln_karten`). Wer das härten will, verlagert ihn
|
||||
in den QGIS-Authentifizierungsmanager (`QgsApplication.authManager()`).
|
||||
- **Requests** laufen synchron (blockierend) — für sehr große
|
||||
Datensätze wäre `QgsNetworkAccessManager` mit Tasks der nächste Schritt.
|
||||
- Die gewählte VKZ liegt in der Projektdatei (`writeEntry`-Scope
|
||||
`vln_karten`) — verschiedene Projekte können verschiedene Verfahren
|
||||
vorausgewählt haben.
|
||||
|
||||
## Entwicklung
|
||||
|
||||
Für schnelles Iterieren empfiehlt sich das Plugin **Plugin Reloader**
|
||||
aus dem offiziellen QGIS-Repository — Codeänderungen wirken dann ohne
|
||||
QGIS-Neustart. Der Plugin-Ordner kann dafür per Symlink direkt aus dem
|
||||
Git-Arbeitsverzeichnis eingebunden bleiben (siehe Installation).
|
||||
|
||||
@@ -0,0 +1,3 @@
|
||||
def classFactory(iface):
|
||||
from .plugin import VlnKartenPlugin
|
||||
return VlnKartenPlugin(iface)
|
||||
@@ -0,0 +1,255 @@
|
||||
"""HTTP-Client für die VLN-Manager-API (Karten-Plugin map.php + Stammdaten).
|
||||
|
||||
API-Vertrag laut Doku im Wissensdatenbank-Vault
|
||||
("API.md", "API – Karten (map.php).md", "API – Schnittstellen (Referenz).md",
|
||||
Stand 2026-06-08):
|
||||
|
||||
Basis-URL: https://api.flurneuordnung-sachsen.de/v2
|
||||
|
||||
POST /person/login
|
||||
Body: {"mail": "...", "password": "..."}
|
||||
Antwort: {"data": {"userauth": "<url_token>", "id": ...}}
|
||||
Das userauth-Token ist der API-Key (= PERSON.url_token).
|
||||
|
||||
GET /tgen -> Teilnehmergemeinschaften (Verfahren)
|
||||
|
||||
GET /maps/<layer>/{vkz} -> GeoJSON FeatureCollection (EPSG:25833)
|
||||
PUT /maps/<layer>/{vkz} <- GeoJSON (ersetzt den Layer dieser VKZ)
|
||||
Antwort: {"data": {"vkz", "layer", "art", "speicher_id"}, "status": "ok"}
|
||||
|
||||
Layer: umringe, p41, st, kas, we
|
||||
Auth nach Login über den Header "X-API-Key".
|
||||
Fehlerformat: RFC 7807 (application/problem+json).
|
||||
|
||||
Es wird QgsBlockingNetworkRequest verwendet, damit Proxy- und
|
||||
Zertifikatseinstellungen aus QGIS automatisch greifen.
|
||||
"""
|
||||
|
||||
import json
|
||||
|
||||
from qgis.PyQt.QtCore import QUrl
|
||||
from qgis.PyQt.QtNetwork import QNetworkRequest
|
||||
from qgis.core import QgsBlockingNetworkRequest
|
||||
|
||||
# Fest verdrahtete Produktiv-API des VLN Managers.
|
||||
DEFAULT_BASE_URL = "https://api.flurneuordnung-sachsen.de/v2"
|
||||
|
||||
|
||||
class ApiError(Exception):
|
||||
"""Fehler bei der Kommunikation mit der Karten-API."""
|
||||
|
||||
|
||||
class AuthError(ApiError):
|
||||
"""API-Key fehlt, ist abgelaufen oder hat keine Berechtigung (401/403)."""
|
||||
|
||||
|
||||
class KartenApiClient:
|
||||
def __init__(self, base_url=DEFAULT_BASE_URL, api_key=None, mail=None):
|
||||
self.base_url = base_url.rstrip("/")
|
||||
self.api_key = api_key
|
||||
self.mail = mail
|
||||
|
||||
@property
|
||||
def is_authenticated(self):
|
||||
return bool(self.api_key)
|
||||
|
||||
def login(self, mail, password):
|
||||
"""Meldet den Benutzer an. Die API liefert das userauth-Token
|
||||
(= API-Key), das für alle weiteren Requests verwendet wird."""
|
||||
data = self._request(
|
||||
"POST",
|
||||
"/person/login",
|
||||
payload={"mail": mail, "password": password},
|
||||
with_auth=False,
|
||||
)
|
||||
payload = data.get("data") if isinstance(data, dict) else None
|
||||
if not isinstance(payload, dict):
|
||||
payload = data if isinstance(data, dict) else {}
|
||||
api_key = payload.get("userauth")
|
||||
if not api_key:
|
||||
raise ApiError(
|
||||
"Login-Antwort enthielt kein userauth-Token. "
|
||||
"Erhaltene Antwort: %s" % json.dumps(data)[:300]
|
||||
)
|
||||
self.api_key = api_key
|
||||
self.mail = mail
|
||||
return api_key
|
||||
|
||||
def logout(self):
|
||||
self.api_key = None
|
||||
self.mail = None
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# Verfahren (Teilnehmergemeinschaften)
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def get_verfahren(self):
|
||||
"""Liste der Verfahren (TGs) als [{"vkz": ..., "name": ...}, ...],
|
||||
sortiert nach VKZ. Quelle: GET /tgen."""
|
||||
data = self._request("GET", "/tgen")
|
||||
|
||||
rows = None
|
||||
if isinstance(data, list):
|
||||
rows = data
|
||||
elif isinstance(data, dict):
|
||||
for key in ("data", "rows"):
|
||||
value = data.get(key)
|
||||
if isinstance(value, list):
|
||||
rows = value
|
||||
break
|
||||
if isinstance(value, dict) and isinstance(value.get("rows"), list):
|
||||
rows = value["rows"]
|
||||
break
|
||||
if rows is None:
|
||||
raise ApiError(
|
||||
"Antwort von /tgen hat ein unerwartetes Format: %s"
|
||||
% json.dumps(data)[:200]
|
||||
)
|
||||
|
||||
verfahren = []
|
||||
for row in rows:
|
||||
if not isinstance(row, dict):
|
||||
continue
|
||||
vkz = row.get("vkz") or row.get("VKZ")
|
||||
name = (
|
||||
row.get("kurzname")
|
||||
or row.get("Kurzname")
|
||||
or row.get("name")
|
||||
or row.get("Name")
|
||||
or ""
|
||||
)
|
||||
if vkz:
|
||||
verfahren.append({"vkz": str(vkz), "name": str(name)})
|
||||
verfahren.sort(key=lambda v: v["vkz"])
|
||||
return verfahren
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# Karten-Layer (die konkreten Pfade stehen in plugin.DATASETS)
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def load_layer_complete(self, layer_key, vkz, page_size=2000):
|
||||
"""Lädt ALLE Objekte eines Layers für eine VKZ aus KARTE_OBJEKT.
|
||||
|
||||
Nutzt wie das Web-GIS (0_map, db_p41) den Listen-Endpunkt
|
||||
/maps/<layer>?vkz=…, aber mit limit/offset-Paging, bis keine
|
||||
weitere Seite mehr kommt — so ist der Abruf auch bei mehr als
|
||||
2000 Objekten garantiert vollständig."""
|
||||
from urllib.parse import quote
|
||||
|
||||
features = []
|
||||
offset = 0
|
||||
while True:
|
||||
data = self._request(
|
||||
"GET",
|
||||
"/maps/%s?vkz=%s&limit=%d&offset=%d"
|
||||
% (quote(str(layer_key)), quote(str(vkz)), page_size, offset),
|
||||
)
|
||||
if not isinstance(data, dict) or data.get("type") != "FeatureCollection":
|
||||
raise ApiError(
|
||||
"Antwort von /maps/%s ist keine GeoJSON FeatureCollection."
|
||||
% layer_key
|
||||
)
|
||||
page = data.get("features", [])
|
||||
features.extend(page)
|
||||
if len(page) < page_size:
|
||||
break
|
||||
offset += page_size
|
||||
return {"type": "FeatureCollection", "features": features}
|
||||
|
||||
def load_feature_collection(self, path):
|
||||
"""Lädt einen Layer als GeoJSON FeatureCollection (dict)."""
|
||||
data = self._request("GET", path)
|
||||
# Die Karten-Endpunkte liefern die FeatureCollection direkt,
|
||||
# andere Routen verpacken sie in {"data": ...}.
|
||||
if isinstance(data, dict):
|
||||
if data.get("type") == "FeatureCollection":
|
||||
return data
|
||||
inner = data.get("data")
|
||||
if isinstance(inner, dict) and inner.get("type") == "FeatureCollection":
|
||||
return inner
|
||||
raise ApiError(
|
||||
"Antwort von %s ist keine GeoJSON FeatureCollection." % path
|
||||
)
|
||||
|
||||
def save_feature_collection(self, path, feature_collection):
|
||||
"""Schreibt eine GeoJSON FeatureCollection zurück an die API
|
||||
(PUT ersetzt den kompletten Layer der jeweiligen VKZ)."""
|
||||
return self._request("PUT", path, payload=feature_collection)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# Intern
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def _request(self, method, path, payload=None, with_auth=True):
|
||||
if with_auth and not self.is_authenticated:
|
||||
raise AuthError("Nicht angemeldet — bitte zuerst einloggen.")
|
||||
|
||||
request = QNetworkRequest(QUrl(self.base_url + path))
|
||||
request.setHeader(
|
||||
QNetworkRequest.KnownHeaders.ContentTypeHeader, "application/json"
|
||||
)
|
||||
request.setRawHeader(b"Accept", b"application/json")
|
||||
if with_auth:
|
||||
request.setRawHeader(b"X-API-Key", self.api_key.encode("utf-8"))
|
||||
|
||||
body = b""
|
||||
if payload is not None:
|
||||
body = json.dumps(payload).encode("utf-8")
|
||||
|
||||
blocking = QgsBlockingNetworkRequest()
|
||||
if method == "GET":
|
||||
error = blocking.get(request)
|
||||
elif method == "POST":
|
||||
error = blocking.post(request, body)
|
||||
elif method == "PUT":
|
||||
error = blocking.put(request, body)
|
||||
else:
|
||||
raise ApiError("Nicht unterstützte HTTP-Methode: %s" % method)
|
||||
|
||||
reply = blocking.reply()
|
||||
content = bytes(reply.content())
|
||||
status = reply.attribute(
|
||||
QNetworkRequest.Attribute.HttpStatusCodeAttribute
|
||||
)
|
||||
status = int(status) if status is not None else None
|
||||
|
||||
if status in (401, 403):
|
||||
raise AuthError(
|
||||
"Keine Berechtigung für %s %s (HTTP %s)%s"
|
||||
% (method, path, status, self._error_detail(content))
|
||||
)
|
||||
if error != QgsBlockingNetworkRequest.NoError:
|
||||
raise ApiError(
|
||||
"%s %s fehlgeschlagen: %s%s"
|
||||
% (method, path, blocking.errorMessage(), self._error_detail(content))
|
||||
)
|
||||
if status is not None and status >= 400:
|
||||
raise ApiError(
|
||||
"%s %s lieferte HTTP %s%s"
|
||||
% (method, path, status, self._error_detail(content))
|
||||
)
|
||||
|
||||
if not content:
|
||||
return None
|
||||
try:
|
||||
return json.loads(content.decode("utf-8"))
|
||||
except ValueError:
|
||||
raise ApiError(
|
||||
"Antwort von %s ist kein gültiges JSON." % path
|
||||
)
|
||||
|
||||
@staticmethod
|
||||
def _error_detail(content):
|
||||
"""Liest title/detail aus einer RFC-7807-Fehlerantwort."""
|
||||
if not content:
|
||||
return ""
|
||||
try:
|
||||
problem = json.loads(content.decode("utf-8"))
|
||||
parts = [
|
||||
str(problem[k]) for k in ("title", "detail") if problem.get(k)
|
||||
]
|
||||
if parts:
|
||||
return " — " + ": ".join(parts)
|
||||
except (ValueError, AttributeError):
|
||||
pass
|
||||
return " — " + content[:300].decode("utf-8", "replace")
|
||||
@@ -0,0 +1,178 @@
|
||||
"""Konvertierung GeoJSON <-> QGIS-Layer.
|
||||
|
||||
Die API liefert je Layer (z.B. Plan 41) gemischte Geometrietypen in einer
|
||||
FeatureCollection: Punkte, Linien, Polygone sowie deren Multi-Varianten.
|
||||
QGIS-Memory-Layer können nur einen Geometrietyp halten — beim Laden wird
|
||||
deshalb nach Geometrie-Familie (Punkte/Linien/Flächen) in bis zu drei
|
||||
Layer gesplittet; Single-Geometrien werden zu Multi befördert.
|
||||
|
||||
Beim Hochladen müssen alle Teil-Layer desselben Datensatzes wieder zu
|
||||
EINER FeatureCollection vereint werden, weil der PUT der API den
|
||||
kompletten Layer-Bestand der VKZ ersetzt (siehe layers_to_feature_collection
|
||||
und plugin_layers).
|
||||
|
||||
Welcher Datensatz/welche VKZ zu einem Layer gehört, steht als
|
||||
Custom-Property am Layer.
|
||||
"""
|
||||
|
||||
import json
|
||||
|
||||
from qgis.core import (
|
||||
QgsJsonExporter,
|
||||
QgsJsonUtils,
|
||||
QgsProject,
|
||||
QgsVectorLayer,
|
||||
)
|
||||
|
||||
try: # QGIS >= 3.30 und QGIS 4
|
||||
from qgis.core import Qgis
|
||||
_GEOM_POINT = Qgis.GeometryType.Point
|
||||
_GEOM_LINE = Qgis.GeometryType.Line
|
||||
_GEOM_POLYGON = Qgis.GeometryType.Polygon
|
||||
except (ImportError, AttributeError): # ältere QGIS-3-Versionen
|
||||
from qgis.core import QgsWkbTypes
|
||||
_GEOM_POINT = QgsWkbTypes.PointGeometry
|
||||
_GEOM_LINE = QgsWkbTypes.LineGeometry
|
||||
_GEOM_POLYGON = QgsWkbTypes.PolygonGeometry
|
||||
|
||||
# Die VLN-API liefert Koordinaten unverändert aus der DB (ST_AsGeoJSON):
|
||||
# ETRS89 / UTM Zone 33 — verifiziert per GET /maps/umringe (Werte wie
|
||||
# [406924, 5658760] liegen im Raum Dresden, nicht in Grad).
|
||||
GEOJSON_CRS = "EPSG:25833"
|
||||
|
||||
PROP_DATASET = "vln_karten/dataset"
|
||||
PROP_VERFAHREN = "vln_karten/verfahren"
|
||||
PROP_PATH = "vln_karten/api_path"
|
||||
|
||||
# Geometrie-Familie -> (Memory-Provider-Typ, Namenszusatz).
|
||||
# Reihenfolge = Lade-Reihenfolge: Flächen zuerst, damit Punkte im
|
||||
# Layerbaum oben landen und nicht verdeckt werden.
|
||||
_FAMILIES = (
|
||||
("polygon", "MultiPolygon", "Flächen"),
|
||||
("line", "MultiLineString", "Linien"),
|
||||
("point", "MultiPoint", "Punkte"),
|
||||
("none", "None", "ohne Geometrie"),
|
||||
)
|
||||
|
||||
|
||||
def _family(feature):
|
||||
"""Geometrie-Familie eines Features. Unbekannte Typen (z.B.
|
||||
GeometryCollection) landen bei 'none' — Attribute bleiben erhalten,
|
||||
die Geometrie ginge beim Hochladen verloren."""
|
||||
if not feature.hasGeometry() or feature.geometry().isNull():
|
||||
return "none"
|
||||
gtype = feature.geometry().type()
|
||||
if gtype == _GEOM_POINT:
|
||||
return "point"
|
||||
if gtype == _GEOM_LINE:
|
||||
return "line"
|
||||
if gtype == _GEOM_POLYGON:
|
||||
return "polygon"
|
||||
return "none"
|
||||
|
||||
|
||||
def memory_layer_from_features(
|
||||
uri_type, name, fields, features, dataset_key, verfahren_nr, api_path, promote=True
|
||||
):
|
||||
"""Baut einen Memory-Layer aus fertigen QgsFeatures, setzt die
|
||||
Plugin-Custom-Properties und hängt ihn ins Projekt."""
|
||||
uri = uri_type if uri_type == "None" else "%s?crs=%s" % (uri_type, GEOJSON_CRS)
|
||||
layer = QgsVectorLayer(uri, name, "memory")
|
||||
if not layer.isValid():
|
||||
raise RuntimeError("Memory-Layer '%s' konnte nicht erzeugt werden." % name)
|
||||
provider = layer.dataProvider()
|
||||
provider.addAttributes(fields.toList())
|
||||
layer.updateFields()
|
||||
prepared = []
|
||||
for feature in features:
|
||||
geometry = feature.geometry()
|
||||
if promote and not geometry.isNull() and not geometry.isMultipart():
|
||||
geometry.convertToMultiType()
|
||||
feature.setGeometry(geometry)
|
||||
prepared.append(feature)
|
||||
provider.addFeatures(prepared)
|
||||
layer.updateExtents()
|
||||
layer.setCustomProperty(PROP_DATASET, dataset_key)
|
||||
layer.setCustomProperty(PROP_VERFAHREN, verfahren_nr)
|
||||
layer.setCustomProperty(PROP_PATH, api_path)
|
||||
return layer
|
||||
|
||||
|
||||
def feature_collection_to_layers(
|
||||
feature_collection, base_name, default_geometry, dataset_key, verfahren_nr, api_path
|
||||
):
|
||||
"""Erzeugt aus einer GeoJSON FeatureCollection je vorkommender
|
||||
Geometrie-Familie einen Memory-Layer und hängt sie ins Projekt.
|
||||
Liefert die Liste der erzeugten Layer."""
|
||||
text = json.dumps(feature_collection)
|
||||
fields = QgsJsonUtils.stringToFields(text)
|
||||
features = QgsJsonUtils.stringToFeatureList(text, fields)
|
||||
|
||||
groups = {}
|
||||
for feature in features:
|
||||
groups.setdefault(_family(feature), []).append(feature)
|
||||
|
||||
layers = []
|
||||
multiple = len(groups) > 1
|
||||
|
||||
def make_layer(uri_type, name, group_features, promote):
|
||||
layers.append(
|
||||
memory_layer_from_features(
|
||||
uri_type, name, fields, group_features,
|
||||
dataset_key, verfahren_nr, api_path, promote=promote,
|
||||
)
|
||||
)
|
||||
|
||||
if not groups:
|
||||
# Leerer Datensatz: ein leerer Layer im Default-Typ, damit der
|
||||
# Nutzer digitalisieren und hochladen kann.
|
||||
make_layer(default_geometry, base_name, [], promote=False)
|
||||
else:
|
||||
for family, uri_type, suffix in _FAMILIES:
|
||||
if family not in groups:
|
||||
continue
|
||||
name = "%s — %s" % (base_name, suffix) if multiple else base_name
|
||||
make_layer(uri_type, name, groups[family], promote=(family != "none"))
|
||||
|
||||
QgsProject.instance().addMapLayers(layers)
|
||||
return layers
|
||||
|
||||
|
||||
def plugin_layers(dataset_key, verfahren_nr):
|
||||
"""Alle Layer im Projekt, die zu diesem Datensatz und dieser VKZ
|
||||
gehören (z.B. die Punkte-/Linien-/Flächen-Teil-Layer von Plan 41)."""
|
||||
result = []
|
||||
for layer in QgsProject.instance().mapLayers().values():
|
||||
if (
|
||||
isinstance(layer, QgsVectorLayer)
|
||||
and layer.customProperty(PROP_DATASET) == dataset_key
|
||||
and str(layer.customProperty(PROP_VERFAHREN)) == str(verfahren_nr)
|
||||
):
|
||||
result.append(layer)
|
||||
return result
|
||||
|
||||
|
||||
def layers_to_feature_collection(layers):
|
||||
"""Vereint die Features mehrerer Layer zu EINER GeoJSON
|
||||
FeatureCollection (für den PUT, der den Serverbestand ersetzt).
|
||||
|
||||
Keine Transformation nach WGS84: Die API erwartet die Koordinaten im
|
||||
selben CRS, in dem sie sie liefert (EPSG:25833)."""
|
||||
merged = []
|
||||
for layer in layers:
|
||||
exporter = QgsJsonExporter(layer)
|
||||
exporter.setSourceCrs(layer.crs())
|
||||
exporter.setTransformGeometries(False)
|
||||
collection = json.loads(
|
||||
exporter.exportFeatures(list(layer.getFeatures()))
|
||||
)
|
||||
merged.extend(collection.get("features", []))
|
||||
return {"type": "FeatureCollection", "features": merged}
|
||||
|
||||
|
||||
def layer_api_path(layer):
|
||||
"""API-Pfad, unter dem dieser Layer geladen wurde — oder None,
|
||||
wenn der Layer nicht von diesem Plugin stammt."""
|
||||
if not isinstance(layer, QgsVectorLayer):
|
||||
return None
|
||||
return layer.customProperty(PROP_PATH) or None
|
||||
@@ -0,0 +1,59 @@
|
||||
"""Anmeldedialog: E-Mail und Passwort.
|
||||
|
||||
Die Server-URL ist fest im Client hinterlegt (api_client.DEFAULT_BASE_URL).
|
||||
Die E-Mail-Adresse wird in den QGIS-Einstellungen gemerkt, das Passwort
|
||||
bewusst nicht.
|
||||
"""
|
||||
|
||||
from qgis.PyQt.QtCore import QSettings
|
||||
from qgis.PyQt.QtWidgets import (
|
||||
QDialog,
|
||||
QDialogButtonBox,
|
||||
QFormLayout,
|
||||
QLineEdit,
|
||||
)
|
||||
|
||||
SETTINGS_GROUP = "vln_karten"
|
||||
|
||||
|
||||
class LoginDialog(QDialog):
|
||||
def __init__(self, parent=None):
|
||||
super().__init__(parent)
|
||||
self.setWindowTitle("VLN Karten — Anmeldung")
|
||||
self.setMinimumWidth(380)
|
||||
|
||||
settings = QSettings()
|
||||
settings.beginGroup(SETTINGS_GROUP)
|
||||
self.mail_edit = QLineEdit(settings.value("mail", ""))
|
||||
settings.endGroup()
|
||||
|
||||
self.mail_edit.setPlaceholderText("vorname.name@beispiel.de")
|
||||
self.password_edit = QLineEdit()
|
||||
self.password_edit.setEchoMode(QLineEdit.EchoMode.Password)
|
||||
|
||||
layout = QFormLayout(self)
|
||||
layout.addRow("E-Mail:", self.mail_edit)
|
||||
layout.addRow("Passwort:", self.password_edit)
|
||||
|
||||
buttons = QDialogButtonBox(
|
||||
QDialogButtonBox.StandardButton.Ok
|
||||
| QDialogButtonBox.StandardButton.Cancel,
|
||||
parent=self,
|
||||
)
|
||||
buttons.button(QDialogButtonBox.StandardButton.Ok).setText("Anmelden")
|
||||
buttons.accepted.connect(self.accept)
|
||||
buttons.rejected.connect(self.reject)
|
||||
layout.addRow(buttons)
|
||||
|
||||
def accept(self):
|
||||
settings = QSettings()
|
||||
settings.beginGroup(SETTINGS_GROUP)
|
||||
settings.setValue("mail", self.mail())
|
||||
settings.endGroup()
|
||||
super().accept()
|
||||
|
||||
def mail(self):
|
||||
return self.mail_edit.text().strip()
|
||||
|
||||
def password(self):
|
||||
return self.password_edit.text()
|
||||
@@ -0,0 +1,17 @@
|
||||
[general]
|
||||
name=VLN Karten
|
||||
qgisMinimumVersion=3.22
|
||||
qgisMaximumVersion=4.99
|
||||
supportsQt6=True
|
||||
description=Beispiel-Plugin: Verfahrensumringe, Wege- und Gewässerplan und Karte alter Stamm über die VLN Karten-API laden und zurückschreiben.
|
||||
about=Demonstriert Login (Benutzername/Passwort -> API-Key), Laden von GeoJSON-Daten in editierbare Memory-Layer und Zurückschreiben der Änderungen an die API.
|
||||
version=0.1.0
|
||||
author=Erik
|
||||
email=phpwelt@gmail.com
|
||||
tracker=https://entwicklung.flurneuordnung-sachsen.de/VLN_SN/API_Karte_QGISDemo/issues
|
||||
repository=https://entwicklung.flurneuordnung-sachsen.de/VLN_SN/API_Karte_QGISDemo
|
||||
tags=vln,flurbereinigung,api,geojson
|
||||
category=Web
|
||||
icon=
|
||||
experimental=True
|
||||
deprecated=False
|
||||
@@ -0,0 +1,416 @@
|
||||
"""Hauptklasse des Plugins.
|
||||
|
||||
Ablauf:
|
||||
- Der API-Key wird nach dem ersten Login in den QGIS-Einstellungen
|
||||
gespeichert und bei jedem QGIS-Start wiederverwendet (kein erneuter
|
||||
Login nötig, solange der Key gültig ist).
|
||||
- Eine Toolbar-Auswahlliste zeigt alle Verfahren (TGs) aus GET /tgen.
|
||||
Das gewählte Verfahren wird in der Projektdatei gemerkt und beim
|
||||
Öffnen des Projekts wiederhergestellt.
|
||||
- Buttons laden Verfahrensgebiet / Plan 41 / Karte alter Stand für das
|
||||
gewählte Verfahren bzw. laden den aktiven Layer zur API hoch.
|
||||
"""
|
||||
|
||||
from functools import partial
|
||||
|
||||
from qgis.PyQt.QtCore import QSettings, QTimer
|
||||
from qgis.PyQt.QtWidgets import QAction, QComboBox, QDialog, QMessageBox
|
||||
from qgis.core import Qgis, QgsProject
|
||||
|
||||
from .api_client import ApiError, AuthError, KartenApiClient
|
||||
from .layer_manager import (
|
||||
PROP_DATASET,
|
||||
PROP_VERFAHREN,
|
||||
feature_collection_to_layers,
|
||||
layer_api_path,
|
||||
layers_to_feature_collection,
|
||||
plugin_layers,
|
||||
)
|
||||
from .login_dialog import LoginDialog, SETTINGS_GROUP
|
||||
|
||||
MENU_TITLE = "&VLN Karten"
|
||||
|
||||
# Schlüssel für die VKZ-Ablage in der Projektdatei (*.qgz).
|
||||
PROJECT_SCOPE = "vln_karten"
|
||||
PROJECT_KEY_VKZ = "/vkz"
|
||||
|
||||
# Karten-Layer der VLN-Manager-API (map.php). Weitere Layer (st, we)
|
||||
# wären je ein Eintrag mehr. {vkz} = Verfahrenskennzeichen der TG.
|
||||
DATASETS = {
|
||||
"umringe": {
|
||||
"label": "Verfahrensgebiet",
|
||||
"path": "/maps/umringe/{vkz}",
|
||||
"geometry": "MultiPolygon",
|
||||
},
|
||||
"p41": {
|
||||
"label": "Plan 41 (Wege- und Gewässerplan)",
|
||||
"path": "/maps/p41/{vkz}",
|
||||
"geometry": "MultiPolygon",
|
||||
},
|
||||
# DB-Art KAS: enthält, was im Web-GIS über "in Datenbank speichern"
|
||||
# abgelegt wurde — kann je VKZ leer sein, solange dort noch nichts
|
||||
# gespeichert wurde.
|
||||
"kas": {
|
||||
"label": "Karte alter Stand (KAS)",
|
||||
"path": "/maps/kas/{vkz}",
|
||||
"geometry": "MultiPolygon",
|
||||
},
|
||||
# DB-Art WE: Wertermittlung (Web-GIS-Modul we.js).
|
||||
"we": {
|
||||
"label": "Wertermittlung (WE)",
|
||||
"path": "/maps/we/{vkz}",
|
||||
"geometry": "MultiPolygon",
|
||||
},
|
||||
}
|
||||
|
||||
|
||||
class VlnKartenPlugin:
|
||||
def __init__(self, iface):
|
||||
self.iface = iface
|
||||
self.client = KartenApiClient()
|
||||
self.actions = []
|
||||
self.toolbar = None
|
||||
self.combo = None
|
||||
# Unterdrückt das Mitschreiben in die Projektdatei, während die
|
||||
# Auswahl programmatisch gesetzt wird.
|
||||
self._restoring = False
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# Plugin-Lebenszyklus
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def initGui(self):
|
||||
self.toolbar = self.iface.addToolBar("VLN Karten")
|
||||
self.toolbar.setObjectName("VlnKartenToolbar")
|
||||
|
||||
self.login_action = self._add_action("Anmelden …", self.run_login)
|
||||
|
||||
self.combo = QComboBox()
|
||||
self.combo.setMinimumWidth(280)
|
||||
self.combo.setToolTip("Verfahren (Teilnehmergemeinschaft) auswählen")
|
||||
self.combo.currentIndexChanged.connect(self.on_verfahren_changed)
|
||||
self.toolbar.addWidget(self.combo)
|
||||
|
||||
self.dataset_actions = []
|
||||
for key, dataset in DATASETS.items():
|
||||
self.dataset_actions.append(
|
||||
self._add_action(
|
||||
"%s laden" % dataset["label"],
|
||||
partial(self.run_load_dataset, key),
|
||||
)
|
||||
)
|
||||
self.dataset_actions.append(
|
||||
self._add_action("Aktiven Layer hochladen", self.run_upload_active_layer)
|
||||
)
|
||||
|
||||
self.iface.projectRead.connect(self.restore_project_selection)
|
||||
self.iface.newProjectCreated.connect(self.restore_project_selection)
|
||||
|
||||
self._set_logged_out_ui()
|
||||
|
||||
# Gespeicherten API-Key aus früherer Sitzung wiederverwenden;
|
||||
# die Verfahrensliste erst nach dem QGIS-Start abrufen.
|
||||
settings = QSettings()
|
||||
settings.beginGroup(SETTINGS_GROUP)
|
||||
api_key = settings.value("api_key", "")
|
||||
mail = settings.value("mail", "")
|
||||
settings.endGroup()
|
||||
if api_key:
|
||||
self.client = KartenApiClient(api_key=api_key, mail=mail)
|
||||
QTimer.singleShot(0, self.populate_verfahren)
|
||||
|
||||
def unload(self):
|
||||
for signal in (self.iface.projectRead, self.iface.newProjectCreated):
|
||||
try:
|
||||
signal.disconnect(self.restore_project_selection)
|
||||
except (TypeError, RuntimeError):
|
||||
pass
|
||||
for action in self.actions:
|
||||
self.iface.removePluginWebMenu(MENU_TITLE, action)
|
||||
self.actions = []
|
||||
if self.toolbar is not None:
|
||||
self.toolbar.deleteLater()
|
||||
self.toolbar = None
|
||||
self.combo = None
|
||||
|
||||
def _add_action(self, text, callback):
|
||||
action = QAction(text, self.iface.mainWindow())
|
||||
action.triggered.connect(callback)
|
||||
self.iface.addPluginToWebMenu(MENU_TITLE, action)
|
||||
self.toolbar.addAction(action)
|
||||
self.actions.append(action)
|
||||
return action
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# Anmeldung & Verfahrensliste
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def run_login(self):
|
||||
dialog = LoginDialog(self.iface.mainWindow())
|
||||
if dialog.exec() != QDialog.DialogCode.Accepted:
|
||||
return
|
||||
|
||||
client = KartenApiClient()
|
||||
try:
|
||||
client.login(dialog.mail(), dialog.password())
|
||||
except ApiError as exc:
|
||||
self._show_error("Anmeldung fehlgeschlagen", str(exc))
|
||||
return
|
||||
|
||||
self.client = client
|
||||
settings = QSettings()
|
||||
settings.beginGroup(SETTINGS_GROUP)
|
||||
settings.setValue("api_key", client.api_key)
|
||||
settings.endGroup()
|
||||
|
||||
self.iface.messageBar().pushMessage(
|
||||
"VLN Karten",
|
||||
"Anmeldung erfolgreich — die Zugangsdaten werden für künftige "
|
||||
"QGIS-Sitzungen gemerkt.",
|
||||
level=Qgis.Success,
|
||||
duration=5,
|
||||
)
|
||||
self.populate_verfahren()
|
||||
|
||||
def populate_verfahren(self):
|
||||
"""Füllt die Auswahlliste mit allen Verfahren aus GET /tgen."""
|
||||
try:
|
||||
verfahren = self.client.get_verfahren()
|
||||
except AuthError:
|
||||
self._handle_session_expired()
|
||||
return
|
||||
except ApiError as exc:
|
||||
self.iface.messageBar().pushMessage(
|
||||
"VLN Karten",
|
||||
"Verfahrensliste konnte nicht geladen werden: %s" % exc,
|
||||
level=Qgis.Warning,
|
||||
duration=10,
|
||||
)
|
||||
return
|
||||
|
||||
self._restoring = True
|
||||
self.combo.clear()
|
||||
self.combo.addItem("— Verfahren wählen —", None)
|
||||
for v in verfahren:
|
||||
label = "%s — %s" % (v["vkz"], v["name"]) if v["name"] else v["vkz"]
|
||||
self.combo.addItem(label, v["vkz"])
|
||||
self.combo.setEnabled(True)
|
||||
self._restoring = False
|
||||
|
||||
self.login_action.setText(
|
||||
"Angemeldet: %s" % (self.client.mail or "API-Key")
|
||||
)
|
||||
for action in self.dataset_actions:
|
||||
action.setEnabled(True)
|
||||
self.restore_project_selection()
|
||||
|
||||
def _handle_session_expired(self):
|
||||
settings = QSettings()
|
||||
settings.beginGroup(SETTINGS_GROUP)
|
||||
settings.remove("api_key")
|
||||
settings.endGroup()
|
||||
self.client.logout()
|
||||
self._set_logged_out_ui()
|
||||
self._show_error(
|
||||
"Sitzung abgelaufen",
|
||||
"Der gespeicherte API-Key ist nicht mehr gültig — "
|
||||
"bitte neu anmelden.",
|
||||
)
|
||||
self.run_login()
|
||||
|
||||
def _set_logged_out_ui(self):
|
||||
self.login_action.setText("Anmelden …")
|
||||
self._restoring = True
|
||||
self.combo.clear()
|
||||
self.combo.addItem("— bitte anmelden —", None)
|
||||
self.combo.setEnabled(False)
|
||||
self._restoring = False
|
||||
for action in self.dataset_actions:
|
||||
action.setEnabled(False)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# Verfahrens-Auswahl <-> Projektdatei
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def on_verfahren_changed(self, index):
|
||||
if self._restoring or self.combo is None:
|
||||
return
|
||||
vkz = self.combo.currentData()
|
||||
if vkz:
|
||||
QgsProject.instance().writeEntry(
|
||||
PROJECT_SCOPE, PROJECT_KEY_VKZ, vkz
|
||||
)
|
||||
|
||||
def restore_project_selection(self):
|
||||
if self.combo is None or self.combo.count() == 0:
|
||||
return
|
||||
vkz, _ok = QgsProject.instance().readEntry(
|
||||
PROJECT_SCOPE, PROJECT_KEY_VKZ, ""
|
||||
)
|
||||
self._restoring = True
|
||||
index = self.combo.findData(vkz) if vkz else -1
|
||||
self.combo.setCurrentIndex(index if index >= 0 else 0)
|
||||
self._restoring = False
|
||||
|
||||
def _current_vkz(self):
|
||||
return self.combo.currentData() if self.combo is not None else None
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# Laden & Hochladen
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def run_load_dataset(self, dataset_key):
|
||||
if not self._ensure_logged_in():
|
||||
return
|
||||
vkz = self._current_vkz()
|
||||
if not vkz:
|
||||
self._show_error(
|
||||
"Kein Verfahren gewählt",
|
||||
"Bitte zuerst in der Auswahlliste ein Verfahren wählen.",
|
||||
)
|
||||
return
|
||||
|
||||
dataset = DATASETS[dataset_key]
|
||||
if not self._confirm_replace_existing(dataset_key, vkz, dataset["label"]):
|
||||
return
|
||||
|
||||
path = dataset["path"].format(vkz=vkz)
|
||||
try:
|
||||
# Vollständig aus KARTE_OBJEKT: Listen-Endpunkt mit Paging
|
||||
# (wie das Web-GIS, aber ohne 2000er-Limit).
|
||||
feature_collection = self.client.load_layer_complete(dataset_key, vkz)
|
||||
layers = feature_collection_to_layers(
|
||||
feature_collection,
|
||||
"%s %s" % (dataset["label"], vkz),
|
||||
dataset["geometry"],
|
||||
dataset_key,
|
||||
vkz,
|
||||
path,
|
||||
)
|
||||
except AuthError:
|
||||
self._handle_session_expired()
|
||||
return
|
||||
except (ApiError, RuntimeError) as exc:
|
||||
self._show_error("Laden fehlgeschlagen", str(exc))
|
||||
return
|
||||
|
||||
total = sum(layer.featureCount() for layer in layers)
|
||||
if total == 0:
|
||||
self.iface.messageBar().pushMessage(
|
||||
"VLN Karten",
|
||||
"Der Server hat für VKZ %s keine Objekte im Layer "
|
||||
"'%s' — es wurde ein leerer Layer zum Digitalisieren "
|
||||
"angelegt." % (vkz, dataset["label"]),
|
||||
level=Qgis.Warning,
|
||||
duration=10,
|
||||
)
|
||||
return
|
||||
|
||||
detail = ", ".join(
|
||||
"%s: %d" % (layer.name(), layer.featureCount()) for layer in layers
|
||||
)
|
||||
self.iface.messageBar().pushMessage(
|
||||
"VLN Karten",
|
||||
"%d Layer geladen (%s)." % (len(layers), detail),
|
||||
level=Qgis.Success,
|
||||
duration=5,
|
||||
)
|
||||
|
||||
def _confirm_replace_existing(self, dataset_key, vkz, label):
|
||||
"""Ersetzt bereits geladene Layer dieses Datensatzes nach
|
||||
Rückfrage — doppelte Layer-Sätze würden beim Hochladen alle
|
||||
Objekte dupliziert an den Server schicken."""
|
||||
existing = plugin_layers(dataset_key, vkz)
|
||||
if not existing:
|
||||
return True
|
||||
answer = QMessageBox.question(
|
||||
self.iface.mainWindow(),
|
||||
"Layer ersetzen",
|
||||
"%s für VKZ %s ist bereits im Projekt geladen (%s).\n"
|
||||
"Beim Neuladen werden diese Layer ersetzt — nicht "
|
||||
"hochgeladene Änderungen gehen verloren. Fortfahren?"
|
||||
% (label, vkz, ", ".join(layer.name() for layer in existing)),
|
||||
)
|
||||
if answer != QMessageBox.StandardButton.Yes:
|
||||
return False
|
||||
QgsProject.instance().removeMapLayers([layer.id() for layer in existing])
|
||||
return True
|
||||
|
||||
def run_upload_active_layer(self):
|
||||
if not self._ensure_logged_in():
|
||||
return
|
||||
|
||||
active = self.iface.activeLayer()
|
||||
path = layer_api_path(active) if active else None
|
||||
if not path:
|
||||
self._show_error(
|
||||
"Hochladen nicht möglich",
|
||||
"Der aktive Layer wurde nicht über dieses Plugin geladen. "
|
||||
"Bitte einen über 'VLN Karten' geladenen Layer auswählen.",
|
||||
)
|
||||
return
|
||||
|
||||
# Der PUT ersetzt den kompletten Layer-Bestand der VKZ auf dem
|
||||
# Server. Deshalb werden immer alle Teil-Layer dieses Datensatzes
|
||||
# (Punkte/Linien/Flächen) gemeinsam hochgeladen — nur den aktiven
|
||||
# zu senden, würde die anderen Geometrietypen serverseitig löschen.
|
||||
dataset_key = active.customProperty(PROP_DATASET)
|
||||
vkz = active.customProperty(PROP_VERFAHREN)
|
||||
siblings = plugin_layers(dataset_key, vkz)
|
||||
|
||||
for layer in siblings:
|
||||
if layer.isEditable() and not layer.commitChanges():
|
||||
self._show_error(
|
||||
"Hochladen nicht möglich",
|
||||
"Die offene Bearbeitungssitzung von '%s' konnte nicht "
|
||||
"gespeichert werden." % layer.name(),
|
||||
)
|
||||
return
|
||||
|
||||
dataset = DATASETS.get(dataset_key, {})
|
||||
total = sum(layer.featureCount() for layer in siblings)
|
||||
listing = "\n".join(
|
||||
"• %s (%d Objekte)" % (layer.name(), layer.featureCount())
|
||||
for layer in siblings
|
||||
)
|
||||
answer = QMessageBox.question(
|
||||
self.iface.mainWindow(),
|
||||
"Zum Server hochladen",
|
||||
"%s für VKZ %s übertragen?\n\n%s\n\n"
|
||||
"Die Layer werden zusammengeführt (%d Objekte); der bisherige "
|
||||
"Bestand dieser VKZ auf dem Server wird ersetzt."
|
||||
% (dataset.get("label", dataset_key), vkz, listing, total),
|
||||
)
|
||||
if answer != QMessageBox.StandardButton.Yes:
|
||||
return
|
||||
|
||||
try:
|
||||
feature_collection = layers_to_feature_collection(siblings)
|
||||
self.client.save_feature_collection(path, feature_collection)
|
||||
except AuthError:
|
||||
self._handle_session_expired()
|
||||
return
|
||||
except ApiError as exc:
|
||||
self._show_error("Hochladen fehlgeschlagen", str(exc))
|
||||
return
|
||||
|
||||
self.iface.messageBar().pushMessage(
|
||||
"VLN Karten",
|
||||
"%d Objekte aus %d Layer(n) erfolgreich hochgeladen."
|
||||
% (total, len(siblings)),
|
||||
level=Qgis.Success,
|
||||
duration=5,
|
||||
)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# Hilfsfunktionen
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def _ensure_logged_in(self):
|
||||
if self.client.is_authenticated:
|
||||
return True
|
||||
self.run_login()
|
||||
return self.client.is_authenticated
|
||||
|
||||
def _show_error(self, title, message):
|
||||
QMessageBox.warning(self.iface.mainWindow(), title, message)
|
||||
Reference in New Issue
Block a user