view clients/editor/plugins/objectedit.py @ 217:68ae8f4234ca

* More compact ObjectSelector dialog
author cheesesucker@33b003aa-7bff-0310-803a-e67f0ece8222
date Thu, 19 Mar 2009 01:40:51 +0000
parents f10a2e78a0e1
children
line wrap: on
line source

#!/usr/bin/env python
# coding: utf-8
# ###################################################
# Copyright (C) 2008 The Zero-Projekt team
# http://zero-projekt.net
# info@zero-projekt.net
# This file is part of Zero "Was vom Morgen blieb"
#
# The Zero-Projekt codebase is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the
# Free Software Foundation, Inc.,
# 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
# ###################################################

""" a tool for FIFEdit to edit object and instance attributes """

import fife
import plugin
import pychan
import pychan.widgets as widgets
from pychan.tools import callbackWithArguments as cbwa

import settings as Settings
import math

class ObjectEdit(plugin.Plugin):
	""" The B{ObjectEdit} module is a plugin for FIFedit and allows to edit
	attributes of an selected instance - like offset, instance id or rotation
	(namespaces and object id editing is excluded)
	
	current features:
		- click instance and get all known data
		- edit offsets, rotation, instance id
		- outline highlighting of the selected object
			
	missing features:
		- blocking flag (flag doesn't work yet from FIFE side)
		- static flag (flag doesn't work yet from FIFE side)		
		- object saving
		- a lot of bug fixing concerning the rotation
		- use sliders to allow offset changes
		- the module should be able to use the editors global undo history

	FIXME:
		- this module owns a pointer to the mapedit module - this shouldn't be
		  necessary; a better plugin system of fifedit should only hand over the needed
		  data (selected instance)
		- we also need to edit run.py of the editor core to make this plugin work (shouldn't be necessary, too)
	"""
	def __init__(self, engine, mapedit):
		# Fifedit plugin data
		self.menu_items = { 'ObjectEdit' : self.toggle_offsetedit }
		
		self._mapedit = mapedit

		# this is _very bad_ - but I need to change the current rotation code by providing
		# project specific rotation angles. FIFE later should provide a list of the loaded
		# object rotations (they are provided by the xml files, so we just need to use them...)
		self._mapedit._objectedit_rotations = None
	
		self.active = False
		self._camera = None
		self._layer = None
		
		self.offset_slider = {}
		self.offset_slider['x'] = False
		self.offset_slider['y'] = False
		
		self.imagepool = engine.getImagePool()
		self.animationpool = engine.getAnimationPool()
		
		self.guidata = {}
		self.objectdata = {}

		self._reset()		
		self.create_gui()

	def _reset(self):
		"""
			resets all dynamic vars, but leaves out static ones (e.g. camera, layer)

		"""
		self._instances = None
		self._image = None
		self._image_default_x_offset = None
		self._image_default_y_offset = None
		self._animation = False
		self._rotation = None
		self._avail_rotations = []
		self._namespace = None	
		self._blocking = 0
		self._static = 0
		self._object_id = None	
		self._instance_id = None
		self._fixed_rotation = None
		
		if self._camera is not None:
			self.renderer.removeAllOutlines()		
		
	def create_gui(self):
		"""
			- creates the gui skeleton by loading the xml file
			- finds some important childs and saves their widget in the object
		"""
		self.container = pychan.loadXML('gui/objectedit.xml')
		self.container.mapEvents({
			'x_offset_up' 	: cbwa(self.change_offset_x, 1),
			'x_offset_dn' 	: cbwa(self.change_offset_x, -1),
			
			'y_offset_up' 	: cbwa(self.change_offset_y, 1),
			'y_offset_dn' 	: cbwa(self.change_offset_y, -1),
			
			'x_offset_slider' : cbwa(self.get_slider_value, "x"),
			'y_offset_slider' : cbwa(self.get_slider_value, "y"),
			
			'use_data'		: self.use_user_data,
			
		})

		self._gui_anim_panel_wrapper = self.container.findChild(name="animation_panel_wrapper")
		self._gui_anim_panel = self._gui_anim_panel_wrapper.findChild(name="animation_panel")
		
		self._gui_anim_panel_wrapper.removeChild(self._gui_anim_panel)

		self._gui_rotation_dropdown = self.container.findChild(name="select_rotations")
		
		self._gui_xoffset_textfield = self.container.findChild(name="x_offset")
		self._gui_yoffset_textfield = self.container.findChild(name="y_offset")
		
		self._gui_instance_id_textfield = self.container.findChild(name="instance_id")
		
		print "Steplength x slider", self.container.findChild(name="x_offset_slider").getStepLength()
		print "Steplength y slider", self.container.findChild(name="y_offset_slider").getStepLength()
		self.container.findChild(name="x_offset_slider").setStepLength(0.01)
		self.container.findChild(name="y_offset_slider").setStepLength(0.01)
		print "New steplength x slider", self.container.findChild(name="x_offset_slider").getStepLength()
		print "New steplength y slider", self.container.findChild(name="y_offset_slider").getStepLength()
		
	def get_slider_value(self, orientation):
		""" get current slider value for offset manipulation """
		
		slider_name = orientation + "_offset_slider"
		widget = self.container.findChild(name=slider_name)
		value = widget.getValue()

		print "%s slider value: %s" % (orientation, str(value))
		
		if value < 0: 
			self.offset_slider[orientation] = False
			return
		
		callback = getattr(self, "change_offset_" + orientation)

		if self.offset_slider[orientation] == widget.getScaleStart():
			self.set_default_offset(orientation)
			self.offset_slider[orientation] = False
			return	
		elif self.offset_slider[orientation] >= widget.getScaleEnd():
			pass
		elif self.offset_slider[orientation] < value:
			callback(1)
		elif self.offset_slider[orientation] > value :
			callback(-1)

		self.offset_slider[orientation] = value

	def set_default_offset(self, axis):
		""" set default image offset for given axis """
		if axis == 'x':
			self._image.setXShift(self._image_default_x_offset)
		elif axis == 'y':
			self._image.setYShift(self._image_default_y_offset)

	def _get_gui_size(self):
		"""
			gets the current size of the gui window and calculates new position
			(atm top right corner)
		"""
		size = self.container._getSize()
		self.position = ((Settings.ScreenWidth - 10 - size[0]), 10)
		
	def update_gui(self):
		"""
			updates the gui widgets with current instance data
			
			FIXME: 
				- drop animation support or turn it into something useful
		"""
		#if self._animation is False:
			#try:
				#self._gui_anim_panel_wrapper.removeChild(self._gui_anim_panel)
			#except:
				#pass
		#elif self._animation is True:
			#try:
				#self._gui_anim_panel_wrapper.resizeToContent()				
				#self._gui_anim_panel_wrapper.addChild(self._gui_anim_panel)
				#self._gui_anim_panel_wrapper.resizeToContent()
			#except:
				#pass

		self.container.distributeInitialData({
			'select_rotations' 	: self._avail_rotations,
			'instance_id'		: str( self._instances[0].getId() ),
			'object_id'			: str( self._object_id ),
			'x_offset'			: str( self._image.getXShift() ),
			'y_offset'			: str( self._image.getYShift() ),
			'instance_rotation' : str( self._instances[0].getRotation() ),
			'object_namespace'	: str( self._namespace ),
			'object_blocking'	: str( self._blocking ),
			'object_static'		: str( self._static ),
		})
		try:
			print self._avail_rotations
			print self._fixed_rotation
			index = self._avail_rotations.index( str(self._fixed_rotation) )
			self._gui_rotation_dropdown._setSelected(index)
		except:
#			pass
			print "Angle (", self._fixed_rotation, ") not supported by this instance"
		
	def toggle_gui(self):
		"""
			show / hide the gui
			
			FIXME:
				- ATM not in use, needs some additional code when showing / hiding the gui (see input() )
		"""
		if self.container.isVisible():
			self.container.hide()
		else:
			self.container.show()
			
	def toggle_offsetedit(self):
		"""
			- toggles the object editor activ / inactiv - just in case the user don't want to have
			  the gui popping up all the time while mapping :-)
			- hides gui
		"""
		if self.active is True:
			self.active = False
			if self.container.isVisible():
				self.container.hide()
		else:
			self.active = True	

	def highlight_selected_instance(self):
		"""
			just highlights selected instance
		"""
		self.renderer.removeAllOutlines() 
		self.renderer.addOutlined(self._instances[0], 205, 205, 205, 1)
			
	def change_offset_x(self, value=1):
		"""
			- callback for changing x offset
			- changes x offset of current instance (image)
			- updates gui
			
			@type	value:	int
			@param	value:	the modifier for the x offset
		"""		
		if self._image is not None:
			self._image.setXShift(self._image.getXShift() + value)
			self.update_gui()

	def change_offset_y(self, value=1):
		"""
			- callback for changing y offset
			- changes y offset of current instance (image)
			- updates gui
			
			@type	value:	int
			@param	value:	the modifier for the y offset
		"""
		if self._image is not None:
			self._image.setYShift(self._image.getYShift() + value)
			self.update_gui()

	def use_user_data(self):
		"""
			- takes the users values and applies them directly to the current ._instance
			- writes current data record
			- writes previous data record
			- updates gui
		
			FIXME:
			- parse user data in case user think strings are considered to be integer offset values...
		"""
		xoffset = self._gui_xoffset_textfield._getText()
		yoffset = self._gui_yoffset_textfield._getText()
		
		instance_id = self._gui_instance_id_textfield._getText()
		if instance_id is not None and instance_id is not "None":
			existing_instances = self._mapedit._layer.getInstances(instance_id)
			if existing_instances == ():
				self._instances[0].setId(instance_id)
				print "Set new instance id: ", instance_id		
			else:
				for i in existing_instances:
					print i
		
		# workaround - dropdown list only has 2 entries, but sends 3 -> pychan bug?
		if len(self._avail_rotations) < self._gui_rotation_dropdown._getSelected():
			index = len(self._avail_rotations)
		else:
			index = self._gui_rotation_dropdown._getSelected()
		
		# strange, but this helps to rotate the image correctly to the value the user selected
		angle = int( self._avail_rotations[index] )
		angle = int(angle - abs( self._camera.getTilt() ) )
		if angle == 360:
			angle = 0
		
		self._instances[0].setRotation(angle)
		self.get_instance_data(None, None, angle)
		
		try:
			self._image.setXShift( int(xoffset) )
		except:
			pass
#		print "x offset must me of type int!"
		try:
			self._image.setYShift( int(yoffset) )
		except:
			pass
#		print "y offset must be of type int!"

		self.update_gui()
		
	def get_instance_data(self, timestamp=None, frame=None, angle=-1, instance=None):
		"""
			- grabs all available data from both object and instance
			- checks if we already hold a record (namespace + object id)
			
			FIXME:
				1.) we need to fix the instance rotation / rotation issue
				2.) use correct instance rotations to store data for _each_ available rotation
				3.) move record code out of this method
		"""
		visual = None
		self._avail_rotations = []
			
		if instance is None:
			instance = self._instances[0]
			
		object = instance.getObject()
		self._namespace = object.getNamespace()
		self._object_id = object.getId()

		self._instance_id = instance.getId()
	
		if self._instance_id == '':
			self._instance_id = 'None'

		if angle == -1:
			angle = int(instance.getRotation())
		else:
			angle = int(angle)	
			
		self._rotation = angle
		
		if object.isBlocking():
			self._blocking = 1
			
		if object.isStatic():
			self._static = 1
		
		try:
			visual = object.get2dGfxVisual()
		except:
			print 'Fetching visual of object - failed. :/'
			raise			

#		print "Camera Tilt: ", self._camera.getTilt()
#		print "Camera Rotation: ", self._camera.getRotation()

		self._fixed_rotation = int(instance.getRotation() + abs( self._camera.getTilt() ) )		
		self._fixed_rotation = visual.getClosestMatchingAngle(self._fixed_rotation)	

		index = visual.getStaticImageIndexByAngle(self._fixed_rotation)

		if index == -1:
			# object is an animation
			self._animation = True
			# no static image available, try default action
			action = object.getDefaultAction()
			if action:
				animation_id = action.get2dGfxVisual().getAnimationIndexByAngle(self._fixed_rotation)
				animation = self.animationpool.getAnimation(animation_id)
#				if timestamp is None and frame is not None:
#					self._image = animation.getFrame(frame)	
#				elif timestamp is not None and frame is None:
#					self._image = animation.getFrameByTimestamp(timestamp)
#				else:
				self._image = animation.getFrameByTimestamp(0)
				index = self._image.getPoolId()
		elif index != -1:
			# object is a static image
			self._animation = False
			self._image = self.imagepool.getImage(index)

		if not self._animation:
			rotation_tuple = visual.getStaticImageAngles()
			for angle in rotation_tuple:
				self._avail_rotations.append( str(angle) )
				
		self._image_default_x_offset = self._image.getXShift()
		self._image_default_y_offset = self._image.getYShift()

# FIXME: see l. 40
		self._mapedit._objectedit_rotations = self._avail_rotations
# end FIXME
		
	def input(self):
		"""
			if called _and_ the user wishes to edit offsets,
			gets instance data and show gui
			
			(see run.py, pump() )
		"""
		if self._mapedit._instances != self._instances:
			if self.active is True:
				self._reset()
				self._instances = self._mapedit._instances
				
				if self._camera is None:
					self._camera = self._mapedit._camera
					self.renderer = fife.InstanceRenderer.getInstance(self._camera)				
					
				self._layer = self._mapedit._layer
			
				if self._instances != ():
					self.highlight_selected_instance()
					self.get_instance_data()
					self.update_gui()
					self.container.adaptLayout()
					self.container.show()
					self._get_gui_size()
					self.container._setPosition(self.position)
				else:
					self._reset()
					self.container.hide()