vnet_data.py 50 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561
  1. """
  2. @package vnet.vnet_data
  3. @brief Vector network analysis classes for data managment.
  4. Classes:
  5. - vnet_data::VNETData
  6. - vnet_data::VNETPointsData
  7. - vnet_data::VNETAnalysisParameters
  8. - vnet_data::VNETAnalysesProperties
  9. - vnet_data::VNETTmpVectMaps
  10. - vnet_data::VectMap
  11. - vnet_data::History
  12. - vnet_data::VNETGlobalTurnsData
  13. (C) 2013-2014 by the GRASS Development Team
  14. This program is free software under the GNU General Public License
  15. (>=v2). Read the file COPYING that comes with GRASS for details.
  16. @author Stepan Turek <stepan.turek seznam.cz> (GSoC 2012, mentor: Martin Landa)
  17. @author Lukas Bocan <silent_bob centrum.cz> (turn costs support)
  18. @author Eliska Kyzlikova <eliska.kyzlikova gmail.com> (turn costs support)
  19. """
  20. import os
  21. import types
  22. from copy import deepcopy
  23. from grass.script.utils import try_remove
  24. from grass.script import core as grass
  25. from grass.script.task import cmdlist_to_tuple
  26. import wx
  27. from core import utils
  28. from core.gcmd import RunCommand, GMessage
  29. from core.settings import UserSettings
  30. from core.utils import _
  31. from vnet.vnet_utils import ParseMapStr, SnapToNode
  32. from gui_core.gselect import VectorDBInfo
  33. from grass.pydispatch.signal import Signal
  34. from vnet.vnet_utils import DegreesToRadians, RadiansToDegrees
  35. class VNETData:
  36. def __init__(self, guiparent, mapWin):
  37. # setting initialization
  38. self._initSettings()
  39. self.guiparent = guiparent
  40. self.an_props = VNETAnalysesProperties()
  41. self.an_params = VNETAnalysisParameters(self.an_props)
  42. self.an_points = VNETPointsData(mapWin, self.an_props, self.an_params)
  43. self.global_turns = VNETGlobalTurnsData()
  44. self.pointsChanged = self.an_points.pointsChanged
  45. self.parametersChanged = self.an_params.parametersChanged
  46. def CleanUp(self):
  47. self.an_points.CleanUp()
  48. def GetAnalyses(self):
  49. return self.an_props.used_an
  50. def GetPointsData(self):
  51. return self.an_points
  52. def GetGlobalTurnsData(self):
  53. return self.global_turns
  54. def GetRelevantParams(self, analysis=None):
  55. if analysis:
  56. return self.an_props.GetRelevantParams(analysis)
  57. else:
  58. analysis, valid = self.an_params.GetParam("analysis")
  59. return self.an_props.GetRelevantParams(analysis)
  60. def GetAnalysisProperties(self, analysis=None):
  61. if analysis:
  62. return self.an_props[analysis]
  63. else:
  64. analysis, valid = self.an_params.GetParam("analysis")
  65. return self.an_props[analysis]
  66. def GetParam(self, param):
  67. return self.an_params.GetParam(param)
  68. def GetParams(self):
  69. return self.an_params.GetParams()
  70. def SetParams(self, params, flags):
  71. return self.an_params.SetParams(params, flags)
  72. def SetSnapping(self, activate):
  73. self.an_points.SetSnapping(activate)
  74. def GetSnapping(self):
  75. return self.an_points.GetSnapping()
  76. def GetLayerStyle(self):
  77. """Returns cmd for d.vect, with set style for analysis result"""
  78. analysis, valid = self.an_params.GetParam("analysis")
  79. resProps = self.an_props[analysis]["resultProps"]
  80. width = UserSettings.Get(
  81. group='vnet',
  82. key='res_style',
  83. subkey="line_width")
  84. layerStyleCmd = ["layer=1", 'width=' + str(width)]
  85. if "catColor" in resProps:
  86. layerStyleCmd.append('flags=c')
  87. elif "singleColor" in resProps:
  88. col = UserSettings.Get(
  89. group='vnet',
  90. key='res_style',
  91. subkey="line_color")
  92. layerStyleCmd.append(
  93. 'color=' + str(col[0]) + ':' + str(col[1]) + ':' + str(col[2]))
  94. layerStyleVnetColors = []
  95. if "attrColColor" in resProps:
  96. colorStyle = UserSettings.Get(
  97. group='vnet', key='res_style', subkey="color_table")
  98. invert = UserSettings.Get(
  99. group='vnet',
  100. key='res_style',
  101. subkey="invert_colors")
  102. layerStyleVnetColors = [
  103. "v.colors",
  104. "color=" + colorStyle,
  105. "column=" + resProps["attrColColor"],
  106. ]
  107. if invert:
  108. layerStyleVnetColors.append("-n")
  109. return layerStyleCmd, layerStyleVnetColors
  110. def InputsErrorMsgs(self, msg, analysis, params, flags,
  111. inv_params, relevant_params):
  112. """Checks input data in Parameters tab and shows messages if some value is not valid
  113. :param str msg: message added to start of message string
  114. :return: True if checked inputs are OK
  115. :return: False if some of checked inputs is not ok
  116. """
  117. if flags["t"] and "turn_layer" not in relevant_params:
  118. GMessage(
  119. parent=self.guiparent, message=_(
  120. "Module <%s> does not support turns costs." %
  121. analysis))
  122. return False
  123. errMapStr = ""
  124. if 'input' in inv_params:
  125. if params['input']:
  126. errMapStr = _("Vector map '%s' does not exist.") % (
  127. params['input'])
  128. else:
  129. errMapStr = _("Vector map was not chosen.")
  130. if errMapStr:
  131. GMessage(parent=self.guiparent,
  132. message=msg + "\n" + errMapStr)
  133. return False
  134. errLayerStr = ""
  135. for layer, layerLabel in {
  136. 'arc_layer': _("arc layer"),
  137. 'node_layer': _("node layer"),
  138. 'turn_layer': _("turntable layer"),
  139. 'turn_cat_layer': _("unique categories layer")}.iteritems():
  140. if layer in ["turn_layer", "turn_cat_layer"] and not flags["t"]:
  141. continue
  142. if layer in inv_params:
  143. if params[layer]:
  144. errLayerStr += _("Chosen %s '%s' does not exist in vector map '%s'.\n") % (
  145. layerLabel, params[layer], params['input'])
  146. else:
  147. errLayerStr += _("Choose existing %s.\n") % \
  148. (layerLabel)
  149. if errLayerStr:
  150. GMessage(parent=self.guiparent,
  151. message=msg + "\n" + errLayerStr)
  152. return False
  153. errColStr = ""
  154. for col in ["arc_column", "arc_backward_column", "node_column"]:
  155. if params[col] and col in inv_params and col in relevant_params:
  156. errColStr += _("Chosen column '%s' does not exist in attribute table of layer '%s' of vector map '%s'.\n") % (
  157. params[col], params[layer], params['input'])
  158. if errColStr:
  159. GMessage(parent=self.guiparent,
  160. message=msg + "\n" + errColStr)
  161. return False
  162. return True
  163. def _initSettings(self):
  164. """Initialization of settings (if not already defined)"""
  165. # initializes default settings
  166. initSettings = [
  167. ['res_style', 'line_width', 5],
  168. ['res_style', 'line_color', (192, 0, 0)],
  169. ['res_style', 'color_table', 'byr'],
  170. ['res_style', 'invert_colors', False],
  171. ['point_symbol', 'point_size', 10],
  172. ['point_symbol', 'point_width', 2],
  173. ['point_colors', "unused", (131, 139, 139)],
  174. ['point_colors', "used1cat", (192, 0, 0)],
  175. ['point_colors', "used2cat", (0, 0, 255)],
  176. ['point_colors', "selected", (9, 249, 17)],
  177. ['other', "snap_tresh", 10],
  178. ['other', "max_hist_steps", 5]
  179. ]
  180. for init in initSettings:
  181. UserSettings.ReadSettingsFile()
  182. UserSettings.Append(dict=UserSettings.userSettings,
  183. group='vnet',
  184. key=init[0],
  185. subkey=init[1],
  186. value=init[2],
  187. overwrite=False)
  188. class VNETPointsData:
  189. def __init__(self, mapWin, an_data, an_params):
  190. self.mapWin = mapWin
  191. self.an_data = an_data
  192. self.an_params = an_params
  193. # information, whether mouse event handler is registered in map window
  194. self.handlerRegistered = False
  195. self.pointsChanged = Signal('VNETPointsData.pointsChanged')
  196. self.an_params.parametersChanged.connect(self.ParametersChanged)
  197. self.snapping = False
  198. self.data = []
  199. self.cols = {"name": ['use', 'type', 'topology', 'e', 'n'],
  200. "label": [_('use'), _('type'), _('topology'), 'e', 'n'],
  201. # TDO
  202. "type": [None, ["", _("Start point"), _("End Point")], None, float, float],
  203. "def_vals": [False, 0, "new point", 0, 0]
  204. }
  205. # registration graphics for drawing
  206. self.pointsToDraw = self.mapWin.RegisterGraphicsToDraw(
  207. graphicsType="point", setStatusFunc=self.SetPointStatus)
  208. self.SetPointDrawSettings()
  209. self.AddPoint()
  210. self.AddPoint()
  211. self.SetPointData(0, {'use': True, 'type': 1})
  212. self.SetPointData(1, {'use': True, 'type': 2})
  213. self.selected = 0
  214. def __del__(self):
  215. self.CleanUp()
  216. def CleanUp(self):
  217. self.mapWin.UnregisterGraphicsToDraw(self.pointsToDraw)
  218. if self.handlerRegistered:
  219. self.mapWin.UnregisterMouseEventHandler(wx.EVT_LEFT_DOWN,
  220. self.OnMapClickHandler)
  221. def SetSnapping(self, activate):
  222. self.snapping = activate
  223. def GetSnapping(self):
  224. return self.snapping
  225. def AddPoint(self):
  226. self.pointsToDraw.AddItem(
  227. coords=(
  228. self.cols["def_vals"][3],
  229. self.cols["def_vals"][4]))
  230. self.data.append(self.cols["def_vals"][:])
  231. self.pointsChanged.emit(method="AddPoint", kwargs={})
  232. def DeletePoint(self, pt_id):
  233. item = self.pointsToDraw.GetItem(pt_id)
  234. if item:
  235. self.pointsToDraw.DeleteItem(item)
  236. self.data.pop(pt_id)
  237. self.pointsChanged.emit(method="DeletePoint", kwargs={"pt_id": pt_id})
  238. def SetPoints(self, pts_data):
  239. for item in self.pointsToDraw.GetAllItems():
  240. self.pointsToDraw.DeleteItem(item)
  241. self.data = []
  242. for pt_data in pts_data:
  243. pt_data_list = self._ptDataToList(pt_data)
  244. self.data.append(pt_data_list)
  245. self.pointsToDraw.AddItem(
  246. coords=(pt_data_list[3], pt_data_list[4]))
  247. self.pointsChanged.emit(
  248. method="SetPoints", kwargs={
  249. "pts_data": pts_data})
  250. def SetPointData(self, pt_id, data):
  251. for col, v in data.iteritems():
  252. if col == 'use':
  253. continue
  254. idx = self.cols["name"].index(col)
  255. self.data[pt_id][idx] = v
  256. # if type is changed checked columns must be recalculated by _usePoint
  257. if 'type' in data and 'use' not in data:
  258. data["use"] = self.GetPointData(pt_id)['use']
  259. if 'use' in data:
  260. if self._usePoint(pt_id, data["use"]) == -1:
  261. data["use"] = False
  262. idx = self.cols["name"].index("use")
  263. self.data[pt_id][idx] = data["use"]
  264. self.pointsChanged.emit(
  265. method="SetPointData", kwargs={
  266. "pt_id": pt_id, "data": data})
  267. def GetPointData(self, pt_id):
  268. return self._ptListDataToPtData(self.data[pt_id])
  269. def GetPointsCount(self):
  270. return len(self.data)
  271. def SetPointStatus(self, item, itemIndex):
  272. """Before point is drawn, decides properties of drawing style"""
  273. analysis, valid = self.an_params.GetParam("analysis")
  274. cats = self.an_data[analysis]["cmdParams"]["cats"]
  275. if itemIndex == self.selected:
  276. wxPen = "selected"
  277. elif not self.data[itemIndex][0]:
  278. wxPen = "unused"
  279. item.hide = False
  280. elif len(cats) > 1:
  281. idx = self.data[itemIndex][1]
  282. if idx == 2: # End/To/Sink point
  283. wxPen = "used2cat"
  284. else:
  285. wxPen = "used1cat"
  286. else:
  287. wxPen = "used1cat"
  288. item.SetPropertyVal('label', str(itemIndex + 1))
  289. item.SetPropertyVal('penName', wxPen)
  290. def SetSelected(self, pt_id):
  291. self.selected = pt_id
  292. self.pointsChanged.emit(method="SetSelected", kwargs={"pt_id": pt_id})
  293. def GetSelected(self):
  294. return self.selected
  295. def SetPointDrawSettings(self):
  296. """Set settings for drawing of points"""
  297. ptSize = int(
  298. UserSettings.Get(
  299. group='vnet',
  300. key='point_symbol',
  301. subkey='point_size'))
  302. self.pointsToDraw.SetPropertyVal("size", ptSize)
  303. colors = UserSettings.Get(group='vnet', key='point_colors')
  304. ptWidth = int(
  305. UserSettings.Get(
  306. group='vnet',
  307. key='point_symbol',
  308. subkey='point_width'))
  309. textProp = self.pointsToDraw.GetPropertyVal("text")
  310. textProp["font"].SetPointSize(ptSize + 2)
  311. for colKey, col in colors.iteritems():
  312. pen = self.pointsToDraw.GetPen(colKey)
  313. if pen:
  314. pen.SetColour(wx.Colour(col[0], col[1], col[2], 255))
  315. pen.SetWidth(ptWidth)
  316. else:
  317. self.pointsToDraw.AddPen(
  318. colKey,
  319. wx.Pen(
  320. colour=wx.Colour(
  321. col[0],
  322. col[1],
  323. col[2],
  324. 255),
  325. width=ptWidth))
  326. def ParametersChanged(self, method, kwargs):
  327. if "analysis" in kwargs["changed_params"].keys():
  328. self._updateTypeCol()
  329. if self.an_params.GetParam("analysis")[0] == "v.net.path":
  330. self._vnetPathUpdateUsePoints(None)
  331. def _updateTypeCol(self):
  332. """Rename category values when module is changed. Expample: Start point -> Sink point"""
  333. colValues = [""]
  334. analysis, valid = self.an_params.GetParam("analysis")
  335. anParamsCats = self.an_data[analysis]["cmdParams"]["cats"]
  336. for ptCat in anParamsCats:
  337. colValues.append(ptCat[1])
  338. type_idx = self.cols["name"].index("type")
  339. self.cols['type'][type_idx] = colValues
  340. def _ptDataToList(self, pt_data):
  341. pt_list_data = [None] * len(self.cols['name'])
  342. for k, val in pt_data.iteritems():
  343. pt_list_data[self.cols["name"].index(k)] = val
  344. return pt_list_data
  345. def _ptListDataToPtData(self, pt_list_data):
  346. pt_data = {}
  347. for i, val in enumerate(pt_list_data):
  348. pt_data[self.cols["name"][i]] = val
  349. return pt_data
  350. def _usePoint(self, pt_id, use):
  351. """Item is checked/unchecked"""
  352. analysis, valid = self.an_params.GetParam("analysis")
  353. cats = self.an_data[analysis]["cmdParams"]["cats"]
  354. # TODO move
  355. # if self.updateMap:
  356. # up_map_evt = gUpdateMap(render = False, renderVector = False)
  357. # wx.PostEvent(self.dialog.mapWin, up_map_evt)
  358. if len(cats) <= 1:
  359. return 0
  360. use_idx = self.cols["name"].index("use")
  361. checkedVal = self.data[pt_id][1]
  362. # point without given type cannot be selected
  363. if checkedVal == 0:
  364. self.data[pt_id][use_idx] = False
  365. self.pointsChanged.emit(
  366. method="SetPointData", kwargs={
  367. "pt_id": pt_id, "data": {
  368. "use": False}})
  369. return -1
  370. if analysis == "v.net.path" and use:
  371. self._vnetPathUpdateUsePoints(pt_id)
  372. def _vnetPathUpdateUsePoints(self, checked_pt_id):
  373. alreadyChecked = []
  374. type_idx = self.cols["name"].index("type")
  375. use_idx = self.cols["name"].index("use")
  376. if checked_pt_id is not None:
  377. checkedKey = checked_pt_id
  378. alreadyChecked.append(self.data[checked_pt_id][type_idx])
  379. else:
  380. checkedKey = -1
  381. for iKey, dt in enumerate(self.data):
  382. pt_type = dt[type_idx]
  383. if ((pt_type in alreadyChecked and checkedKey != iKey)
  384. or pt_type == 0) and self.data[iKey][use_idx]:
  385. self.data[iKey][use_idx] = False
  386. self.pointsChanged.emit(
  387. method="SetPointData", kwargs={
  388. "pt_id": iKey, "data": {
  389. "use": False}})
  390. elif self.data[iKey][use_idx]:
  391. alreadyChecked.append(pt_type)
  392. def EditPointMode(self, activate):
  393. """Registers/unregisters mouse handler into map window"""
  394. if activate == self.handlerRegistered:
  395. return
  396. if activate:
  397. self.mapWin.RegisterMouseEventHandler(wx.EVT_LEFT_DOWN,
  398. self.OnMapClickHandler,
  399. 'cross')
  400. self.handlerRegistered = True
  401. else:
  402. self.mapWin.UnregisterMouseEventHandler(wx.EVT_LEFT_DOWN,
  403. self.OnMapClickHandler)
  404. self.handlerRegistered = False
  405. self.pointsChanged.emit(
  406. method="EditMode", kwargs={
  407. "activated": activate})
  408. def IsEditPointModeActive(self):
  409. return self.handlerRegistered
  410. def OnMapClickHandler(self, event):
  411. """Take coordinates from map window"""
  412. # TODO update snapping after input change
  413. if event == 'unregistered':
  414. self.handlerRegistered = False
  415. return
  416. if not self.data:
  417. self.AddPoint()
  418. e, n = self.mapWin.GetLastEN()
  419. if self.snapping:
  420. # compute threshold
  421. snapTreshPix = int(UserSettings.Get(group='vnet',
  422. key='other',
  423. subkey='snap_tresh'))
  424. res = max(
  425. self.mapWin.Map.region['nsres'],
  426. self.mapWin.Map.region['ewres'])
  427. snapTreshDist = snapTreshPix * res
  428. params, err_params, flags = self.an_params.GetParams()
  429. vectMap = params["input"]
  430. if "input" in err_params:
  431. msg = _("new point")
  432. coords = SnapToNode(e, n, snapTreshDist, vectMap)
  433. if coords:
  434. e = coords[0]
  435. n = coords[1]
  436. msg = ("snapped to node")
  437. else:
  438. msg = _("new point")
  439. else:
  440. msg = _("new point")
  441. self.SetPointData(self.selected,
  442. {'topology': msg,
  443. 'e': e,
  444. 'n': n})
  445. self.pointsToDraw.GetItem(self.selected).SetCoords([e, n])
  446. if self.selected == len(self.data) - 1:
  447. self.SetSelected(0)
  448. else:
  449. self.SetSelected(self.GetSelected() + 1)
  450. def GetColumns(self, only_relevant=True):
  451. cols_data = deepcopy(self.cols)
  452. hidden_cols = []
  453. hidden_cols.append(self.cols["name"].index("e"))
  454. hidden_cols.append(self.cols["name"].index("n"))
  455. analysis, valid = self.an_params.GetParam("analysis")
  456. if only_relevant and len(self.an_data[analysis][
  457. "cmdParams"]["cats"]) <= 1:
  458. hidden_cols.append(self.cols["name"].index("type"))
  459. i_red = 0
  460. hidden_cols.sort()
  461. for idx in hidden_cols:
  462. for dt in cols_data.itervalues():
  463. dt.pop(idx - i_red)
  464. i_red += 1
  465. return cols_data
  466. class VNETAnalysisParameters:
  467. def __init__(self, an_props):
  468. self.an_props = an_props
  469. self.params = {"analysis": self.an_props.used_an[0],
  470. "input": "",
  471. "arc_layer": "",
  472. "node_layer": "",
  473. "arc_column": "",
  474. "arc_backward_column": "",
  475. "node_column": "",
  476. "turn_layer": "",
  477. "turn_cat_layer": "",
  478. "iso_lines": "", # TODO check validity
  479. "max_dist": 0} # TODO check validity
  480. self.flags = {"t": False}
  481. self.parametersChanged = Signal(
  482. 'VNETAnalysisParameters.parametersChanged')
  483. def SetParams(self, params, flags):
  484. changed_params = {}
  485. for p, v in params.iteritems():
  486. if p == "analysis" and v not in self.an_props.used_an:
  487. continue
  488. if p == "input":
  489. mapName, mapSet = ParseMapStr(v)
  490. v = mapName + "@" + mapSet
  491. if p in self.params:
  492. if isinstance(v, str):
  493. v = v.strip()
  494. self.params[p] = v
  495. changed_params[p] = v
  496. changed_flags = {}
  497. for p, v in flags.iteritems():
  498. if p in self.flags:
  499. self.flags[p] = v
  500. changed_flags[p] = v
  501. self.parametersChanged.emit(
  502. method="SetParams",
  503. kwargs={
  504. "changed_params": changed_params,
  505. "changed_flags": changed_flags})
  506. return changed_params, changed_flags
  507. def GetParam(self, param):
  508. invParams = []
  509. if param in [
  510. "input", "arc_layer", "node_layer", "arc_column",
  511. "arc_backward_column", "node_column", "turn_layer",
  512. "turn_cat_layer"]:
  513. invParams = self._getInvalidParams(self.params)
  514. if invParams:
  515. return self.params[param], False
  516. return self.params[param], True
  517. def GetParams(self):
  518. invParams = self._getInvalidParams(self.params)
  519. return self.params, invParams, self.flags
  520. def _getInvalidParams(self, params):
  521. """Check of analysis input data for invalid values (Parameters tab)"""
  522. # dict of invalid values {key from self.itemData (comboboxes from
  523. # Parameters tab) : invalid value}
  524. invParams = []
  525. # check vector map
  526. if params["input"]:
  527. mapName, mapSet = params["input"].split("@")
  528. if mapSet in grass.list_grouped('vector'):
  529. vectMaps = grass.list_grouped('vector')[mapSet]
  530. if not params["input"] or mapName not in vectMaps:
  531. invParams = params.keys()[:]
  532. return invParams
  533. # check arc/node layer
  534. layers = utils.GetVectorNumberOfLayers(params["input"])
  535. for l in ['arc_layer', 'node_layer', 'turn_layer', 'turn_cat_layer']:
  536. if not layers or params[l] not in layers:
  537. invParams.append(l)
  538. dbInfo = VectorDBInfo(params["input"])
  539. try:
  540. table = dbInfo.GetTable(int(params["arc_layer"]))
  541. columnchoices = dbInfo.GetTableDesc(table)
  542. except (KeyError, ValueError):
  543. table = None
  544. # check costs columns
  545. for col in ["arc_column", "arc_backward_column", "node_column"]:
  546. if col == "node_column":
  547. try:
  548. table = dbInfo.GetTable(int(params["node_layer"]))
  549. columnchoices = dbInfo.GetTableDesc(table)
  550. except (KeyError, ValueError):
  551. table = None
  552. if not table or not params[col] in columnchoices.keys():
  553. invParams.append(col)
  554. continue
  555. if columnchoices[
  556. params[col]]['type'] not in [
  557. 'integer', 'double precision']:
  558. invParams.append(col)
  559. continue
  560. return invParams
  561. class VNETAnalysesProperties:
  562. def __init__(self):
  563. """Initializes parameters for different v.net.* modules """
  564. # initialization of v.net.* analysis parameters (data which
  565. # characterizes particular analysis)
  566. self.attrCols = {
  567. 'arc_column': {
  568. "label": _("Arc forward/both direction(s) cost column:"),
  569. "name": _("arc forward/both")
  570. },
  571. 'arc_backward_column': {
  572. "label": _("Arc backward direction cost column:"),
  573. "name": _("arc backward")
  574. },
  575. 'acolumn': {
  576. "label": _("Arcs' cost column (for both directions):"),
  577. "name": _("arc"),
  578. "inputField": 'arc_column',
  579. },
  580. 'node_column': {
  581. "label": _("Node cost column:"),
  582. "name": _("node")
  583. }
  584. }
  585. self.vnetProperties = {
  586. "v.net.path": {
  587. "label": _("Shortest path %s") % "(v.net.path)",
  588. "cmdParams": {
  589. "cats": [
  590. ["st_pt", _("Start point")],
  591. ["end_pt", _("End point")]
  592. ],
  593. "cols": [
  594. 'arc_column',
  595. 'arc_backward_column',
  596. 'node_column'
  597. ],
  598. },
  599. "resultProps": {
  600. "singleColor": None,
  601. "dbMgr": True # TODO delete this property, this information can be get from result
  602. },
  603. "turns_support": True
  604. },
  605. "v.net.salesman": {
  606. "label": _("Traveling salesman %s") % "(v.net.salesman)",
  607. "cmdParams": {
  608. "cats": [["center_cats", None]],
  609. "cols": [
  610. 'arc_column',
  611. 'arc_backward_column'
  612. ],
  613. },
  614. "resultProps": {
  615. "singleColor": None,
  616. "dbMgr": False
  617. },
  618. "turns_support": True
  619. },
  620. "v.net.flow": {
  621. "label": _("Maximum flow %s") % "(v.net.flow)",
  622. "cmdParams": {
  623. "cats": [
  624. ["source_cats", _("Source point")],
  625. ["sink_cats", _("Sink point")]
  626. ],
  627. "cols": [
  628. 'arc_column',
  629. 'arc_backward_column',
  630. 'node_column'
  631. ]
  632. },
  633. "resultProps": {
  634. "attrColColor": "flow",
  635. "dbMgr": True
  636. },
  637. "turns_support": False
  638. },
  639. "v.net.alloc": {
  640. "label": _("Subnets for nearest centers %s") % "(v.net.alloc)",
  641. "cmdParams": {
  642. "cats": [["center_cats", None]],
  643. "cols": [
  644. 'arc_column',
  645. 'arc_backward_column',
  646. 'node_column'
  647. ]
  648. },
  649. "resultProps": {
  650. "catColor": None,
  651. "dbMgr": False
  652. },
  653. "turns_support": True
  654. },
  655. "v.net.steiner": {
  656. "label": _("Steiner tree for the network and given terminals %s") % "(v.net.steiner)",
  657. "cmdParams": {
  658. "cats": [["terminal_cats", None]],
  659. "cols": [
  660. 'acolumn',
  661. ]
  662. },
  663. "resultProps": {
  664. "singleColor": None,
  665. "dbMgr": False
  666. },
  667. "turns_support": True
  668. },
  669. "v.net.distance": {
  670. "label": _("Shortest distance via the network %s") % "(v.net.distance)",
  671. "cmdParams": {
  672. "cats": [
  673. ["from_cats", "From point"],
  674. ["to_cats", "To point"]
  675. ],
  676. "cols": [
  677. 'arc_column',
  678. 'arc_backward_column',
  679. 'node_column'
  680. ],
  681. },
  682. "resultProps": {
  683. "catColor": None,
  684. "dbMgr": True
  685. },
  686. "turns_support": False
  687. },
  688. "v.net.iso": {
  689. "label": _("Cost isolines %s") % "(v.net.iso)",
  690. "cmdParams": {
  691. "cats": [["center_cats", None]],
  692. "cols": [
  693. 'arc_column',
  694. 'arc_backward_column',
  695. 'node_column'
  696. ]
  697. },
  698. "resultProps": {
  699. "catColor": None,
  700. "dbMgr": False
  701. },
  702. "turns_support": True
  703. }
  704. }
  705. self.used_an = ["v.net.path",
  706. "v.net.salesman",
  707. "v.net.flow",
  708. "v.net.alloc",
  709. "v.net.distance",
  710. "v.net.iso",
  711. #"v.net.steiner"
  712. ]
  713. for an in self.vnetProperties.keys():
  714. if an not in self.used_an:
  715. del self.vnetProperties[an]
  716. continue
  717. cols = self.vnetProperties[an]["cmdParams"]["cols"]
  718. self.vnetProperties[an]["cmdParams"]["cols"] = {}
  719. for c in cols:
  720. self.vnetProperties[an]["cmdParams"][
  721. "cols"][c] = self.attrCols[c]
  722. def has_key(self, key):
  723. return key in self.vnetProperties
  724. def __getitem__(self, key):
  725. return self.vnetProperties[key]
  726. def GetRelevantParams(self, analysis):
  727. if analysis not in self.vnetProperties:
  728. return None
  729. relevant_params = ["input", "arc_layer", "node_layer"]
  730. if self.vnetProperties[analysis]["turns_support"]:
  731. relevant_params += ["turn_layer", "turn_cat_layer"]
  732. cols = self.vnetProperties[analysis]["cmdParams"]["cols"]
  733. for col, v in cols.iteritems():
  734. if "inputField" in col:
  735. colInptF = v["inputField"]
  736. else:
  737. colInptF = col
  738. relevant_params.append(colInptF)
  739. return relevant_params
  740. class VNETTmpVectMaps:
  741. """Class which creates, stores and destroys all tmp maps created during analysis"""
  742. def __init__(self, parent, mapWin):
  743. self.tmpMaps = [] # temporary maps
  744. self.parent = parent
  745. self.mapWin = mapWin
  746. def AddTmpVectMap(self, mapName, msg):
  747. """New temporary map
  748. :return: instance of VectMap representing temporary map
  749. """
  750. currMapSet = grass.gisenv()['MAPSET']
  751. tmpMap = grass.find_file(name=mapName,
  752. element='vector',
  753. mapset=currMapSet)
  754. fullName = tmpMap["fullname"]
  755. # map already exists
  756. if fullName:
  757. # TODO move dialog out of class, AddTmpVectMap(self, mapName,
  758. # overvrite = False)
  759. dlg = wx.MessageDialog(parent=self.parent,
  760. message=msg,
  761. caption=_("Overwrite map layer"),
  762. style=wx.YES_NO | wx.NO_DEFAULT |
  763. wx.ICON_QUESTION | wx.CENTRE)
  764. ret = dlg.ShowModal()
  765. dlg.Destroy()
  766. if ret == wx.ID_NO:
  767. return None
  768. else:
  769. fullName = mapName + "@" + currMapSet
  770. newVectMap = VectMap(self.mapWin, fullName)
  771. self.tmpMaps.append(newVectMap)
  772. return newVectMap
  773. def HasTmpVectMap(self, vectMapName):
  774. """
  775. :param: vectMapName name of vector map
  776. :return: True if it contains the map
  777. :return: False if not
  778. """
  779. mapValSpl = vectMapName.strip().split("@")
  780. if len(mapValSpl) > 1:
  781. mapSet = mapValSpl[1]
  782. else:
  783. mapSet = grass.gisenv()['MAPSET']
  784. mapName = mapValSpl[0]
  785. fullName = mapName + "@" + mapSet
  786. for vectTmpMap in self.tmpMaps:
  787. if vectTmpMap.GetVectMapName() == fullName:
  788. return True
  789. return False
  790. def GetTmpVectMap(self, vectMapName):
  791. """Get instance of VectMap with name vectMapName"""
  792. for vectMap in self.tmpMaps:
  793. if vectMap.GetVectMapName() == vectMapName.strip():
  794. return vectMap
  795. return None
  796. def RemoveFromTmpMaps(self, vectMap):
  797. """Temporary map is removed from the class instance however it is not deleted
  798. :param vectMap: instance of VectMap class to be removed
  799. :return: True if was removed
  800. :return: False if does not contain the map
  801. """
  802. try:
  803. self.tmpMaps.remove(vectMap)
  804. return True
  805. except ValueError:
  806. return False
  807. def DeleteTmpMap(self, vectMap):
  808. """Temporary map is removed from the class and it is deleted
  809. :param vectMap: instance of VectMap class to be deleted
  810. :return: True if was removed
  811. :return: False if does not contain the map
  812. """
  813. if vectMap:
  814. vectMap.DeleteRenderLayer()
  815. RunCommand('g.remove', flags='f', type='vector',
  816. name=vectMap.GetVectMapName())
  817. self.RemoveFromTmpMaps(vectMap)
  818. return True
  819. return False
  820. def DeleteAllTmpMaps(self):
  821. """Delete all temporary maps in the class"""
  822. update = False
  823. for tmpMap in self.tmpMaps:
  824. RunCommand('g.remove', flags='f', type='vector',
  825. name=tmpMap.GetVectMapName())
  826. if tmpMap.DeleteRenderLayer():
  827. update = True
  828. return update
  829. class VectMap:
  830. """Represents map
  831. It can check if it was modified or render it
  832. """
  833. def __init__(self, mapWin, fullName):
  834. self.fullName = fullName
  835. self.mapWin = mapWin
  836. self.renderLayer = None
  837. self.modifTime = None # time, for modification check
  838. def __del__(self):
  839. self.DeleteRenderLayer()
  840. def AddRenderLayer(self, cmd=None, colorsCmd=None):
  841. """Add map from map window layers to render """
  842. if not self.mapWin:
  843. return False
  844. existsMap = grass.find_file(name=self.fullName,
  845. element='vector',
  846. mapset=grass.gisenv()['MAPSET'])
  847. if not existsMap["name"]:
  848. self.DeleteRenderLayer()
  849. return False
  850. if not cmd:
  851. cmd = []
  852. cmd.insert(0, 'd.vect')
  853. cmd.append('map=%s' % self.fullName)
  854. if self.renderLayer:
  855. self.DeleteRenderLayer()
  856. if colorsCmd:
  857. colorsCmd.append('map=%s' % self.fullName)
  858. layerStyleVnetColors = cmdlist_to_tuple(colorsCmd)
  859. RunCommand(layerStyleVnetColors[0],
  860. **layerStyleVnetColors[1])
  861. self.renderLayer = self.mapWin.Map.AddLayer(
  862. ltype="vector", command=cmd, name=self.fullName, active=True,
  863. opacity=1.0, render=False, pos=-1)
  864. return True
  865. def DeleteRenderLayer(self):
  866. """Remove map from map window layers to render"""
  867. if not self.mapWin:
  868. return False
  869. if self.renderLayer:
  870. self.mapWin.Map.DeleteLayer(self.renderLayer)
  871. self.renderLayer = None
  872. return True
  873. return False
  874. def GetRenderLayer(self):
  875. return self.renderLayer
  876. def GetVectMapName(self):
  877. return self.fullName
  878. def SaveVectMapState(self):
  879. """Save modification time for vector map"""
  880. self.modifTime = self.GetLastModified()
  881. def VectMapState(self):
  882. """Checks if map was modified
  883. :return: -1 - if no modification time was saved
  884. :return: 0 - if map was modified
  885. :return: 1 - if map was not modified
  886. """
  887. if self.modifTime is None:
  888. return -1
  889. if self.modifTime != self.GetLastModified():
  890. return 0
  891. return 1
  892. def GetLastModified(self):
  893. """Get modification time
  894. :return: MAP DATE time string from vector map head file
  895. """
  896. mapValSpl = self.fullName.split("@")
  897. mapSet = mapValSpl[1]
  898. mapName = mapValSpl[0]
  899. headPath = os.path.join(grass.gisenv()['GISDBASE'],
  900. grass.gisenv()['LOCATION_NAME'],
  901. mapSet,
  902. "vector",
  903. mapName,
  904. "head")
  905. try:
  906. head = open(headPath, 'r')
  907. for line in head.readlines():
  908. i = line.find('MAP DATE:', )
  909. if i == 0:
  910. head.close()
  911. return line.split(':', 1)[1].strip()
  912. head.close()
  913. return ""
  914. except IOError:
  915. return ""
  916. class History:
  917. """Class which reads and saves history data (based on gui.core.settings Settings class file save/load)
  918. .. todo::
  919. Maybe it could be useful for other GRASS wxGUI tools.
  920. """
  921. def __init__(self):
  922. # max number of steps in history (zero based)
  923. self.maxHistSteps = 3
  924. # current history step
  925. self.currHistStep = 0
  926. # number of steps saved in history
  927. self.histStepsNum = 0
  928. # dict contains data saved in history for current history step
  929. self.currHistStepData = {}
  930. # buffer for data to be saved into history
  931. self.newHistStepData = {}
  932. self.histFile = grass.tempfile()
  933. # key/value separator
  934. self.sep = ';'
  935. def __del__(self):
  936. try_remove(self.histFile)
  937. def GetNext(self):
  938. """Go one step forward in history"""
  939. self.currHistStep -= 1
  940. self.currHistStepData.clear()
  941. self.currHistStepData = self._getHistStepData(self.currHistStep)
  942. return self.currHistStepData
  943. def GetPrev(self):
  944. """Go one step back in history"""
  945. self.currHistStep += 1
  946. self.currHistStepData.clear()
  947. self.currHistStepData = self._getHistStepData(self.currHistStep)
  948. return self.currHistStepData
  949. def GetStepsNum(self):
  950. """Get number of steps saved in history"""
  951. return self.histStepsNum
  952. def GetCurrHistStep(self):
  953. """Get current history step"""
  954. return self.currHistStep
  955. def Add(self, key, subkey, value):
  956. """Add new data into buffer"""
  957. if key not in self.newHistStepData:
  958. self.newHistStepData[key] = {}
  959. if isinstance(subkey, types.ListType):
  960. if subkey[0] not in self.newHistStepData[key]:
  961. self.newHistStepData[key][subkey[0]] = {}
  962. self.newHistStepData[key][subkey[0]][subkey[1]] = value
  963. else:
  964. self.newHistStepData[key][subkey] = value
  965. def SaveHistStep(self):
  966. """Create new history step with data in buffer"""
  967. self.maxHistSteps = UserSettings.Get(group='vnet',
  968. key='other',
  969. subkey='max_hist_steps')
  970. self.currHistStep = 0
  971. newHistFile = grass.tempfile()
  972. newHist = open(newHistFile, "w")
  973. self._saveNewHistStep(newHist)
  974. oldHist = open(self.histFile)
  975. removedHistData = self._savePreviousHist(newHist, oldHist)
  976. oldHist.close()
  977. newHist.close()
  978. try_remove(self.histFile)
  979. self.histFile = newHistFile
  980. self.newHistStepData.clear()
  981. return removedHistData
  982. def _savePreviousHist(self, newHist, oldHist):
  983. """Save previous history into new file"""
  984. newHistStep = False
  985. removedHistData = {}
  986. newHistStepsNum = self.histStepsNum
  987. for line in oldHist.readlines():
  988. if not line.strip():
  989. newHistStep = True
  990. newHistStepsNum += 1
  991. continue
  992. if newHistStep:
  993. newHistStep = False
  994. line = line.split("=")
  995. line[1] = str(newHistStepsNum)
  996. line = "=".join(line)
  997. if newHistStepsNum >= self.maxHistSteps:
  998. removedHistStep = removedHistData[line] = {}
  999. continue
  1000. else:
  1001. newHist.write('%s%s%s' % (os.linesep, line, os.linesep))
  1002. self.histStepsNum = newHistStepsNum
  1003. else:
  1004. if newHistStepsNum >= self.maxHistSteps:
  1005. self._parseLine(line, removedHistStep)
  1006. else:
  1007. newHist.write('%s' % line)
  1008. return removedHistData
  1009. def _saveNewHistStep(self, newHist):
  1010. """Save buffer (new step) data into file"""
  1011. newHist.write('%s%s%s' % (os.linesep, "history step=0", os.linesep))
  1012. for key in self.newHistStepData.keys():
  1013. subkeys = self.newHistStepData[key].keys()
  1014. newHist.write('%s%s' % (key, self.sep))
  1015. for idx in range(len(subkeys)):
  1016. value = self.newHistStepData[key][subkeys[idx]]
  1017. if isinstance(value, types.DictType):
  1018. if idx > 0:
  1019. newHist.write('%s%s%s' % (os.linesep, key, self.sep))
  1020. newHist.write('%s%s' % (subkeys[idx], self.sep))
  1021. kvalues = self.newHistStepData[key][subkeys[idx]].keys()
  1022. srange = range(len(kvalues))
  1023. for sidx in srange:
  1024. svalue = self._parseValue(
  1025. self.newHistStepData[key][
  1026. subkeys[idx]][
  1027. kvalues[sidx]])
  1028. newHist.write(
  1029. '%s%s%s' %
  1030. (kvalues[sidx], self.sep, svalue))
  1031. if sidx < len(kvalues) - 1:
  1032. newHist.write('%s' % self.sep)
  1033. else:
  1034. if idx > 0 and isinstance(
  1035. self.newHistStepData[key][subkeys[idx - 1]],
  1036. types.DictType):
  1037. newHist.write('%s%s%s' % (os.linesep, key, self.sep))
  1038. value = self._parseValue(
  1039. self.newHistStepData[key][subkeys[idx]])
  1040. newHist.write('%s%s%s' % (subkeys[idx], self.sep, value))
  1041. if idx < len(subkeys) - 1 and not isinstance(
  1042. self.newHistStepData[key][subkeys[idx + 1]],
  1043. types.DictType):
  1044. newHist.write('%s' % self.sep)
  1045. newHist.write(os.linesep)
  1046. self.histStepsNum = 0
  1047. def _parseValue(self, value, read=False):
  1048. """Parse value"""
  1049. if read: # -> read data (cast values)
  1050. if value:
  1051. if value[
  1052. 0] == '[' and value[-1] == ']': # TODO, possible wrong interpretation
  1053. value = value[1:-1].split(',')
  1054. value = map(self._castValue, value)
  1055. return value
  1056. if value == 'True':
  1057. value = True
  1058. elif value == 'False':
  1059. value = False
  1060. elif value == 'None':
  1061. value = None
  1062. elif ':' in value: # -> color
  1063. try:
  1064. value = tuple(map(int, value.split(':')))
  1065. except ValueError: # -> string
  1066. pass
  1067. else:
  1068. try:
  1069. value = int(value)
  1070. except ValueError:
  1071. try:
  1072. value = float(value)
  1073. except ValueError:
  1074. pass
  1075. else: # -> write data
  1076. if isinstance(value, type(())): # -> color
  1077. value = str(value[0]) + ':' +\
  1078. str(value[1]) + ':' + \
  1079. str(value[2])
  1080. return value
  1081. def _castValue(self, value):
  1082. """Cast value"""
  1083. try:
  1084. value = int(value)
  1085. except ValueError:
  1086. try:
  1087. value = float(value)
  1088. except ValueError:
  1089. value = value[1:-1]
  1090. return value
  1091. def _getHistStepData(self, histStep):
  1092. """Load data saved in history step"""
  1093. hist = open(self.histFile)
  1094. histStepData = {}
  1095. newHistStep = False
  1096. isSearchedHistStep = False
  1097. for line in hist.readlines():
  1098. if not line.strip() and isSearchedHistStep:
  1099. break
  1100. elif not line.strip():
  1101. newHistStep = True
  1102. continue
  1103. elif isSearchedHistStep:
  1104. self._parseLine(line, histStepData)
  1105. if newHistStep:
  1106. line = line.split("=")
  1107. if int(line[1]) == histStep:
  1108. isSearchedHistStep = True
  1109. newHistStep = False
  1110. hist.close()
  1111. return histStepData
  1112. def _parseLine(self, line, histStepData):
  1113. """Parse line in file with history"""
  1114. line = line.rstrip('%s' % os.linesep).split(self.sep)
  1115. key = line[0]
  1116. kv = line[1:]
  1117. idx = 0
  1118. subkeyMaster = None
  1119. if len(kv) % 2 != 0: # multiple (e.g. nviz)
  1120. subkeyMaster = kv[0]
  1121. del kv[0]
  1122. idx = 0
  1123. while idx < len(kv):
  1124. if subkeyMaster:
  1125. subkey = [subkeyMaster, kv[idx]]
  1126. else:
  1127. subkey = kv[idx]
  1128. value = kv[idx + 1]
  1129. value = self._parseValue(value, read=True)
  1130. if key not in histStepData:
  1131. histStepData[key] = {}
  1132. if isinstance(subkey, types.ListType):
  1133. if subkey[0] not in histStepData[key]:
  1134. histStepData[key][subkey[0]] = {}
  1135. histStepData[key][subkey[0]][subkey[1]] = value
  1136. else:
  1137. histStepData[key][subkey] = value
  1138. idx += 2
  1139. def DeleteNewHistStepData(self):
  1140. """Delete buffer data for new history step"""
  1141. self.newHistStepData.clear()
  1142. class VNETGlobalTurnsData:
  1143. """Turn Data"""
  1144. def __init__(self):
  1145. # Definition of four basic directions
  1146. self.turn_data = [
  1147. ["Straight", DegreesToRadians(-30), DegreesToRadians(+30), 0.0],
  1148. ["Right Turn", DegreesToRadians(+30), DegreesToRadians(+150), 0.0],
  1149. ["Reverse", DegreesToRadians(+150), DegreesToRadians(-150), 0.0],
  1150. ["Left Turn", DegreesToRadians(-150), DegreesToRadians(-30), 0.0]
  1151. ]
  1152. def GetData(self):
  1153. data = []
  1154. for ival in self.turn_data:
  1155. data.append(ival[1:])
  1156. return data
  1157. def GetValue(self, line, col):
  1158. return self.turn_data[line][col]
  1159. def GetLinesCount(self):
  1160. return len(self.turn_data)
  1161. def SetValue(self, value, line, col):
  1162. self.DataValidator(line, col, value)
  1163. self.turn_data[line][col] = value
  1164. def SetUTurns(self, value):
  1165. """Checked if checeBox is checed"""
  1166. useUTurns = value
  1167. def AppendRow(self, values):
  1168. self.turn_data.append(values)
  1169. def InsertRow(self, line, values):
  1170. self.turn_data.insert(line, values)
  1171. def PopRow(self, values):
  1172. self.RemoveDataValidator(values)
  1173. self.turn_data.pop(values)
  1174. def DataValidator(self, row, col, value):
  1175. """Angle recalculation due to value changing"""
  1176. if col not in [1, 2]:
  1177. return
  1178. if col == 1:
  1179. new_from_angle = value
  1180. old_from_angle = self.turn_data[row][1]
  1181. new_to_angle = self.turn_data[row][2]
  1182. if self.IsInInterval(old_from_angle, new_to_angle, new_from_angle):
  1183. prev_row = row - 1
  1184. if prev_row == -1:
  1185. prev_row = len(self.turn_data) - 1
  1186. self.turn_data[prev_row][2] = new_from_angle
  1187. return
  1188. if col == 2:
  1189. new_to_angle = value
  1190. old_to_angle = self.turn_data[row][2]
  1191. new_from_angle = self.turn_data[row][1]
  1192. if self.IsInInterval(new_from_angle, old_to_angle, new_to_angle):
  1193. next_row = row + 1
  1194. if len(self.turn_data) == next_row:
  1195. next_row = 0
  1196. self.turn_data[next_row][1] = new_to_angle
  1197. return
  1198. inside_new = []
  1199. overlap_new_from = []
  1200. overlap_new_to = []
  1201. for i in range(self.GetLinesCount()):
  1202. if i == row:
  1203. continue
  1204. from_angle = self.turn_data[i][1]
  1205. is_in_from = self.IsInInterval(
  1206. new_from_angle, new_to_angle, from_angle)
  1207. to_angle = self.turn_data[i][2]
  1208. is_in_to = self.IsInInterval(
  1209. new_from_angle, new_to_angle, to_angle)
  1210. if is_in_from and is_in_to:
  1211. inside_new.append(i)
  1212. if is_in_from:
  1213. overlap_new_to.append(i)
  1214. if is_in_to:
  1215. overlap_new_from.append(i)
  1216. for i_row in overlap_new_from:
  1217. self.turn_data[i_row][2] = new_from_angle
  1218. for i_row in overlap_new_to:
  1219. self.turn_data[i_row][1] = new_to_angle
  1220. for i_row in inside_new:
  1221. if col == 1:
  1222. angle = new_from_angle
  1223. else:
  1224. angle = new_to_angle
  1225. self.turn_data[i_row][1] = angle
  1226. self.turn_data[i_row][2] = angle
  1227. def RemoveDataValidator(self, row):
  1228. """Angle recalculation due to direction remove"""
  1229. if row == 0:
  1230. prev_row = self.GetLinesCount() - 1
  1231. else:
  1232. prev_row = row - 1
  1233. remove_to_angle = self.turn_data[row][2]
  1234. self.turn_data[prev_row][2] = remove_to_angle
  1235. def IsInInterval(self, from_angle, to_angle, angle):
  1236. """Test if a direction includes or not includes a value"""
  1237. if to_angle < from_angle:
  1238. to_angle = math.pi * 2 + to_angle
  1239. if angle < from_angle:
  1240. angle = math.pi * 2 + angle
  1241. if angle > from_angle and angle < to_angle:
  1242. return True
  1243. return False