Mercurial > fife-parpg
diff tools/editor/scripts/gui/mapeditor.py @ 418:70ba57cd9d18
Documented, commented, cleaned and updated mapeditor.py to match new python coding guidelines.
author | cheesesucker@33b003aa-7bff-0310-803a-e67f0ece8222 |
---|---|
date | Sat, 06 Feb 2010 18:50:36 +0000 |
parents | 64738befdf3b |
children | b2feacaed53c |
line wrap: on
line diff
--- a/tools/editor/scripts/gui/mapeditor.py Wed Feb 03 20:04:39 2010 +0000 +++ b/tools/editor/scripts/gui/mapeditor.py Sat Feb 06 18:50:36 2010 +0000 @@ -21,14 +21,27 @@ # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA # #################################################################### -# MapEditor is a plugin for Fifedit. It allows for selection and visual editing of maps. +""" +Visual Map Editor +================= + +The map editor provides the user with an interface for editing +maps visually. -import math, os, time +To edit a map through code, use MapController. + +""" + +import math +import os +import time from datetime import date from fife import fife from fife.extensions import pychan import fife.extensions.pychan.widgets as widgets +from fife.extensions import fife_utils + import scripts import scripts.events as events import action @@ -36,48 +49,75 @@ from menubar import Menu, MenuBar from action import Action, ActionGroup from scripts.mapcontroller import MapController -from fife.extensions import fife_utils - -states = (u'SELECTING', u'INSERTING', u'REMOVING', u'MOVING', u'OBJECTPICKER') -for s in states: - globals()[s] = s -NOT_INITIALIZED = -9999999 class MapEditor: - def __init__(self): + """ This class provides a basic user interface for map editing. It allows the user + to visually edit a map. + """ + + # Editor states + SELECTING = u"Selecting" + INSERTING = u"Inserting" + REMOVING = u"Removing" + MOVING = u"Moving" + OBJECTPICKER = u"Objectpicking" + + def __init__(self): + """ Set up all variables and call some initial functions """ self._ignoreToggles = False # A hack to avoid infinite recursion when toggling a button self._controller = None - self._mode = SELECTING + self._mode = MapEditor.SELECTING + self._debug = False # TODO: We should have a central system for activating debug messages + # GUI self._editor = scripts.editor.getEditor() self._eventlistener = self._editor.getEventListener() self._statusbar = self._editor.getStatusBar() self._toolbar = self._editor.getToolBar() - self._object = None - self._startDragPos = None - self._lastDragPos = None - self._lastDragPosExact = None - self._toolbox = self._editor.getToolbox() - self._initToolbuttons() + # Currently selected object type + self._object = None + + # Variables used for moving instances + self._last_drag_pos = None + self._last_drag_pos_exact = None + + self._selected_instances = [] + self._undogroup = False + + # Variables for scrolling the map + self._dragx = 0 + self._dragy = 0 + self._scrollX = 0 + self._scrollY = 0 + + self._initToolboxButtons() self._toolbox.show() - self._instances = [] events.postMapShown.connect(self._mapChanged) events.preMapClosed.connect(self._mapClosed) events.onObjectSelected.connect(self.setObject) - self._undogroup = False + + def setObject(self, object): + """ Set the object type to be paint onto map """ + self._object = object - self._scrollX = 0 - self._scrollY = 0 + def setController(self, controller): + """ Set the controller to use. """ + if self._controller is not None: + self._clear() + + self._controller = controller + + if self._controller is not None: + self._init() def _init(self): - self._dragx = NOT_INITIALIZED - self._dragy = NOT_INITIALIZED - - self._setMode(SELECTING) + """ Sets up the mapeditor to work with the selected controller """ + self._debug = self._controller.debug + self._setMode(MapEditor.SELECTING) self._initToolbarbuttons() @@ -95,6 +135,7 @@ events.onPump.connect(self.pump) def _clear(self): + """ Remove any functionality set up by _init """ self._clearToolbarButtons() events.keyPressed.disconnect(self.keyPressed) @@ -111,37 +152,47 @@ events.onPump.disconnect(self.pump) def _mapChanged(self, sender, mapview): + """ Called when a new map is selected. + Sets the mapeditor to control the mapcontroller in the new map + """ self.setController(mapview.getController()) def _mapClosed(self, sender, mapview): - self.setController(None) + """ Called when a map is closed. """ + if mapview.getMap().getId() == self._controller.getMap().getId(): + self.setController(None) - def _setCursor(self): + def _updateCursor(self): + """ Updates the cursor to reflect the mode of the editor """ engine = self._editor.getEngine() cursor = engine.getCursor() id = -1 - if self._mode == SELECTING: + if self._mode == MapEditor.SELECTING: id = engine.getImagePool().addResourceFromFile("gui/icons/select_instance.png") image = engine.getImagePool().getImage(id) image.setXShift(-7) image.setYShift(-7) - elif self._mode == INSERTING: + + elif self._mode == MapEditor.INSERTING: id = engine.getImagePool().addResourceFromFile("gui/icons/add_instance.png") image = engine.getImagePool().getImage(id) image.setXShift(-2) image.setYShift(-20) - elif self._mode == REMOVING: + + elif self._mode == MapEditor.REMOVING: id = engine.getImagePool().addResourceFromFile("gui/icons/erase_instance.png") image = engine.getImagePool().getImage(id) image.setXShift(-2) image.setYShift(-19) - elif self._mode == MOVING: + + elif self._mode == MapEditor.MOVING: id = engine.getImagePool().addResourceFromFile("gui/icons/move_instance.png") image = engine.getImagePool().getImage(id) image.setXShift(-11) image.setYShift(-11) - elif self._mode == OBJECTPICKER: + + elif self._mode == MapEditor.OBJECTPICKER: id = engine.getImagePool().addResourceFromFile("gui/icons/objectpicker.png") image = engine.getImagePool().getImage(id) image.setXShift(-0) @@ -153,22 +204,13 @@ cursor.set(fife.CURSOR_IMAGE, id) def _resetCursor(self): + """ Reset the cursor to the standard native one """ cursor = self._editor.getEngine().getCursor() cursor.set(fife.CURSOR_NATIVE, fife.NC_ARROW) - def setObject(self, object): - self._object = object + def _initToolboxButtons(self): + """ Sets up and connects buttons related to the toolbox """ - def setController(self, controller): - if self._controller is not None: - self._clear() - - if controller is not None: - self._init() - - self._controller = controller - - def _initToolbuttons(self): self._selectAction = Action(text=u"Select", icon="gui/icons/select_instance.png", checkable=True) self._drawAction = Action(text=u"Draw", icon="gui/icons/add_instance.png", checkable=True) self._removeAction = Action(text=u"Remove", icon="gui/icons/erase_instance.png", checkable=True) @@ -194,13 +236,14 @@ self._toolgroup.addAction(self._removeAction) self._toolgroup.addAction(self._objectpickerAction) - self._toolbox.addAction(self._toolgroup) self._toolbox.adaptLayout() self._editor._edit_menu.addAction(self._toolgroup) def _initToolbarbuttons(self): + """ Sets up and connects buttons related to the toolbar """ + rotateLeftAction = Action(text=u"Rotate counterclockwise", icon="gui/icons/rotate_countercw.png") rotateRightAction = Action(text=u"Rotate clockwise", icon="gui/icons/rotate_clockwise.png") zoomInAction = Action(text=u"Zoom in", icon="gui/icons/zoom_in.png") @@ -215,12 +258,12 @@ zoomResetAction.helptext = u"Reset zoom to default level" screenshotAction.helptext = u"Take screenshot (F7)" - action.activated.connect(self.rotateCounterClockwise, sender=rotateLeftAction) - action.activated.connect(self.rotateClockwise, sender=rotateRightAction) - action.activated.connect(self.zoomIn, sender=zoomInAction) - action.activated.connect(self.zoomOut, sender=zoomOutAction) - action.activated.connect(self.resetZoom, sender=zoomResetAction) - action.activated.connect(self.captureScreen, sender=screenshotAction) + action.activated.connect(self._rotateCounterClockwise, sender=rotateLeftAction) + action.activated.connect(self._rotateClockwise, sender=rotateRightAction) + action.activated.connect(self._zoomIn, sender=zoomInAction) + action.activated.connect(self._zoomOut, sender=zoomOutAction) + action.activated.connect(self._resetZoom, sender=zoomResetAction) + action.activated.connect(self._captureScreen, sender=screenshotAction) self._viewGroup = ActionGroup(name=u"View group") self._viewGroup.addAction(rotateLeftAction) @@ -234,52 +277,84 @@ self._toolbar.adaptLayout() def _clearToolbarButtons(self): + """ Remove toolbar buttons """ self._toolbar.removeAction(self._viewGroup) self._toolbar.adaptLayout() self._viewGroup = None - def _setMode(self, mode): - if (mode == INSERTING) and (not self._object): + """ Set the editor mode """ + if (mode == MapEditor.INSERTING) and (not self._object): self._statusbar.setText(u'Please select object first') mode = self._mode self._ignoreToggles = True # Update toolbox buttons - if (mode == INSERTING): + if (mode == MapEditor.INSERTING): self._drawAction.setChecked(True) - elif mode == REMOVING: + elif mode == MapEditor.REMOVING: self._removeAction.setChecked(True) - elif mode == MOVING: + elif mode == MapEditor.MOVING: self._moveAction.setChecked(True) - elif mode == OBJECTPICKER: + elif mode == MapEditor.OBJECTPICKER: self._objectpickerAction.setChecked(True) else: self._selectAction.setChecked(True) self._ignoreToggles = False self._mode = mode - print "Entered mode " + mode - self._statusbar.setText(mode.replace('_', ' ').capitalize()) - self._setCursor() + if self._debug: print "Entered mode " + mode + self._statusbar.setText(mode) + self._updateCursor() + + def _zoomIn(self, zoom=1.10): + self._controller.setZoom(self._controller.getZoom()*zoom) + + def _zoomOut(self, zoom=1.10): + self._controller.setZoom(self._controller.getZoom()/zoom) + + def _resetZoom(self): + """ Resets zoom level to 1:1 """ + self._controller.setZoom(1) + + def _rotateCounterClockwise(self): + """ Rotates map counterclockwise """ + self._controller.rotateCounterClockwise() + + def _rotateClockwise(self): + """ Rotates map clockwise """ + self._controller.rotateClockwise() + + def _captureScreen(self): + """ Saves a screenshot to the users data directory """ + userDir = fife_utils.getUserDataDirectory("fife", "editor") + t = userDir+"/screenshots" + if not os.path.isdir(t): + os.makedirs(t) + t += "/screen-%s-%s.png" % (date.today().strftime('%Y-%m-%d'), + time.strftime('%H-%M-%S')) + + self._editor.getEngine().getRenderBackend().captureScreen(t) + print "Saved screenshot to:", t def _buttonToggled(self, sender, toggled): + """ Called when a button controlling the editor mode was activated """ if self._controller is None: return if self._ignoreToggles is True: return - mode = SELECTING + mode = MapEditor.SELECTING if toggled: if sender == self._selectAction: - mode = SELECTING + mode = MapEditor.SELECTING elif sender == self._moveAction: - mode = MOVING + mode = MapEditor.MOVING elif sender == self._drawAction: - mode = INSERTING + mode = MapEditor.INSERTING elif sender == self._removeAction: - mode = REMOVING + mode = MapEditor.REMOVING elif sender == self._objectpickerAction: - mode = OBJECTPICKER + mode = MapEditor.OBJECTPICKER self._setMode(mode) @@ -288,7 +363,7 @@ return if not self._controller._layer: - if self._controller.debug: print 'No layers active. Cancelling map action' + if self._debug: print 'No layers active. Cancelling map action' return realCoords = self._getRealCoords(sender, event) @@ -301,7 +376,7 @@ if event.getButton() == fife.MouseEvent.RIGHT: self._controller.deselectSelection() - if self._mode == SELECTING: + if self._mode == MapEditor.SELECTING: if event.getButton() == fife.MouseEvent.LEFT: if self._eventlistener.shiftPressed: self._controller.deselectCell(realCoords[0], realCoords[1]) @@ -313,7 +388,7 @@ elif event.getButton() == fife.MouseEvent.RIGHT: self._controller.deselectSelection() - elif self._mode == INSERTING: + elif self._mode == MapEditor.INSERTING: if event.getButton() == fife.MouseEvent.LEFT: self._controller.deselectSelection() self._controller.selectCell(realCoords[0], realCoords[1]) @@ -326,7 +401,7 @@ self._controller.selectCell(realCoords[0], realCoords[1]) self._controller.placeInstance(position, self._object) - elif self._mode == REMOVING: + elif self._mode == MapEditor.REMOVING: if event.getButton() == fife.MouseEvent.LEFT: self._controller.deselectSelection() self._controller.selectCell(realCoords[0], realCoords[1]) @@ -335,27 +410,27 @@ self._controller.removeInstances(self._controller.getInstancesFromSelection()) - elif self._mode == MOVING: + elif self._mode == MapEditor.MOVING: if event.getButton() == fife.MouseEvent.LEFT: position = self._controller._camera.toMapCoordinates(fife.ScreenPoint(realCoords[0], realCoords[1]), False) - self._lastDragPos = self._controller._layer.getCellGrid().toLayerCoordinates(position) - self._lastDragPosExact = self._controller._layer.getCellGrid().toExactLayerCoordinates(position) + self._last_drag_pos = self._controller._layer.getCellGrid().toLayerCoordinates(position) + self._last_drag_pos_exact = self._controller._layer.getCellGrid().toExactLayerCoordinates(position) for loc in self._controller._selection: - if loc.getLayerCoordinates() == self._lastDragPos: + if loc.getLayerCoordinates() == self._last_drag_pos: break else: self._controller.deselectSelection() self._controller.selectCell(realCoords[0], realCoords[1]) - self._instances = self._controller.getInstancesFromSelection() + self._selected_instances = self._controller.getInstancesFromSelection() self._controller.getUndoManager().startGroup("Moved instances") self._undogroup = True - elif self._mode == OBJECTPICKER: + elif self._mode == MapEditor.OBJECTPICKER: position = self._controller._camera.toMapCoordinates(fife.ScreenPoint(realCoords[0], realCoords[1]), False) exact = self._controller._layer.getCellGrid().toExactLayerCoordinates(position) instances = self._controller.getInstancesFromPosition(exact) @@ -369,7 +444,7 @@ return if not self._controller._layer: - if self._controller.debug: print 'No layers active. Cancelling map action' + if self._debug: print 'No layers active. Cancelling map action' return realCoords = self._getRealCoords(sender, event) @@ -377,49 +452,51 @@ if event.getButton() == fife.MouseEvent.MIDDLE: self._scrollX = (self._dragx-realCoords[0])/10.0 self._scrollY = (self._dragy-realCoords[1])/10.0 + else: - if self._mode != SELECTING: + if self._mode != MapEditor.SELECTING: self._controller.deselectSelection() - if self._mode == SELECTING: + if self._mode == MapEditor.SELECTING: if event.getButton() == fife.MouseEvent.LEFT: if self._eventlistener.shiftPressed: self._controller.deselectCell(realCoords[0], realCoords[1]) else: self._controller.selectCell(realCoords[0], realCoords[1]) - elif self._mode == INSERTING: + elif self._mode == MapEditor.INSERTING: position = self._controller._camera.toMapCoordinates(fife.ScreenPoint(realCoords[0], realCoords[1]), False) position = self._controller._layer.getCellGrid().toLayerCoordinates(position) self._controller.selectCell(realCoords[0], realCoords[1]) self._controller.placeInstance(position, self._object) - elif self._mode == REMOVING: + elif self._mode == MapEditor.REMOVING: self._controller.selectCell(realCoords[0], realCoords[1]) self._controller.removeInstances(self._controller.getInstancesFromSelection()) - elif self._mode == MOVING: + elif self._mode == MapEditor.MOVING: position = self._controller._camera.toMapCoordinates(fife.ScreenPoint(realCoords[0], realCoords[1]), False) positionExact = self._controller._layer.getCellGrid().toExactLayerCoordinates(position) position = self._controller._layer.getCellGrid().toLayerCoordinates(position) if self._eventlistener.shiftPressed: - self._controller.moveInstances(self._instances, positionExact-self._lastDragPosExact, True) + self._controller.moveInstances(self._selected_instances, positionExact-self._last_drag_pos_exact, True) else: - self._controller.moveInstances(self._instances, position-self._lastDragPos, False) - self._lastDragPos = position - self._lastDragPosExact = positionExact + self._controller.moveInstances(self._selected_instances, position-self._last_drag_pos, False) + self._last_drag_pos = position + self._last_drag_pos_exact = positionExact # Update selection self._controller.deselectSelection() - for i in self._instances: + for i in self._selected_instances: pos = i.getLocation().getMapCoordinates() pos = self._controller._camera.toScreenCoordinates(pos) self._controller.selectCell(pos.x, pos.y) - elif self._mode == OBJECTPICKER: + + elif self._mode == MapEditor.OBJECTPICKER: pass def mouseReleased(self, sender, event): @@ -427,10 +504,10 @@ return if not self._controller._layer: - if self._controller.debug: print 'No layers active. Cancelling map action' + if self._debug: print 'No layers active. Cancelling map action' return - if self._mode == SELECTING or self._mode == MOVING: + if self._mode == MapEditor.SELECTING or self._mode == MapEditor.MOVING: instances = self._controller.getInstancesFromSelection() if len(instances) > 0: events.onInstancesSelected.send(sender=self, instances=instances) @@ -445,27 +522,27 @@ self._controller.getUndoManager().endGroup() self._undogroup = False - self._dragx = NOT_INITIALIZED - self._dragy = NOT_INITIALIZED - def mouseMoved(self, sender, event): pass def mouseEntered(self, sender, event): - self._setCursor() + # Mouse has entered map area. Set cursor to reflect current mode + self._updateCursor() def mouseExited(self, sender, event): + # Mouse has exited the map area. Set the cursor to native arrow self._resetCursor() def mouseWheelMovedUp(self, event): + # Zoom in if self._eventlistener.controlPressed and self._controller._camera: self._controller._camera.setZoom(self._controller._camera.getZoom() * 1.10) def mouseWheelMovedDown(self, event): + # Zoom out if self._eventlistener.controlPressed and self._controller._camera: self._controller._camera.setZoom(self._controller._camera.getZoom() / 1.10) - def keyPressed(self, event): keyval = event.getKey().getValue() keystr = event.getKey().getAsString().lower() @@ -489,25 +566,25 @@ self.captureScreen() elif keystr == "s": - self._setMode(SELECTING) + self._setMode(MapEditor.SELECTING) elif keystr == "i": - if self._mode != INSERTING: + if self._mode != MapEditor.INSERTING: self._setMode(INSERTING) else: - self._setMode(SELECTING) + self._setMode(MapEditor.SELECTING) elif keystr == "r": - if self._mode != REMOVING: - self._setMode(REMOVING) + if self._mode != MapEditor.REMOVING: + self._setMode(MapEditor.REMOVING) else: - self._setMode(SELECTING) + self._setMode(MapEditor.SELECTING) elif keystr == 'm': - if self._mode != MOVING: - self._setMode(MOVING) + if self._mode != MapEditor.MOVING: + self._setMode(MapEditor.MOVING) else: - self._setMode(SELECTING) + self._setMode(MapEditor.SELECTING) elif keystr == 't': gridrenderer = self._controller._camera.getRenderer('GridRenderer') @@ -534,22 +611,8 @@ def keyReleased(self, event): pass - def zoomIn(self, zoom=1.10): - self._controller.setZoom(self._controller.getZoom()*zoom) - - def zoomOut(self, zoom=1.10): - self._controller.setZoom(self._controller.getZoom()/zoom) - - def resetZoom(self): - self._controller.setZoom(1) - - def rotateCounterClockwise(self): - self._controller.rotateCounterClockwise() - - def rotateClockwise(self): - self._controller.rotateClockwise() - def _getRealCoords(self, sender, event): + """ Converts relative widget coordinate to absolute coordinates """ cw = sender offsetX = event.getX() offsetY = event.getY() @@ -565,16 +628,7 @@ return (offsetX, offsetY) - def captureScreen(self): - userDir = fife_utils.getUserDataDirectory("fife", "editor") - t = userDir+"/screenshots" - if not os.path.isdir(t): - os.makedirs(t) - t += "/screen-%s-%s.png" % (date.today().strftime('%Y-%m-%d'), - time.strftime('%H-%M-%S')) - - self._editor.getEngine().getRenderBackend().captureScreen(t) - print "Saved screenshot to:", t - def pump(self): + """ Called each frame """ + # Scroll camera self._controller.moveCamera(self._scrollX, self._scrollY) \ No newline at end of file