Scripts/resources/[standalone]/dpemotes/client/Emote.lua
2024-12-29 21:02:20 +01:00

667 lines
23 KiB
Lua

-- You probably shouldnt touch these.
local AnimationDuration = -1
local ChosenAnimation = ""
local ChosenDict = ""
local ChosenAnimOptions = false
local MovementType = 0
local PlayerGender = "male"
local PlayerHasProp = false
local PlayerProps = {}
local PlayerParticles = {}
local SecondPropEmote = false
local lang = Config.MenuLanguage
local PtfxNotif = false
local PtfxPrompt = false
local PtfxWait = 500
local PtfxCanHold = false
local PtfxNoProp = false
local AnimationThreadStatus = false
local CanCancel = true
local InExitEmote = false
IsInAnimation = false
-- Remove emotes if needed
local emoteTypes = {
"Shared",
"Dances",
"AnimalEmotes",
"Emotes",
"PropEmotes",
}
for i = 1, #emoteTypes do
local emoteType = emoteTypes[i]
for emoteName, emoteData in pairs(RP[emoteType]) do
local shouldRemove = false
if Config.AdultEmotesDisabled and emoteData.AdultAnimation then shouldRemove = true end
if not Config.AnimalEmotesEnabled and emoteData.AnimalEmote then shouldRemove = true RP.AnimalEmotes = {} end
if emoteData[1] and not ((emoteData[1] == 'Scenario') or (emoteData[1] == 'ScenarioObject') or (emoteData[1] == 'MaleScenario')) and not DoesAnimDictExist(emoteData[1]) then shouldRemove = true end
if shouldRemove then RP[emoteType][emoteName] = nil end
end
end
local function IsPlayerAiming(player)
return IsPlayerFreeAiming(player) or IsAimCamActive() or IsAimCamThirdPersonActive()
end
local function RunAnimationThread()
local playerId = PlayerId()
if AnimationThreadStatus then return end
AnimationThreadStatus = true
CreateThread(function()
local sleep
while AnimationThreadStatus and (IsInAnimation or PtfxPrompt) do
sleep = 500
if IsInAnimation then
sleep = 0
if IsPlayerAiming(playerId) then
EmoteCancel()
end
end
if PtfxPrompt then
sleep = 0
if not PtfxNotif then
SimpleNotify(PtfxInfo)
PtfxNotif = true
end
if IsControlPressed(0, 47) then
PtfxStart()
Wait(PtfxWait)
if PtfxCanHold then
while IsControlPressed(0, 47) and IsInAnimation and AnimationThreadStatus do
Wait(5)
end
end
PtfxStop()
end
end
Wait(sleep)
end
end)
end
if Config.EnableXtoCancel then
RegisterKeyMapping("emotecancel", "Cancel current emote", "keyboard", Config.CancelEmoteKey)
end
if Config.HandsupKeybindEnabled then
RegisterKeyMapping("handsup", "Put your arms up", "keyboard", Config.HandsupKeybind)
end
-----------------------------------------------------------------------------------------------------
-- Commands / Events --------------------------------------------------------------------------------
-----------------------------------------------------------------------------------------------------
Citizen.CreateThread(function()
TriggerEvent('chat:addSuggestion', '/e', 'Play an emote',
{ { name = "emotename", help = "dance, camera, sit or any valid emote." },
{ name = "texturevariation", help = "(Optional) 1, 2, 3 or any number. Will change the texture of some props used in emotes, for example the color of a phone. Enter -1 to see a list of variations." } })
TriggerEvent('chat:addSuggestion', '/em', 'Open dpemotes menu . This may differ from server to server.')
end)
RegisterCommand('e', function(source, args, raw) EmoteCommandStart(source, args, raw) end, false)
RegisterCommand('emote', function(source, args, raw) EmoteCommandStart(source, args, raw) end, false)
if Config.SqlKeybinding then
RegisterCommand('emotebind', function(source, args, raw) EmoteBindStart(source, args, raw) end, false)
RegisterCommand('emotebinds', function(source, args, raw) EmoteBindsStart(source, args, raw) end, false)
end
if Config.MenuKeybindEnabled then
RegisterCommand('emoteui', function() OpenEmoteMenu() end, false)
RegisterKeyMapping("emoteui", "Open dpemotes menu", "keyboard", Config.MenuKeybind)
else
RegisterCommand('emotemenu', function() OpenEmoteMenu() end, false)
end
RegisterCommand('emotes', function() EmotesOnCommand() end, false)
RegisterCommand('walk', function(source, args, raw) WalkCommandStart(source, args, raw) end, false)
RegisterCommand('walks', function() WalksOnCommand() end, false)
RegisterCommand('emotecancel', function() EmoteCancel() end, false)
RegisterCommand('handsup', function()
if Config.HandsupKeybindEnabled then
if IsEntityPlayingAnim(PlayerPedId(), "missminuteman_1ig_2", "handsup_base", 51) then
EmoteCancel()
else
EmoteCommandStart(nil, {"handsup"}, nil)
end
end
end, false)
AddEventHandler('onResourceStop', function(resource)
if resource == GetCurrentResourceName() then
local ply = PlayerPedId()
DestroyAllProps()
ClearPedTasksImmediately(ply)
DetachEntity(ply, true, false)
ResetPedMovementClipset(ply, 0.8)
AnimationThreadStatus = false
end
end)
-----------------------------------------------------------------------------------------------------
------ Functions and stuff --------------------------------------------------------------------------
-----------------------------------------------------------------------------------------------------
function EmoteCancel(force)
-- Don't cancel if we are in an exit emote
if InExitEmote then
return
end
local ply = PlayerPedId()
if not CanCancel and force ~= true then return end
if ChosenDict == "MaleScenario" and IsInAnimation then
ClearPedTasksImmediately(ply)
IsInAnimation = false
DebugPrint("Forced scenario exit")
elseif ChosenDict == "Scenario" and IsInAnimation then
ClearPedTasksImmediately(ply)
IsInAnimation = false
DebugPrint("Forced scenario exit")
end
PtfxNotif = false
PtfxPrompt = false
Pointing = false
if IsInAnimation then
if LocalPlayer.state.ptfx then
PtfxStop()
end
DetachEntity(ply, true, false)
CancelSharedEmote(ply)
if ChosenAnimOptions and ChosenAnimOptions.ExitEmote then
-- If the emote exit type is not spesifed it defaults to Emotes
local options = ChosenAnimOptions
local ExitEmoteType = options.ExitEmoteType or "Emotes"
-- Checks that the exit emote actually exists
if not RP[ExitEmoteType] or not RP[ExitEmoteType][options.ExitEmote] then
DebugPrint("Exit emote was invalid")
ClearPedTasks(ply)
IsInAnimation = false
return
end
OnEmotePlay(RP[ExitEmoteType][options.ExitEmote])
DebugPrint("Playing exit animation")
-- Check that the exit emote has a duration, and if so, set InExitEmote variable
local animationOptions = RP[ExitEmoteType][options.ExitEmote].AnimationOptions
if animationOptions and animationOptions.EmoteDuration then
InExitEmote = true
SetTimeout(animationOptions.EmoteDuration, function()
InExitEmote = false
DestroyAllProps()
ClearPedTasks(ply)
end)
return
end
else
ClearPedTasks(ply)
IsInAnimation = false
end
DestroyAllProps()
end
AnimationThreadStatus = false
end
function EmoteChatMessage(msg, multiline)
if msg then
TriggerEvent("chat:addMessage", { multiline = multiline == true or false, color = { 255, 255, 255 }, args = { "^5Help^0", tostring(msg) } })
end
end
function DebugPrint(args)
if Config.DebugDisplay then
print(args)
end
end
--#region ptfx
function PtfxThis(asset)
while not HasNamedPtfxAssetLoaded(asset) do
RequestNamedPtfxAsset(asset)
Wait(10)
end
UseParticleFxAsset(asset)
end
function PtfxStart()
LocalPlayer.state:set('ptfx', true, true)
end
function PtfxStop()
LocalPlayer.state:set('ptfx', false, true)
end
AddStateBagChangeHandler('ptfx', nil, function(bagName, key, value, _unused, replicated)
local plyId = tonumber(bagName:gsub('player:', ''), 10)
-- We stop here if we don't need to go further
-- We don't need to start or stop the ptfx twice
if (PlayerParticles[plyId] and value) or (not PlayerParticles[plyId] and not value) then return end
-- Only allow ptfx change on players
local ply = GetPlayerFromServerId(plyId)
if ply == 0 then return end
local plyPed = GetPlayerPed(ply)
if not DoesEntityExist(plyPed) then return end
local stateBag = Player(plyId).state
if value then
-- Start ptfx
local asset = stateBag.ptfxAsset
local name = stateBag.ptfxName
local offset = stateBag.ptfxOffset
local rot = stateBag.ptfxRot
local boneIndex = stateBag.ptfxBone and GetPedBoneIndex(plyPed, stateBag.ptfxBone) or GetEntityBoneIndexByName(name, "VFX")
local scale = stateBag.ptfxScale or 1
local color = stateBag.ptfxColor
local propNet = stateBag.ptfxPropNet
local entityTarget = plyPed
-- Only do for valid obj
if propNet then
local propObj = NetToObj(propNet)
if DoesEntityExist(propObj) then
entityTarget = propObj
end
end
PtfxThis(asset)
PlayerParticles[plyId] = StartNetworkedParticleFxLoopedOnEntityBone(name, entityTarget, offset.x, offset.y, offset.z, rot.x, rot.y, rot.z, boneIndex, scale + 0.0, 0, 0, 0, 1065353216, 1065353216, 1065353216, 0)
if color then
if color[1] and type(color[1]) == 'table' then
local randomIndex = math.random(1, #color)
color = color[randomIndex]
end
SetParticleFxLoopedAlpha(PlayerParticles[plyId], color.A)
SetParticleFxLoopedColour(PlayerParticles[plyId], color.R / 255, color.G / 255, color.B / 255, false)
end
DebugPrint("Started PTFX: " .. PlayerParticles[plyId])
else
-- Stop ptfx
DebugPrint("Stopped PTFX: " .. PlayerParticles[plyId])
StopParticleFxLooped(PlayerParticles[plyId], false)
RemoveNamedPtfxAsset(stateBag.ptfxAsset)
PlayerParticles[plyId] = nil
end
end)
--#endregion ptfx
function EmotesOnCommand(source, args, raw)
local EmotesCommand = ""
for a in pairsByKeys(RP.Emotes) do
EmotesCommand = EmotesCommand .. "" .. a .. ", "
end
EmoteChatMessage(EmotesCommand)
EmoteChatMessage(Config.Languages[lang]['emotemenucmd'])
end
function pairsByKeys(t, f)
local a = {}
for n in pairs(t) do
table.insert(a, n)
end
table.sort(a, f)
local i = 0 -- iterator variable
local iter = function() -- iterator function
i = i + 1
if a[i] == nil then
return nil
else
return a[i], t[a[i]]
end
end
return iter
end
function EmoteMenuStart(args, hard, textureVariation)
local name = args
local etype = hard
if etype == "dances" then
if RP.Dances[name] ~= nil then
OnEmotePlay(RP.Dances[name])
end
elseif etype == "animals" then
if RP.AnimalEmotes[name] ~= nil then
OnEmotePlay(RP.AnimalEmotes[name])
end
elseif etype == "props" then
if RP.PropEmotes[name] ~= nil then
OnEmotePlay(RP.PropEmotes[name], textureVariation)
end
elseif etype == "emotes" then
if RP.Emotes[name] ~= nil then
OnEmotePlay(RP.Emotes[name])
end
elseif etype == "expression" then
if RP.Expressions[name] ~= nil then
SetPlayerPedExpression(RP.Expressions[name][1], true)
end
end
end
function EmoteCommandStart(source, args, raw)
if #args > 0 then
local name = string.lower(args[1])
if name == "c" then
if IsInAnimation then
EmoteCancel()
else
EmoteChatMessage(Config.Languages[lang]['nocancel'])
end
return
elseif name == "help" then
EmotesOnCommand()
return
end
if RP.Emotes[name] ~= nil then
OnEmotePlay(RP.Emotes[name])
return
elseif RP.Dances[name] ~= nil then
OnEmotePlay(RP.Dances[name])
return
elseif RP.AnimalEmotes[name] ~= nil then
OnEmotePlay(RP.AnimalEmotes[name])
return
elseif RP.Exits[name] ~= nil then
OnEmotePlay(RP.Exits[name])
return
elseif RP.PropEmotes[name] ~= nil then
if RP.PropEmotes[name].AnimationOptions.PropTextureVariations then
if #args > 1 then
local textureVariation = tonumber(args[2])
if (RP.PropEmotes[name].AnimationOptions.PropTextureVariations[textureVariation] ~= nil) then
OnEmotePlay(RP.PropEmotes[name], textureVariation - 1)
return
else
local str = ""
for k, v in ipairs(RP.PropEmotes[name].AnimationOptions.PropTextureVariations) do
str = str .. string.format("\n(%s) - %s", k, v.Name)
end
EmoteChatMessage(string.format(Config.Languages[lang]['invalidvariation'], str), true)
OnEmotePlay(RP.PropEmotes[name], 0)
return
end
end
end
OnEmotePlay(RP.PropEmotes[name])
return
else
EmoteChatMessage("'" .. name .. "' " .. Config.Languages[lang]['notvalidemote'] .. "")
end
end
end
function LoadAnim(dict)
if not DoesAnimDictExist(dict) then
return false
end
while not HasAnimDictLoaded(dict) do
RequestAnimDict(dict)
Wait(10)
end
return true
end
function LoadPropDict(model)
while not HasModelLoaded(joaat(model)) do
RequestModel(joaat(model))
Wait(10)
end
end
function DestroyAllProps()
for _, v in pairs(PlayerProps) do
DeleteEntity(v)
end
PlayerHasProp = false
DebugPrint("Destroyed Props")
end
function AddPropToPlayer(prop1, bone, off1, off2, off3, rot1, rot2, rot3, textureVariation)
local Player = PlayerPedId()
local x, y, z = table.unpack(GetEntityCoords(Player))
if not IsModelValid(prop1) then
DebugPrint(tostring(prop1).." is not a valid model!")
return false
end
if not HasModelLoaded(prop1) then
LoadPropDict(prop1)
end
prop = CreateObject(joaat(prop1), x, y, z + 0.2, true, true, true)
if textureVariation ~= nil then
SetObjectTextureVariation(prop, textureVariation)
end
AttachEntityToEntity(prop, Player, GetPedBoneIndex(Player, bone), off1, off2, off3, rot1, rot2, rot3, true, true,
false, true, 1, true)
table.insert(PlayerProps, prop)
PlayerHasProp = true
SetModelAsNoLongerNeeded(prop1)
return true
end
-----------------------------------------------------------------------------------------------------
-- V -- This could be a whole lot better, i tried messing around with "IsPedMale(ped)"
-- V -- But i never really figured it out, if anyone has a better way of gender checking let me know.
-- V -- Since this way doesnt work for ped models.
-- V -- in most cases its better to replace the scenario with an animation bundled with prop instead.
-----------------------------------------------------------------------------------------------------
function CheckGender()
local playerPed = PlayerPedId()
if GetEntityModel(playerPed) == joaat("mp_f_freemode_01") then
PlayerGender = "female"
else
PlayerGender = "male"
end
DebugPrint("Set gender as = (" .. PlayerGender .. ")")
end
-----------------------------------------------------------------------------------------------------
------ This is the major function for playing emotes! -----------------------------------------------
-----------------------------------------------------------------------------------------------------
function OnEmotePlay(EmoteName, textureVariation)
InVehicle = IsPedInAnyVehicle(PlayerPedId(), true)
Pointing = false
if not Config.AllowedInCars and InVehicle == 1 then
return
end
if not DoesEntityExist(PlayerPedId()) then
return false
end
-- Don't play a new animation if we are in an exit emote
if InExitEmote then
return false
end
local animOption = EmoteName.AnimationOptions
if animOption and animOption.NotInVehicle and InVehicle then
return EmoteChatMessage("You can't play this animation while in vehicle.")
end
if ChosenAnimOptions and ChosenAnimOptions.ExitEmote then
if RP.Exits[ChosenAnimOptions.ExitEmote][2] ~= EmoteName[2] then
return
end
end
if IsProne then
EmoteChatMessage("You can't play animations while crawling.")
return false
end
ChosenDict, ChosenAnimation, ename = table.unpack(EmoteName)
ChosenAnimOptions = animOption
AnimationDuration = -1
if Config.DisarmPlayer then
if IsPedArmed(PlayerPedId(), 7) then
SetCurrentPedWeapon(PlayerPedId(), joaat('WEAPON_UNARMED'), true)
end
end
if animOption and animOption.Prop and PlayerHasProp then
DestroyAllProps()
end
if ChosenDict == "MaleScenario" or "Scenario" then
CheckGender()
if ChosenDict == "MaleScenario" then if InVehicle then return end
if PlayerGender == "male" then
ClearPedTasks(PlayerPedId())
TaskStartScenarioInPlace(PlayerPedId(), ChosenAnimation, 0, true)
DebugPrint("Playing scenario = (" .. ChosenAnimation .. ")")
IsInAnimation = true
RunAnimationThread()
else
EmoteChatMessage(Config.Languages[lang]['maleonly'])
end
return
elseif ChosenDict == "ScenarioObject" then if InVehicle then return end
BehindPlayer = GetOffsetFromEntityInWorldCoords(PlayerPedId(), 0.0, 0 - 0.5, -0.5);
ClearPedTasks(PlayerPedId())
TaskStartScenarioAtPosition(PlayerPedId(), ChosenAnimation, BehindPlayer['x'], BehindPlayer['y'], BehindPlayer['z'], GetEntityHeading(PlayerPedId()), 0, true, false)
DebugPrint("Playing scenario = (" .. ChosenAnimation .. ")")
IsInAnimation = true
RunAnimationThread()
return
elseif ChosenDict == "Scenario" then if InVehicle then return end
ClearPedTasks(PlayerPedId())
TaskStartScenarioInPlace(PlayerPedId(), ChosenAnimation, 0, true)
DebugPrint("Playing scenario = (" .. ChosenAnimation .. ")")
IsInAnimation = true
RunAnimationThread()
return
end
end
-- Small delay at the start
if animOption and animOption.StartDelay then
Wait(animOption.StartDelay)
end
if not LoadAnim(ChosenDict) then
EmoteChatMessage("'" .. ename .. "' " .. Config.Languages[lang]['notvalidemote'] .. "")
return
end
MovementType = 0 -- Default movement type
if InVehicle == 1 then
MovementType = 51
elseif animOption then
if animOption.EmoteMoving then
MovementType = 51
elseif animOption.EmoteLoop then
MovementType = 1
elseif animOption.EmoteStuck then
MovementType = 50
end
end
if animOption then
if animOption.EmoteDuration == nil then
animOption.EmoteDuration = -1
AttachWait = 0
else
AnimationDuration = animOption.EmoteDuration
AttachWait = animOption.EmoteDuration
end
if animOption.PtfxAsset then
PtfxAsset = animOption.PtfxAsset
PtfxName = animOption.PtfxName
if animOption.PtfxNoProp then
PtfxNoProp = animOption.PtfxNoProp
else
PtfxNoProp = false
end
Ptfx1, Ptfx2, Ptfx3, Ptfx4, Ptfx5, Ptfx6, PtfxScale = table.unpack(animOption.PtfxPlacement)
PtfxBone = animOption.PtfxBone
PtfxColor = animOption.PtfxColor
PtfxInfo = animOption.PtfxInfo
PtfxWait = animOption.PtfxWait
PtfxCanHold = animOption.PtfxCanHold
PtfxNotif = false
PtfxPrompt = true
-- RunAnimationThread() -- ? This call should not be required, see if needed with tests
TriggerServerEvent("dpemotes:ptfx:sync", PtfxAsset, PtfxName, vector3(Ptfx1, Ptfx2, Ptfx3), vector3(Ptfx4, Ptfx5, Ptfx6), PtfxBone, PtfxScale, PtfxColor)
else
DebugPrint("Ptfx = none")
PtfxPrompt = false
end
end
TaskPlayAnim(PlayerPedId(), ChosenDict, ChosenAnimation, 5.0, 5.0, AnimationDuration, MovementType, 0, false, false, false)
RemoveAnimDict(ChosenDict)
IsInAnimation = true
RunAnimationThread()
MostRecentDict = ChosenDict
MostRecentAnimation = ChosenAnimation
if animOption and animOption.Prop then
PropName = animOption.Prop
PropBone = animOption.PropBone
PropPl1, PropPl2, PropPl3, PropPl4, PropPl5, PropPl6 = table.unpack(animOption.PropPlacement)
if animOption.SecondProp then
SecondPropName = animOption.SecondProp
SecondPropBone = animOption.SecondPropBone
SecondPropPl1, SecondPropPl2, SecondPropPl3, SecondPropPl4, SecondPropPl5, SecondPropPl6 = table.unpack(animOption.SecondPropPlacement)
SecondPropEmote = true
else
SecondPropEmote = false
end
Wait(AttachWait)
if not AddPropToPlayer(PropName, PropBone, PropPl1, PropPl2, PropPl3, PropPl4, PropPl5, PropPl6, textureVariation) then return end
if SecondPropEmote then
if not AddPropToPlayer(SecondPropName, SecondPropBone, SecondPropPl1, SecondPropPl2, SecondPropPl3, SecondPropPl4, SecondPropPl5, SecondPropPl6, textureVariation) then
DestroyAllProps()
return
end
end
-- Ptfx is on the prop, then we need to sync it
if animOption.PtfxAsset and not PtfxNoProp then
TriggerServerEvent("dpemotes:ptfx:syncProp", ObjToNet(prop))
end
end
end
-----------------------------------------------------------------------------------------------------
------ Some exports to make the script more standalone! (by Clem76) ---------------------------------
-----------------------------------------------------------------------------------------------------
exports("EmoteCommandStart", function(emoteName, textureVariation)
EmoteCommandStart(nil, {emoteName, textureVariation}, nil)
end)
exports("EmoteCancel", EmoteCancel)
exports("CanCancelEmote", function(State)
CanCancel = State == true
end)
exports('IsPlayerInAnim', function()
return IsInAnimation
end)