12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577 |
- """!
- @package wxvdigit.py
- @brief wxGUI vector digitizer (base class)
- Code based on wxVdigit C++ component from GRASS 6.4.0
- (gui/wxpython/vdigit). Converted to Python in 2010/12-2011/01.
- List of classes:
- - VDigitError
- - IVDigit
- (C) 2007-2011 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 Martin Landa <landa.martin gmail.com>
- """
- from gcmd import GError
- from debug import Debug
- from preferences import globalSettings as UserSettings
- from wxvdriver import DisplayDriver
- from grass.lib.grass import *
- from grass.lib.vector import *
- from grass.lib.vedit import *
- from grass.lib.dbmi import *
- class VDigitError:
- def __init__(self, parent):
- """!Class for managing error messages of vector digitizer
- @param parent parent window for dialogs
- """
- self.parent = parent
- self.caption = _('Digitization Error')
-
- def NoMap(self, name = None):
- """!No map for editing"""
- if name:
- message = _('Unable to open vector map <%s>.') % name
- else:
- message = _('No vector map open for editing.')
- GError(message + ' ' + _('Operation cancelled.'),
- parent = self.parent,
- caption = self.caption)
- def WriteLine(self):
- """!Writing line failed
- """
- GError(message = _('Writing new feature failed. '
- 'Operation cancelled.'),
- parent = self.parent,
- caption = self.caption)
- def ReadLine(self, line):
- """!Reading line failed
- """
- GError(message = _('Reading feature id %d failed. '
- 'Operation cancelled.') % line,
- parent = self.parent,
- caption = self.caption)
- def DbLink(self, dblink):
- """!No dblink available
- """
- GError(message = _('Database link %d not available. '
- 'Operation cancelled.') % dblink,
- parent = self.parent,
- caption = self.caption)
- def Driver(self, driver):
- """!Staring driver failed
- """
- GError(message = _('Unable to start database driver <%s>. '
- 'Operation cancelled.') % driver,
- parent = self.parent,
- caption = self.caption)
- def Database(self, driver, database):
- """!Opening database failed
- """
- GError(message = _('Unable to open database <%s> by driver <%s>. '
- 'Operation cancelled.') % (database, driver),
- parent = self.parent,
- caption = self.caption)
- def DbExecute(self, sql):
- """!Sql query failed
- """
- GError(message = _("Unable to execute SQL query '%s'. "
- "Operation cancelled.") % sql,
- parent = self.parent,
- caption = self.caption)
- def DeadLine(self, line):
- """!Dead line
- """
- GError(message = _("Feature id %d is marked as dead. "
- "Operation cancelled.") % line,
- parent = self.parent,
- caption = self.caption)
- def FeatureType(self, ftype):
- """!Unknown feature type
- """
- GError(message = _("Unsupported feature type %d. "
- "Operation cancelled.") % ftype,
- parent = self.parent,
- caption = self.caption)
-
- class IVDigit:
- def __init__(self, mapwindow):
- """!Base class for vector digitizer (ctypes interface)
-
- @parem mapwindow reference for map window (BufferedWindow)
- """
- self.poMapInfo = None # pointer to Map_info
- self.mapWindow = mapwindow
- # background map
- self.bgMapInfo = Map_info()
- self.poBgMapInfo = self.popoBgMapInfo = None
-
- if not mapwindow.parent.IsStandalone():
- goutput = mapwindow.parent.GetLayerManager().GetLogWindow()
- log = goutput.GetLog(err = True)
- progress = goutput.GetProgressBar()
- else:
- log = sys.stderr
- progress = None
-
- self.toolbar = mapwindow.parent.toolbars['vdigit']
-
- self._error = VDigitError(parent = self.mapWindow)
-
- self._display = DisplayDriver(device = mapwindow.pdcVector,
- deviceTmp = mapwindow.pdcTmp,
- mapObj = mapwindow.Map,
- window = mapwindow,
- glog = log,
- gprogress = progress)
-
- # GRASS lib
- self.poPoints = Vect_new_line_struct()
- self.poCats = Vect_new_cats_struct()
-
- # self.SetCategory()
-
- # layer / max category
- self.cats = dict()
-
- self._settings = dict()
- self.UpdateSettings() # -> self._settings
-
- # undo/redo
- self.changesets = dict()
- self.changesetCurrent = -1 # first changeset to apply
- self.changesetEnd = -1 # last changeset to be applied
-
- if self.poMapInfo:
- self.InitCats()
-
- def __del__(self):
- Vect_destroy_line_struct(self.poPoints)
- self.poPoints = None
- Vect_destroy_cats_struct(self.poCats)
- self.poCats = None
-
- if self.poBgMapInfo:
- Vect_close(self.poBgMapInfo)
- self.poBgMapInfo = self.popoBgMapInfo = None
- del self.bgMapInfo
-
- def CloseBackgroundMap(self):
- """!Close background vector map"""
- if not self.poBgMapInfo:
- return
-
- Vect_close(self.poBgMapInfo)
- self.poBgMapInfo = self.popoBgMapInfo = None
-
- def OpenBackgroundMap(self, bgmap):
- """!Open background vector map
- @todo support more background maps then only one
-
- @param bgmap name of vector map to be opened
- @return pointer to map_info
- @return None on error
- """
- name = create_string_buffer(GNAME_MAX)
- mapset = create_string_buffer(GMAPSET_MAX)
- if not G_name_is_fully_qualified(bgmap, name, mapset):
- name = str(bgmap)
- mapset = str(G_find_vector2(bgmap, ''))
- else:
- name = str(name.value)
- mapset = str(mapset.value)
-
- if (name == Vect_get_name(self.poMapInfo) and \
- mapset == Vect_get_mapset(self.poMapInfo)):
- self.poBgMapInfo = self.popoBgMapInfo = None
- self._error.NoMap(bgmap)
- return
-
- self.poBgMapInfo = pointer(self.bgMapInfo)
- self.popoBgMapInfo = pointer(self.poBgMapInfo)
- if Vect_open_old(self.poBgMapInfo, name, mapset) == -1:
- self.poBgMapInfo = self.popoBgMapInfo = None
- self._error.NoMap(bgmap)
- return
-
- def _getSnapMode(self):
- """!Get snapping mode
- - snap to vertex
- - snap to nodes
- - no snapping
-
- @return snap mode
- """
- threshold = self._display.GetThreshold()
- if threshold > 0.0:
- if UserSettings.Get(group = 'vdigit', key = 'snapToVertex', subkey = 'enabled'):
- return SNAPVERTEX
- else:
- return SNAP
- else:
- return NO_SNAP
-
- def _breakLineAtIntersection(self, line, pointsLine, changeset):
- """!Break given line at intersection
- \param line line id
- \param pointsLine line geometry
- \param changeset id
-
- \return number of modified lines
- """
- if not self._checkMap():
- return -1
-
- if not Vect_line_alive(self.poMapInfo, line):
- return 0
-
- if not pointsLine:
- if Vect_read_line(self.poMapInfo, self.poPoints, None, line) < 0:
- self._error.ReadLine(line)
- return -1
- points = self.poPoints
- else:
- points = pointsLine
-
- listLine = Vect_new_list()
- listRef = Vect_new_list()
- listBreak = Vect_new_list()
-
- pointsCheck = Vect_new_line_struct()
-
- lineBox = bound_box()
- # find all relevant lines
- Vect_get_line_box(self.poMapInfo, line, byref(lineBox))
- Vect_select_lines_by_box(self.poMapInfo, byref(lineBox),
- GV_LINES, listLine)
-
- # check for intersection
- Vect_list_append(listBreak, line)
- Vect_list_append(listRef, line)
- for i in range(listLine.contents.n_values):
- lineBreak = listLine.contents.value[i]
- if lineBreak == line:
- continue
-
- ltype = Vect_read_line(self.poMapInfo, pointsCheck, None, lineBreak)
- if not (ltype & GV_LINES):
- continue
-
- if Vect_line_check_intersection(self.poPoints, pointsCheck,
- WITHOUT_Z):
- Vect_list_append(listBreak, lineBreak)
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- for i in range(listBreak.contents.n_values):
- self._addActionToChangeset(changeset, listBreak.contents.value[i], add = False)
-
- ret = Vect_break_lines_list(self.poMapInfo, listBreak, listRef,
- GV_LINES, None)
-
- for i in range(listBreak.contents.n_values):
- if Vect_line_alive(self.poMapInfo, listBreak.contents.value[i]):
- self._removeActionFromChangeset(changeset, listBreak.contents.value[i],
- add = False)
-
- for line in range(nlines + 1, Vect_get_num_lines(self.poMapInfo) + 1):
- self._addActionToChangeset(changeset, line, add = True)
-
- Vect_destroy_line_struct(pointsCheck)
- Vect_destroy_list(listLine)
- Vect_destroy_list(listBreak)
- Vect_destroy_list(listRef)
-
- return ret
-
- def _addActionsBefore(self):
- """!Register action before operation
-
- @return changeset id
- """
- changeset = len(self.changesets)
- for line in self._display.selected['ids']:
- if Vect_line_alive(self.poMapInfo, line):
- self._addActionToChangeset(changeset, line, add = False)
-
- return changeset
- def _applyChangeset(self, changeset, undo):
- """!Apply changeset (undo/redo changeset)
-
- @param changeset changeset id
- @param undo True for undo otherwise redo
- @return 1 changeset applied
- @return 0 changeset not applied
- @return -1 on error
- """
- if changeset < 0 or changeset > len(self.changesets.keys()):
- return -1
-
- if self.changesetEnd < 0:
- self.changesetEnd = changeset
-
- ret = 0
- actions = self.changesets[changeset]
- for action in actions:
- add = action['add']
- line = action['line']
- if (undo and add) or \
- (not undo and not add):
- if Vect_line_alive(self.poMapInfo, line):
- Debug.msg(3, "IVDigit._applyChangeset(): changeset=%d, action=add, line=%d -> deleted",
- changeset, line)
- Vect_delete_line(self.poMapInfo, line)
- ret = 1
- else:
- Debug.msg(3, "Digit.ApplyChangeset(): changeset=%d, action=add, line=%d dead",
- changeset, line)
- else: # delete
- offset = action['offset']
- if not Vect_line_alive(self.poMapInfo, line):
- Debug.msg(3, "Digit.ApplyChangeset(): changeset=%d, action=delete, line=%d -> added",
- changeset, line)
- if Vect_restore_line(self.poMapInfo, line, offset) < 0:
- return -1
- ret = 1
- else:
- Debug.msg(3, "Digit.ApplyChangeset(): changeset=%d, action=delete, line=%d alive",
- changeset, line)
-
- return ret
-
- def _addActionsAfter(self, changeset, nlines):
- """!Register action after operation
- @param changeset changeset id
- @param nline number of lines
- """
- for line in self._display.selected['ids']:
- if Vect_line_alive(self.poMapInfo, line):
- self._removeActionFromChangeset(changeset, line, add = False)
-
- for line in range(nlines + 1, Vect_get_num_lines(self.poMapInfo)):
- if Vect_line_alive(self.poMapInfo, line):
- self._addActionToChangeset(changeset, line, add = True)
-
- def _addActionToChangeset(self, changeset, line, add):
- """!Add action to changeset
-
- @param changeset id of changeset
- @param line feature id
- @param add True to add, otherwise delete
- """
- if not self._checkMap():
- return
-
- if not Vect_line_alive(self.poMapInfo, line):
- return
-
- offset = Vect_get_line_offset(self.poMapInfo, line)
-
- if not self.changesets.has_key(changeset):
- self.changesets[changeset] = list()
- self.changesetCurrent = changeset
-
- self.changesets[changeset].append({ 'add' : add,
- 'line' : line,
- 'offset' : offset })
-
- Debug.msg(3, "IVDigit._addActionToChangeset(): changeset=%d, type=%d, line=%d, offset=%d",
- changeset, type, line, offset)
-
- def _removeActionFromChangeset(self, changeset, line, add):
- """!Remove action from changeset
-
- @param changeset changeset id
- @param line line id
- @param add True for add, False for delete
-
- @return number of actions in changeset
- @return -1 on error
- """
- if changeset not in self.changesets.keys():
- return -1
-
- alist = self.changesets[changeset]
- for action in alist:
- if action['add'] == add and action['line'] == line:
- alist.remove(action)
-
- return len(alist)
- def AddFeature(self, ftype, points):
- """!Add new feature
-
- @param ftype feature type (point, line, centroid, boundary)
- @param points tuple of points ((x, y), (x, y), ...)
-
- @return tuple (number of added features, feature ids)
- """
- if UserSettings.Get(group = 'vdigit', key = "categoryMode", subkey = 'selection') == 2:
- layer = -1 # -> no category
- cat = -1
- else:
- layer = UserSettings.Get(group = 'vdigit', key = "layer", subkey = 'value')
- cat = self.SetCategory()
-
- if ftype == 'point':
- vtype = GV_POINT
- elif ftype == 'line':
- vtype = GV_LINE
- elif ftype == 'centroid':
- vtype = GV_CENTROID
- elif ftype == 'boundary':
- vtype = GV_BOUNDARY
- elif ftype == 'area':
- vtype = GV_AREA
- else:
- GError(parent = self.mapWindow,
- message = _("Unknown feature type '%s'") % ftype)
- return (-1, None)
-
- if vtype & GV_LINES and len(points) < 2:
- GError(parent = self.mapWindow,
- message = _("Not enough points for line"))
- return (-1, None)
-
- self.toolbar.EnableUndo()
-
- return self._addFeature(vtype, points, layer, cat,
- self._getSnapMode(), self._display.GetThreshold())
-
- def DeleteSelectedLines(self):
- """!Delete selected features
- @return number of deleted features
- """
- deleteRec = UserSettings.Get(group = 'vdigit', key = 'delRecord', subkey = 'enabled')
- if not self._checkMap():
- return -1
-
- n_dblinks = Vect_get_num_dblinks(self.poMapInfo)
- Cats_del = None
-
- # collect categories for delete if requested
- if deleteRec:
- poCats = Vect_new_cats_struct()
- poCatsDel = Vect_new_cats_struct()
- for i in self._display.selected['ids']:
- if Vect_read_line(self.poMapInfo, None, poCats, i) < 0:
- Vect_destroy_cats_struct(poCatsDel)
- self._error.ReadLine(i)
- return -1
-
- cats = poCats.contents
- for j in range(cats.n_cats):
- Vect_cat_set(poCatsDel, cats.field[j], cats.cat[j])
-
- Vect_destroy_cats_struct(poCats)
-
- # register changeset
- changeset = self._addActionsBefore()
-
- poList = self._display.GetSelectedIList()
- nlines = Vedit_delete_lines(self.poMapInfo, poList)
- Vect_destroy_list(poList)
- self._display.selected['ids'] = list()
-
- if nlines > 0 and deleteRec:
- handle = dbHandle()
- poHandle = pointer(handle)
- stmt = dbString()
- poStmt = pointer(stmt)
-
- for dblink in range(n_dblinks):
- poFi = Vect_get_dblink(self.poMapInfo, dblink)
- if poFi is None:
- self._error.DbLink(dblink)
- return -1
-
- Fi = poFi.contents
- poDriver = db_start_driver(Fi.driver)
- if poDriver is None:
- self._error.Driver(Fi.driver)
- return -1
-
- db_init_handle(poHandle)
- db_set_handle(poHandle, Fi.database, None)
- if db_open_database(poDriver, poHandle) != DB_OK:
- self._error.Database(Fi.driver, Fi.database)
- return -1
-
- db_init_string(poStmt)
- db_set_string(poStmt, "DELETE FROM %s WHERE" % Fi.table)
- n_cats = 0;
- catsDel = poCatsDel.contents
- for c in range(catsDel.n_cats):
- if catsDel.field[c] == Fi.number:
- if n_cats > 0:
- db_append_string(poStmt, " or")
-
- db_append_string(poStmt, " %s = %d" % (Fi.key, catsDel.cat[c]))
- n_cats += 1
-
- Vect_cat_del(poCatsDel, Fi.number)
-
- if n_cats and \
- db_execute_immediate(poDriver, poStmt) != DB_OK:
- self._error.DbExecute(db_get_string(poStmt))
- return -1
-
- db_close_database(poDriver)
- db_shutdown_driver(poDriver)
-
- if poCatsDel:
- Vect_destroy_cats_struct(poCatsDel)
-
- if nlines > 0:
- self.toolbar.EnableUndo()
-
- return nlines
- def MoveSelectedLines(self, move):
- """!Move selected features
- @param move direction (x, y)
- """
- if not self._checkMap():
- return -1
-
- thresh = self._display.GetThreshold()
- snap = self._getSnapMode()
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- # register changeset
- changeset = self._addActionsBefore()
-
- poList = self._display.GetSelectedIList()
- nlines = Vedit_move_lines(self.poMapInfo, self.popoBgMapInfo, int(self.poBgMapInfo is not None),
- poList,
- move[0], move[1], 0,
- snap, thresh)
- Vect_destroy_list(poList)
-
- if nlines > 0:
- self._addActionsAfter(changeset, nlines)
- else:
- del self.changesets[changeset]
-
- if nlines > 0 and self._settings['breakLines']:
- for i in range(1, nlines):
- self._breakLineAtIntersection(nlines + i, None, changeset)
-
- if nlines > 0:
- self.toolbar.EnableUndo()
-
- return nlines
- def MoveSelectedVertex(self, point, move):
- """!Move selected vertex of the line
- @param point location point
- @param move x,y direction
-
- @return id of new feature
- @return 0 vertex not moved (not found, line is not selected)
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- if len(self._display.selected['ids']) != 1:
- return -1
-
- Vect_reset_line(self.poPoints)
- Vect_append_point(self.poPoints, point[0], point[1], 0.0)
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- changeset = self._addActionsBefore()
-
- # move only first found vertex in bbox
- poList = self._display.GetSelectedIList()
- moved = Vedit_move_vertex(self.poMapInfo, self.popoBgMapInfo, int(self.poBgMapInfo is not None),
- poList, self.poPoints,
- self._display.GetThreshold(type = 'selectThresh'),
- self._display.GetThreshold(),
- move[0], move[1], 0.0,
- 1, self._getSnapMode())
- Vect_destroy_list(poList)
-
- if moved > 0:
- self._addActionsAfter(changeset, nlines)
- else:
- del self.changesets[changeset]
-
- if moved > 0 and self._settings['breakLines']:
- self._breakLineAtIntersection(Vect_get_num_lines(self.poMapInfo),
- None, changeset)
-
- if moved > 0:
- self.toolbar.EnableUndo()
-
- return moved
- def AddVertex(self, coords):
- """!Add new vertex to the selected line/boundary on position 'coords'
- @param coords coordinates to add vertex
- @return id of new feature
- @return 0 nothing changed
- @return -1 on failure
- """
- added = self._ModifyLineVertex(coords, add = True)
-
- if added > 0:
- self.toolbar.EnableUndo()
- return added
- def RemoveVertex(self, coords):
- """!Remove vertex from the selected line/boundary on position 'coords'
- @param coords coordinates to remove vertex
- @return id of new feature
- @return 0 nothing changed
- @return -1 on failure
- """
- deleted = self._ModifyLineVertex(coords, add = False)
-
- if deleted > 0:
- self.toolbar.EnableUndo()
- return deleted
- def SplitLine(self, point):
- """!Split/break selected line/boundary on given position
- @param point point where to split line
-
- @return 1 line modified
- @return 0 nothing changed
- @return -1 error
- """
- thresh = self._display.GetThreshold('selectThresh')
- if not self._checkMap():
- return -1
-
- poList = self._display.GetSelectedIList()
- Vect_reset_line(self.poPoints)
- Vect_append_point(self.poPoints, point[0], point[1], 0.0)
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- changeset = self._addActionsBefore()
-
- ret = Vedit_split_lines(self.poMapInfo, poList,
- self.poPoints, thresh, poList)
- Vect_destroy_list(poList)
-
- if ret > 0:
- self._addActionsAfter(changeset, nlines)
- self.toolbar.EnableUndo()
- else:
- del self.changesets[changeset]
-
- return ret
- def EditLine(self, line, coords):
- """!Edit existing line/boundary
- @param line id of line to be modified
- @param coords list of coordinates of modified line
- @return feature id of new line
- @return -1 on error
- """
- if self._checkMap():
- return -1
-
- try:
- lineid = line[0]
- except:
- lineid = -1
-
- if len(coords) < 2:
- self.DeleteSelectedLines()
- return 0
-
- ret = self.digit.RewriteLine(lineid, listCoords,
- bgmap, self._getSnapMode(),
- self._display.GetThreshold())
-
- if ret > 0:
- self.toolbar.EnableUndo()
-
- return ret
- def FlipLine(self):
- """!Flip selected lines/boundaries
- @return number of modified lines
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- # register changeset
- changeset = self._addActionsBefore()
-
- poList = self._display.GetSelectedIList()
- ret = Vedit_flip_lines(self.poMapInfo, poList)
- Vect_destroy_list(poList)
-
- if ret > 0:
- self._addActionsAfter(changeset, nlines)
- self.toolbar.EnableUndo()
- else:
- del self.changesets[changeset]
-
- return ret
- def MergeLine(self):
- """!Merge selected lines/boundaries
- @return number of modified lines
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- changeset = self._addActionsBefore()
-
- poList = self._display.GetSelectedIList()
- ret = Vedit_merge_lines(self.poMapInfo, poList)
- Vect_destroy_list(poList)
-
- if ret > 0:
- self._addActionsAfter(changeset, nlines)
- self.toolbar.EnableUndo()
- else:
- del self.changesets[changeset]
-
- return ret
- def BreakLine(self):
- """!Break selected lines/boundaries
- @return number of modified lines
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- changeset = self._addActionsBefore()
-
- poList = self._display.GetSelectedIList()
- ret = Vect_break_lines_list(self.poMapInfo, poList, None,
- GV_LINES, None)
- Vect_destroy_list(poList)
-
- if ret > 0:
- self._addActionsAfter(changeset, nlines)
- self.toolbar.EnableUndo()
- else:
- del self.changesets[changeset]
-
- return ret
- def SnapLine(self):
- """!Snap selected lines/boundaries
- @return on success
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- changeset = self._addActionsBefore()
-
- poList = self._display.GetSelectedIList()
- Vect_snap_lines_list(self.poMapInfo, poList,
- self._display.GetThreshold(), None)
- Vect_destroy_list(poList)
-
- if nlines < Vect_get_num_lines(self.poMapInfo):
- self._addActionsAfter(changeset, nlines)
- self.toolbar.EnableUndo()
- else:
- del self.changesets[changeset]
-
- def ConnectLine(self):
- """!Connect selected lines/boundaries
- @return 1 lines connected
- @return 0 lines not connected
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- # register changeset
- changeset = self._addActionsBefore()
-
- poList = self._display.GetSelectedIList()
- ret = Vedit_connect_lines(self.poMapInfo, poList,
- self._display.GetThreshold())
- Vect_destroy_list(poList)
-
- if ret > 0:
- self._addActionsAfter(changeset, nlines)
- self.toolbar.EnableUndo()
- else:
- del self.changesets[changeset]
-
- return ret
-
- def CopyLine(self, ids = []):
- """!Copy features from (background) vector map
- @param ids list of line ids to be copied
- @return number of copied features
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- poList = self._display.GetSelectedIList(ids)
- ret = Vedit_copy_lines(self.poMapInfo, self.poBgMapInfo,
- poList)
- Vect_destroy_list(poList)
-
- if ret > 0:
- changeset = len(self.changesets)
- for line in (range(nlines + 1, Vect_get_num_lines(self.poMapInfo))):
- self._addActionToChangeset(changeset, line, add = True)
- self.toolbar.EnableUndo()
- else:
- del self.changesets[changeset]
- if ret > 0 and self.poBgMapInfo and self._settings['breakLines']:
- for i in range(1, ret):
- self._breakLineAtIntersection(nlines + i, None, changeset)
-
- return ret
- def CopyCats(self, fromId, toId, copyAttrb=False):
- """!Copy given categories to objects with id listed in ids
- @param cats ids of 'from' feature
- @param ids ids of 'to' feature(s)
- @return number of modified features
- @return -1 on error
- """
- if len(fromId) == 0 or len(toId) == 0:
- return 0
-
- ret = self.digit.CopyCats(fromId, toId, copyAttrb)
- if ret > 0:
- self.toolbar.EnableUndo()
- return ret
- def _selectLinesByQueryThresh(self):
- """!Generic method used for SelectLinesByQuery() -- to get
- threshold value"""
- thresh = 0.0
- if UserSettings.Get(group = 'vdigit', key = 'query', subkey = 'selection') == 0:
- thresh = UserSettings.Get(group = 'vdigit', key = 'queryLength', subkey = 'thresh')
- if UserSettings.Get(group = 'vdigit', key = "queryLength", subkey = 'than-selection') == 0:
- thresh = -1 * thresh
- else:
- thresh = UserSettings.Get(group = 'vdigit', key = 'queryDangle', subkey = 'thresh')
- if UserSettings.Get(group = 'vdigit', key = "queryDangle", subkey = 'than-selection') == 0:
- thresh = -1 * thresh
-
- return thresh
- def SelectLinesByQuery(self, bbox):
- """!Select features by query
-
- @todo layer / 3D
-
- @param bbox bounding box definition
- """
- if not self._checkMap():
- return -1
-
- thresh = self._selectLinesByQueryThresh()
-
- query = QUERY_UNKNOWN
- if UserSettings.Get(group = 'vdigit', key = 'query', subkey = 'selection') == 0:
- query = QUERY_LENGTH
- else:
- query = QUERY_DANGLE
-
- ftype = GV_POINTS | GV_LINES # TODO: 3D
- layer = 1 # TODO
-
- ids = list()
- poList = Vect_new_list()
- coList = poList.contents
- if UserSettings.Get(group = 'vdigit', key = 'query', subkey = 'box'):
- Vect_reset_line(self.poPoints)
- x1, y1 = bbox[0]
- x2, y2 = bbox[1]
- z1 = z2 = 0.0
-
- Vect_append_point(self.poPoints, x1, y1, z1)
- Vect_append_point(self.poPoints, x2, y1, z2)
- Vect_append_point(self.poPoints, x2, y2, z1)
- Vect_append_point(self.poPoints, x1, y2, z2)
- Vect_append_point(self.poPoints, x1, y1, z1)
-
- Vect_select_lines_by_polygon(self.poMapInfo, self.poPoints, 0, None,
- ftype, poList)
-
- if coList.n_values == 0:
- return ids
-
- Vedit_select_by_query(self.poMapInfo,
- ftype, layer, thresh, query,
- poList)
-
- for i in range(coList.n_values):
- ids.append(int(coList.value[i]))
-
- Debug.msg(3, "IVDigit.SelectLinesByQuery(): lines=%d", coList.n_values)
- Vect_destroy_list(poList)
-
- return ids
- def IsVector3D(self):
- """!Check if open vector map is 3D
- """
- if not self._checkMap():
- return False
-
- return Vect_is_3d(self.poMapInfo)
-
- def GetLineCats(self, line=-1):
- """!Get layer/category pairs from given (selected) line
-
- @param line feature id (-1 for first selected line)
- """
- return dict(self.digit.GetLineCats(line))
- def GetLineLength(self, line):
- """!Get line length
- @param line feature id
- @return line length
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- if not Vect_line_alive(self.poMapInfo, line):
- return -1
-
- ltype = Vect_read_line(self.poMapInfo, self.poPoints, None, line)
- if ltype < 0:
- self._error.ReadLine(line)
- return ret
-
- length = -1
- if ltype & GV_LINES: # lines & boundaries
- length = Vect_line_length(self.poPoints)
-
- return length
- def GetAreaSize(self, centroid):
- """!Get area size
- @param centroid centroid id
- @return area size
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- ltype = Vect_read_line(self.poMapInfo, None, None, centroid)
- if ltype < 0:
- self._error.ReadLine(line)
- return ret
-
- if ltype != GV_CENTROID:
- return -1
-
- area = Vect_get_centroid_area(self.poMapInfo, centroid)
- size = -1
- if area > 0:
- if not Vect_area_alive(self.poMapInfo, area):
- return size
-
- size = Vect_get_area_area(self.poMapInfo, area)
-
- return size
-
- def GetAreaPerimeter(self, centroid):
- """!Get area perimeter
-
- @param centroid centroid id
-
- @return area size
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- ltype = Vect_read_line(self.poMapInfo, None, None, centroid)
- if ltype < 0:
- self._error.ReadLine(line)
- return ret
-
- if ltype != GV_CENTROID:
- return -1
-
- area = Vect_get_centroid_area(self.poMapInfo, centroid)
- perimeter = -1
- if area > 0:
- if not Vect_area_alive(self,poMapInfo, area):
- return -1
-
- Vect_get_area_points(self.poMapInfo, area, self.poPoints)
- perimeter = Vect_area_perimeter(self.poPoints)
-
- return perimeter
-
- def SetLineCats(self, line, layer, cats, add=True):
- """!Set categories for given line and layer
- @param line feature id
- @param layer layer number (-1 for first selected line)
- @param cats list of categories
- @param add if True to add, otherwise do delete categories
- @return new feature id (feature need to be rewritten)
- @return -1 on error
- """
- ret = self.digit.SetLineCats(line, layer, cats, add)
- if ret > 0:
- self.toolbar.EnableUndo()
- return ret
- def GetLayers(self):
- """!Get list of layers"""
- return self.digit.GetLayers()
- def TypeConvForSelectedLines(self):
- """!Feature type conversion for selected objects.
- Supported conversions:
- - point <-> centroid
- - line <-> boundary
- @return number of modified features
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- # register changeset
- changeset = self._addActionsBefore()
-
- poList = self._display.GetSelectedIList()
- ret = Vedit_chtype_lines(self.poMapInfo, poList)
- Vect_destroy_list(poList)
-
- if ret > 0:
- self._addActionsAfter(changeset, nlines)
- self.toolbar.EnableUndo()
- else:
- del self.changesets[changeset]
-
- return ret
- def Undo(self, level = -1):
- """!Undo action
- @param level levels to undo (0 to revert all)
- @return id of current changeset
- """
- changesetLast = len(self.changesets.keys()) - 1
- if changesetLast < 0:
- return changesetLast
-
- if self.changesetCurrent == -2: # value uninitialized
- self.changesetCurrent = changesetLast
-
- if level > 0 and self.changesetCurrent < 0:
- self.changesetCurrent = 0
-
- if level == 0:
- # 0 -> undo all
- level = -1 * changesetLast + 1
- Debug.msg(2, "Digit.Undo(): changeset_last=%d, changeset_current=%d, level=%d",
- changesetLast, self.changesetCurrent, level)
-
- if level < 0: # undo
- if self.changesetCurrent + level < -1:
- return changesetCurrent;
- for changeset in range(self.changesetCurrent, self.changesetCurrent + level, -1):
- self._applyChangeset(changeset, undo = True)
- elif level > 0: # redo
- if self.changesetCurrent + level > len(self.changesets.keys()):
- return self.changesetCurrent
- for changeset in range(self.changesetCurrent, self.changesetCurrent + level):
- self._applyChangeset(changeset, undo = False)
-
- self.changesetCurrent += level
- Debug.msg(2, "Digit.Undo(): changeset_current=%d, changeset_last=%d, changeset_end=%d",
- self.changesetCurrent, changesetLast, self.changesetEnd)
-
- if self.changesetCurrent == self.changesetEnd:
- self.changesetEnd = changesetLast
- return -1
-
- self.mapWindow.UpdateMap(render = False)
-
- if self.changesetCurrent < 0: # disable undo tool
- self.toolbar.EnableUndo(False)
- def ZBulkLines(self, pos1, pos2, start, step):
- """!Z-bulk labeling
- @param pos1 reference line (start point)
- @param pos1 reference line (end point)
- @param start starting value
- @param step step value
- @return number of modified lines
- @return -1 on error
- """
- if not self._checkMap():
- return -1
-
- nlines = Vect_get_num_lines(self.poMapInfo)
-
- # register changeset
- changeset = self._addActionsBefore()
-
- poList = self._display.GetSelectedIList()
- ret = Vedit_bulk_labeling(self.poMapInfo, poList,
- pos1[0], pos1[1], pos2[0], pos2[1],
- start, step)
- Vect_destroy_list(poList)
-
- if ret > 0:
- self._addActionsAfter(changeset, nlines)
- self.toolbar.EnableUndo()
- else:
- del self.changesets[changeset]
-
- return ret
-
- def GetDisplay(self):
- """!Get display driver instance"""
- return self._display
-
- def OpenMap(self, name):
- """!Open vector map for editing
-
- @param map name of vector map to be set up
- """
- Debug.msg (3, "AbstractDigit.SetMapName map=%s" % name)
-
- name, mapset = name.split('@')
-
- self.poMapInfo = self._display.OpenMap(str(name), str(mapset), True)
-
- if self.poMapInfo:
- self.InitCats()
-
- return self.poMapInfo
-
- def CloseMap(self):
- """!Close currently open vector map
- """
- if not self._checkMap():
- return
-
- self._display.CloseMap()
- def InitCats(self):
- """!Initialize categories information
-
- @return 0 on success
- @return -1 on error
- """
- self.cats.clear()
- if not self._checkMap():
- return -1
-
- ndblinks = Vect_get_num_dblinks(self.poMapInfo)
- for i in range(ndblinks):
- fi = Vect_get_dblink(self.poMapInfo, i).contents
- if fi:
- self.cats[fi.number] = None
-
- # find max category
- nfields = Vect_cidx_get_num_fields(self.poMapInfo)
- Debug.msg(2, "wxDigit.InitCats(): nfields=%d", nfields)
-
- for i in range(nfields):
- field = Vect_cidx_get_field_number(self.poMapInfo, i)
- ncats = Vect_cidx_get_num_cats_by_index(self.poMapInfo, i)
- if field <= 0:
- continue
- for j in range(ncats):
- cat = c_int()
- type = c_int()
- id = c_int()
- Vect_cidx_get_cat_by_index(self.poMapInfo, i, j,
- byref(cat), byref(type), byref(id))
- if self.cats.has_key(field):
- if cat > self.cats[field]:
- self.cats[field] = cat.value
- else:
- self.cats[field] = cat.value
- Debug.msg(3, "wxDigit.InitCats(): layer=%d, cat=%d", field, self.cats[field])
-
- # set default values
- for field, cat in self.cats.iteritems():
- if cat == None:
- self.cats[field] = 0 # first category 1
- Debug.msg(3, "wxDigit.InitCats(): layer=%d, cat=%d", field, self.cats[field])
-
- def _checkMap(self):
- """!Check if map is open
- """
- if not self.poMapInfo:
- self._error.NoMap()
- return False
-
- return True
- def _addFeature(self, ftype, coords, layer, cat, snap, threshold):
- """!Add new feature(s) to the vector map
- @param ftype feature type (GV_POINT, GV_LINE, GV_BOUNDARY, ...)
- @coords tuple of coordinates ((x, y), (x, y), ...)
- @param layer layer number (-1 for no cat)
- @param cat category number
- @param snap snap to node/vertex
- @param threshold threshold for snapping
-
- @return tuple (number of added features, list of fids)
- @return number of features -1 on error
- """
- fids = list()
- if not self._checkMap():
- return (-1, None)
-
- is3D = bool(Vect_is_3d(self.poMapInfo))
-
- Debug.msg(2, "IVDigit._addFeature(): npoints=%d, layer=%d, cat=%d, snap=%d",
- len(coords), layer, cat, snap)
-
- if not (ftype & (GV_POINTS | GV_LINES | GV_AREA)): # TODO: 3D
- self._error.FeatureType(ftype)
- return (-1, None)
-
- # set category
- Vect_reset_cats(self.poCats)
- if layer > 0 and ftype != GV_AREA:
- Vect_cat_set(self.poCats, layer, cat)
- self.cats[layer] = max(cat, self.cats.get(layer, 1))
-
- # append points
- Vect_reset_line(self.poPoints)
- for c in coords:
- Vect_append_point(self.poPoints, c[0], c[1], 0.0)
-
- if ftype & (GV_BOUNDARY | GV_AREA):
- # close boundary
- points = self.poPoints.contents
- last = points.n_points - 1
- if Vect_points_distance(points.x[0], points.x[0], points.z[0],
- points.x[last], points.x[last], points.z[last],
- is3D) <= threshold:
- points.x[last] = points.x[0]
- points.y[last] = points.y[0]
- points.z[last] = points.z[0]
-
- if snap != NO_SNAP:
- # apply snapping (node or vertex)
- modeSnap = not (snap == SNAP)
- Vedit_snap_line(self.poMapInfo, self.popoBgMapInfo, int(self.poBgMapInfo is not None),
- -1, self.poPoints, threshold, modeSnap)
-
- if ftype == GV_AREA:
- ltype = GV_BOUNDARY
- else:
- ltype = ftype
- newline = Vect_write_line(self.poMapInfo, ltype, self.poPoints, self.poCats)
- if newline < 0:
- self._error.WriteLine()
- return (-1, None)
- else:
- fids.append(newline)
-
- left = right = -1
- if ftype & GV_AREA:
- # add centroids for left/right area
- bpoints = Vect_new_line_struct()
- cleft = c_int()
- cright = c_int()
-
- Vect_get_line_areas(self.poMapInfo, newline,
- byref(cleft), byref(cright))
- left = cleft.value
- right = cright.value
-
- # check if area exists and has no centroid inside
- if layer > 0 and (left > 0 or right > 0):
- Vect_cat_set(self.poCats, layer, cat)
- self.cats[layer] = max(cat, self.cats.get(layer, 0))
-
- x = c_double()
- y = c_double()
- if left > 0 and \
- Vect_get_area_centroid(self.poMapInfo, left) == 0:
- if Vect_get_area_points(self.poMapInfo, left, bpoints) > 0 and \
- Vect_find_poly_centroid(bpoints, byref(x), byref(y)) == 0:
- Vect_reset_line(bpoints)
- Vect_append_point(bpoints, x.value, y.value, 0.0)
- newline = Vect_write_line(self.poMapInfo, GV_CENTROID,
- bpoints, self.poCats)
- if newline < 0:
- self._error.WriteLine()
- return (len(fids), fids)
- else:
- fids.append(newline)
-
- if right > 0 and \
- Vect_get_area_centroid(self.poMapInfo, right) == 0:
- if Vect_get_area_points(byref(self.poMapInfo), right, bpoints) > 0 and \
- Vect_find_poly_centroid(bpoints, byref(x), byref(y)) == 0:
- Vect_reset_line(bpoints)
- Vect_append_point(bpoints, x.value, y.value, 0.0)
- newline = Vect_write_line(byref(self.poMapInfo), GV_CENTROID,
- bpoints, self.poCats)
- if newline < 0:
- self._error.WriteLine()
- return (len(fids, fids))
- else:
- fids.append(newline)
-
- Vect_destroy_line_struct(bpoints)
-
- # register changeset
- changeset = len(self.changesets)
- self._addActionToChangeset(changeset, newline, add = True)
-
- # break at intersection
- if self._settings['breakLines']:
- self._breakLineAtIntersection(newline, self.poPoints, changeset)
-
- return (len(fids), fids)
-
- def _ModifyLineVertex(self, coords, add = True):
- """!Add or remove vertex
-
- Shape of line/boundary is not changed when adding new vertex.
-
- @param coords coordinates of point
- @param add True to add, False to remove
-
- @return id id of the new feature
- @return 0 nothing changed
- @return -1 error
- """
- if not self._checkMap():
- return -1
-
- selected = self._display.selected
- if len(selected['ids']) != 1:
- return 0
-
- poList = self._display.GetSelectedIList()
- Vect_reset_line(self.poPoints)
- Vect_append_point(self.poPoints, coords[0], coords[1], 0.0)
-
- nlines = Vect_get_num_lines(self.poMapInfo)
- thresh = self._display.GetThreshold(type = 'selectThresh')
-
- changeset = self._addActionsBefore()
-
- if add:
- ret = Vedit_add_vertex(self.poMapInfo, poList,
- self.poPoints, thresh)
- else:
- ret = Vedit_remove_vertex(self.poMapInfo, poList,
- self.poPoints, thresh)
- Vect_destroy_list(poList)
-
- if ret > 0:
- self._addActionsAfter(changeset, nlines)
- else:
- del self.changesets[changeset]
-
- if not add and ret > 0 and self._settings['breakLines']:
- self._breakLineAtIntersection(Vect_get_num_lines(self.poMapInfo),
- None, changeset)
-
- return nlines + 1 # feature is write at the end of the file
-
- def GetLineCats(self, line):
- """!Get list of layer/category(ies) for selected feature.
- @param line feature id (-1 for first selected feature)
- @return list of layer/cats
- """
- ret = dict()
- if not self._checkMap():
- return ret
-
- if line == -1 and len(self._display.selected['ids']) < 1:
- return ret
-
- if line == -1:
- line = self._display.selected['ids'][0]
-
- if not Vect_line_alive(self.poMapInfo, line):
- self._error.DeadLine(line)
- return ret
-
- if Vect_read_line(self.poMapInfo, None, self.poCats, line) < 0:
- self._error.ReadLine(line)
- return ret
-
- cats = self.poCats.contents
- for i in range(cats.n_cats):
- field = cats.field[i]
- if field not in ret:
- ret[field] = list()
- ret[field].append(cats.cat[i])
-
- return ret
- def GetLayers(self):
- """!Get list of layers
-
- Requires self.InitCats() to be called.
- @return list of layers
- """
- return self.cats.keys()
-
- def UpdateSettings(self):
- """!Update digit (and display) settings
- """
- self._display.UpdateSettings()
-
- self._settings['breakLines'] = bool(UserSettings.Get(group = 'vdigit', key = "breakLines",
- subkey = 'enabled'))
-
- def SetCategory(self):
- """!Update self.cats based on settings"""
- sel = UserSettings.Get(group = 'vdigit', key = 'categoryMode', subkey = 'selection')
- cat = None
- if sel == 0: # next to usep
- cat = self._setCategoryNextToUse()
- elif sel == 1:
- cat = UserSettings.Get(group = 'vdigit', key = 'category', subkey = 'value')
-
- if cat:
- layer = UserSettings.Get(group = 'vdigit', key = 'layer', subkey = 'value')
- self.cats[layer] = cat
-
- return cat
-
- def _setCategoryNextToUse(self):
- """!Find maximum category number for the given layer and
- update the settings
- @return category to be used
- """
- # get max category number for given layer and update the settings
- layer = UserSettings.Get(group = 'vdigit', key = 'layer', subkey = 'value')
- cat = self.cats.get(layer, 0) + 1
- UserSettings.Set(group = 'vdigit', key = 'category', subkey = 'value',
- value = cat)
- Debug.msg(1, "IVDigit._setCategoryNextToUse(): cat=%d", cat)
-
- return cat
- def SelectLinesFromBackgroundMap(self, bbox):
- """!Select features from background map
- @param bbox bounding box definition
-
- @return list of selected feature ids
- """
- ret = list()
-
- # try select features by Box
- ids = self._display.SelectLinesByBox(bbox, poMapInfo = self.poBgMapInfo)
- if not ids:
- ids = [self._display.SelectLineByPoint(bbox[0], poMapInfo = self.poBgMapInfo)['line'], ]
-
- return ids
- def GetUndoLevel(self):
- """!Get undo level (number of active changesets)
-
- Note: Changesets starts wiht 0
- """
- return self.changesetCurrent
-
|