Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(modules): Scaleform class #584

Open
wants to merge 10 commits into
base: main
Choose a base branch
from
11 changes: 11 additions & 0 deletions modules/lib.lua
Original file line number Diff line number Diff line change
@@ -1,5 +1,8 @@
local isServer = IsDuplicityVersion()

---@class Scaleform
local scaleform = require 'modules.scaleform'

local qbx = {}
qbx.string = {}
qbx.math = {}
Expand Down Expand Up @@ -560,6 +563,14 @@ else

ReleaseSoundId(soundId)
end

-- Create a new scaleform instance
-- Uses ox lib class system to create an easy to use scaleform handler
---@param scaleformDetails string | detailsTable
---@return Scaleform
function qbx.newScaleform(scaleformDetails)
return scaleform:new(scaleformDetails)
end
end

_ENV.qbx = qbx
235 changes: 235 additions & 0 deletions modules/scaleform.lua
Original file line number Diff line number Diff line change
@@ -0,0 +1,235 @@
---@class renderTargetTable
---@field name string
---@field model string|number

---@class detailsTable
---@field name string
---@field fullScreen? boolean
---@field x? number
---@field y? number
---@field width? number
---@field height? number
---@field renderTarget? renderTargetTable

---@class Scaleform : OxClass
---@field scaleform number
---@field draw boolean
---@field target number
---@field targetName string
---@field handle number
---@field fullScreen boolean
local Scaleform = lib.class('Scaleform')

---@param details detailsTable | string
---@return nil
---@description Create a new scaleform class
function Scaleform:constructor(details)
details = type(details) == "table" and details or {name = details} -- Set the details to a table if it is not already

local scaleform = lib.requestScaleformMovie(details.name) -- Request the scaleform movie

if not scaleform then -- If the scaleform is nil
return error(('Failed to request scaleform movie - [%s]'):format(details.name)) -- Error the failed scaleform request
end

self.handle = scaleform -- Set the scaleform handle
self.draw = false -- Set the draw to false

-- Set Default Values if not provided
self.fullScreen = details.fullScreen ~= nil and details.fullScreen or true
self.x = details.x or 0
self.y = details.y or 0
self.width = details.width or 0
self.height = details.height or 0

if details.renderTarget then
self:setRenderTarget(details.renderTarget.name, details.renderTarget.model)
end
end

local function convertArgs(argsTable)
for i=1, #argsTable do -- loop through the args
local arg = argsTable[i] -- Set the value to the current arg
if type(arg) == 'string' then -- If the type of v is a string
ScaleformMovieMethodAddParamPlayerNameString(arg) -- Add the player name string
elseif type(arg) == 'number' then -- If the type of v is a number
if math.type(arg) == 'integer' then -- If the math type of v is an integer
ScaleformMovieMethodAddParamInt(arg) -- Add the integer
else -- If the math type of v is not an integer
ScaleformMovieMethodAddParamFloat(arg) -- Add the float
end
elseif type(arg) == 'boolean' then -- If the type of v is a boolean
ScaleformMovieMethodAddParamBool(arg) -- Add the boolean
else
error(('Unsupported Parameter type [%s]'):format(type(arg))) -- Error unsupported type
end
end
end

---@param type 'boolean' | 'integer' | 'string'
---@return boolean | integer | string
---@description Awaits the return value, and converts it to a usable data type
local function retrieveReturnValue(type)
local result = EndScaleformMovieMethodReturnValue() -- End the scaleform movie method with a return value

lib.waitFor(function()
if IsScaleformMovieMethodReturnValueReady(result) then
return true
end
end, "Failed to retrieve return value", 1000)

if type == "integer" then -- If the type is an integer
return GetScaleformMovieMethodReturnValueInt(result) -- Get the return value as an integer
elseif type == "boolean" then
return GetScaleformMovieMethodReturnValueBool(result) -- Get the return value as a boolean
else -- If the type is not an integer
return GetScaleformMovieMethodReturnValueString(result) -- Get the return value as a string
end
end

---@param name string
---@param args? table
---@param returnValue? string
---@return any
---@description Call a scaleform function, with optional args or return value.
function Scaleform:callMethod(name, args, returnValue)
if not self.handle then -- If the scaleform handle is nil
return error('Scaleform handle is nil') -- Error the scaleform handle is nil
end

if args and type(args) ~= 'table' then -- If the type of args is not a table
return error('Args must be a table') -- Error args must be a table
end

BeginScaleformMovieMethod(self.handle, name) -- Begin the scaleform movie method

-- Converts the arguments into data types usable by scaleform
if args then
convertArgs(args)
end

-- When wanting a return value, the scaleform has to be ended with a different native
if returnValue then
return retrieveReturnValue(returnValue)
end

EndScaleformMovieMethod() -- End the scaleform movie method
end

---@param isFullscreen boolean
---@return nil
---@description Set the scaleform to render in full screen
function Scaleform:setFullScreen(isFullscreen)
self.fullScreen = isFullscreen
end

---@param x number
---@param y number
---@param width number
---@param height number
---@return nil
---@description Set the properties of the scaleform (Requires SetFullScreen to be false)
function Scaleform:setProperties(x, y, width, height)
if self.fullScreen then -- If the full screen is true
return error('Cannot set properties when full screen is enabled') -- Error cannot set properties when full screen is enabled
end
self.x = x
self.y = y
self.width = width
self.height = height
end

---@param name string
---@param model string|number
---@return nil
---@description Create a render target for the scaleform - optional , only if you want to render the scaleform in 3D
function Scaleform:setRenderTarget(name, model)

-- ensures theres no Targets still active, since this could cause a memory leak
-- if the render targets are not released.
if self.target then
ReleaseNamedRendertarget(self.targetName)
end

if type(model) == 'string' then -- If the type of model is a string
model = joaat(model) -- Convert the model to a hash
end

if not IsNamedRendertargetRegistered(name) then -- If the named render target is not registered
RegisterNamedRendertarget(name, false) -- Register the named render target

if not IsNamedRendertargetLinked(model) then -- If the named render target is not linked
LinkNamedRendertarget(model) -- Link the named render target
end

self.target = GetNamedRendertargetRenderId(name) -- Get the named render target render id
self.targetName = name -- Set the target name
end
end

---@return nil
---@description Set The Scaleform to draw
function Scaleform:startDrawing()
if self.draw then -- If the draw is equal to should draw
return error("Scaleform Already Drawing")
end

self.draw = true
CreateThread(function() -- Create a thread
while self.draw do -- While the draw is true

if self.target then -- If the render target is true
SetTextRenderId(self.target) -- Set the text render id
SetScriptGfxDrawOrder(4) -- Set the script gfx draw order
SetScriptGfxDrawBehindPausemenu(true) -- allow it to draw behind pause menu
SetScaleformFitRendertarget(self.handle, true)
end

if self.fullScreen then
DrawScaleformMovieFullscreen(self.handle, 255, 255, 255, 255, 0)
else
if not self.x or not self.y or not self.width or not self.height then
error('Properties not set for scaleform') -- Error properties not set for scaleform
DrawScaleformMovieFullscreen(self.handle, 255, 255, 255, 255, 0)
else
DrawScaleformMovie(self.handle, self.x, self.y, self.width, self.height, 255, 255, 255, 255, 0)
end
end

if self.target then -- If the render target is true
SetTextRenderId(1) -- Reset the text render id
end

Wait(0)
end
end)
end

---@return nil
---@description stop the scaleform from drawing, use this to only temporarily disable it, use Dispose otherwise.
function Scaleform:stopDrawing()
if not self.draw then
return
end
self.draw = false
end

---@return nil
---@description Dispose of the scaleform
function Scaleform:dispose()
if self.handle then -- If the handle exists
SetScaleformMovieAsNoLongerNeeded(self.handle) -- Set the scaleform movie as no longer needed
end

if self.target then -- If the render target exists
ReleaseNamedRendertarget(self.targetName) -- Release the named render target
Mycroft-Studios marked this conversation as resolved.
Show resolved Hide resolved
end

-- Reset the values
self.handle = nil -- Set the handle to nil
self.target = nil -- Set the render target to nil
self.draw = false -- Set the draw to false
end

---@return Scaleform
return Scaleform