Вопросы по 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
 
Последнее редактирование:

Beer Stone

Участник
32
3
Да, но есть единственный минус, что команда /stats станет локальной и ты больше не сможешь ее отправить серверу пока LUA скрипт загружен.
Если ты хочешь сделать активацию/деактивацию, то тут надо хукать onSendCommand() и проверять на включение + нужную команду.
Спасибо.
 

Andrinall

Известный
677
529
через цикл for в основном беск. цикле.
В начале беск. цикла объявляешь переменную равную 0
В цикле проверяешь на нажатую клавишу и если она нажата добавляешь к ранее объявленной переменной нужное число

sampRegisterChatCommand
А каким образом я буду их проверять, если это не клавиши винды)
Они не проверяются в isKeyDown или прочем.
Это кнопки геймпада, в onWindowMessage что-то они тоже не захотели проверяться адекватно.
Эти числа (0 - 1023) берётся из joyinfo.wButtons (получаю через winmm.dll)

И мне нужно как-то чекать какие кнопки нажаты из дефолтного списка.
1, 2, 4, 8, 16, 32, 64, 128, 256, 512
Простой цикл тут бессилен, увы. Там есть множество ключей, которые могут получаться при нажатии разных пар кнопок, например.
+ они не поштучно ведь идут, значения суммируются. при нажатии A+Y, например, получится значение 9.
Мой мозг тут бессилен(
 

meowprd

Тот самый Котовский
Проверенный
1,283
710
А каким образом я буду их проверять, если это не клавиши винды)
Они не проверяются в isKeyDown или прочем.
Это кнопки геймпада, в onWindowMessage что-то они тоже не захотели проверяться адекватно.
Эти числа (0 - 1023) берётся из joyinfo.wButtons (получаю через winmm.dll)

И мне нужно как-то чекать какие кнопки нажаты из дефолтного списка.
1, 2, 4, 8, 16, 32, 64, 128, 256, 512
Простой цикл тут бессилен, увы. Там есть множество ключей, которые могут получаться при нажатии разных пар кнопок, например.
Мой мозг тут бессилен(
Как это onWindowMessage бессилен?
Пойду проверю, придется доставать пыльный геймпад.
 
  • Нравится
Реакции: Andrinall

meowprd

Тот самый Котовский
Проверенный
1,283
710
Ну лично у меня ничего не вышло, конечно я могу где-то накосячить, но, вроде как, всё было ровно.
Да, onWindowMessage не видит геймпад, значит я ошибся.
То есть у тебя, на данный момент, нет функции для проверки нажата/зажата ли клавиша, правильно?
 
  • Нравится
Реакции: Andrinall

Andrinall

Известный
677
529
Да, onWindowMessage не видит геймпад, значит я ошибся.
То есть у тебя, на данный момент, нет функции для проверки нажата/зажата ли клавиша, правильно?
Не совсем. Я могу только поштучно проверять нажатые клавиши, получая значение joyinfo.dwButtons
Но, мне нужно, чтобы я могу не по 1 клавише тыкать, а по несколько, и всё это регалось.
Типо... Если я нажимаю больше 1 клавиши разом - оно не будет писать какие клавиши нажаты, будет просто сумма значений и всё.
С крестовиной (POV) всё чуть проще, там всего 4 кнопки)))
Вот код, если интересно: ( я там некоторое немного не допилил и кое-где тестовые куски )
 

Вложения

  • gp.lua
    343.7 KB · Просмотры: 6

Beer Stone

Участник
32
3
Как сделать скрипт lua с меню настроек?
Можно ссылку на гайд? А то никак не найду.
 

meowprd

Тот самый Котовский
Проверенный
1,283
710
Не совсем. Я могу только поштучно проверять нажатые клавиши, получая значение joyinfo.dwButtons
Но, мне нужно, чтобы я могу не по 1 клавише тыкать, а по несколько, и всё это регалось.
Типо... Если я нажимаю больше 1 клавиши разом - оно не будет писать какие клавиши нажаты, будет просто сумма значений и всё.
С крестовиной (POV) всё чуть проще, там всего 4 кнопки)))
Вот код, если интересно: ( я там некоторое немного не допилил и кое-где тестовые куски )
К сожалению, не знаю чем помочь
Сам с геймпадом я на lua не работал, видимо только методом тыка)
 
  • Нравится
Реакции: Andrinall

Andrinall

Известный
677
529
К сожалению, не знаю чем помочь
Сам с геймпадом я на lua не работал, видимо только методом тыка)
Жаль.
Ну, по сути это применимо не только к геймпаду.
Типо из суммы надо как-то получить раздельные числа, каждое которых будет равно одному из дефолтных.
Пойду методом тыка пробовать что-то делать. Спасибо за попытку)

upd: для основных кнопок A B X Y придумал хрень через bit.tohex(joyinfo.wButtons) , как раз он туда не берёт в счёт другие кнопки и все 4 вместе нажатых кнопки дают 0xF, что упрощает мне задачу. Просто расписать массив и проходиться по нему циклом. А вот с остальными кнопками надо всё ещё думать. остались RB, LB, back, start и кнопки под стиками
 
Последнее редактирование:
  • Нравится
Реакции: meowprd

SamperJostkiy

Участник
169
19
Ничего у меняне получается, вот мой код:

binder:
if not isCursorActive() and not sampIsChatInputActive() then
    if isKeyJustPressed(VK_Z) then
        sampSendChat("/mask")
    end
    if isKeyJustPressed(VK_X) then
        sampSendChat("/heal")
    end
    if isKeyJustPressed(VK_B) then
        sampSendChat("/tp")
    end
end

вот консоль сф:
Консоль СФ:
[ML] (error) GWmonserBinder.lua: E:\DaVid\GTA SA\moonloader\GWmonserBinder.lua:10: attempt to call global 'isCursorActive' (a nil value)
stack traceback:
    E:\DaVid\GTA SA\moonloader\GWmonserBinder.lua: in function <E:\DaVid\GTA SA\moonloader\GWmonserBinder.lua:4>
[ML] (error) GWmonserBinder.lua: Script died due to an error. (0FAB78CC)
 

Andrinall

Известный
677
529
Жаль.
Ну, по сути это применимо не только к геймпаду.
Типо из суммы надо как-то получить раздельные числа, каждое которых будет равно одному из дефолтных.
Пойду методом тыка пробовать что-то делать. Спасибо за попытку)

upd: для основных кнопок A B X Y придумал хрень через bit.tohex(joyinfo.wButtons) , как раз он туда не берёт в счёт другие кнопки и все 4 вместе нажатых кнопки дают 0xF, что упрощает мне задачу. Просто расписать массив и проходиться по нему циклом. А вот с остальными кнопками надо всё ещё думать. остались RB, LB, back, start и кнопки под стиками
Решил проблему через костыль, разделив значения и преобразовав в HEX, теперь вместо 1024 ключей получается всего 0xF на каждый сегмент из
1) кнопки: A, B, X, Y
2) кнопки: RB, LB, back, start
3) кнопки: *нажатие левого стика*, *нажатие правого стика* (здесь максимальное значение 3, а не 0xF т.к. кнопки всего 2)
Из этого слепил такой вот говнокод и использую его

Lua:
function getGamepadKeys()
    local keys = {
        [1] = {
            ["0x1"] = 1,              -- A
            ["0x2"] = 2,             -- B
            ["0x3"] = { 1, 2 },      -- A + B
            ["0x4"] = 4,               -- X
            ["0x5"] = { 1, 4 },      -- A + X
            ["0x6"] = { 2, 4 },      -- B + X
            ["0x7"] = { 1, 2, 4 },   -- A + B + X
            ["0x8"] = 8,             -- Y
            ["0x9"] = { 1, 8 },      -- A + Y
            ["0xa"] = { 2, 8 },      -- B + Y
            ["0xb"] = { 1, 2, 8 },   -- A + B + Y
            ["0xc"] = { 4, 8 },      -- X + Y
            ["0xd"] = { 1, 4, 8 },   -- A + X + Y
            ["0xe"] = { 2, 4, 8 },   -- B + X + Y
            ["0xf"] = { 1, 2, 4, 8 } -- A + B + X + Y
        },
        [2] = {
            ["0x1"] = 1,              -- LB
            ["0x2"] = 2,             -- RB
            ["0x3"] = { 1, 2 },      -- LB + RB
            ["0x4"] = 4,               -- Back
            ["0x5"] = { 1, 4 },      -- LB + Back
            ["0x6"] = { 2, 4 },      -- RB + Back
            ["0x7"] = { 1, 2, 4 },   -- LB + RB + Back
            ["0x8"] = 8,             -- Start
            ["0x9"] = { 1, 8 },      -- LB + Start
            ["0xa"] = { 2, 8 },      -- RB + Start
            ["0xb"] = { 1, 2, 8 },   -- LB + RB + Start
            ["0xc"] = { 4, 8 },      -- Back + Start
            ["0xd"] = { 1, 4, 8 },   -- LB + Back + Start
            ["0xe"] = { 2, 4, 8 },   -- RB + Back + Start
            ["0xf"] = { 1, 2, 4, 8 } -- LB + RB + Back + Start
        },
        [3] = {
            ["0x1"] = 1,              -- LStickButton
            ["0x2"] = 2,             -- RStickButton
            ["0x3"] = { 1, 2 },      -- LStickButton + RStickButton
        }
    }

    local b, ret = {}, {}
    b[1] = ("0x%s"):format( bit.tohex( joyinfo.wButtons, 1 ) )
    b[2] = ("%s"):format( bit.tohex( joyinfo.wButtons, 2 ) ):gsub( "(.).", "0x%1" )
    b[3] = ("%s"):format( bit.tohex( joyinfo.wButtons, 3 ) ):gsub( "(.)..", "0x%1" )
    
    ret[1] = keys[1][ b[1] ]
    ret[2] = keys[2][ b[2] ]
    ret[3] = keys[3][ b[3] ]
    
    return ret
end
 

meowprd

Тот самый Котовский
Проверенный
1,283
710
Ничего у меняне получается, вот мой код:

binder:
if not isCursorActive() and not sampIsChatInputActive() then
    if isKeyJustPressed(VK_Z) then
        sampSendChat("/mask")
    end
    if isKeyJustPressed(VK_X) then
        sampSendChat("/heal")
    end
    if isKeyJustPressed(VK_B) then
        sampSendChat("/tp")
    end
end

вот консоль сф:
Консоль СФ:
[ML] (error) GWmonserBinder.lua: E:\DaVid\GTA SA\moonloader\GWmonserBinder.lua:10: attempt to call global 'isCursorActive' (a nil value)
stack traceback:
    E:\DaVid\GTA SA\moonloader\GWmonserBinder.lua: in function <E:\DaVid\GTA SA\moonloader\GWmonserBinder.lua:4>
[ML] (error) GWmonserBinder.lua: Script died due to an error. (0FAB78CC)
нет такой функции как isCursorActive(), потому что она называется по другому sampIsCursorActive()
 

Sanchez.

Известный
704
186
Lua:
require 'lib.moonloader'

local hook = require 'lib.samp.events'
local encoding = require 'encoding'
local imgui = require 'imgui'
encoding.default = 'CP1251'
u8 = encoding.UTF8

local messages = {}

local imguiokno = imgui.ImBool(false)

local sw, sh = getScreenResolution()

local clipper = imgui.ImGuiListClipper(#messages)

function main()
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end

    sampRegisterChatCommand('lua.j', function()
        imguiokno.v = not imguiokno.v
        imgui.Process = imguiokno.v
    end)
  
    imgui.Process = false

    while true do
        wait(0)
        if imguiokno.v == false then
            imgui.Process = false
        end
    end
end

function blue()
    imgui.SwitchContext()
    local style = imgui.GetStyle()
    local colors = style.Colors
    local clr = imgui.Col
    local ImVec4 = imgui.ImVec4

    style.WindowRounding = 2.0
    style.WindowTitleAlign = imgui.ImVec2(0.5, 0.84)
    style.ChildWindowRounding = 2.0
    style.FrameRounding = 2.0
    style.ItemSpacing = imgui.ImVec2(5.0, 4.0)
    style.ScrollbarSize = 13.0
    style.ScrollbarRounding = 0
    style.GrabMinSize = 8.0
    style.GrabRounding = 1.0

    colors[clr.FrameBg]                = ImVec4(0.16, 0.29, 0.48, 0.54)
    colors[clr.FrameBgHovered]         = ImVec4(0.26, 0.59, 0.98, 0.40)
    colors[clr.FrameBgActive]          = ImVec4(0.26, 0.59, 0.98, 0.67)
    colors[clr.TitleBg]                = ImVec4(0.04, 0.04, 0.04, 1.00)
    colors[clr.TitleBgActive]          = ImVec4(0.16, 0.29, 0.48, 1.00)
    colors[clr.TitleBgCollapsed]       = ImVec4(0.00, 0.00, 0.00, 0.51)
    colors[clr.CheckMark]              = ImVec4(0.26, 0.59, 0.98, 1.00)
    colors[clr.SliderGrab]             = ImVec4(0.24, 0.52, 0.88, 1.00)
    colors[clr.SliderGrabActive]       = ImVec4(0.26, 0.59, 0.98, 1.00)
    colors[clr.Button]                 = ImVec4(0.26, 0.59, 0.98, 0.40)
    colors[clr.ButtonHovered]          = ImVec4(0.26, 0.59, 0.98, 1.00)
    colors[clr.ButtonActive]           = ImVec4(0.06, 0.53, 0.98, 1.00)
    colors[clr.Header]                 = ImVec4(0.26, 0.59, 0.98, 0.31)
    colors[clr.HeaderHovered]          = ImVec4(0.26, 0.59, 0.98, 0.80)
    colors[clr.HeaderActive]           = ImVec4(0.26, 0.59, 0.98, 1.00)
    colors[clr.Separator]              = colors[clr.Border]
    colors[clr.SeparatorHovered]       = ImVec4(0.26, 0.59, 0.98, 0.78)
    colors[clr.SeparatorActive]        = ImVec4(0.26, 0.59, 0.98, 1.00)
    colors[clr.ResizeGrip]             = ImVec4(0.26, 0.59, 0.98, 0.25)
    colors[clr.ResizeGripHovered]      = ImVec4(0.26, 0.59, 0.98, 0.67)
    colors[clr.ResizeGripActive]       = ImVec4(0.26, 0.59, 0.98, 0.95)
    colors[clr.TextSelectedBg]         = ImVec4(0.26, 0.59, 0.98, 0.35)
    colors[clr.Text]                   = ImVec4(1.00, 1.00, 1.00, 1.00)
    colors[clr.TextDisabled]           = ImVec4(0.50, 0.50, 0.50, 1.00)
    colors[clr.WindowBg]               = ImVec4(0.06, 0.06, 0.06, 0.94)
    colors[clr.ChildWindowBg]          = ImVec4(1.00, 1.00, 1.00, 0.00)
    colors[clr.PopupBg]                = ImVec4(0.08, 0.08, 0.08, 0.94)
    colors[clr.ComboBg]                = colors[clr.PopupBg]
    colors[clr.Border]                 = ImVec4(0.43, 0.43, 0.50, 0.50)
    colors[clr.BorderShadow]           = ImVec4(0.00, 0.00, 0.00, 0.00)
    colors[clr.MenuBarBg]              = ImVec4(0.14, 0.14, 0.14, 1.00)
    colors[clr.ScrollbarBg]            = ImVec4(0.02, 0.02, 0.02, 0.53)
    colors[clr.ScrollbarGrab]          = ImVec4(0.31, 0.31, 0.31, 1.00)
    colors[clr.ScrollbarGrabHovered]   = ImVec4(0.41, 0.41, 0.41, 1.00)
    colors[clr.ScrollbarGrabActive]    = ImVec4(0.51, 0.51, 0.51, 1.00)
    colors[clr.CloseButton]            = ImVec4(0.41, 0.41, 0.41, 0.50)
    colors[clr.CloseButtonHovered]     = ImVec4(0.98, 0.39, 0.36, 1.00)
    colors[clr.CloseButtonActive]      = ImVec4(0.98, 0.39, 0.36, 1.00)
    colors[clr.PlotLines]              = ImVec4(0.61, 0.61, 0.61, 1.00)
    colors[clr.PlotLinesHovered]       = ImVec4(1.00, 0.43, 0.35, 1.00)
    colors[clr.PlotHistogram]          = ImVec4(0.90, 0.70, 0.00, 1.00)
    colors[clr.PlotHistogramHovered]   = ImVec4(1.00, 0.60, 0.00, 1.00)
    colors[clr.ModalWindowDarkening]   = ImVec4(0.80, 0.80, 0.80, 0.35)
end
blue()

function hook.onShowDialog(dialogId, style, title, button1, button2, text)
        if dialogId == 235 then
            --[[if text:find('Статус: {......}(.+)') then
                status = text:match('Статус: {......}(%S+)')
                sampAddChatMessage('Status: ' .. status,-1)
            end--]]
            if text:find('Имя: {......}(%S+)') then
                name = text:match('Имя: {......}(%S+)')
                sampAddChatMessage(name,-1)
            end
        end
end

function hook.onServerMessage(color, text)
    --[[if text:find('%[Таксист%] .+%[%d+%]: .+') then
        name, id, message = text:match('(.+)%[(%d+)%]: (.+)')
        --sampAddChatMessage("{15edc9}" .. name .. '[' ..id .. ']' .. ': ' .. message, 0x15edc9)
        print('\n' .. name .. '[' .. id .. ']' .. ': ' .. message)
        return false
    end--]]
    --[[if text:find('(.+)') then
        message = text:match('(.+)')
        print('\n\n' .. message)
        return false
    end--]]
    if text:find('%[Таксист%] .+%[%d+%]: .+') then
        name, id, message = text:match('(.+)%[(%d+)%]: (.+)')
        table.insert(messages,{
            name = name,
            id = id,
            message = message
        })
        messages = {}
        return false
    end
end

function imgui.OnDrawFrame()
    imgui.SetNextWindowSize(imgui.ImVec2(300, 150), imgui.Cond.FirstUseEver)

    imgui.SetNextWindowPos(imgui.ImVec2((sw / 2), sh / 2), imgui.Cond.FirstUseEver, imgui.ImVec2(0.5, 0.5))

    imgui.Begin('Taxi-Chat', imguiokno, imgui.WindowFlags.NoResize)
    imgui.BeginChild('##chatj', imgui.ImVec2(-1, -1), true)
    while clipper:Step() do
        for i = clipper.DisplayStart + 1, clipper.DisplayEnd do
            imgui.Text(messages[i])
         end
    end
    imgui.EndChild()
    imgui.End()
end

У меня почему-то не работает. Логов в мунлоадер лог нету, просто пишет что тип мунлоадер грузит этот скрипт, а дальше ничего. Что я не так сделал?
p.s Заработало, я переместил
local clipper = imgui.ImGuiListClipper(#messages) в имгуи он драф фрейм. Только сообщения в имгуи с чата таксистов не появляются. Как фиксануть?
актуал (Котовский помоги)