Вопросы по Lua скриптингу

Общая тема для вопросов по разработке скриптов на языке программирования Lua, в частности под MoonLoader.
  • Задавая вопрос, убедитесь, что его нет в списке частых вопросов и что на него ещё не отвечали (воспользуйтесь поиском).
  • Поищите ответ в теме посвященной разработке Lua скриптов в MoonLoader
  • Отвечая, убедитесь, что ваш ответ корректен.
  • Старайтесь как можно точнее выразить мысль, а если проблема связана с кодом, то обязательно прикрепите его к сообщению, используя блок [code=lua]здесь мог бы быть ваш код[/code].
  • Если вопрос связан с MoonLoader-ом первым делом желательно поискать решение на wiki.

Частые вопросы

Как научиться писать скрипты? С чего начать?
Информация - Гайд - Всё о Lua скриптинге для MoonLoader(https://blast.hk/threads/22707/)
Как вывести текст на русском? Вместо русского текста у меня какие-то каракули.
Изменить кодировку файла скрипта на Windows-1251. В Atom: комбинация клавиш Ctrl+Shift+U, в Notepad++: меню Кодировки -> Кодировки -> Кириллица -> Windows-1251.
Как получить транспорт, в котором сидит игрок?
Lua:
local veh = storeCarCharIsInNoSave(PLAYER_PED)
Как получить свой id или id другого игрока?
Lua:
local _, id = sampGetPlayerIdByCharHandle(PLAYER_PED) -- получить свой ид
local _, id = sampGetPlayerIdByCharHandle(ped) -- получить ид другого игрока. ped - это хендл персонажа
Как проверить, что строка содержит какой-то текст?
Lua:
if string.find(str, 'текст', 1, true) then
-- строка str содержит "текст"
end
Как эмулировать нажатие игровой клавиши?
Lua:
local game_keys = require 'game.keys' -- где-нибудь в начале скрипта вне функции main

setGameKeyState(game_keys.player.FIREWEAPON, -1) -- будет сэмулировано нажатие клавиши атаки
Все иды клавиш находятся в файле moonloader/lib/game/keys.lua.
Подробнее о функции setGameKeyState здесь: lua - setgamekeystate | BlastHack — DEV_WIKI(https://www.blast.hk/wiki/lua:setgamekeystate)
Как получить id другого игрока, в которого целюсь я?
Lua:
local valid, ped = getCharPlayerIsTargeting(PLAYER_HANDLE) -- получить хендл персонажа, в которого целится игрок
if valid and doesCharExist(ped) then -- если цель есть и персонаж существует
  local result, id = sampGetPlayerIdByCharHandle(ped) -- получить samp-ид игрока по хендлу персонажа
  if result then -- проверить, прошло ли получение ида успешно
    -- здесь любые действия с полученным идом игрока
  end
end
Как зарегистрировать команду чата SAMP?
Lua:
-- До бесконечного цикла/задержки
sampRegisterChatCommand("mycommand", function (param)
     -- param будет содержать весь текст введенный после команды, чтобы разделить его на аргументы используйте string.match()
    sampAddChatMessage("MyCMD", -1)
end)
Крашит игру при вызове sampSendChat. Как это исправить?
Это происходит из-за бага в SAMPFUNCS, когда производится попытка отправки пакета определенными функциями изнутри события исходящих RPC и пакетов. Исправления для этого бага нет, но есть способ не провоцировать его. Вызов sampSendChat изнутри обработчика исходящих RPC/пакетов нужно обернуть в скриптовый поток с нулевой задержкой:
Lua:
function onSendRpc(id)
  -- крашит:
  -- sampSendChat('Send RPC: ' .. id)

  -- норм:
  lua_thread.create(function()
    wait(0)
    sampSendChat('Send RPC: ' .. id)
  end)
end
 
Последнее редактирование:

Неадекватная сова

Известный
Проверенный
278
245

MSIshka

Новичок
9
0
Всем привет! Я пытаюсь переделать скрипт KeySync под свои нужды. А именно чтобы ид автоматом подтягивался во время спека за игроком:
Lua:
local target = -1
local spectatingPlayerId = -1
local isTracking = false

function sampev.onPlayerSync(playerId, data)
    local result, id = sampGetPlayerIdByCharHandle(target)
    if result and id == playerId then
            SyncKeys["OnFoot"] = {}

            SyncKeys["OnFoot"]["W"] = (data.upDownKeys == 65408) or nil
            SyncKeys["OnFoot"]["A"] = (data.leftRightKeys == 65408) or nil
            SyncKeys["OnFoot"]["S"] = (data.upDownKeys == 00128) or nil
            SyncKeys["OnFoot"]["D"] = (data.leftRightKeys == 00128) or nil

            SyncKeys["OnFoot"]["Alt"] = (bit.band(data.keysData, 1024) == 1024) or nil
            SyncKeys["OnFoot"]["Shift"] = (bit.band(data.keysData, 8) == 8) or nil
            SyncKeys["OnFoot"]["Space"] = (bit.band(data.keysData, 32) == 32) or nil
            SyncKeys["OnFoot"]["F"] = (bit.band(data.keysData, 16) == 16) or nil
            SyncKeys["OnFoot"]["C"] = (bit.band(data.keysData, 2) == 2) or nil

            SyncKeys["OnFoot"]["RKM"] = (bit.band(data.keysData, 4) == 4) or nil
            SyncKeys["OnFoot"]["LKM"] = (bit.band(data.keysData, 128) == 128) or nil
    end
end

function sampev.onVehicleSync(playerId, vehicleId, data)
    local result, id = sampGetPlayerIdByCharHandle(target)
    print(id)
    if result and id == playerId then
            SyncKeys["Vehicle"] = {}

            SyncKeys["Vehicle"]["W"] = (bit.band(data.keysData, 8) == 8) or nil
            SyncKeys["Vehicle"]["A"] = (data.leftRightKeys == 65408) or nil
            SyncKeys["Vehicle"]["S"] = (bit.band(data.keysData, 32) == 32) or nil
            SyncKeys["Vehicle"]["D"] = (data.leftRightKeys == 00128) or nil

            SyncKeys["Vehicle"]["H"] = (bit.band(data.keysData, 2) == 2) or nil
            SyncKeys["Vehicle"]["Space"] = (bit.band(data.keysData, 128) == 128) or nil
            SyncKeys["Vehicle"]["Ctrl"] = (bit.band(data.keysData, 1) == 1) or nil
            SyncKeys["Vehicle"]["Alt"] = (bit.band(data.keysData, 4) == 4) or nil
            SyncKeys["Vehicle"]["Q"] = (bit.band(data.keysData, 256) == 256) or nil
            SyncKeys["Vehicle"]["E"] = (bit.band(data.keysData, 64) == 64) or nil
            SyncKeys["Vehicle"]["F"] = (bit.band(data.keysData, 16) == 16) or nil

            SyncKeys["Vehicle"]["Up"] = (data.upDownKeys == 65408) or nil
            SyncKeys["Vehicle"]["Down"] = (data.upDownKeys == 00128) or nil
    end
end

function updateTarget(playerId)
    if playerId and playerId >= 0 then
        local pedExist, ped = sampGetCharHandleBySampPlayerId(playerId)
        if pedExist and doesCharExist(ped) then
            target = ped
            spectatingPlayerId = playerId
            print(string.format(utext("Следим за игроком с ID: %d"), playerId))
        else
            target = -1
            spectatingPlayerId = -1
            print(utext("Игрока не существует или он вышел."))
        end
    else
        target = -1
        spectatingPlayerId = -1
        print(utext("ID игрока не задан."))
    end
end


function sampev.onTogglePlayerSpectating(state)
    if state then
        print(utext("Наблюдение включено."))
        lua_thread.create(function()
            while true do
                for i = 0, sampGetMaxPlayerId(false) do
                    local pedExist, ped = sampGetCharHandleBySampPlayerId(i)
                    if pedExist and doesCharExist(ped) then
                        updateTarget(i)
                        return
                    end
                end
                wait(0)
            end
        end)
    else
        print(utext("Наблюдение отключено."))
        updateTarget(-1)
    end
end

function sampev.onSpectatePlayer(playerId, camtype)
    -- Обновляем цель при срабатывании события
    if playerId and playerId >= 0 then
        updateTarget(playerId)
    else
        print(utext("onSpectatePlayer: Неверный ID игрока."))
        updateTarget(-1)
    end
end

function sampev.onRequestSpawnResponse()
    updateTarget(-1)
end

function sampev.onSetPlayerPos()
    updateTarget(-1)
end

function KeyCap(keyName, isPressed, size)
    local DL = imgui.GetWindowDrawList()
    local p = imgui.GetCursorScreenPos()
    local colors = {
        [true] = imgui.ImVec4(0.60, 0.60, 1.00, 1.00),
        [false] = imgui.ImVec4(0.60, 0.60, 1.00, 0.10)
    }

    if KEYCAP == nil then KEYCAP = {} end
    if KEYCAP[keyName] == nil then
        KEYCAP[keyName] = {
            status = isPressed,
            color = colors[isPressed],
            timer = nil
        }
    end

    local K = KEYCAP[keyName]
    if isPressed ~= K.status then
        K.status = isPressed
        K.timer = os.clock()
    end

    local rounding = 3.0
    local A = imgui.ImVec2(p.x, p.y)
    local B = imgui.ImVec2(p.x + size.x, p.y + size.y)
    if K.timer ~= nil then
        K.color = bringVec4To(colors[not isPressed], colors[isPressed], K.timer, 0.1)
    end
    local ts = imgui.CalcTextSize(keyName)
    local text_pos = imgui.ImVec2(p.x + (size.x / 2) - (ts.x / 2), p.y + (size.y / 2) - (ts.y / 2))

    imgui.Dummy(size)
    DL:AddRectFilled(A, B, u32(K.color), rounding)
    DL:AddRect(A, B, u32(colors[true]), rounding, _, 1)
    DL:AddText(text_pos, 0xFFFFFFFF, keyName)
end

function bringVec4To(from, dest, start_time, duration)
    local timer = os.clock() - start_time
    if timer >= 0.00 and timer <= duration then
        local count = timer / (duration / 100)
        return imgui.ImVec4(
            from.x + (count * (dest.x - from.x) / 100),
            from.y + (count * (dest.y - from.y) / 100),
            from.z + (count * (dest.z - from.z) / 100),
            from.w + (count * (dest.w - from.w) / 100)
        ), true
    end
    return (timer > duration) and dest or from, false
end

imgui.OnInitialize(function()
    theme.change()
    style.change()
    sW, sH = getScreenResolution()
    u32 = imgui.ColorConvertFloat4ToU32
    imgui.GetIO().IniFilename = nil
    local config = imgui.ImFontConfig()
    config.MergeMode = true
    config.PixelSnapH = true
end)

local spectateSyncKeys = imgui.OnFrame(
    function() return target ~= -1 end,
    function(self)
            self.HideCursor = true
            imgui.SetNextWindowPos(imgui.ImVec2(sW / 1.16, sH - 90), imgui.Cond.Always, imgui.ImVec2(0.5, 0.5))
            imgui.Begin("##KEYS", nil, imgui.WindowFlags.NoTitleBar + imgui.WindowFlags.AlwaysAutoResize)
            if doesCharExist(target) then
                local plState = (isCharOnFoot(target) and "OnFoot" or "Vehicle")
                imgui.BeginGroup()
                    imgui.SetCursorPosX(10 + 30 + 5)
                    KeyCap("W", (SyncKeys[plState]["W"] ~= nil), imgui.ImVec2(30, 30))
                    KeyCap("A", (SyncKeys[plState]["A"] ~= nil), imgui.ImVec2(30, 30)); imgui.SameLine()
                    KeyCap("S", (SyncKeys[plState]["S"] ~= nil), imgui.ImVec2(30, 30)); imgui.SameLine()
                    KeyCap("D", (SyncKeys[plState]["D"] ~= nil), imgui.ImVec2(30, 30))
                imgui.EndGroup()
                imgui.SameLine(nil, 20)

                if plState == "OnFoot" then
                    imgui.BeginGroup()
                        KeyCap("Shift", (SyncKeys[plState]["Shift"] ~= nil), imgui.ImVec2(75, 30)); imgui.SameLine()
                        KeyCap("Alt", (SyncKeys[plState]["Alt"] ~= nil), imgui.ImVec2(55, 30))
                        KeyCap("Space", (SyncKeys[plState]["Space"] ~= nil), imgui.ImVec2(135, 30))
                    imgui.EndGroup()
                    imgui.SameLine()
                    imgui.BeginGroup()
                        KeyCap("C", (SyncKeys[plState]["C"] ~= nil), imgui.ImVec2(30, 30)); imgui.SameLine()
                        KeyCap("F", (SyncKeys[plState]["F"] ~= nil), imgui.ImVec2(30, 30))
                        KeyCap("RM", (SyncKeys[plState]["RKM"] ~= nil), imgui.ImVec2(30, 30)); imgui.SameLine()
                        KeyCap("LM", (SyncKeys[plState]["LKM"] ~= nil), imgui.ImVec2(30, 30))    
                    imgui.EndGroup()
                else
                    imgui.BeginGroup()
                        KeyCap("Ctrl", (SyncKeys[plState]["Ctrl"] ~= nil), imgui.ImVec2(65, 30)); imgui.SameLine()
                        KeyCap("Alt", (SyncKeys[plState]["Alt"] ~= nil), imgui.ImVec2(65, 30))
                        KeyCap("Space", (SyncKeys[plState]["Space"] ~= nil), imgui.ImVec2(135, 30))
                    imgui.EndGroup()
                    imgui.SameLine()
                    imgui.BeginGroup()
                        KeyCap("Up", (SyncKeys[plState]["Up"] ~= nil), imgui.ImVec2(40, 30))
                        KeyCap("Down", (SyncKeys[plState]["Down"] ~= nil), imgui.ImVec2(40, 30))
                    imgui.EndGroup()
                    imgui.SameLine()
                    imgui.BeginGroup()
                        KeyCap("H", (SyncKeys[plState]["H"] ~= nil), imgui.ImVec2(30, 30)); imgui.SameLine()
                        KeyCap("F", (SyncKeys[plState]["F"] ~= nil), imgui.ImVec2(30, 30))
                        KeyCap("Q", (SyncKeys[plState]["Q"] ~= nil), imgui.ImVec2(30, 30)); imgui.SameLine()
                        KeyCap("E", (SyncKeys[plState]["E"] ~= nil), imgui.ImVec2(30, 30))
                    imgui.EndGroup()
                end
            end
        imgui.End()
    end
)
Код работает, но столкнулся с проблемой. Что при входе в игру, первый игрок в зоне стрима сразу хукается и появляются кнопки. Как я понял это из-за перебора в
function sampev.onTogglePlayerSpectating(state). Прошу вашей помощи, как это можно пофиксить, а то столько костылей сделал, а пофиксить никак не могу
 

ARMOR

Мне кажется, я всё проебал
Модератор
5,021
7,078
как перезарядить оружие моментально? (через writeMemory)
 

Cloud.

Активный
389
58
ладно, попробую, но видел в 1 строчку ( но там анимация перезарядки )
 

Naito

Активный
161
35
How to simulate adding a tuning component to my vehicle? by component id, for example the nitro.

How to get PacketLoss from a player? by id
 
Последнее редактирование:

D1sheq

Новичок
6
1
Ребята помогите, мне нужен скрипт который убирает ограничение скорости заднего хода, раньше он работал через RPC пакеты но 21 декабря 2024 скрипт перестал работать, я не могу понять как найти адресс в памяти который отвечает за это у меня вообще из знаний это Chat GPT и как написать Хеллоу Ворлд для начала я бы просто хотел узнать адресс который отвечает за ограничение и попытатся изменить значение или просто заморозить его а дальше с луа я сам как нибудь
 

.Oinoas

Новичок
6
0
How do I check the surfingVehicle ID
Lua:
function event.onPlayerSync(id, data)
    if state and data.surfingVehicleId >= 2000 then
        sampAddChatMessage("Wrong SurfingVehicle ID["..id.."]")
    end
end

error:
[19:36:14.135761] (error)    test.lua: F:\GTA San Andreas\moonloader\test.lua:592: attempt to compare nil with number
stack traceback:
    F:\GTA San Andreas\moonloader\test.lua:592: in function 'callback'
    ...\GTA San Andreas\moonloader\lib\samp\events\core.lua:77: in function 'process_event'
    ...\GTA San Andreas\moonloader\lib\samp\events\core.lua:100: in function 'process_packet'
    ...\GTA San Andreas\moonloader\lib\samp\events\core.lua:131: in function <...\GTA San Andreas\moonloader\lib\samp\events\core.lua:130>
 

ChаtGPT

Активный
403
97
Как увеличить дальность прорисовки информации о т/с через /dl ?
И можно как-то вырезать ненужную информацию?
 
  • Нравится
Реакции: Cloud.

Jefferson.!2.

Новичок
22
1
Этот код отключает выбор класса, но как отключить курсор?

sampToggleCursor(bool show) кажется, не работает.

Lua:
local memory = require('memory')
local samp = getModuleHandle("samp.dll")
local bEnable = true;
function enableClassSelection(bValue)
    local pClassSelection = memory.getuint32(samp + 0x21A18C, true)
    memory.setint8(pClassSelection + 0x13, bValue, true)
end
function main()
    while not isSampAvailable() do wait(0) end
    while true do
        wait(0)
        if wasKeyPressed(49) then
            bEnable = not bEnable
            enableClassSelection(bEnable and 1 or 0)
        end
    end
end
 

meowprd

Тот самый Котовский
Проверенный
1,302
731
How do I check the surfingVehicle ID
Lua:
function event.onPlayerSync(id, data)
    if state and data.surfingVehicleId >= 2000 then
        sampAddChatMessage("Wrong SurfingVehicle ID["..id.."]")
    end
end

error:
[19:36:14.135761] (error)    test.lua: F:\GTA San Andreas\moonloader\test.lua:592: attempt to compare nil with number
stack traceback:
    F:\GTA San Andreas\moonloader\test.lua:592: in function 'callback'
    ...\GTA San Andreas\moonloader\lib\samp\events\core.lua:77: in function 'process_event'
    ...\GTA San Andreas\moonloader\lib\samp\events\core.lua:100: in function 'process_packet'
    ...\GTA San Andreas\moonloader\lib\samp\events\core.lua:131: in function <...\GTA San Andreas\moonloader\lib\samp\events\core.lua:130>
Lua:
function event.onPlayerSync(id, data)
    if state and data.surfingVehicleId >= 2000 then
        sampAddChatMessage(string.format("Wrong SurfingVehicle ID[%d]", tonumber(data.surfingVehicleId)))
    end
end

Этот код отключает выбор класса, но как отключить курсор?

sampToggleCursor(bool show) кажется, не работает.

Lua:
local memory = require('memory')
local samp = getModuleHandle("samp.dll")
local bEnable = true;
function enableClassSelection(bValue)
    local pClassSelection = memory.getuint32(samp + 0x21A18C, true)
    memory.setint8(pClassSelection + 0x13, bValue, true)
end
function main()
    while not isSampAvailable() do wait(0) end
    while true do
        wait(0)
        if wasKeyPressed(49) then
            bEnable = not bEnable
            enableClassSelection(bEnable and 1 or 0)
        end
    end
end
Попробуй эту функцию
 

Jefferson.!2.

Новичок
22
1
Lua:
function event.onPlayerSync(id, data)
    if state and data.surfingVehicleId >= 2000 then
        sampAddChatMessage(string.format("Wrong SurfingVehicle ID[%d]", tonumber(data.surfingVehicleId)))
    end
end


Попробуй эту функцию
Не сработало. Мышь остаётся на экране и мешает двигать камеру.

aa.png