2026-04-14 17:41:39 +02:00

431 lines
13 KiB
Lua

local Jobs = setmetatable({}, {__index = function(_, key)
return ESX.GetJobs()[key]
end
})
local RegisteredSocieties = {}
local SocietiesByName = {}
function GetSociety(name)
return SocietiesByName[name]
end
exports("GetSociety", GetSociety)
function registerSociety(name, label, account, datastore, inventory, data)
if SocietiesByName[name] then
print(('[^3WARNING^7] society already registered, name: ^5%s^7'):format(name))
return
end
local society = {
name = name,
label = label,
account = account,
datastore = datastore,
inventory = inventory,
data = data
}
SocietiesByName[name] = society
table.insert(RegisteredSocieties, society)
end
AddEventHandler('esx_society:registerSociety', registerSociety)
exports("registerSociety", registerSociety)
AddEventHandler('esx_society:getSocieties', function(cb)
cb(RegisteredSocieties)
end)
AddEventHandler('esx_society:getSociety', function(name, cb)
cb(GetSociety(name))
end)
RegisterServerEvent('esx_society:checkSocietyBalance')
AddEventHandler('esx_society:checkSocietyBalance', function(society)
local xPlayer = ESX.Player(source)
local society = GetSociety(society)
if xPlayer.getJob().name ~= society.name then
print(('esx_society: %s attempted to call checkSocietyBalance!'):format(xPlayer.getIdentifier()))
return
end
TriggerEvent('esx_addonaccount:getSharedAccount', society.account, function(account)
TriggerClientEvent("esx:showNotification", xPlayer.src, TranslateCap('check_balance', ESX.Math.GroupDigits(account.money)))
end)
end)
RegisterServerEvent('esx_society:withdrawMoney')
AddEventHandler('esx_society:withdrawMoney', function(societyName, amount)
local source = source
local society = GetSociety(societyName)
if not society then
print(('[^3WARNING^7] Player ^5%s^7 attempted to withdraw from non-existing society - ^5%s^7!'):format(source, societyName))
return
end
local xPlayer = ESX.Player(source)
amount = ESX.Math.Round(tonumber(amount))
if xPlayer.getJob().name ~= society.name then
return print(('[^3WARNING^7] Player ^5%s^7 attempted to withdraw from society - ^5%s^7!'):format(source, society.name))
end
TriggerEvent('esx_addonaccount:getSharedAccount', society.account, function(account)
if amount > 0 and account.money >= amount then
account.removeMoney(amount)
xPlayer.addMoney(amount, TranslateCap('money_add_reason'))
xPlayer.showNotification(TranslateCap('have_withdrawn', ESX.Math.GroupDigits(amount)))
else
xPlayer.showNotification(TranslateCap('invalid_amount'))
end
end)
end)
RegisterServerEvent('esx_society:depositMoney')
AddEventHandler('esx_society:depositMoney', function(societyName, amount)
local source = source
local xPlayer = ESX.Player(source)
local society = GetSociety(societyName)
if not society then
print(('[^3WARNING^7] Player ^5%s^7 attempted to deposit to non-existing society - ^5%s^7!'):format(source, societyName))
return
end
amount = ESX.Math.Round(tonumber(amount))
if xPlayer.getJob().name ~= society.name then
return print(('[^3WARNING^7] Player ^5%s^7 attempted to deposit to society - ^5%s^7!'):format(source, society.name))
end
if amount > 0 and xPlayer.getMoney() >= amount then
TriggerEvent('esx_addonaccount:getSharedAccount', society.account, function(account)
xPlayer.removeMoney(amount, TranslateCap('money_remove_reason'))
xPlayer.showNotification(TranslateCap('have_deposited', ESX.Math.GroupDigits(amount)))
account.addMoney(amount)
end)
else
xPlayer.showNotification(TranslateCap('invalid_amount'))
end
end)
RegisterServerEvent('esx_society:washMoney')
AddEventHandler('esx_society:washMoney', function(society, amount)
local source = source
local xPlayer = ESX.Player(source)
local account = xPlayer.getAccount('black_money')
amount = ESX.Math.Round(tonumber(amount))
if xPlayer.getJob().name ~= society then
return print(('[^3WARNING^7] Player ^5%s^7 attempted to wash money in society - ^5%s^7!'):format(source, society))
end
if amount and amount > 0 and account.money >= amount then
xPlayer.removeAccountMoney('black_money', amount, "Washing")
MySQL.insert('INSERT INTO society_moneywash (identifier, society, amount) VALUES (?, ?, ?)', {xPlayer.getIdentifier(), society, amount},
function(rowsChanged)
xPlayer.showNotification(TranslateCap('you_have', ESX.Math.GroupDigits(amount)))
end)
else
xPlayer.showNotification(TranslateCap('invalid_amount'))
end
end)
RegisterServerEvent('esx_society:putVehicleInGarage')
AddEventHandler('esx_society:putVehicleInGarage', function(societyName, vehicle)
local source = source
local society = GetSociety(societyName)
if not society then
print(('[^3WARNING^7] Player ^5%s^7 attempted to put vehicle in non-existing society garage - ^5%s^7!'):format(source, societyName))
return
end
TriggerEvent('esx_datastore:getSharedDataStore', society.datastore, function(store)
local garage = store.get('garage') or {}
table.insert(garage, vehicle)
store.set('garage', garage)
end)
end)
RegisterServerEvent('esx_society:removeVehicleFromGarage')
AddEventHandler('esx_society:removeVehicleFromGarage', function(societyName, vehicle)
local source = source
local society = GetSociety(societyName)
if not society then
print(('[^3WARNING^7] Player ^5%s^7 attempted to remove vehicle from non-existing society garage - ^5%s^7!'):format(source, societyName))
return
end
TriggerEvent('esx_datastore:getSharedDataStore', society.datastore, function(store)
local garage = store.get('garage') or {}
for i=1, #garage, 1 do
if garage[i].plate == vehicle.plate then
table.remove(garage, i)
break
end
end
store.set('garage', garage)
end)
end)
ESX.RegisterServerCallback('esx_society:getSocietyMoney', function(source, cb, societyName)
local society = GetSociety(societyName)
if not society then
print(('[^3WARNING^7] Player ^5%s^7 attempted to get money from non-existing society - ^5%s^7!'):format(source, societyName))
return cb(0)
end
TriggerEvent('esx_addonaccount:getSharedAccount', society.account, function(account)
cb(account.money or 0)
end)
end)
ESX.RegisterServerCallback('esx_society:getEmployees', function(source, cb, society)
local employees = {}
local xPlayers = ESX.ExtendedPlayers('job', society)
for i=1, #(xPlayers) do
local xPlayer = xPlayers[i]
local name = xPlayer.getName()
if Config.EnableESXIdentity and name == GetPlayerName(xPlayer.src) then
name = xPlayer.get('firstName') .. ' ' .. xPlayer.get('lastName')
end
local job = xPlayer.getJob()
table.insert(employees, {
name = name,
identifier = xPlayer.getIdentifier(),
job = {
name = society,
label = job.label,
grade = job.grade,
grade_name = job.grade_name,
grade_label = job.grade_label
}
})
end
local query = "SELECT identifier, job_grade FROM `users` WHERE `job`= ? ORDER BY job_grade DESC"
if Config.EnableESXIdentity then
query = "SELECT identifier, job_grade, firstname, lastname FROM `users` WHERE `job`= ? ORDER BY job_grade DESC"
end
MySQL.query(query, {society},
function(result)
for k, row in pairs(result) do
local alreadyInTable
local identifier = row.identifier
for k, v in pairs(employees) do
if v.identifier == identifier then
alreadyInTable = true
end
end
if not alreadyInTable then
local name = TranslateCap('name_not_found')
if Config.EnableESXIdentity then
name = row.firstname .. ' ' .. row.lastname
end
table.insert(employees, {
name = name,
identifier = identifier,
job = {
name = society,
label = Jobs[society].label,
grade = row.job_grade,
grade_name = Jobs[society].grades[tostring(row.job_grade)].name,
grade_label = Jobs[society].grades[tostring(row.job_grade)].label
}
})
end
end
cb(employees)
end)
end)
ESX.RegisterServerCallback('esx_society:getJob', function(source, cb, society)
if not Jobs[society] then
return cb(false)
end
local job = json.decode(json.encode(Jobs[society]))
local grades = {}
for k,v in pairs(job.grades) do
table.insert(grades, v)
end
table.sort(grades, function(a, b)
return a.grade < b.grade
end)
job.grades = grades
cb(job)
end)
ESX.RegisterServerCallback('esx_society:setJob', function(source, cb, identifier, job, grade, actionType)
local xPlayer = ESX.Player(source)
local isBoss = Config.BossGrades[xPlayer.getJob().grade_name]
local xTarget = ESX.Player(identifier)
if not isBoss then
print(('[^3WARNING^7] Player ^5%s^7 attempted to setJob for Player ^5%s^7!'):format(source, xTarget.src))
return cb()
end
if not xTarget then
MySQL.update('UPDATE users SET job = ?, job_grade = ? WHERE identifier = ?', {job, grade, identifier},
function()
cb()
end)
return
end
xTarget.setJob(job, grade)
local xTargetName, xTargetJob = xTarget.getName(), xTarget.getJob()
if actionType == 'hire' then
xTarget.showNotification(TranslateCap('you_have_been_hired', job))
xPlayer.showNotification(TranslateCap("you_have_hired", xTargetName))
elseif actionType == 'promote' then
xTarget.showNotification(TranslateCap('you_have_been_promoted'))
xPlayer.showNotification(TranslateCap("you_have_promoted", xTargetName, xTargetJob.grade_label))
elseif actionType == 'fire' then
xTarget.showNotification(TranslateCap('you_have_been_fired', xTargetJob.label))
xPlayer.showNotification(TranslateCap("you_have_fired", xTargetName))
end
cb()
end)
ESX.RegisterServerCallback('esx_society:setJobSalary', function(source, cb, job, grade, salary)
local xPlayer = ESX.Player(source)
local xPlayerJob = xPlayer.getJob()
if xPlayerJob.name == job and Config.BossGrades[xPlayerJob.grade_name] then
if salary <= Config.MaxSalary then
MySQL.update('UPDATE job_grades SET salary = ? WHERE job_name = ? AND grade = ?', {salary, job, grade},
function(rowsChanged)
Jobs[job].grades[tostring(grade)].salary = salary
ESX.RefreshJobs()
Wait(1)
local xPlayers = ESX.ExtendedPlayers('job', job)
for _, xTarget in pairs(xPlayers) do
if xTarget.getJob().grade == grade then
xTarget.setJob(job, grade)
end
end
cb()
end)
else
print(('[^3WARNING^7] Player ^5%s^7 attempted to setJobSalary over the config limit for ^5%s^7!'):format(source, job))
cb()
end
else
print(('[^3WARNING^7] Player ^5%s^7 attempted to setJobSalary for ^5%s^7!'):format(source, job))
cb()
end
end)
ESX.RegisterServerCallback('esx_society:setJobLabel', function(source, cb, job, grade, label)
local xPlayer = ESX.Player(source)
local xPlayerJob = xPlayer.getJob()
if xPlayerJob.name == job and Config.BossGrades[xPlayerJob.grade_name] then
MySQL.update('UPDATE job_grades SET label = ? WHERE job_name = ? AND grade = ?', {label, job, grade},
function(rowsChanged)
Jobs[job].grades[tostring(grade)].label = label
ESX.RefreshJobs()
Wait(1)
local xPlayers = ESX.ExtendedPlayers('job', job)
for _, xTarget in pairs(xPlayers) do
if xTarget.getJob().grade == grade then
xTarget.setJob(job, grade)
end
end
cb()
end)
else
print(('[^3WARNING^7] Player ^5%s^7 attempted to setJobLabel for ^5%s^7!'):format(source, job))
cb()
end
end)
local getOnlinePlayers, onlinePlayers = false, nil
ESX.RegisterServerCallback('esx_society:getOnlinePlayers', function(source, cb)
if getOnlinePlayers == false and onlinePlayers == nil then -- Prevent multiple xPlayer loops from running in quick succession
getOnlinePlayers, onlinePlayers = true, {}
local xPlayers = ESX.ExtendedPlayers() -- Returns all xPlayers
for _, xPlayer in pairs(xPlayers) do
table.insert(onlinePlayers, {
source = xPlayer.src,
identifier = xPlayer.getIdentifier(),
name = xPlayer.getName(),
job = xPlayer.getJob()
})
end
cb(onlinePlayers)
getOnlinePlayers = false
Wait(1000) -- For the next second any extra requests will receive the cached list
onlinePlayers = nil
return
end
while getOnlinePlayers do Wait(0) end -- Wait for the xPlayer loop to finish
cb(onlinePlayers)
end)
ESX.RegisterServerCallback('esx_society:getVehiclesInGarage', function(source, cb, societyName)
local society = GetSociety(societyName)
if not society then
print(('[^3WARNING^7] Attempting To get a non-existing society - %s!'):format(societyName))
return
end
TriggerEvent('esx_datastore:getSharedDataStore', society.datastore, function(store)
local garage = store.get('garage') or {}
cb(garage)
end)
end)
ESX.RegisterServerCallback('esx_society:isBoss', function(source, cb, job)
cb(isPlayerBoss(source, job))
end)
function isPlayerBoss(playerId, job)
local xPlayer = ESX.Player(playerId)
local xPlayerJob = xPlayer.getJob()
if xPlayerJob.name == job and Config.BossGrades[xPlayerJob.grade_name] then
return true
else
print(('esx_society: %s attempted open a society boss menu!'):format(xPlayer.getIdentifier()))
return false
end
end
function WashMoneyCRON(d, h, m)
MySQL.query('SELECT * FROM society_moneywash', function(result)
for i=1, #result, 1 do
local society = GetSociety(result[i].society)
local xPlayer = ESX.Player(result[i].identifier)
-- add society money
TriggerEvent('esx_addonaccount:getSharedAccount', society.account, function(account)
account.addMoney(result[i].amount)
end)
-- send notification if player is online
if xPlayer then
xPlayer.showNotification(TranslateCap('you_have_laundered', ESX.Math.GroupDigits(result[i].amount)))
end
end
MySQL.update('DELETE FROM society_moneywash')
end)
end
TriggerEvent('cron:runAt', 3, 0, WashMoneyCRON)