1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238
|
"""
@package iclass.digit
@brief wxIClass digitizer classes
Classes:
- digit::IClassVDigit
- digit::IClassVDigitWindow
(C) 2006-2012 by the GRASS Development Team
This program is free software under the GNU General Public
License (>=v2). Read the file COPYING that comes with GRASS
for details.
@author Vaclav Petras <wenzeslaus gmail.com>
@author Anna Kratochvilova <kratochanna gmail.com>
"""
import wx
from vdigit.mapwindow import VDigitWindow
from vdigit.wxdigit import IVDigit
from vdigit.wxdisplay import DisplayDriver, TYPE_AREA
from core.gcmd import GWarning
try:
from grass.lib.gis import G_verbose, G_set_verbose
from grass.lib.vector import *
from grass.lib.vedit import *
except ImportError:
pass
import grass.script as grass
class IClassVDigitWindow(VDigitWindow):
"""Class similar to VDigitWindow but specialized for wxIClass."""
def __init__(self, parent, giface, map, properties):
"""
@a parent should has toolbar providing current class (category).
:param parent: gui parent
:param map: map renderer instance
"""
VDigitWindow.__init__(
self, parent=parent, giface=giface, Map=map, properties=properties
)
def _onLeftDown(self, event):
action = self.toolbar.GetAction()
if not action:
return
region = grass.region()
e, n = self.Pixel2Cell(event.GetPosition())
if not (
(region["s"] <= n <= region["n"]) and (region["w"] <= e <= region["e"])
):
GWarning(
parent=self.parent,
message=_(
"You are trying to create a training area "
"outside the computational region. "
"Please, use g.region to set the appropriate region first."
),
)
return
cat = self.GetCurrentCategory()
if cat is None and action == "addLine":
dlg = wx.MessageDialog(
parent=self.parent,
message=_(
"In order to create a training area, "
"you have to select class first.\n\n"
"There is no class yet, "
"do you want to create one?"
),
caption=_("No class selected"),
style=wx.YES_NO,
)
if dlg.ShowModal() == wx.ID_YES:
self.parent.OnCategoryManager(None)
dlg.Destroy()
event.Skip()
return
super()._onLeftDown(event)
def _addRecord(self):
return False
def _updateATM(self):
pass
def _onRightUp(self, event):
super()._onRightUp(event)
self.parent.UpdateChangeState(changes=True)
def GetCurrentCategory(self):
"""Returns current category (class).
Category should be assigned to new features (areas).
It is taken from parent's toolbar.
"""
return self.parent.GetToolbar("iClass").GetSelectedCategoryIdx()
def GetCategoryColor(self, cat):
"""Get color associated with given category"""
r, g, b = [int(x) for x in self.parent.GetClassColor(cat).split(":")][:3]
return wx.Colour(r, g, b)
class IClassVDigit(IVDigit):
"""Class similar to IVDigit but specialized for wxIClass."""
def __init__(self, giface, mapwindow):
IVDigit.__init__(self, giface, mapwindow, driver=IClassDisplayDriver)
self._settings["closeBoundary"] = True # snap to the first node
def _getNewFeaturesLayer(self):
return 1
def _getNewFeaturesCat(self):
cat = self.mapWindow.GetCurrentCategory()
return cat
def DeleteAreasByCat(self, cats):
"""Delete areas (centroid+boundaries) by categories
:param cats: list of categories
"""
for cat in cats:
Vedit_delete_areas_cat(self.poMapInfo, 1, cat)
def CopyMap(self, name, tmp=False, update=False):
"""Make a copy of open vector map
Note: Attributes are not copied
:param name: name for a copy
:param tmp: True for temporary map
:param bool update: True if copy target vector map (poMapInfoNew)
exist
:return: number of copied features
:return: -1 on error
"""
if not self.poMapInfo:
# nothing to copy
return -1
poMapInfoNew = pointer(Map_info())
if not tmp:
if update:
open_fn = Vect_open_update
else:
open_fn = Vect_open_new
else:
if update:
open_fn = Vect_open_tmp_update
else:
open_fn = Vect_open_tmp_new
if update:
if open_fn(poMapInfoNew, name, "") == -1:
return -1
else:
is3D = bool(Vect_is_3d(self.poMapInfo))
if open_fn(poMapInfoNew, name, is3D) == -1:
return -1
verbose = G_verbose()
G_set_verbose(-1) # be silent
if Vect_copy_map_lines(self.poMapInfo, poMapInfoNew) == 1:
G_set_verbose(verbose)
return -1
Vect_build(poMapInfoNew)
G_set_verbose(verbose)
ret = Vect_get_num_lines(poMapInfoNew)
Vect_close(poMapInfoNew)
return ret
def GetMapInfo(self):
"""Returns Map_info() struct of open vector map"""
return self.poMapInfo
class IClassDisplayDriver(DisplayDriver):
"""Class similar to DisplayDriver but specialized for wxIClass
.. todo::
needs refactoring (glog, gprogress)
"""
def __init__(self, device, deviceTmp, mapObj, window, glog, gprogress):
DisplayDriver.__init__(self, device, deviceTmp, mapObj, window, glog, gprogress)
self._cat = -1
def _drawObject(self, robj):
"""Draw given object to the device
:param robj: object to draw
"""
if robj.type == TYPE_AREA:
self._cat = Vect_get_area_cat(self.poMapInfo, robj.fid, 1)
elif robj.type == TYPE_CENTROIDIN:
return # skip centroids
DisplayDriver._drawObject(self, robj)
def _definePen(self, rtype):
"""Define pen/brush based on rendered object)
:param rtype: type of the object
:return: pen, brush
"""
pen, brush = DisplayDriver._definePen(self, rtype)
if self._cat > 0 and rtype == TYPE_AREA:
brush = wx.Brush(self.window.GetCategoryColor(self._cat), wx.SOLID)
return pen, brush
def CloseMap(self):
"""Close training areas map - be quiet"""
verbosity = G_verbose()
G_set_verbose(0)
DisplayDriver.CloseMap(self)
G_set_verbose(verbosity)
|