frame.py 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626
  1. """!
  2. @package mapswipe.frame
  3. @brief Map Swipe Frame
  4. Classes:
  5. - dialogs::SwipeMapDialog
  6. (C) 2012 by the GRASS Development Team
  7. This program is free software under the GNU General Public License
  8. (>=v2). Read the file COPYING that comes with GRASS for details.
  9. @author Anna Kratochvilova <kratochanna gmail.com>
  10. """
  11. import os
  12. import sys
  13. import wx
  14. import time
  15. import grass.script as grass
  16. from gui_core.mapdisp import DoubleMapFrame
  17. from gui_core.dialogs import GetImageHandlers
  18. from core.render import Map
  19. from mapdisp import statusbar as sb
  20. from core.debug import Debug
  21. from core.gcmd import RunCommand, GError, GMessage
  22. from mapdisp.statusbar import EVT_AUTO_RENDER
  23. from mapswipe.toolbars import SwipeMapToolbar, SwipeMainToolbar, SwipeMiscToolbar
  24. from mapswipe.mapwindow import SwipeBufferedWindow
  25. from mapswipe.dialogs import SwipeMapDialog
  26. class SwipeMapFrame(DoubleMapFrame):
  27. def __init__(self, parent = None, giface = None,
  28. title = _("GRASS GIS Map Swipe"), name = "swipe", **kwargs):
  29. DoubleMapFrame.__init__(self, parent = parent, title = title, name = name,
  30. firstMap = Map(), secondMap = Map(), **kwargs)
  31. Debug.msg (1, "SwipeMapFrame.__init__()")
  32. #
  33. # Add toolbars
  34. #
  35. toolbars = ['swipeMisc', 'swipeMap', 'swipeMain']
  36. if sys.platform == 'win32':
  37. self.AddToolbar(toolbars.pop(1))
  38. toolbars.reverse()
  39. else:
  40. self.AddToolbar(toolbars.pop(0))
  41. for toolb in toolbars:
  42. self.AddToolbar(toolb)
  43. self._giface = giface
  44. #
  45. # create widgets
  46. #
  47. self.splitter = MapSplitter(parent = self, id = wx.ID_ANY)
  48. self.sliderH = wx.Slider(self, id = wx.ID_ANY, style = wx.SL_HORIZONTAL)
  49. self.sliderV = wx.Slider(self, id = wx.ID_ANY, style = wx.SL_VERTICAL)
  50. self.firstMapWindow = SwipeBufferedWindow(parent = self.splitter, giface = self._giface,
  51. Map = self.firstMap, frame = self)
  52. self.secondMapWindow = SwipeBufferedWindow(parent = self.splitter, giface = self._giface,
  53. Map = self.secondMap, frame = self)
  54. self.MapWindow = self.firstMapWindow # current by default
  55. self.firstMapWindow.zoomhistory = self.secondMapWindow.zoomhistory
  56. self.SetBindRegions(True)
  57. self._mode = 'swipe'
  58. self.splitter.SplitVertically(self.firstMapWindow, self.secondMapWindow, 0)
  59. self._addPanes()
  60. self._bindWindowsActivation()
  61. self._mgr.GetPane('sliderV').Hide()
  62. self._mgr.GetPane('sliderH').Show()
  63. self.slider = self.sliderH
  64. self.InitStatusbar()
  65. self.Bind(wx.EVT_SIZE, self.OnSize)
  66. self.Bind(EVT_AUTO_RENDER, self.OnAutoRenderChanged)
  67. self.Bind(wx.EVT_IDLE, self.OnIdle)
  68. self.Bind(wx.EVT_CLOSE, self.OnCloseWindow)
  69. self.SetSize((800, 600))
  70. self._mgr.Update()
  71. self.rasters = {'first': None, 'second': None}
  72. # default action in map toolbar
  73. self.OnPan(event = None)
  74. self.resize = False
  75. wx.CallAfter(self.CallAfterInit)
  76. def CallAfterInit(self):
  77. self.InitSliderBindings()
  78. if not (self.rasters['first'] and self.rasters['second']):
  79. self.OnSelectRasters(event = None)
  80. def InitStatusbar(self):
  81. """!Init statusbar (default items)."""
  82. # items for choice
  83. self.statusbarItems = [sb.SbCoordinates,
  84. sb.SbRegionExtent,
  85. sb.SbCompRegionExtent,
  86. sb.SbShowRegion,
  87. sb.SbAlignExtent,
  88. sb.SbResolution,
  89. sb.SbDisplayGeometry,
  90. sb.SbMapScale,
  91. sb.SbGoTo,
  92. sb.SbProjection]
  93. # create statusbar and its manager
  94. statusbar = self.CreateStatusBar(number = 4, style = 0)
  95. statusbar.SetStatusWidths([-5, -2, -1, -1])
  96. self.statusbarManager = sb.SbManager(mapframe = self, statusbar = statusbar)
  97. # fill statusbar manager
  98. self.statusbarManager.AddStatusbarItemsByClass(self.statusbarItems, mapframe = self, statusbar = statusbar)
  99. self.statusbarManager.AddStatusbarItem(sb.SbMask(self, statusbar = statusbar, position = 2))
  100. self.statusbarManager.AddStatusbarItem(sb.SbRender(self, statusbar = statusbar, position = 3))
  101. self.statusbarManager.Update()
  102. def ResetSlider(self):
  103. if self.splitter.GetSplitMode() == wx.SPLIT_VERTICAL:
  104. size = self.splitter.GetSize()[0]
  105. else:
  106. size = self.splitter.GetSize()[1]
  107. self.slider.SetRange(0, size)
  108. self.slider.SetValue(self.splitter.GetSashPosition())
  109. def InitSliderBindings(self):
  110. self.sliderH.Bind(wx.EVT_SPIN, self.OnSliderPositionChanging)
  111. self.sliderH.Bind(wx.EVT_SCROLL_THUMBRELEASE, self.OnSliderPositionChanged)
  112. self.sliderV.Bind(wx.EVT_SPIN, self.OnSliderPositionChanging)
  113. self.sliderV.Bind(wx.EVT_SCROLL_THUMBRELEASE, self.OnSliderPositionChanged)
  114. self.splitter.Bind(wx.EVT_SPLITTER_SASH_POS_CHANGING, self.OnSashChanging)
  115. self.splitter.Bind(wx.EVT_SPLITTER_SASH_POS_CHANGED, self.OnSashChanged)
  116. def OnSliderPositionChanging(self, event):
  117. """!Slider changes its position, sash must be moved too."""
  118. Debug.msg (5, "SwipeMapFrame.OnSliderPositionChanging()")
  119. self.GetFirstWindow().movingSash = True
  120. self.GetSecondWindow().movingSash = True
  121. pos = event.GetPosition()
  122. if pos > 0:
  123. self.splitter.SetSashPosition(pos)
  124. self.splitter.OnSashChanging(None)
  125. def OnSliderPositionChanged(self, event):
  126. """!Slider position changed, sash must be moved too."""
  127. Debug.msg (5, "SwipeMapFrame.OnSliderPositionChanged()")
  128. self.splitter.SetSashPosition(event.GetPosition())
  129. self.splitter.OnSashChanged(None)
  130. def OnSashChanging(self, event):
  131. """!Sash position is changing, slider must be moved too."""
  132. Debug.msg (5, "SwipeMapFrame.OnSashChanging()")
  133. self.slider.SetValue(self.splitter.GetSashPosition())
  134. event.Skip()
  135. def OnSashChanged(self, event):
  136. """!Sash position changed, slider must be moved too."""
  137. Debug.msg (5, "SwipeMapFrame.OnSashChanged()")
  138. self.OnSashChanging(event)
  139. event.Skip()
  140. def OnSize(self, event):
  141. Debug.msg (4, "SwipeMapFrame.OnSize()")
  142. self.resize = time.clock()
  143. def OnIdle(self, event):
  144. if self.resize and time.clock() - self.resize > 0.2:
  145. w1 = self.GetFirstWindow()
  146. w2 = self.GetSecondWindow()
  147. sizeAll = self.splitter.GetSize()
  148. w1.SetClientSize(sizeAll)
  149. w2.SetClientSize(sizeAll)
  150. w1.OnSize(event)
  151. w2.OnSize(event)
  152. self.ResetSlider()
  153. self.resize = False
  154. def OnAutoRenderChanged(self, event):
  155. """!Auto rendering state changed."""
  156. style = self.splitter.GetWindowStyle()
  157. style ^= wx.SP_LIVE_UPDATE
  158. self.splitter.SetWindowStyle(style)
  159. def AddToolbar(self, name):
  160. """!Add defined toolbar to the window
  161. Currently known toolbars are:
  162. - 'swipeMap' - basic map toolbar
  163. - 'swipeMain' - swipe functionality
  164. """
  165. if name == "swipeMap":
  166. self.toolbars[name] = SwipeMapToolbar(self)
  167. self._mgr.AddPane(self.toolbars[name],
  168. wx.aui.AuiPaneInfo().
  169. Name(name).Caption(_("Map Toolbar")).
  170. ToolbarPane().Top().
  171. LeftDockable(False).RightDockable(False).
  172. BottomDockable(False).TopDockable(True).
  173. CloseButton(False).Layer(2).Row(1).
  174. BestSize((self.toolbars[name].GetBestSize())))
  175. if name == "swipeMain":
  176. self.toolbars[name] = SwipeMainToolbar(self)
  177. self._mgr.AddPane(self.toolbars[name],
  178. wx.aui.AuiPaneInfo().
  179. Name(name).Caption(_("Main Toolbar")).
  180. ToolbarPane().Top().
  181. LeftDockable(False).RightDockable(False).
  182. BottomDockable(False).TopDockable(True).
  183. CloseButton(False).Layer(2).Row(1).
  184. BestSize((self.toolbars[name].GetBestSize())))
  185. if name == "swipeMisc":
  186. self.toolbars[name] = SwipeMiscToolbar(self)
  187. self._mgr.AddPane(self.toolbars[name],
  188. wx.aui.AuiPaneInfo().
  189. Name(name).Caption(_("Misc Toolbar")).
  190. ToolbarPane().Top().
  191. LeftDockable(False).RightDockable(False).
  192. BottomDockable(False).TopDockable(True).
  193. CloseButton(False).Layer(2).Row(1).
  194. BestSize((self.toolbars[name].GetBestSize())))
  195. def _addPanes(self):
  196. """!Add splitter window and sliders to aui manager"""
  197. # splitter window
  198. self._mgr.AddPane(self.splitter, wx.aui.AuiPaneInfo().
  199. Name('splitter').CaptionVisible(False).PaneBorder(True).
  200. Dockable(False).Floatable(False).CloseButton(False).
  201. Center().Layer(1).BestSize((self.splitter.GetBestSize())))
  202. # sliders
  203. self._mgr.AddPane(self.sliderH, wx.aui.AuiPaneInfo().
  204. Name('sliderH').CaptionVisible(False).PaneBorder(False).
  205. CloseButton(False).Gripper(True).GripperTop(False).
  206. BottomDockable(True).TopDockable(True).
  207. LeftDockable(False).RightDockable(False).
  208. Bottom().Layer(1).BestSize((self.sliderH.GetBestSize())))
  209. self._mgr.AddPane(self.sliderV, wx.aui.AuiPaneInfo().
  210. Name('sliderV').CaptionVisible(False).PaneBorder(False).
  211. CloseButton(False).Gripper(True).GripperTop(True).
  212. BottomDockable(False).TopDockable(False).
  213. LeftDockable(True).RightDockable(True).
  214. Right().Layer(1).BestSize((self.sliderV.GetBestSize())))
  215. def ZoomToMap(self):
  216. """!
  217. Set display extents to match selected raster (including NULLs)
  218. or vector map.
  219. """
  220. if self.rasters['first']:
  221. self.GetFirstWindow().ZoomToMap(layers = self.firstMap.GetListOfLayers())
  222. if self.rasters['second']:
  223. self.GetSecondWindow().ZoomToMap(layers = self.secondMap.GetListOfLayers())
  224. def OnZoomToMap(self, event):
  225. """!Zoom to map"""
  226. self.ZoomToMap()
  227. def OnZoomBack(self, event):
  228. self.GetFirstWindow().ZoomBack()
  229. self.secondMap.region = self.firstMap.region
  230. self.Render(self.GetSecondWindow())
  231. def OnSelectRasters(self, event):
  232. """!Choose raster maps and rerender."""
  233. dlg = SwipeMapDialog(self, first = self.rasters['first'], second = self.rasters['second'])
  234. if dlg.ShowModal() == wx.ID_OK:
  235. maps = dlg.GetValues()
  236. res1 = self.SetFirstRaster(name = maps[0])
  237. res2 = self.SetSecondRaster(name = maps[1])
  238. if not (res1 and res2):
  239. message = ''
  240. if not res1:
  241. message += _("Map <%s> not found. ") % maps[0]
  242. if not res2:
  243. message += _("Map <%s> not found.") % maps[1]
  244. GError(parent = self, message = message)
  245. dlg.Destroy()
  246. self.SetRasterNames()
  247. self.ZoomToMap()
  248. dlg.Destroy()
  249. self.OnRender(event = None)
  250. def SetFirstRaster(self, name):
  251. """!Set raster map to first Map"""
  252. raster = grass.find_file(name = name, element = 'cell')
  253. if raster['fullname']:
  254. self.rasters['first'] = raster['fullname']
  255. self.SetLayer(name = raster['fullname'], mapInstance = self.GetFirstMap())
  256. return True
  257. return False
  258. def SetSecondRaster(self, name):
  259. """!Set raster map to second Map"""
  260. raster = grass.find_file(name = name, element = 'cell')
  261. if raster['fullname']:
  262. self.rasters['second'] = raster['fullname']
  263. self.SetLayer(name = raster['fullname'], mapInstance = self.GetSecondMap())
  264. return True
  265. return False
  266. def SetLayer(self, name, mapInstance):
  267. """!Sets layer in Map.
  268. @param name layer (raster) name
  269. """
  270. Debug.msg (3, "SwipeMapFrame.SetLayer(): name=%s" % name)
  271. # this simple application enables to keep only one raster
  272. mapInstance.DeleteAllLayers()
  273. cmdlist = ['d.rast', 'map=%s' % name]
  274. # add layer to Map instance (core.render)
  275. newLayer = mapInstance.AddLayer(type = 'raster', command = cmdlist, l_active = True,
  276. name = name, l_hidden = False, l_opacity = 1.0,
  277. l_render = True)
  278. def OnSwitchWindows(self, event):
  279. """!Switch windows position."""
  280. Debug.msg(3, "SwipeMapFrame.OnSwitchWindows()")
  281. splitter = self.splitter
  282. w1, w2 = splitter.GetWindow1(), splitter.GetWindow2()
  283. splitter.ReplaceWindow(w1, w2)
  284. splitter.ReplaceWindow(w2, w1)
  285. # self.OnSize(None)
  286. splitter.OnSashChanged(None)
  287. def _saveToFile(self, fileName, fileType):
  288. """!Creates composite image by rendering both images and
  289. pasting them into the new one.
  290. @todo specify size of the new image (problem is inaccurate scaling)
  291. @todo make dividing line width and color optional
  292. """
  293. w1 = self.splitter.GetWindow1()
  294. w2 = self.splitter.GetWindow2()
  295. lineWidth = 1
  296. # render to temporary files
  297. filename1 = grass.tempfile(False) + '1'
  298. filename2 = grass.tempfile(False) + '2'
  299. width, height = self.splitter.GetClientSize()
  300. if self._mode == 'swipe':
  301. x, y = w2.GetImageCoords()
  302. w1.SaveToFile(filename1, fileType, width, height)
  303. w2.SaveToFile(filename2, fileType, width, height)
  304. else:
  305. fw, fh = w1.GetClientSize()
  306. w1.SaveToFile(filename1, fileType, fw, fh)
  307. sw, sh = w2.GetClientSize()
  308. w2.SaveToFile(filename2, fileType, sw, sh)
  309. # create empty white image - needed for line
  310. im = wx.EmptyImage(width, height)
  311. im.Replace(0, 0, 0, 255, 255, 255)
  312. # paste images
  313. if self._mode == 'swipe':
  314. if self.splitter.GetSplitMode() == wx.SPLIT_HORIZONTAL:
  315. im1 = wx.Image(filename1).GetSubImage((0, 0, width, -y))
  316. im.Paste(im1, 0, 0)
  317. im.Paste(wx.Image(filename2), -x, -y + lineWidth)
  318. else:
  319. im1 = wx.Image(filename1).GetSubImage((0, 0, -x, height))
  320. im.Paste(im1, 0, 0)
  321. im.Paste(wx.Image(filename2), -x + lineWidth, -y)
  322. else:
  323. if self.splitter.GetSplitMode() == wx.SPLIT_HORIZONTAL:
  324. im1 = wx.Image(filename1)
  325. im.Paste(im1, 0, 0)
  326. im.Paste(wx.Image(filename2), 0, fh + lineWidth)
  327. else:
  328. im1 = wx.Image(filename1)
  329. im.Paste(im1, 0, 0)
  330. im.Paste(wx.Image(filename2), fw + lineWidth, 0)
  331. im.SaveFile(fileName, fileType)
  332. # remove temporary files
  333. grass.try_remove(filename1)
  334. grass.try_remove(filename2)
  335. def SaveToFile(self, event):
  336. """!Save map to image
  337. """
  338. img = self.firstMapWindow.img or self.secondMapWindow.img
  339. if not img:
  340. GMessage(parent = self,
  341. message = _("Nothing to render (empty map). Operation canceled."))
  342. return
  343. filetype, ltype = GetImageHandlers(img)
  344. # get filename
  345. dlg = wx.FileDialog(parent = self,
  346. message = _("Choose a file name to save the image "
  347. "(no need to add extension)"),
  348. wildcard = filetype,
  349. style = wx.SAVE | wx.FD_OVERWRITE_PROMPT)
  350. if dlg.ShowModal() == wx.ID_OK:
  351. path = dlg.GetPath()
  352. if not path:
  353. dlg.Destroy()
  354. return
  355. base, ext = os.path.splitext(path)
  356. fileType = ltype[dlg.GetFilterIndex()]['type']
  357. extType = ltype[dlg.GetFilterIndex()]['ext']
  358. if ext != extType:
  359. path = base + '.' + extType
  360. self._saveToFile(path, fileType)
  361. dlg.Destroy()
  362. def OnSwitchOrientation(self, event):
  363. """!Switch orientation of the sash."""
  364. Debug.msg(3, "SwipeMapFrame.OnSwitchOrientation()")
  365. splitter = self.splitter
  366. splitter.Unsplit()
  367. if splitter.GetSplitMode() == wx.SPLIT_HORIZONTAL:
  368. splitter.SplitVertically(self.firstMapWindow, self.secondMapWindow, 0)
  369. self.slider = self.sliderH
  370. if self._mode == 'swipe':
  371. self._mgr.GetPane('sliderH').Show()
  372. self._mgr.GetPane('sliderV').Hide()
  373. else:
  374. splitter.SplitHorizontally(self.firstMapWindow, self.secondMapWindow, 0)
  375. self.slider = self.sliderV
  376. if self._mode == 'swipe':
  377. self._mgr.GetPane('sliderV').Show()
  378. self._mgr.GetPane('sliderH').Hide()
  379. self._mgr.Update()
  380. splitter.OnSashChanged(None)
  381. self.OnSize(None)
  382. self.SetRasterNames()
  383. def OnAddText(self, event):
  384. """!Double click on text overlay
  385. So far not implemented.
  386. """
  387. pass
  388. def SetViewMode(self, mode):
  389. """!Sets view mode.
  390. @param mode view mode ('swipe', 'mirror')
  391. """
  392. if self._mode == mode:
  393. return
  394. self._mode = mode
  395. self.toolbars['swipeMain'].SetMode(mode)
  396. # set window mode
  397. self.GetFirstWindow().SetMode(mode)
  398. self.GetSecondWindow().SetMode(mode)
  399. # hide/show slider
  400. if self.splitter.GetSplitMode() == wx.SPLIT_HORIZONTAL:
  401. self._mgr.GetPane('sliderV').Show(mode == 'swipe')
  402. size = self.splitter.GetSize()[1] / 2
  403. else:
  404. self._mgr.GetPane('sliderH').Show(mode == 'swipe')
  405. size = self.splitter.GetSize()[0] / 2
  406. # set sash in the middle
  407. self.splitter.SetSashPosition(size)
  408. self.slider.SetValue(size)
  409. self._mgr.Update()
  410. # enable / disable sash
  411. self.splitter.EnableSash(mode == 'swipe')
  412. # hack to make it work
  413. self.splitter.OnSashChanged(None)
  414. self.SendSizeEvent()
  415. def SetRasterNames(self):
  416. if self.rasters['first']:
  417. self.GetFirstWindow().SetRasterNameText(self.rasters['first'], 101)
  418. if self.rasters['second']:
  419. self.GetSecondWindow().SetRasterNameText(self.rasters['second'], 102)
  420. def GetMapToolbar(self):
  421. """!Returns toolbar with zooming tools"""
  422. return self.toolbars['swipeMap']
  423. def IsStandalone(self):
  424. """!Since we do not need layer manager, we are standalone"""
  425. return True
  426. def OnHelp(self, event):
  427. self._giface.Help(entry = 'wxGUI.mapswipe')
  428. def OnCloseWindow(self, event):
  429. self.GetFirstMap().Clean()
  430. self.GetSecondMap().Clean()
  431. self.Destroy()
  432. class MapSplitter(wx.SplitterWindow):
  433. """!Splitter window for displaying two maps"""
  434. def __init__(self, parent, id):
  435. wx.SplitterWindow.__init__(self, parent = parent, id = id,
  436. style = wx.SP_LIVE_UPDATE
  437. )
  438. Debug.msg(2, "MapSplitter.__init__()")
  439. self.sashWidthMin = 1
  440. self.sashWidthMax = 10
  441. self.Bind(wx.EVT_SPLITTER_SASH_POS_CHANGED, self.OnSashChanged)
  442. self.Bind(wx.EVT_SPLITTER_SASH_POS_CHANGING, self.OnSashChanging)
  443. self._moveSash = True
  444. wx.CallAfter(self.Init)
  445. def EnableSash(self, enable):
  446. self._moveSash = enable
  447. def Init(self):
  448. self.OnSashChanged(evt = None)
  449. self.SetMinimumPaneSize(0)
  450. self.SetSashSize(self.sashWidthMin)
  451. # def OnMotion(self, event):
  452. # w = self.GetSashSize()
  453. # w1, w2 = self.GetWindow1(), self.GetWindow2()
  454. # if self.SashHitTest(event.GetX(), event.GetY(), tolerance = 20):
  455. # if w == self.sashWidthMin:
  456. # self.SetSashSize(self.sashWidthMax)
  457. # self.SetNeedUpdating(True)
  458. # w1.movingSash = True
  459. # w2.movingSash = True
  460. # else:
  461. # w1.movingSash = False
  462. # w1.movingSash = False
  463. # else:
  464. # if w == self.sashWidthMax:
  465. # self.SetSashSize(self.sashWidthMin)
  466. # self.SetNeedUpdating(True)
  467. # w1.movingSash = True
  468. # w2.movingSash = True
  469. # else:
  470. # w1.movingSash = False
  471. # w2.movingSash = False
  472. # event.Skip()
  473. def OnSashChanged(self, evt):
  474. Debug.msg(5, "MapSplitter.OnSashChanged()")
  475. if not self._moveSash:
  476. return
  477. w1, w2 = self.GetWindow1(), self.GetWindow2()
  478. w1.movingSash = False
  479. w2.movingSash = False
  480. wx.CallAfter(self.SashChanged)
  481. def SashChanged(self):
  482. Debug.msg(5, "MapSplitter.SashChanged()")
  483. w1, w2 = self.GetWindow1(), self.GetWindow2()
  484. w1.SetImageCoords((0, 0))
  485. if self.GetSplitMode() == wx.SPLIT_VERTICAL:
  486. w = w1.GetSize()[0]
  487. w2.SetImageCoords((-w, 0))
  488. else:
  489. h = w1.GetSize()[1]
  490. w2.SetImageCoords((0, -h))
  491. w1.UpdateMap(render = False, renderVector = False)
  492. w2.UpdateMap(render = False, renderVector = False)
  493. pos = self.GetSashPosition()
  494. self.last = pos
  495. def OnSashChanging(self, event):
  496. Debug.msg(5, "MapSplitter.OnSashChanging()")
  497. if not self._moveSash:
  498. event.SetSashPosition(-1)
  499. return
  500. if not (self.GetWindowStyle() & wx.SP_LIVE_UPDATE):
  501. if event:
  502. event.Skip()
  503. return
  504. pos = self.GetSashPosition()
  505. dpos = pos - self.last
  506. self.last = pos
  507. if self.GetSplitMode() == wx.SPLIT_VERTICAL:
  508. dx = -dpos
  509. dy = 0
  510. else:
  511. dx = 0
  512. dy = -dpos
  513. self.GetWindow2().TranslateImage(dx, dy)
  514. self.GetWindow1().movingSash = True
  515. self.GetWindow2().movingSash = True