431 lines
13 KiB
Lua
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)
|