--
--

Logic = {}

Logic.Activities = {"Follow Route", "Patrol", "Repeat Road", "Deploy", "Wander", "Rest In Zone", "Feed In Zone", "Hunt In Zone", "Guard Zone", "Stand Still", "Stand On Start Point", "Go To Start Point"}
Logic.TimeLimits = {"No Limit", "Until", "Few Sec", "Few Min", "Chat", "Season", "Day", "Month"}
Logic.Seasons = {"Spring", "Summer", "Autumn", "Winter"}
Logic.Days = {"Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday", "Sunday"}
Logic.Months = {"January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"}
Logic.ScriptActions ={"Sit Down", "Stand Up", "Go To Step", "Set Activity", "Stop Actions", "Start Mission"}

Logic.EventModes ={"Once Only", "Repeating"}
Logic.EventTypes ={"Time", "Place", "Entity or Group", "State Machine", "Counter"}
Logic.ConditionTypes ={
							{},
							{"Any player in zone", "All players in zone", "No players in zone", "Some players in zone"},
							{"At Destination", "Enters State", "Leaves State", "Activity Is", "Die", "Is Attacked"},
							{},
							{"v0_changed", "v3_changed", "Equals", "Lesser", "Greater", ""},
						}

Logic.ChatTypes = {"None", "Repeating", "Non Repeating", "Continue"}
Logic.InteractionTypes = {"Give", "Say"}

Logic.Name = "Activity"
Logic.Description = "A feature for describing activities sequences"




Logic.Components = {}




	-- ACTION -------------------
Logic.Components.LogicEntityAction = 
	{
		BaseClass ="BaseClass",
		Name = "LogicEntityAction",
		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "logicEntityActionPropertySheetDisplayer",
		Prop =
		{
			{Name ="Name", Type ="String"}, 
			{Name ="Event", Type ="EventType"}, 
			{Name ="Actions", Type ="Table"}, -- "ActionStep" list
			{Name ="Conditions", Type ="Table"}, -- "ConditionStep" list
		},

		getName = function(this)

			local logicEntity = this.Parent.Parent.Parent
			assert(logicEntity)

			if logicEntity:isGrouped() then
				logicEntity = logicEntity.ParentInstance
			end

			local coloredName = (logicEntity.InstanceId == r2.events.filteredLogicEntityId)

			local eventType = r2.events.keyWordsColor .. i18n.get("uiR2EdEventTxtPreEvent"):toUtf8() .. " "

			if coloredName then		
				eventType = eventType .. r2.events.filterWordColor 
			else
				eventType = eventType .. r2.events.entityWordColor 
			end
			local name = "No name"
			if logicEntity.getName then
				name = logicEntity:getName()
			elseif logicEntity.Name then
				name = logicEntity.Name
			end
			eventType = eventType .. name .. " " 
			eventType = eventType .. r2.events.communWordsColor 

			eventType = eventType ..r2.getLogicAttribute(logicEntity.Class, "Events", this.Event.Type).text
			
			if this.Event.Value~="" then
				local instance = r2:getInstanceFromId(this.Event.Value)
				assert(instance)

				eventType = eventType .. " " .. instance:getShortName()
			end

			
			if this.Event.ValueString and this.Event.ValueString ~= "" then
				if string.gfind(eventType, "%%1")() then
					eventType = string.gsub(eventType, "%%1", "'"..tostring(this.Event.ValueString).."'")
				else
					eventType = eventType .. " '" .. tostring(this.Event.ValueString).. "'"
				end
			end

			eventType = eventType .. ", "

			
			-- conditions
			local conditionsList = ""
			for i=0, this.Conditions.Size-1 do
				local conditionInst = this.Conditions[i]
				
				if conditionInst.Entity~="" and conditionInst.Condition.Type~="" then

					conditionsList = conditionsList .. "\n" .. r2.events.keyWordsColor 
						..string.upper(i18n.get("uiR2EdEventTxtPreCondition"):toUtf8()).." " 
						.. r2.events.communWordsColor 

					local conditionLogicEntity = r2:getInstanceFromId(tostring(conditionInst.Entity))
					assert(conditionLogicEntity)

					coloredName = (conditionLogicEntity.InstanceId == r2.events.filteredLogicEntityId)
					if coloredName then 
						conditionsList = conditionsList .. r2.events.filterWordColor 
					else
						conditionsList = conditionsList .. r2.events.entityWordColor
					end
					conditionsList = conditionsList .. conditionLogicEntity.Name .. " " 
					conditionsList = conditionsList .. r2.events.communWordsColor 
			
					conditionsList = conditionsList ..r2.getLogicAttribute(conditionLogicEntity.Class, "Conditions", conditionInst.Condition.Type).text

					if conditionInst.Condition.Value~="" then
						local instance = r2:getInstanceFromId(conditionInst.Condition.Value)
						assert(instance)
						conditionsList = conditionsList .. " '" .. instance:getShortName() .."'"
					end

					conditionsList = conditionsList .. ", "
				end
			end

			-- actions
			local actionsList = ""
			for i=0, this.Actions.Size-1 do
				local actionInst = this.Actions[i]
				
				if actionInst.Entity~="" and actionInst.Action.Type~="" then
					
					actionsList = actionsList .. "\n" .. r2.events.keyWordsColor 
						..string.upper(i18n.get("uiR2EdEventTxtPreActions"):toUtf8()).." "
						.. r2.events.communWordsColor 

					local actionLogicEntity = r2:getInstanceFromId(tostring(actionInst.Entity))
					assert(actionLogicEntity)
					
					coloredName = (actionLogicEntity.InstanceId == r2.events.filteredLogicEntityId)
					if coloredName then 
						actionsList = actionsList .. r2.events.filterWordColor 
					else
						actionsList = actionsList .. r2.events.entityWordColor 
					end
					local name = "No name"
					if actionLogicEntity.getName then
						name = actionLogicEntity:getName()
					elseif actionLogicEntity.Name then
						name = actionLogicEntity.Name
					end
					actionsList = actionsList .. name .. " " 
					actionsList = actionsList .. r2.events.communWordsColor 
					
					actionsList = actionsList ..r2.getLogicAttribute(actionLogicEntity.Class, "ApplicableActions", actionInst.Action.Type).text
					if actionInst.Action.Value~="" then
						local instance = r2:getInstanceFromId(actionInst.Action.Value)
						assert(instance)
						actionsList = actionsList .. " '" .. instance:getShortName() .. "'"
					end

					if actionInst.Action.ValueString and actionInst.Action.ValueString ~= "" then

						if string.gfind(actionsList, "%%1")() then
							actionsList = string.gsub(actionsList, "%%1", "'"..tostring(actionInst.Action.ValueString).."'")
						else
							actionsList = actionsList .. " '" .. tostring(actionInst.Action.ValueString).. "'"
						end
					end

					if i~=this.Actions.Size-1 then
						actionsList = actionsList .. ","
					else
						actionsList = actionsList .. "."
					end
				end
			end

			if actionsList=="" then
				actionsList = "\n..."
			end
		
			return eventType .. conditionsList .. actionsList --.. "\n"
		end,

		getLogicEntityParent = function(this)

			local logicEntity = this.Parent.Parent.Parent
			if logicEntity:isGrouped() then logicEntity = logicEntity.Parent.Parent end

			return logicEntity
		end,

		-- The act of the event
		getLogicAct = function(this)
			
			local currentAct = r2:getCurrentAct()
			local baseAct =  r2.Scenario:getBaseAct()
			local isBaseAct = false
			local presentActs = {}
			local isCurrentAct = false
			local counter = 0
			

			local entity = this:getLogicEntityParent()
			if entity and entity.getParentAct then
				local act = entity:getParentAct()
				if act then
					if act==currentAct then 
						isCurrentAct=true 
					elseif act==baseAct then
						isBaseAct = true
					else
						presentActs[act.InstanceId] = act.InstanceId
						counter = counter+1
					end
				end
			end


			local function verifyAct ( container)
				
				local key, value = next(container, nil)
				while key do
					if value.Action and value.Action.Type == "Start Act" then
					else
						local entityId = value.Entity
						local entity= r2:getInstanceFromId(entityId)
						if entity and entity.getParentAct then
							local act = entity:getParentAct()
							if act then
								if act==currentAct then 
									isCurrentAct=true 
								elseif act==baseAct then
									isBaseAct = true
								else
									presentActs[act.InstanceId] = act.InstanceId
									counter = counter+1
								end
							end
						end
					end
					key, value = next(container, key)
				end
			end

			verifyAct(this.Actions)
			verifyAct(this.Conditions)

			if counter>=2 or (counter==1 and isCurrentAct) then
				return nil
			elseif counter==1 and not isCurrentAct and isBaseAct then
				local act = nil
				for k, actId in pairs(presentActs) do
					act = r2:getInstanceFromId(actId)
					break
				end
				--inspect(act)
				return act
			else
				--inspect(currentAct)
				return currentAct
			end
		end,
		



		----------
		-- called upon translation
		getLogicActForTranslate = function(this)
			local currentAct = this:getLogicEntityParent():getParentAct()
			local baseAct =  r2.Scenario:getBaseAct()

			local entity = this.Event.Entity
			if entity and entity.getParentAct then
				local act = entity:getParentAct()
				if currentAct == nil then
					currentAct = act
				elseif act ~= currentAct then
					if currentAct == baseAct then
						currentAct = act
					elseif act == baseAct then
						-- do nothing
					else 
						return nil
					end
				end
			end


			
			local function verifyAct ( container)
				local key, value = next(container, nil)
				while key do
					if value.Action and value.Action.Type == "Start Act" then
					else
						local entityId = value.Entity
						local entity= r2:getInstanceFromId(entityId)
						if entity and entity.getParentAct then
							local act = entity:getParentAct()
							if act then
								if currentAct == nil then
									currentAct = act
								elseif act ~= currentAct then
									if currentAct == baseAct then
										currentAct = act
									elseif act == baseAct then
										-- do nothing
									else 																				return false
									end
								end
							end
						end
					end
					key, value = next(container, key)
				end
				return true
			end


			if not verifyAct(this.Actions) then return false end
			if not verifyAct(this.Conditions) then return false end
			return currentAct		
		end,
		--
		getLogicActInstanceId = function(this)
			--local ok, act = pcall(Logic.Components.LogicEntityAction.getLogicAct, this)
			local ok, act = pcall(Logic.Components.LogicEntityAction.getLogicActForTranslate, this)
			if not ok or not act then
				return r2.RefId("")
			end
			
			return act.InstanceId
		end,
	}

Logic.Components.EventType = 
	{
		BaseClass ="BaseClass",
		Name = "EventType",
		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "eventTypePropertySheetDisplayer",
		Prop =
		{
			{Name ="Type", Type ="String"}, 
			{Name ="Value", Type ="RefId"}, -- if type is end of activity/chat step/sequence, instance id
			{Name ="ValueString", Type ="String", DefaultValue="", DefaultInBase=1}, -- if type is emit user event, sub timer...

		}
	}

local classActionStepVersion = 2
Logic.Components.ActionStep = {
		BaseClass ="BaseClass",
		Name = "ActionStep",
		Version = classActionStepVersion,
		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "actionStepPropertySheetDisplayer",
		Prop =
		{
			{Name ="Entity", Type ="RefId"}, -- entity target id
			{Name ="Action", Type ="ActionType"} -- sequence id
		},

		updateVersion = function(this, scenarioValue, currentValue )
			
			local patchValue = scenarioValue
			if patchValue < 1 then
				local invalidActions = {}
				invalidActions["desactivate"] = "deactivate"
				invalidActions["Desactivate"] = "deactivate"
				r2.updateLogicActions(this, invalidActions, "BanditCamp")
				patchValue = 1
			end

			if patchValue < 2 then

				local action = this.Action
				if action.Type == "Add 10 Seconds" then
					r2.requestSetNode(action.InstanceId, "Type", "add seconds")
					r2.requestInsertNode(action.InstanceId, "", -1, "ValueString", "10")
				end

				if action.Type == "Sub 10 seconds" then
					r2.requestSetNode(action.InstanceId, "Type", "sub seconds")
					r2.requestInsertNode(action.InstanceId, "", -1, "ValueString", "10")
				end
				if action.Type == "Add 1 minute" then
					r2.requestSetNode(action.InstanceId, "Type", "add seconds")
					r2.requestInsertNode(action.InstanceId, "", -1, "ValueString", "60")
				end

				if action.Type == "Sub 1 minute" then
					r2.requestSetNode(action.InstanceId, "Type", "sub seconds")
					r2.requestInsertNode(action.InstanceId, "", -1, "ValueString", "60")
				end

				patchValue = 2
			end
			
			if patchValue == currentValue then return true end
			return false
		end,	
	}


Logic.Components.ActionType = 
	{
		BaseClass ="BaseClass",
		Name = "ActionType",
		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "actionTypePropertySheetDisplayer",
		Prop =
		{
			{Name ="Type", Type ="String"}, 
			{Name ="Value", Type ="RefId"}, -- if type is begin of activity/chat sequence, instance id
			{Name ="ValueString", Type ="String", DefaultValue="", DefaultInBase=1}, -- if type is emit user event, sub timer...

		
		}
	}

Logic.Components.ConditionStep =
	{
		BaseClass ="BaseClass",
		Name = "ConditionStep",
		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "conditionStepPropertySheetDisplayer",
		Prop =
		{
			{Name ="Entity", Type ="RefId"}, -- entity id
			{Name ="Condition", Type ="ConditionType"}
			
		}
	}

Logic.Components.ConditionType = 
	{
		BaseClass ="BaseClass",
		Name = "ConditionType",
		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "conditionTypePropertySheetDisplayer",
		Prop =
		{
			{Name ="Type", Type ="String"}, 
			{Name ="Value", Type ="RefId"}, -- if type is being in activity/chat step/sequence, instance id

		
		}
	}

	-- REACTION -----------------

Logic.Components.LogicEntityReaction = 
	{
		BaseClass ="BaseClass",
		Name = "LogicEntityReaction",
		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "logicEntityReactionPropertySheetDisplayer",
		Prop =
		{
			{Name ="Name", Type ="String"}, 
			{Name ="LogicEntityAction", Type ="String"}, -- id of associated LogicEntityAction
			{Name ="ActionStep", Type ="String"},		  -- id of associated ActionStep
		}
	}

Logic.Components.ChatAction=
	{
		BaseClass ="BaseClass",
		Name ="ChatAction",
		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "chatActionPropertySheetDisplayer",
		Prop =
		{
			{Name ="Who", Type ="RefId"},
			{Name ="WhoNoEntity", Type ="String"},
			{Name ="Says", Type ="String"},
			{Name ="Emote", Type ="String"},
			{Name ="Facing", Type ="RefId"},
--			{Name ="FacingNoEntity", Type ="String"},
		}
	}

Logic.Components.ChatStep = {
		BaseClass ="BaseClass",
		Name = "ChatStep",
		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "chatStepPropertySheetDisplayer",
		Prop =
		{
			{Name ="Actions", Type ="Table"},
			{Name ="Time", Type ="Number" },
			{Name ="Name", Type ="String" }
		},

		getName = function(this)

			-- after time
			local minNb, secNb = r2.dialogs:calculMinSec(this.Time)
			local time = ""
			if minNb ~= 0 then
				time = tostring(minNb)..i18n.get("uiR2EdShortMinutes"):toUtf8()
			end
			time = time.." " ..tostring(secNb)..i18n.get("uiR2EdShortSeconds"):toUtf8()
			local afterTime = "(" ..i18n.get("uiR2EdAfter"):toUtf8().." ".. time..") "

			-- says
			local whoToWho = ""
			local action = this.Actions[0]
			local who = action.Who
			if who=="" then who=action.WhoNoEntity end
			if who and who ~= "" then
	
				if r2:getInstanceFromId(who) then
					who=r2:getInstanceFromId(who).Name	
				else
					who = r2.dialogs.whoToWhoTranslation[who]
				end

				whoToWho = who .. " " 

				local facing = action.Facing
				if facing~="" then
					whoToWho = whoToWho ..i18n.get("uiR2EdSaysTo"):toUtf8().. " " .. r2:getInstanceFromId(facing).Name .. " "	
				end

				local emote = action.Emote
				if emote~="" and r2.dialogs.fromEmoteIdToName[emote] ~= nil then
					whoToWho = whoToWho .."(" .. r2.dialogs.fromEmoteIdToName[emote] .. ") :"
				end
			end

			return afterTime..whoToWho

		end,

		getShortName = function(this)

			-- says
			local saysWhat = ""
			local action = this.Actions[0]
			local who = action.Who
			if who ~= "" or WhoNoEntity~="" then
				who = r2:getInstanceFromId(who)
				if who then
					saysWhat = who.Name
				elseif action.WhoNoEntity=="_System" then
					saysWhat = i18n.get("uiR2EdSystem"):toUtf8()
				elseif action.WhoNoEntity=="_DM" then
					saysWhat = i18n.get("uiR2EdDonjonMaster"):toUtf8()
				end
			
				saysWhat = saysWhat .. " " ..i18n.get("uiR2EdSays"):toUtf8().. " " 

				local says = action.Says
				if says ~= "" then
					local inst=r2:getInstanceFromId(says)
					if inst then
						says = inst.Text 
						local uc_says = ucstring()
						uc_says:fromUtf8(says)
						uc_says = uc_says:substr(0, 4)
						says = uc_says:toUtf8()					
					end

				end

				saysWhat = saysWhat .. says .. "..."
			end

			return saysWhat

		end,

		--------------------------------------------------------------------------------------------
		-- from 'BaseClass'
		getParentTreeNode = function(this)				
		end,
	}


local classChatSequenceVersion = 1
Logic.Components.ChatSequence = {
		PropertySheetHeader = r2.getDisplayButtonHeader("r2.dialogs:openEditor()", "uiR2EdEditDialogButton"),
		BaseClass = "LogicEntity",
		Name = "ChatSequence",
		InEventUI = true,
		Menu="ui:interface:r2ed_feature_menu",
		Version = classChatSequenceVersion,

		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "chatSequencePropertySheetDisplayer",

		DisplayerUI = "R2::CDisplayerLua",
		DisplayerUIParams = "defaultUIDisplayer",
		DisplayerVisual = "R2::CDisplayerVisualEntity",

		Parameters = {},
		ApplicableActions = {
								"activate", "deactivate", "starts dialog", "stops dialog", "starts chat"  , "continues dialog"
							},
		Events =			{
								"start of dialog", "end of dialog" ,
								"start of chat", "end of chat",
								"activation", "deactivation"
							},
		Conditions =		{
								"is not in dialog", "is in dialog", "is in chat"
							},
		TextContexts =		{},
		TextParameters =	{},
		LiveParameters =	{},

		
		Prop =
		{	
			{Name = "Type", Type="String",DefaultValue = "None", Visible=false},
			{Name ="Repeating", Type="Number" ,DefaultValue = "0", Visible=false},
			{Name="Components",Type="Table"},
			{Name="SubComponents", Type="Table"},
			{Name="Name", Type="String", MaxNumChar="32"},
			{Name="AutoStart", Type="Number", WidgetStyle="Boolean", DefaultValue="1", Visible=function(this) return this:mustDisplayAutostart() end},
			{Name="Active", Type="Number", WidgetStyle="Boolean", DefaultValue = "1"},
		},
	
		
		mustDisplayAutostart = function(this)
			if this.Class == "ProximityDialog" then
				return false
			end
			return true
		end,

		-- it's a feature
		getParentTreeNode = function(this)
			return this:getFeatureParentTreeNode()
		end,

		onPostCreate = function(this)
			--this:createGhostComponents()
			if this.User.DisplayProp and this.User.DisplayProp == 1 then
				r2:setSelectedInstanceId(this.InstanceId)				
				r2:showProperties(this)		
				this.User.DisplayProp = nil
			end

		end,

		create = function()
			
			--if r2:getLeftQuota() <= 0 then -- ??	
			--	r2:makeRoomMsg()
			--	return
			--end
	
			local function posOk(x, y, z)
				if r2.mustDisplayInfo("ChatSequence") == 1 then 
					r2.displayFeatureHelp("ChatSequence")
				end
				debugInfo(string.format("Validate creation of 'Dialog' at pos (%d, %d, %d)", x, y, z))
				r2.dialogs:newSequenceInst(x, y, z)
			end
			local function posCancel()
				debugInfo("Cancel choice 'Dialog' position")
			end	
			
			local creature = r2.Translator.getDebugCreature("object_component_dialog.creature")
			r2:choosePos(creature, posOk, posCancel, "createDialog")
		end,

		---- get Name ------
		getName = function(this)

			local name = this.Name
			if name == "" then

				local index = r2.logicComponents:searchElementIndex(this)
				if index >= 0 then
					name = i18n.get("uiR2EDDialog"):toUtf8() .. index
				end
			end
			
			return name 
		end,

		---- get Name ------
		getShortName = function(this)

			return this:getName() 
		end,

		---------------------------------------------------------------------------------------------------------

		editDialogs = function(this)
			r2.dialogs:openEditor()
		end,

		getAvailableCommands = function(this, dest)
			r2.Classes.LogicEntity.getAvailableCommands(this, dest)
			table.insert(dest, this:buildCommand(this.editDialogs, "edit_dialogs", "uiR2EDEditDialogs", "r2ed_edit_dialog.tga", true))				
			this:getAvailableDisplayModeCommands(dest)
		end,

		initEventValuesMenu = function(this, menu, categoryEvent)
			for ev=0,menu:getNumLine()-1 do

				local eventType = tostring(menu:getLineId(ev))

				if r2.events.eventTypeWithValue[eventType] == "Number" then
					menu:addSubMenu(ev)
					local subMenu = menu:getSubMenu(ev)
					local func = ""
					for i=0, 9 do
						local uc_name = ucstring()
						uc_name:fromUtf8( tostring(i) )
						func = "r2.events:setEventValue('','" .. categoryEvent .."','".. tostring(i).."')"
						subMenu:addLine(uc_name, "lua", func, tostring(i))
					end

				elseif r2.events.eventTypeWithValue[eventType]=="ChatStep" then

					menu:addSubMenu(ev)
					local subMenu = menu:getSubMenu(ev)

					for c=0, this.Components.Size-1 do
						local chat = this.Components[c]
						local uc_name = ucstring()
						uc_name:fromUtf8(chat:getShortName())
						subMenu:addLine(uc_name, "lua", 
							"r2.events:setEventValue('".. chat.InstanceId .."','" .. categoryEvent .. "')", chat.InstanceId)
					end

					if this.Components.Size==0 then
						subMenu:addLine(i18n.get("uiR2EdNoSelelection"), "", "", "")
					end
				end
			end
		end,
		
		getLogicAction = function(this, context, action)
			return r2.Translator.getDialogLogicAction(this, context, action)
		end,
		getLogicEvent = function(this, context, event)
			return r2.Translator.getDialogLogicEvent(this, context, event)
		end,
		getLogicCondition = function(this, context, condition)
			return r2.Translator.getDialogLogicCondition(this, context, condition)
		end,

		pretranslate = function(this, context)
			r2.Translator.createAiGroup(this, context)
		end,

		translate = function(this, context)
			r2.Translator.translateFeatureActivation(this, context)
			r2.Translator.translateDialog( this, context)
		end,

		updateVersion = function(this, scenarioValue, currentValue )
			local patchValue = scenarioValue
			if patchValue < 1 then
				local subComponents = {}
				r2.requestInsertNode(this.InstanceId, "", -1, "SubComponents", subComponents)
				r2.requestInsertNode(this.InstanceId, "", -1, "AutoStart", this.Active)
				r2.requestSetNode(this.InstanceId, "Active", 1)
				r2.requestSetNode(this.InstanceId, "Base", r2.Translator.getDebugBase("palette.entities.botobjects.dialog"))
				patchValue = 1
			end
			
			if patchValue == currentValue then return true end
			return false
		end,

		initLogicEntitiesInstancesMenu = function(this, subMenu, calledFunction)
			
			local entitiesTable = r2.Scenario:getAllInstancesByType(this.Name)
			for key, entity in pairs(entitiesTable) do
				local uc_name = ucstring()
				uc_name:fromUtf8(entity.Name)
				subMenu:addLine(uc_name, "lua", calledFunction.."('".. entity.InstanceId .."')", entity.InstanceId)
			end

			if table.getn(entitiesTable)==0 then
				subMenu:addLine(i18n.get("uiR2EdNoSelelection"), "", "", "")
			end
		end
		
	}


local ActivityStepVersion = 1
Logic.Components.ActivityStep = {
		BaseClass ="BaseClass",
		Name ="ActivityStep",
		Version = ActivityStepVersion,
		BuildPropertySheet = true,
		Menu ="ui:interface:r2ed_feature_menu",
		
		BuildPropertySheet = false,
		DisplayerVisualParams = 
		{
			Look = r2.PrimRender.GroupLook,		
			ArrayName = "Components"
		},
		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "activityStepPropertySheetDisplayer",	
		
		Prop =
		{
			{Name="Activity", Type="String"},
			{Name="ActivityZoneId",Type="RefId",Category="ActivityStep"},
			{Name="TimeLimit",Type="String",Category="ActivityStep"},
			{Name="TimeLimitValue", Type="String",Category="ActivityStep",Category="ActivityStep"},
			{Name="RoadCountLimit", Type="String",Category="ActivityStep",Category="ActivityStep", DefaultInBase = 1, DefaultValue = "0"},
			{Name="Chat",Type="RefId"},
			{Name = "Type", Type="String",DefaultValue = "None"},  --TEMP TEMP TEMP
			{Name="EventsIds",Type="Table"},
			{Name="Name", Type="String", MaxNumChar="32"}
		},
		-- get sequencable parent for this activity step
		getSequencableParent = function(this)
			local currParent = this.Parent
			while currParent do
				if currParent.InstanceId and currParent:isSequencable() then
					return currParent
				end
				currParent = currParent.Parent
			end
			return nil
		end,
		getVerbLookupString = function(this)
			local seqParent = this:getSequencableParent()			
			if seqParent then
				return seqParent:getActivityVerbLookupName(this.Activity)
			else
				return this.Activity
			end			
		end,
		-- do lookup from the verb lookup string used to know the display name, and button texture
		doLookupFromVerb = function(this, lookupTable)			
			local lookupString = this:getVerbLookupString()			
			-- activity type
			local value = lookupTable[lookupString]
			if  not value then
				debugInfo(colorTag(255,0,0).."The activity '".. lookupString .."' is not properly registred")				
				return nil				
			end
			return value
		end,
		-- get texture for the button displayed in the mini activity view
		getMiniButtonTexture = function(this)						
			local result = this:doLookupFromVerb(r2.miniActivities.activityTextures)
			if result then return result end
			return "brick_default.tga"
		end,
		-- get the verb for this activity (rest, feed, patrol etc...)
		getVerb = function(this)									
			local result = this:doLookupFromVerb(r2.activities.activityTypeTranslation)
			if result then result = result.trans end
			if result then return result end
			return ucstring("Activity not registered : " .. lookupString)
		end,
		-- element name
		getName = function(this)			
			local activityType = this:getVerb() .. " "
			if this.ActivityZoneId~= "" then
				local place = r2:getInstanceFromId(tostring(this.ActivityZoneId))
				if place~=nil then
					activityType = activityType.. place.Name .." "
				end	
			end

			-- activity time
			local activityTime = ""
			if this.TimeLimit == "Few Sec" then
			
				local hourNb, minNb, secNb = r2.logicComponents:calculHourMinSec(tonumber(this.TimeLimitValue))

				local timeLimitText = i18n.get("uiR2EdFor"):toUtf8() .. " "
				if hourNb ~= 0 then timeLimitText = timeLimitText .. hourNb .. i18n.get("uiR2EdShortHours"):toUtf8() .. " " end
				if minNb ~= 0 then timeLimitText = timeLimitText .. minNb .. i18n.get("uiR2EdShortMinutes"):toUtf8() .. " " end
				timeLimitText = timeLimitText .. secNb .. i18n.get("uiR2EdShortSeconds"):toUtf8() 
				
				activityTime = timeLimitText

			elseif r2.activities.timeLimitsTranslation[this.TimeLimit] ~= nil then
				activityTime = string.lower(r2.activities.timeLimitsTranslation[this.TimeLimit])
			end

			return activityType..activityTime
		end,

		-- element name
		getShortName = function(this)

			-- activity type			
			local activityType = this:getVerb()
			if this.ActivityZoneId~= "" then
				local place = r2:getInstanceFromId(tostring(this.ActivityZoneId))
				if place~=nil then
					activityType = activityType.. " " .. place.Name 
				end	
			end

			return activityType
		end,

		updateVersion = function(this, scenarioValue, currentValue )
			local patchValue = scenarioValue

			if patchValue < 1 then
				local activity = this.Activity
				if (activity and activity == "Inactive") then
					r2.requestSetNode(this.InstanceId, "Activity", "Stand Still")
				end
				local name = this.Name
				
				patchValue = 1
			end
			if patchValue == currentValue then return true end
			return false
		end
	}



Logic.Components.ActivitySequence = {
		BaseClass="BaseClass",
		Name="ActivitySequence",
		BuildPropertySheet = true,
		Menu ="ui:interface:r2ed_feature_menu",

		--DisplayerUI = "R2::CDisplayerLua",
		--DisplayerUIParams = "defaultUIDisplayer",
		BuildPropertySheet = false,
		DisplayerVisual = "R2::CDisplayerVisualActivitySequence",
		DisplayerProperties = "R2::CDisplayerLua",
		DisplayerPropertiesParams = "activitySequencePropertySheetDisplayer",	
		
		Prop =
		{
			--{Name ="InstanceId", Type ="String"},
			{Name ="Repeating", Type ="Number" , DefaultValue = "1"},
			{Name ="Components", Type ="Table"},
			{Name ="Name", Type ="String"}
		},

		---- get Name ------
		getName = function(this)

			local name = this.Name
			if name == "" then

				local index = r2.logicComponents:searchElementIndex(this)

				if index >= 0 then
					name = i18n.get("uiR2EdSeq"):toUtf8() .. index
				end
			end
			
			return name 
		end,

		---- get Name ------
		getShortName = function(this)

			return this:getName() 
		end,

		-- get ActiveLogicEntity parent
		getActiveLogicEntityParent = function(this)

			-- ActiveLogicEntity.Behavior.Activities
			local activeLogicEntity = this.Parent.Parent.Parent

			-- NpcGrpFeature.Components
			--if activeLogicEntity:isGrouped() then activeLogicEntity = activeLogicEntity.Parent.Parent end

			return activeLogicEntity
		end,
	}



--this function must be called when we delete a ChatAction,
--to unregister the text
onDeleteChatAction = function(chatAction)
	if chatAction.User.DeleteInProgress == true then return end
	chatAction.User.DeleteInProgress = true
	if chatAction.Says ~= nil and chatAction.Says ~= ""
	then

		debugInfo("onDeleteChatAction")
		r2.unregisterTextFromId(chatAction.Says)
	end
end

--counts the states created, to ensure states names uniqueness

--timer used to time the sequences steps
Logic.activityStepTimer = "t0"
Logic.activityStepTimerId = 0
--timer used to time the chat steps
Logic.chatTimer = "t1"
Logic.chatTimerId = 1
--timer used inside activities
Logic.activityInternalTimer = "t2"
Logic.activityInternalTimerId = 2

--variable used to know which action of the current chat sequence
--must be played
Logic.chatStepVar = "v1"
--variable used to know which chat sequence is currently played
Logic.chatSequenceVar = "v0"
--variable used to know which activity sequence is currently played
Logic.activityStepVar = "v2"
--variable used to know which interrupt action must be executed
Logic.triggerEventVar = "v3"

Logic.EndOfActivitySequenceEvent = 9
Logic.EndOfChatSequenceEvent = 8

--switch_action which hold all chat sequences
Logic.DialogAction = nil
--switch action holding chat sequences transitions
Logic.DialogAction2 = nil

--switch_action which hold all sequence steps
Logic.SequenceAction = nil
Logic.ActionStepCounter =0
Logic.ChatSequenceCounter = 0
--used to know the names of all the states generated
--to translate this activity
Logic.StatesByName = ""

--timer_triggered event, triggered 
--each time we must enter a new chat step
Logic.DialogEvent = nil

Logic.DialogEvent2 = nil

--timer_triggered event, triggered
--each time we must enter a new sequence step
Logic.SequenceEvent = nil

--hold the states used in the current activity.
--all the states are created at the start of the translation,
--before translating the events and actions.
Logic.States = {}

--hold the states used in all the activities
--of the group
Logic.ActivitiesStates ={}

--the activity which is currently being translated.
Logic.CurrentActivity = nil

----------------------------------------------------
--to be called once for each sequence of each group
--complete the AiActivity field of the first state of this sequence
--and create the actions to go to the first step
Logic.initGroupActivitiesTranslation = function(context, hlComponent, sequence, first, activityIndex, aiActivity, rtGrp)
	assert(context and hlComponent and sequence and activityIndex and aiActivity and rtGrp)
	--get the states of this activity
	
	local activityStates = Logic.ActivitiesStates[ tostring(sequence.InstanceId) ]	
	assert(sequence.Class == "ActivitySequence")
	if not activityStates then
		
		if hlComponent then
			printWarning("Error while translating '" .. hlComponent.Name .. "' activities in sequence " .. sequence.Name)			
			-- pretranslate sequence.InstanceId errors instanceId not added in currentAct.ActivitiesId ?
			return nil
		end		
	end
	--get the initial state of this activity
	local aiState = activityStates[sequence.InstanceId][1]
	Logic.StatesByName = ""
	for k, v in pairs(activityStates)
	do
		if v.Class ~= "RtAiState"
		then
			local  k1, v1 = next( v, nil)
			while k1 do
				Logic.StatesByName = r2.Utils.concat(Logic.StatesByName, v1.Name)
				k1, v1 = next(v, k1)
			end
		else
			Logic.StatesByName = r2.Utils.concat(Logic.StatesByName, v.Name)
		end
	end
	aiState.AiActivity = aiActivity
	assert(aiState.AiActivity)
	local event = r2.Translator.createEvent("start_of_state", aiState.Name, rtGrp.Name)
	-- local event = r2.Translator.createEvent("group_spawned", aiState.Name, rtGrp.Name)
	table.insert(context.RtAct.Events, event)
	if first == true
	then
		table.insert(aiState.Children, rtGrp.Id)
	end

	local speed = ""

	if hlComponent.Speed and hlComponent.Speed == "run" then speed = "()addProfileParameter(\"running\");\n" end
	
	local code = "currentActivitySequenceVar = " .. tostring(activityIndex -1 ) ..";\n"			
		.. "oldActivitySequenceVar = currentActivitySequenceVar;\n"
		.. "RoadCountLimit=0;\n"
		.. "oldActivityStepVar = -1;\n" 
		.. "oldActivityStepVar2 = 0;\n" 
		.. Logic.activityStepVar .." = 0;\n" 
		.. speed
		.."alive = 0;\n"
		.."(alive)isAlived();\n"
		.. "if (alive == 1)\n{\n"
		.. "\t()setTimer(30, " .. Logic.activityStepTimerId .. ");\n"
		.. "\tinit = 1;\n"
		.. "}\n"


	-- Fauna activities


	local action = r2.Translator.createAction("code", code)
	action.Name = "startActivitySequence"

	table.insert(context.RtAct.Actions, action)
	table.insert(event.ActionsId, action.Id)



	do
		local event = r2.Translator.createEvent("group_spawned", aiState.Name, rtGrp.Name)
		local action = r2.Translator.createAction("code", "if (init != 1)\n{\n\t()setTimer(30, " .. Logic.activityStepTimerId .. ");\n}\n")
		table.insert(context.RtAct.Events, event)
		table.insert(context.RtAct.Actions, action)
		table.insert(event.ActionsId, action.Id)
	end


	local leader = hlComponent

	if hlComponent:isKindOf("NpcGrpFeature") then
		if hlComponent.Components.Size ~= 0 then
			leader = hlComponent.Components[0]
		else
			leader = nil
		end
	end
end

--insert an activity step at the specified position
Logic.insertActivityStep = function(sequence, step, pos)
	local n = table.getn(sequence.Components)
	if n < pos
	then
		table.insert(sequence.Components, step)
	else
		table.insert(sequence.Components, pos, step)
	end
end

--insert a chat step at the specified position
Logic.insertChatStep = function(chatSequence, chatStep, pos)
	local n = table.getn(chatSequence.Components)
	if n < pos
	then
		table.insert(chatSequence.Components, chatStep)
	else
		table.insert(chatSequence.Components, pos)
	end
end


Logic.createInteraction = function(action, parameter)
	local index = Logic.find(Logic.InteractionTypes, action)
	local interaction
	
	if index == nil or parameter == nil
	then
--			debugInfo(colorTag(255,0,0).."unknown interaction : " .. action .." !")
		return nil
	end

	interaction = r2.newComponent("Interaction")
	interaction.Type = action
	interaction.Parameter = parameter
	return interaction
end

Logic.createActivityStep = function(activity, timeLimit, zoneId, time)
	local activityStep = nil
	
	if activity == nil or timeLimit == nil
	then
		return nil
	end

--checking activity type and time limit type
	if Logic.find(Logic.Activities, activity)~= nil and Logic.find(Logic.TimeLimits, timeLimit)~= nil
	then
		--checking activity parameter
		if ( (activity ~= "Stand Still") and (activity ~= "Inactive") and (zoneId == nil) )
		then
			--this activity need a zone, and no zone id is given
--				debugInfo(colorTag(255,0,0).."activity <" .. activity .."> need a zone id!")
			return nil
		end
		
	--checking time parameter
		--if time parameter needed, check its presence
		if timeLimit~="No Limit" and timeLimit~="Chat" and time == nil
		then
			--this activity needs a time limit and no limit is given
--				debugInfo(colorTag(255,0,0).."Time limit <" .. timeLimit .."> need a time limit parameter!")
			return nil
		end

		--check season validity
		if (timeLimit == "Season") and (Logic.find(Logic.Seasons, time) == nil)
		then
--				debugInfo(colorTag(255,0,0).."Unknown season: " .. time)
			return nil
		end
		
		--check day validity
		if (timeLimit == "Day") and (Logic.find(Logic.Days, time) == nil)
		then
			debugInfo(colorTag(255,0,0).."Uknown week day: " .. time)
			return nil
		end
		
		if (timeLimit == "Month") and (Logic.find(Logic.Months, time)== nil)
		then
--				debugInfo(colorTag(255,0,0,0).."Uknown month: " .. time)
			return nil
		end

		activityStep = r2.newComponent("ActivityStep")
		activityStep.Activity = activity
		activityStep.TimeLimit = timeLimit
		activityStep.ActivityZoneId = zoneId
		activityStep.TimeLimitValue = time
	else
--			debugInfo(colorTag(255,0,0).."Unknown activity !")
	end
	return activityStep
end

Logic.createState = function()
	local aiState = r2.newComponent("RtAiState")
	aiState.Name = aiState.Id
	Logic.StatesByName = Logic.StatesByName .."\n" .. aiState.Name
	return aiState
end

--create the AiStates necessay to translate this step
Logic.createStepStates = function(step, context)
	local states = {}
	local state = Logic.createState()
	table.insert(states, state)
	table.insert(context.RtAct.AiStates, state)
	if step.Activity == "Patrol"
	then
		local state = Logic.createState()
		table.insert(states, state)
		table.insert(context.RtAct.AiStates, state)
	end
	return states
end

--assign a zone or a road to an AiState
--if invert is true, invert the road
Logic.assignZone = function(rtAiState, zone, invert)
	assert( zone and type(zone) == "userdata")
	local points = zone.Points
	assert( points ~= nil)

	local size = points.Size + 1
	rtAiState.Pts = {}
	local i =0
	local j

	local k, v = next(points, nil)
	while k ~= nil
	do
		assert(v ~= nil)
		i = i +1
		if invert == true
		then
			j = size - i
		else
			j = i
		end
		rtAiState.Pts[j]={}
		rtAiState.Pts[j].x = r2.getWorldPos(v).x
		rtAiState.Pts[j].y = r2.getWorldPos(v).y
		rtAiState.Pts[j].z = r2.getWorldPos(v).z
		k, v = next(points, k)
	end
end

--create all the states used in this activity Sequence
Logic.createActivityStates = function(context, sequence)
	assert( sequence ~= nil)

	local activityStates ={}
	--activity's initial stat, aiStopStatee
	
	-- init state
	local aiStartState = r2.newComponent("RtAiState")
	aiStartState.Name = aiStartState.Id
	Logic.StatesByName = aiStartState.Name
	aiStartState.AiMovement = "idle" -- "stand_on_start_point"
	table.insert(context.RtAct.AiStates, aiStartState)
	

	-- end state
	local aiStopState = r2.newComponent("RtAiState")
	aiStopState.Name = aiStopState.Id
	--Logic.StatesByName = aiStopState.Name
	aiStopState.AiMovement = "idle" -- "stand_on_start_point"
	table.insert(context.RtAct.AiStates, aiStopState)

	activityStates[sequence.InstanceId]= { aiStartState, aiStopState}


	if (sequence.Components == nil) then assert(nil) end
	--creation of the states of the step
	local k, v = next(sequence.Components, nil)
	while k do
		local states = Logic.createStepStates(v, context)
		activityStates[v.InstanceId]= states
		 k, v = next(sequence.Components, k)
	end
	Logic.ActivitiesStates[sequence.InstanceId] = activityStates
end

Logic.translateActivitySequence = function(context, hlComponent, activitySequence, sequenceId, rtNpcGrp)
	assert(rtNpcGrp ~= nil)
	if context.Real == false
	then
		return
	end
	Logic.ActionStepCounter = 0
	Logic.ChatSequenceCounter = 0
	--Logic.States = {}
	Logic.CurrentActivity = activitySequence
	local ok = false

	local nbSteps = activitySequence.Components.Size
	local aiState
	local action
	local event
	local sequence = activitySequence


	if sequence.Components.Size ==0
	then	
		return
	end


	Logic.initSequence(context, rtNpcGrp)
	if  Logic.ActivitiesStates[sequence.InstanceId] then
		Logic.States = Logic.ActivitiesStates[sequence.InstanceId]
	else
		Logic.States = ""
	end
	
	

	local i =1
	local k, activityStep = next(sequence.Components, nil)
	while k do
		--translate step
		Logic.translateActivityStep(context,  hlComponent, activitySequence, activityStep, i, rtNpcGrp)
		i = i + 1
		k, activityStep = next(sequence.Components, k)

	end

	--create actions to repeat the sequence
	if sequence.Repeating == 1
	then

		local code =
			"//sequence will repeat"..
			"oldChatSequenceVar = " .. Logic.chatSequenceVar .. ";\n" ..
			"RoadCountLimit=0;\n" ..
			"oldActivityStepVar = " .. Logic.activityStepVar .. ";\n" ..
			"oldActivityStepVar2 = " .. Logic.activityStepVar .. " + 1" .. ";\n" ..
			"v2=0;\n" ..
			"()setTimer(1, " ..Logic.activityStepTimerId .. ");\n"
--			.. "()setEvent(" .. Logic.EndOfActivitySequenceEvent ..");\n"
		
		local action = r2.Translator.createAction("code", code)
		action.Name = "repeat_sequence"
		table.insert(Logic.SequenceAction.Children, action)
		
	else
	
		local backupState =
			"//sequence will not repeat"..
			"oldChatSequenceVar = " .. Logic.chatSequenceVar .. ";\n" ..
			"RoadCountLimit=0;\n"..
			"oldActivityStepVar = " .. Logic.activityStepVar .. ";\n" ..
			"oldActivityStepVar2 = " .. Logic.activityStepVar ..  " + 1" .. ";\n"
		
		local stateName = Logic.ActivitiesStates[sequence.InstanceId][sequence.InstanceId][2].Id

		local setNextActivityState = "()postNextState(\"" .. r2.getNamespace() .. stateName .."\");\n"
		
		local action = r2.Translator.createAction("code", backupState 
			-- .."()setEvent(" .. Logic.EndOfActivitySequenceEvent ..");\n"
			..setNextActivityState)
		
		
		action.Name = "end_of_activity_sequence"
		table.insert(Logic.SequenceAction.Children, action)
	end

	Logic.SequenceEvent.StatesByName = Logic.StatesByName
	Logic.DialogEvent = nil
	Logic.SequenceEvent = nil
end



Logic._obsolete_TranslateScriptAction = function(context, step, nextTime)
	local actionType = Logic.getAction(step)
	assert(actionType~= nil)
--		debugInfo(colorTag(0,255,0).. actionType)
	
	if actionType == "stop_actions"
	then
		local action = r2.Translator.createAction("set_timer_" ..Logic.chatTimer, "0")
		return action
	end
	
	if actionType == "start_mission"
	then
		local counterMission = context.Components[step.Parameter]
		local action = r2.Translator.createAction("trigger_event_1", context.RtCounters[counterMission.InstanceId].RtGroup.Name)
--			debugInfo("action created!!")
		return action
	end

	if 0==1 and actionType == "go_to_step"
	then
		--TODO
		local param = tonumber(step.Parameter)
		debugInfo("param = " .. param)
		if (param < 1) or (Logic.CurrentActivity.Components[param]== nil)
		then
--				debugInfo(colorTag(255,0,0).."Impossible to go to step " .. param .." !")
			return nil
		end
		debugInfo(context.Feature.Class)
		return Logic.selectActivityStep(param)
	end

	if actionType == "set_activity"
	then
		local group = nil
		local param
		local rtGroup
		if step.Who ~= "" and step.Who ~= nil
		then
			group = step.Who
			rtGroup = r2.Translator.getRtGroup(context, group)
		end
		
		param = tonumber(step.Parameter)

--			debugInfo("param = " .. param)

		local action
		if group ~= nil
		then
			action = r2.Translator.createAction("begin_state", Logic.getActivityInitialStateName(group, param, context))
			assert(action ~= nil)
			local index = Logic.addTriggeredAction(context, context.RtGroups[group].Name, action)
			action = r2.Translator.createAction("modify_variable", rtGroup.Name ..":" ..Logic.triggerEventVar .." = " .. index)
			action.Name ="set_activity"
		else
			action = r2.Translator.createAction("begin_state", Logic.getActivityInitialStateName(context.Group.Name, param, context))
			assert(action ~= nil)
		end
		return action
	end

	local action = r2.Translator.createAction(actionType)
	assert(action ~= nil)
	return action
end


	

Logic.translateChatAction = function(context, chatSequenceNumber, chatStepNumber, chatSequence, chatStep, chatAction, rtMAction)
	local getRtId = r2.Features["TextManager"].getRtId
	if chatAction.Class == "ChatAction"
	then	
		local who =  r2.Utils.getNpcParam( tostring( chatAction.Who), context)
		-- create say action
		if not who then

			local whoInstance = r2:getInstanceFromId( tostring(chatAction.Who) )
			local chatSequenceInstance = r2:getInstanceFromId(chatSequence.InstanceId)
			
			local whoName = ""
			if whoInstance then  whoName = "(" .. whoInstance.Name ..")" end

			if false then
				printWarning( "Error in ChatStep '" .. chatStep:getName()
				..	"' of the ChatSequence '" .. chatSequence:getName() .. "'"
				.. 	" the field Who '" ..  chatAction.Who .."' is invalid " )
			end

			return nil
				
		end

		if  chatAction.Says ~= "" and chatAction.Says ~= nil
		then
			local param = getRtId(context, tostring(chatAction.Says))
			if param then
				action = r2.Translator.createAction("npc_say", param,   who); 

			--	action = r2.newComponent("RtNpcEventHandlerAction")
			--	action.Action = "npc_say"
			
			--	action.Parameters = who .."\n" .. param
				table.insert(rtMAction.Children, action)
			else
				debugInfo(colorTag(255,255,0).."WRN: Text " .. chatAction.Says .. " error")
			end
	
		end

			-- create emot action
		if tostring(chatAction.Emote) ~= "" and chatAction.Emote ~= nil
		then
			action = r2.newComponent("RtNpcEventHandlerAction")
			action.Action = "emot"
			action.Parameters = tostring(chatAction.Emote).."\n" .. who
			table.insert(rtMAction.Children, action)
		end

		-- create facing action
		if tostring(chatAction.Facing) ~= "" and chatAction.Facing ~= nil
		then
			local facing = r2.Utils.getNpcParam(tostring(chatAction.Facing), context)
			if facing then
				action = r2.newComponent("RtNpcEventHandlerAction")
				action.Action = "facing"
				action.Parameters = facing .."\n" .. who
				table.insert(rtMAction.Children, action)
			else
				printWarning("Can not find npc")
			end
		end
	else
		printWarning( chatAction.Class .. " not implemented yet")
	end
end

Logic.translateChatStep = function(context, chatSequenceNumber, chatSequence, chatStepNumber, chatStep, rtActionSwitchChatStep)
	
	assert(chatStep)
	assert(chatStep.Class == "ChatStep")
	
	if  chatStep.Actions.Size > 0  then

		local rtMAction = r2.Translator.createAction("multi_actions")		
		table.insert(rtActionSwitchChatStep.Children, rtMAction)
		
		local k, chatAction = next(chatStep.Actions, nil)
		while k do
			Logic.translateChatAction(context, chatSequenceNumber, chatStepNumber, chatSequence, chatStep, chatAction, rtMAction)

			local oldChatAction = chatAction
			-- next wait
			do
				
				if (chatStepNumber < chatSequence.Components.Size and chatSequence.Components[chatStepNumber ]) then
				
					local action = r2.Translator.createAction("code", 
						"oldChatStepVar = " .. tostring(chatStepNumber)..";\n"  ..
						Logic.chatStepVar .." = " .. tostring(chatStepNumber+1)..";\n" ..
						"()setTimer(" .. tostring(1+chatSequence.Components[chatStepNumber].Time*10) .. ", " .. Logic.chatTimerId .. ");\n");
					action.Name = "next"
					table.insert(rtMAction.Children, action)
				
				else
					local action = r2.Translator.createAction("code", 
						"oldChatStepVar = " .. tostring(chatStepNumber)..";\n"  ..
						Logic.chatStepVar .." = " .. tostring(chatStepNumber+1)..";\n" ..
						"()setEvent(" ..Logic.EndOfChatSequenceEvent ..");\n" ..
						"if (repeatSequence == 1)\n{\n  " .. Logic.chatSequenceVar .."=" ..Logic.chatSequenceVar ..";\n" ..
						"}\n")

					action.Name = "end_of_chat_sequence"
					table.insert(rtMAction.Children, action)

				end
	
	

			end

			
			k, chatAction = next(chatStep.Actions, k)

		end

		
	else
		local action = r2.Translator.createAction("null_action")
		action.Name = "empty_chat_step_" .. chatStepNumber
		table.insert(rtActionSwitchChatStep.Children, action)
	end

end

Logic.translateChatSequence = function (context, chatSequenceNumber, chatSequence, rtSwitchChatSequence)
	assert(chatSequence)
	assert(chatSequence.Class == "ChatSequence")
	if  chatSequence.Components.Size  > 0  then
		
		local rtActionSwitchChatStep = r2.Translator.createAction("switch_actions", Logic.chatStepVar)
		rtActionSwitchChatStep.Name = "switch_chat_step" 
		table.insert(rtSwitchChatSequence.Children, rtActionSwitchChatStep)
		
		local chatStepNumber = 0
	
		-- initial wait
		do
			local action = r2.Translator.createAction("code", "oldChatStepVar = 0;\n" 
				.. Logic.chatStepVar .." = 1;\n()setTimer(" .. tostring( tonumber(chatSequence.Components[0].Time)*10 + 1) .. ", " .. Logic.chatTimerId ..");\n");
			action.Name = "initial_wait"
			table.insert(rtActionSwitchChatStep.Children, action)
		end

	
		local k, chatStep = next(chatSequence.Components, nil)
		while k do
			chatStepNumber = chatStepNumber + 1
			Logic.translateChatStep(context, chatSequenceNumber, chatSequence, chatStepNumber, chatStep, rtActionSwitchChatStep)
			k, chatStep = next(chatSequence.Components, k)
		end
	else
		local action = r2.Translator.createAction("null_action")
		action.Name = "empty_chat_sequence_" .. chatSequenceNumber
		table.insert(rtSwitchChatSequence.Children, action)
	end
end

Logic.translateChatSequences = function (context, hlComponent, behavior, rtNpcGrp)
	assert(behavior.ChatSequences)
	if behavior.ChatSequences.Size > 0 then
		assert(rtNpcGrp)

		do
			local	event = r2.Translator.createEvent("variable_" ..Logic.chatSequenceVar .."_changed", "", rtNpcGrp.Name)
			event.Name = "activity_sequence_changed"
			table.insert(context.RtAct.Events, event)
			
			local rtInitChatStep = r2.Translator.createAction("code", "oldChatStepVar = -1;\n" .. Logic.chatStepVar .." = 0;\n()setTimer(1, " ..Logic.chatTimerId .. ")\;\n")
			rtInitChatStep.Name = "init_chat_step"
			table.insert(context.RtAct.Actions, rtInitChatStep)
			table.insert(event.ActionsId, rtInitChatStep.Id)
		end
		

		local	event = r2.Translator.createEvent("timer_" ..Logic.chatTimer .."_triggered", "", rtNpcGrp.Name)
		event.Name = "dialog_event"
		table.insert(context.RtAct.Events, event)
			
		local rtSwitchChatSequence = r2.Translator.createAction("switch_actions", Logic.chatSequenceVar)
		rtSwitchChatSequence.Name = "switch_chat_sequence"
		table.insert(context.RtAct.Actions, rtSwitchChatSequence)
		table.insert(event.ActionsId, rtSwitchChatSequence.Id)
		
		local chatSequenceNumber = 0
		local k, chatSequence = next(behavior.ChatSequences, nil)
		while k do
			context.Feature = nil
			chatSequenceNumber = chatSequenceNumber + 1
			Logic.translateChatSequence(context, chatSequenceNumber, chatSequence, rtSwitchChatSequence)	
			k, chatSequence = next(behavior.ChatSequences, k)
		end
	end
end



function Logic.isWanderActivity(activity)
	return activity == "Wander" or activity == "Go To Zone" or activity == "Rest In Zone" or activity == "Feed In Zone"
		or activity == "Hunt In Zone" or activity == "Guard Zone"
end


-------------------------------------------------
--create a transition to go to step number stepNb
--set a timer to go to next step
--select the step
--goto the step's state
Logic.createTransition = function(context,  hlComponent, activitySequence, activityStep, activityStepIndex, stepAction)
	local findChatSequenceIdByInstanceId = function (chat)

		local parent = chat.ParentInstance.ChatSequences
		assert(parent)

		
		local nbSequence = 0
		local k, sequence = next(parent, nil)
		while k do
			if tostring(sequence.InstanceId) == tostring(chat.InstanceId) then return nbSequence end			
 			nbSequence = nbSequence + 1
			k, sequence = next(parent, k)
		end
		return -1
	end

	assert(activityStep)

	local chat = nil


	if activityStep.Chat ~= "" then
		chat = r2:getInstanceFromId( tostring(activityStep.Chat) )
	end

	local code = ""


	local backupState = ""
	local setChatSequence = ""
	local setNextActivity = ""
	local setNextActivityTimer = ""
	local setNextActivityState = ""

	
	backupState =
		"oldChatSequenceVar = " .. Logic.chatSequenceVar .. ";\n" ..
		"RoadCountLimit=0;\n"..
		"oldActivityStepVar = " .. Logic.activityStepVar .. ";\n" ..
		"oldActivityStepVar2 = " .. Logic.activityStepVar.. " + 1;\n"
	
		
	if chat~= nil then		
		local id = findChatSequenceIdByInstanceId( chat )
		assert(id ~= -1 )
		setChatSequence =  Logic.chatSequenceVar .. " = " .. tostring( id ) .. ";\n"		
		if (activityStep.Type == "Repeating") then 
			setChatSequence = setChatSequence .. "repeatSequence = 1;\n" 
		else
			setChatSequence = setChatSequence .. "repeatSequence = 0;\n" 
		end
	end
	
	-- next activity
	setNextActivity =  Logic.activityStepVar .. " = " .. activityStepIndex ..";\n"	

	-- next activity timer
	local param = Logic.getTimeLimit(activityStep)
	if param ~= nil  then
		if Logic.isWanderActivity(activityStep.Activity) then
			-- The time limite is use by the activity
		else
			setNextActivityTimer  = "()setTimer(" .. param ..", " ..Logic.activityStepTimerId .. ");\n"
		end
	end

	local states = Logic.States[activityStep.InstanceId]
	local stateName = states[1].Name

	-- next activity state
	
	setNextActivityState = "()postNextState(\"" .. r2.getNamespace() .. stateName .."\");\n"
	local action = r2.Translator.createAction("code", backupState .. setChatSequence .. setNextActivity .. setNextActivityTimer .. setNextActivityState)


	action.Name = "process_activity_step" .. activityStepIndex
	
	if stepAction then
		local tmpAction = r2.Translator.createAction("multi_actions")
		table.insert(tmpAction.Children, action)
		table.insert(tmpAction.Children, stepAction)		
		action = tmpAction
	end
	
	action.Name = "process_activity_step" .. activityStepIndex
	return action
	
end


------------------------------------------
--cr�e une action qui permet de 
--s�lectionner une s�quence de chat
Logic.selectDialog = function(dialogNb, rtNpcGrp)
	local action
	local prefix = ""
	if (rtNpcGrp ~= nil) then
		prefix = r2.getNamespace() .. rtNpcGrp .. "." 
	end
		
	action = r2.Translator.createAction("code",
		prefix .."oldChatSequenceVar =" .. prefix..Logic.chatSequenceVar ..";\n" ..
		prefix .. Logic.chatSequenceVar .." = " ..(dialogNb-1)..";\n")
	action.Name = "select_dialog"
	return action
end

--create the actions and events necessary to choose
--the dialog while running sequence
Logic.initDialog = function(context)
	local event = r2.Translator.createEvent("variable_" ..Logic.chatSequenceVar .."_changed", "all_sequence_states", context.Group.Name)
	event.Name = "change_dialog_event"
	Logic.DialogEvent2 = event
	local action
	local mAction = r2.Translator.createAction("multi_actions")
	action = r2.Translator.createAction("code", "oldChatStepVar = " .. Logic.chatStepVar ..";\n" .. Logic.chatStepVar .." = 0;\n")
	action.Name = "reset_dialog"
	table.insert(mAction.Children, action)
	action = r2.Translator.createAction("switch_actions", Logic.chatSequenceVar)
	action.Name = "init_dialog_timer"
	table.insert(mAction.Children, action)
	Logic.DialogAction2 = action
	table.insert(context.RtAct.Events, event)
	table.insert(context.RtAct.Actions, mAction)
	table.insert(event.ActionsId, mAction.Id)


	event = r2.Translator.createEvent("timer_" ..Logic.chatTimer .."_triggered", "all_sequence_states", context.Group.Name)
	event.Name = "dialog_event"
	Logic.DialogEvent = event
	table.insert(context.RtAct.Events, event)

	Logic.DialogAction = r2.Translator.createAction("switch_actions", Logic.chatSequenceVar)
	table.insert(context.RtAct.Actions, Logic.DialogAction)
	table.insert(event.ActionsId, Logic.DialogAction.Id)
end

Logic.initSequence = function(context, rtNpcGrp)
	assert(context)
	assert(rtNpcGrp)

	event = r2.Translator.createEvent("timer_" ..Logic.activityStepTimer .."_triggered", "all_sequence_states", rtNpcGrp.Name)
	event.Name = "sequence_event"
	Logic.SequenceEvent = event
	table.insert(context.RtAct.Events, event)

	Logic.SequenceAction = r2.Translator.createAction("switch_actions", Logic.activityStepVar)
	table.insert(context.RtAct.Actions, Logic.SequenceAction)
	table.insert(event.ActionsId, Logic.SequenceAction.Id)
end

-- ust to implement translateEvents used to implement event created for implementing 
Logic.translateEvent = function( context,  hlComponent, activitySequence, activityStep,  activityStepIndex, event, rtNpcGrp, statesByName)
	local chat = nil
-- context, groups, states, actionToAdd)
	local eventType, eventParam = Logic.getEvent(event)
	local eventHandler = nil

	if context.Events[event.InstanceId] == nil
	then
		eventHandler = r2.Translator.createEvent(eventType, states, groups)
		assert(eventHandler ~= nil)
			
		local stepAction = Logic.translateChatStep(context, event.Action, nil)
			
		if actionToAdd ~= nil
		then
			table.insert(stepAction.Children, actionToAdd)
		end

		assert(stepAction ~= nil)

		table.insert(context.RtAct.Actions, stepAction)
		table.insert(eventHandler.ActionsId, stepAction.Id)
		context.Events[event.InstanceId] = eventHandler
	else
		eventHandler = context.Events[event.InstanceId]
			
		if string.find(eventHandler.StatesByName, states) == nil
		then
			eventHandler.StatesByName = eventHandler.StatesByName .. "\n" .. states
		end

		if string.find(eventHandler.GroupsByName, groups) == nil
		then
			eventHandler.GroupsByName = eventHandler.GroupsByName .. "\n" .. groups
		end
				
	end

	return eventHandler
end

-- translateEvents
Logic.translateEvents = function(context,  hlComponent, activitySequence, activityStep,  activityStepIndex, rtNpcGrp, statesByName)
	
	local activityStep

	local last = nil

	if activityStepIndex == activitySequence.Components.Size then 
		last = true
	end


	local  k, eventId  = next(step.EventsIds, nil)
	do
		local eventHandler = r2:getInstanceFromId(eventId)
		assert(eventHandler ~= nil)
		local action = nil
		
		if  eventHandler.Condition.Type == "At Destination"	then
			if step.Activity == "Follow Route" and last ~= true then
				action = Logic.selectActivityStep(activityStepIndex + 1)
			end
		end

		--the event returned here is a RtNpcEventHandler
		event = translateEvent(context, groups, states, action)
		assert(event ~= nil)
		
		table.insert(context.RtAct.Events, event)
		k, eventId  = next(step.EventsIds, k)
	end
end

--translate an activity step
Logic.translateActivityStep = function(context,  hlComponent, activitySequence, activityStep,  activityStepIndex, rtNpcGrp)

	--translate this activity activityStep's events
	assert(rtNpcGrp)
	local aiState
	local aiState2
	local statesByName
	local groupsByName = rtNpcGrp.Name
	local zone = nil
	local chat = nil
	local aiScriptDebug = false

	local activityStep = activityStep

	if activityStep.Chat ~= "" then
		chat = r2:getInstanceFromId(activityStep.Chat)
		if not chat then
			printErr("translation error: in the activitySequence " .. activitySequence.Name .. " in component " .. hlComponent.Name .. ": " .. " the chat sequence associated to this activity sequence can not be found")
			return nil
		end
	end
		
	local states = Logic.States[activityStep.InstanceId]
	--states creation
	aiState = states[1] -- 1 is first ( a lua object not a instance )

	--setting state AiMovement
	aiState.AiMovement = Logic.getAiMovement(activityStep.Activity)
	assert(aiState.AiMovement ~= "")
	--setting state zone or road


	local positionalActivity= {["Patrol"]		= "road", 
							["Wander"]			= "zone",
							["Repeat Road"]		= "road",		
							["Follow Route"]	= "road",		
							["Rest In Zone"]	= "zone",
							["Feed In Zone"]	= "zone",	
							["Hunt In Zone"]	= "zone",	
							["Guard Zone"]		= "zone",		
							["Go To Zone"]		= "zone",	
							 }
	if positionalActivity[activityStep.Activity] then
		local zone = r2:getInstanceFromId(activityStep.ActivityZoneId)
		if zone == nil then
			aiState = states[1]
			aiState.AiMovement = "idle"
			return aiState
		else
			if positionalActivity[activityStep.Activity] == "zone" then
				if table.getn(zone.Points) < 3 then
					aiState = states[1]
					aiState.AiMovement = "idle"
					return aiState
				end
			else
				if table.getn(zone.Points) < 2 then
					aiState = states[1]
					aiState.AiMovement = "idle"
					return aiState
				end
			end	
		end
	end

	


	if activityStep.Activity == "Patrol" then
		local zone = r2:getInstanceFromId(activityStep.ActivityZoneId)
		if (zone == nil)
		then
			printErr("translation error: in the activitySequence " .. activitySequence.Name .. " in component " .. hlComponent.Name .. ": " .. " the zone associated to the action can not be found")
			return nil
		end

		--un etat aller, un etat retour 
		aiState2 = states[2]
		aiState2.AiMovement = "follow_route"
		Logic.assignZone(aiState, zone, false)
		Logic.assignZone(aiState2, zone, true)

		statesByName = aiState.Name .."\n" .. aiState2.Name
	elseif activityStep.Activity == "Repeat Road" or  activityStep.Activity == "Follow Route" or  activityStep.Activity == "Wander" 
		or activityStep.Activity == "Rest In Zone" or activityStep.Activity == "Feed In Zone" or activityStep.Activity ==  "Hunt In Zone"
		or activityStep.Activity == "Guard Zone" or activityStep.Activity == "Go To Zone" then
		local zone = r2:getInstanceFromId(activityStep.ActivityZoneId)
		if (zone == nil) then
			printErr("translation error: in the activitySequence " .. activitySequence.Name .. " in component " .. hlComponent.Name .. ": " .. " the zone associated to the action can not be found")
			return nil
		end
		Logic.assignZone(aiState, zone, false)
		statesByName = aiState.Name
	else
		statesByName = aiState.Name
	end


	if activityStep.Activity == "Repeat Road" then
		local action
		local event
		--when at end of the road, ...
		event = r2.Translator.createEvent("destination_reached_all", statesByName, groupsByName)
		table.insert(context.RtAct.Events, event)
		--return to the start of the road
		local repeatCount = activityStep.RoadCountLimit
		if not repeatCount then repeatCount = 2 end

		action = r2.Translator.createAction("next_road", groupsByName, aiState.Name, activityStepIndex ,  repeatCount)
		table.insert(context.RtAct.Actions, action)
		table.insert(event.ActionsId, action.Id)

	elseif activityStep.Activity == "Patrol" then
		local action
		local event
		local repeatCount = activityStep.RoadCountLimit
		if not repeatCount then repeatCount = 2 end
		event = r2.Translator.createEvent("destination_reached_all", aiState.Name, groupsByName)
		table.insert(context.RtAct.Events, event)
		action = r2.Translator.createAction("next_road", groupsByName, aiState2.Name, activityStepIndex,  repeatCount)
		action.Name = "return"
		table.insert(context.RtAct.Actions, action)
		table.insert(event.ActionsId, action.Id)

		event = r2.Translator.createEvent("destination_reached_all", aiState2.Name, groupsByName)
		table.insert(context.RtAct.Events, event)
		action = r2.Translator.createAction("next_road", groupsByName, aiState.Name, activityStepIndex ,  repeatCount)
		action.Name = "go"
		table.insert(context.RtAct.Actions, action)
		table.insert(event.ActionsId, action.Id)

	elseif activityStep.Activity == "Follow Route" or activityStep.Activity == "Go To Start Point" or activityStep.Activity == "Go To Zone" then
		local action
		local event
		event = r2.Translator.createEvent("destination_reached_all", aiState.Name, groupsByName)
		table.insert(context.RtAct.Events, event)
		action =  Logic.selectActivityStep(activityStepIndex + 1)
		table.insert(context.RtAct.Actions, action)
		table.insert(event.ActionsId, action.Id)
	elseif activityStep.Activity == "Rest In Zone" then
		local wanderTime = 100
		local restTime = 300
		r2.Translator.createActivityInZone(context, aiState.Name, groupsByName, "Rest", Logic.activityInternalTimerId, wanderTime, restTime, aiScriptDebug)
	elseif activityStep.Activity == "Feed In Zone" then
		local wanderTime = 50
		local feedTime = 150
		r2.Translator.createActivityInZone(context, aiState.Name, groupsByName, "Eat", Logic.activityInternalTimerId, wanderTime, feedTime, aiScriptDebug)
	elseif activityStep.Activity == "Hunt In Zone" then
		local wanderTime = 100
		local alertTime = 25
		local eatTime = 80
		r2.Translator.createHuntActivityInZone(context, aiState.Name, groupsByName, Logic.activityInternalTimerId, wanderTime, alertTime, eatTime, aiScriptDebug)
	elseif activityStep.Activity == "Guard Zone" then
		r2.Translator.createSimpleActivityInZone(context, aiState.Name, groupsByName, "Alert", false, aiScriptDebug)
	end


	if Logic.isWanderActivity(activityStep.Activity) and activityStep.TimeLimit == "Few Sec" then
		local event = r2.Translator.createEvent("destination_reached_all", aiState.Name, groupsByName)
		table.insert(context.RtAct.Events, event)
		local number = tonumber(activityStep.TimeLimitValue)
		if number == nil then
			number = 0
		end
		local action =  r2.Translator.createAction("wander_destination_reached",  groupsByName, aiState.Name, activityStepIndex, number)
		table.insert(context.RtAct.Actions, action)
		table.insert(event.ActionsId, action.Id)
	end

	local stepAction = nil	
	if activityStep.Activity == "Stand Up" then
		stepAction = r2.Translator.createAction("stand_up", rtNpcGrp.Id )		
	elseif activityStep.Activity == "Sit Down" then
		stepAction = r2.Translator.createAction("sit_down", rtNpcGrp.Id )		
	end
	

	-- translate event create by activitySequence (eg patrol)
	if activityStep.EventsIds.Size ~= 0
	then
		Logic.translateEvents(context,  hlComponent, activitySequence, activityStep,  activityStepIndex, rtNpcGrp, statesByName)
	end


	
	--go to next activityStep
	local action = Logic.createTransition(context,  hlComponent, activitySequence, activityStep, activityStepIndex, stepAction)
	table.insert(Logic.SequenceAction.Children, action)


	-- if ActivityStep.TimeLimit == "Chat" creates a eventhandler to go to next state
	
	if activityStep.TimeLimit == "Chat" then
		local action
		local condition
		local event

		local number = Logic.findChatSequenceIdByInstanceId(hlComponent, chat.InstanceId)	
		if (number == nil or number == -1) then
			printWarning("Error in translation:  in component '" .. hlComponent.Name .. " the chat sequence selected as time limit of Activity Step " .. activityStepIndex .. " does not exist.")

		end
	
		

		event = r2.Translator.createEvent("user_event_8", aiState.Name,  rtNpcGrp.Id)
		table.insert(context.RtAct.Events, event)

	
		
		condition =  r2.Translator.createAction("condition_if", "v0  == "..tostring(number-1))
		table.insert(context.RtAct.Actions, condition)

		
		action =  Logic.selectActivityStep(activityStepIndex + 1)
		table.insert(condition.Children, action)
		table.insert(event.ActionsId, condition.Id)


	end

	assert(State ~= "")
	return aiState
end

---------------------------------------------------------------------------------------
--helpers functions--
---------------------------------------------------------------------------------------


Logic.getAiMovement = function(activity)
	assert(activity and type(activity) == "string")
	if (activity == "Follow Route") or (activity == "Patrol") or (activity == "Repeat Road") then
		return "follow_route"
	elseif activity == "Deploy" then
		return "stand_on_vertices"
	elseif activity == "Wander" or activity == "Go To Zone" or activity == "Rest In Zone" or activity == "Feed In Zone"
		or activity == "Hunt In Zone" or activity == "Guard Zone" then
		return "wander"
	elseif activity == "Stand Still" or activity == "Inactive" or activity == "Stand Up" or activity == "Sit Down" then
		return "idle"
	elseif activity == "Stand On Start Point" or activity == "Go To Start Point" then
		return "stand_on_start_point"
	end
	
	printWarning("Undef activity '"..activity.."'")
	assert(nil)
	return ""
end

Logic.getAction = function(action)
local action_type = action.Action

	if action_type == "Sit Down"
	then
		return "sit_down"
	end
	if action_type == "Stand Up"
	then
		return "stand_up"
	end
	if action_type == "Go To Step"
	then
		if action.Parameter == nil or action.Parameter == ""
		then
			return nil
		end
		return "go_to_step"
	end
	if action_type == "Set Activity"
	then
		if action.Parameter == nil or action.Parameter == ""
		then
			return nil
		end
		return "set_activity"
	end

	if action_type == "Stop Actions"
	then
		return "stop_actions"
	end

	if action_type == "Start Mission"
	then	
		return "start_mission"
	end

	return nil
end

Logic.getEvent = function(Event)
	local i = Logic.find(Logic.EventTypes, Event.Condition.SourceType)
	--Time event type
	if i == 0
	then
	end
	
	--place event type
	if i == 1
	then
	end

	--entity event type
	if i == 2
	then
		if Event.Condition.Type == "At Destination"
		then
			return "destination_reached_all"
		end
		
		if Event.Condition.Type == "Is Attacked"
		then
			return "group_under_attack"
		end

		if Event.Condition.Type == "Enters State"
		then
			return "start_of_state"
		end

		if Event.Condition.Type == "Leaves State"
		then
			return "end_of_state"
		end
		if Event.Condition.Type == "Activity Is"
		then				
		end
	end

	--State Machine event type
	if i == 3
	then
	end

	--Counter event type
	if i == 4
	then
		local counterName = Event.Parameter
		local param
		
		if counterName == nil
		then
			
		end

		if string.find(Event.Condition.Type, "v%d_changed") ~= nil
		then
			return "variable_" ..Event.Condition.Type
		end

		if Event.Condition.Type == "Equals"
		then
			param = " = "
		else
			if Event.Condition.Type == "Lesser"
			then
				param = " < "
			else
				if Event.Condition.Type == "Greater"
				then
					param = " > "
				else
					param = nil
				end
			end
		end
		if param == nil
		then
			return nil
		end
		param = param ..Event.Condition.ConditionParameter

		return "variable_changed", param
	end
end

Logic.checkEvent = function(eventMode, sourceType, sourceId, conditionType, parameter)
	local eventTypeInd = Logic.find(Logic.EventTypes, sourceType)
	local conditionTypeInd
	if eventTypeInd == nil
	then
		return false
	end

	--{"Time", "Place", "Entity or Group", "State Machine", "Counter"}

	conditionTypeInd = Logic.find(Logic.ConditionTypes[eventTypeInd], conditionType)
	if eventTypeInd == nil
	then
		return false
	end

	--Time event type
	if eventTypeInd == 0
	then
	end

	--Place event type
	if eventTypeInd == 1
	then
	end

	--Entity or Group event type
	if eventTypeInd == 2
	then
		if parameter == nil or parameter == ""
		then
			if conditionType == "At Destination" or conditionType == "Is Attacked"
			then
				return true
			else
				return false
			end
		else
			if conditionType == "Enters State" or conditionType == "Leaves State"
			then
				return true
			else
				return false
			end
		end
	end

	--State Machine event type
	if eventTypeInd == 3
	then
	end

	--Counter event type
	if eventTypeInd == 4
	then
		if string.find(conditionType, "v%d_changed") ~= nil
		then
			return true
		end
		if conditionType == "Lesser" or conditionType == "Greater" or conditionType == "Equals"
		then
			return true
		end
	end

	return false

end

Logic.fillCondition = function(condition, sourceType, sourceId, conditionType, conditionParameter)
	condition.SourceType = sourceType
	condition.SourceId = sourceId
	condition.Type = conditionType
	condition.ConditionParameter = conditionParameter
end

Logic.fillEvent = function(event, eventMode, sourceType, sourceId, conditionType, conditionParameter)
	if Logic.checkEvent(eventMode, sourceType, sourceId, conditionType, conditionParameter) == false
	then
		assert(0)
		return false
	end

	Logic.fillCondition(event.Condition, sourceType, sourceId, conditionType, conditionParameter)
	event.EventMode = eventMode
	return true
end
-----------------------------------------
--genere le parametre de l'action set_timer
Logic.getTimeLimit = function(step)
	local param = ""

	if step.TimeLimit == "Few Sec" then
		local limit = tonumber(step.TimeLimitValue)
		if limit == nil then
			limit = 0
		end
		limit = 1 + limit * 10
		param = tostring( limit )
		return param
	end
	return nil
end
	
Logic.getActivityInitialStateName = function(groupId, activityNb, context)
	local group = context.Components[groupId]
	local activity = group.Components[1].Behavior.Activities[activityNb]
	
	if activity == nil
	then
		return nil
	end

	local activityStates = Logic.ActivitiesStates[activity.InstanceId]
	return activityStates[activity.InstanceId][0].Name
end

--return the states used to represent this step activity
Logic.getActivityStepStates = function(activity, stepNb)
	local activityStates = Logic.ActivitiesStates[activity.InstanceId]
	local states = activityStates[activity.Components[stepNb].InstanceId]
	return states
end

Logic.find = function(array, elem)
	local i = 0
	for k, v in pairs(array)
	do
		if k ~="Keys"
		then
			if elem == v
			then
				return i
			end
			i = i + 1
		end
	end
	return nil
end



Logic.findChatSequenceIdByInstanceIdImpl = function(instance, instanceId)
	local behvior = instance:getBehavior()
	local k, sequence = next(behvior.ChatSequences, nil)
	local nbSequence = 0
	while k do
		nbSequence = nbSequence + 1
		if tostring(sequence.InstanceId) == instanceId then return nbSequence end
		 k, sequence = next(behvior.ChatSequences, k)
	end
	return -1
end

Logic.findChatSequenceIdByInstanceId = function(target, instanceId)
	local instance =target
	return Logic.findChatSequenceIdByInstanceIdImpl(instance, instanceId)

end

-- target -> hlNpc or hlNpcGrp
Logic.findChatStepIdByInstanceId = function(target, instanceId)
	assert( type(target) == "userdata")
	local nbSequence = 0
	local nbStep = 0

	local instance = target
	assert(instance ~= nil)
	local behavior = instance:getBehavior()
	assert(behavior ~= nil)
	local k2, sequence = next(behavior.ChatSequences, nil)
	while k2 do

		nbSequence = nbSequence + 1
		nbStep =  0
		local k, step = next(sequence.Components, nil)
		while k do
			nbStep = nbStep  +1
			if tostring(step.InstanceId) == instanceId then return {nbStep, nbSequence} end
			k, step = next(sequence.Components, k)
		end
		 k2, sequence = next(behavior.ChatSequences, k2)
	end
	assert(0)
	return {-1, -1}
end

-- target is a NpcGrp or a Npc or a CustomNpc
Logic.findActivitySequenceIdByInstanceId = function(target, instanceId)
	assert( type(target) == "userdata")
	
	local nbSequence = 0
	local behavior = nil

	behavior = target:getBehavior()

	if behavior == nil then
		debugInfo("Error: try to find activity Sequence on an unknown entity of type " .. target.Class )
		assert(nil)
	end

	local npc = behavior.ParentInstance
	local grp = behavior.ParentInstance.ParentInstance
	if not npc:isGrouped() then grp = nil end
	

	local k, sequence  = next(behavior.Activities, nil)
	while (k ~= nil) do

		nbSequence = nbSequence + 1
		if sequence.InstanceId == instanceId then
			if grp then return {nbSequence, grp} end
			if npc then return {nbSequence, npc} end
			assert(nil)
		end
		k, sequence  = next(behavior.Activities, k)
	end
	return {-1, nil}
end

Logic.findActivityStepIdByInstanceId = function(target, instanceId)
	assert( type(target) == "userdata")
	local nbSequence = 0
	local nbStep = 0
	local comp = target
	local behavior = comp:getBehavior()
	
	local k, sequence = next(behavior.Activities, nil)
	while k do

		if tostring(sequence.InstanceId) == instanceId then
			assert(nil) -- use findActivityStep instead of findactivitySequence
		end

		nbSequence = nbSequence + 1
		nbStep =  0
		local k2, step = next(sequence.Components, nil)
		while (k2) do
			nbStep = nbStep  +1
			if tostring(step.InstanceId) == instanceId then return {nbStep, nbSequence} end
			k2, step = next(sequence.Components, k2)
		end
		k, sequence = next(behavior.Activities, k)
	end
	return {-1, -1}
end





Logic.selectActivityStep = function(stepNb)
	local action 
	stepNb = stepNb - 1

	action  = r2.Translator.createAction("code", 
		Logic.activityStepVar .. " = " .. stepNb ..";\n"
		.. "()setTimer(1, " ..Logic.activityStepTimerId .. ");\n")
	action.Name = "select_activity_step " .. stepNb

	return action
end



----------------------------------------------------------------------------
-- add a line to the event menu

local component = Logic.Components.ChatSequence


----------------------------------------------------------------------------
-- add a line to the event menu
function component:getLogicTranslations()

	local logicTranslations = {
		["ApplicableActions"] = {
			["starts dialog"]			= { menu=i18n.get( "uiR2AA0ChatSeqStart"		):toUtf8(), 
											text=i18n.get( "uiR2AA1ChatSeqStart"		):toUtf8()},
			["stops dialog"]			= { menu=i18n.get( "uiR2AA0ChatSeqStop"			):toUtf8(), 
											text=i18n.get( "uiR2AA1ChatSeqStop"			):toUtf8()},
			["starts chat"]				= { menu=i18n.get( "uiR2AA0ChatStepStart"		):toUtf8(), 
											text=i18n.get( "uiR2AA1ChatStepStart"		):toUtf8()},
			["continues dialog"]				= { menu=i18n.get( "uiR2AA0ChatStepContinue"		):toUtf8(), 
											text=i18n.get( "uiR2AA1ChatStepContinue"		):toUtf8()},
		},
		["Events"] = {	
			["start of dialog"]			= { menu=i18n.get( "uiR2Event0ChatSeqStart"		):toUtf8(), 
											text=i18n.get( "uiR2Event1ChatSeqStart"		):toUtf8()},
			["end of dialog"]			= { menu=i18n.get( "uiR2Event0ChatSeqEnd"		):toUtf8(), 
											text=i18n.get( "uiR2Event1ChatSeqEnd"		):toUtf8()},
			["start of chat"]			= { menu=i18n.get( "uiR2Event0ChatStepStart"	):toUtf8(), 
											text=i18n.get( "uiR2Event1ChatStepStart"	):toUtf8()},
			["end of chat"]				= { menu=i18n.get( "uiR2Event0ChatStepEnd"		):toUtf8(), 
											text=i18n.get( "uiR2Event1ChatStepEnd"		):toUtf8()},
		},
		["Conditions"] = {	
			["is in dialog"]			= { menu=i18n.get( "uiR2Test0ChatSeq"			):toUtf8(), 
											text=i18n.get( "uiR2Test1ChatSeq"			):toUtf8()},
			["is not in dialog"]		= { menu=i18n.get( "uiR2Test0ChatNotSeq"		):toUtf8(), 
											text=i18n.get( "uiR2Test1ChatNotSeq"		):toUtf8()},
			["is in chat"]				= { menu=i18n.get( "uiR2Test0ChatStep"			):toUtf8(), 
											text=i18n.get( "uiR2Test1ChatStep"			):toUtf8()},
		}
	}
	return logicTranslations
end


r2.Features["ActivitySequence"] = Logic