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

Sanchez.

Известный
704
187
фу-фу-фу, учишь гадостям.
используй clipper.
Lua:
local clipper = imgui.ImGuiListClipper(#array)
while clipper:Step() do
     for i = clipper.DisplayStart + 1, clipper.DisplayEnd do
           imgui.Text(array[i])
     end
end
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) в имгуи он драф фрейм. Только сообщения в имгуи с чата таксистов не появляются. Как фиксануть?
 

Beer Stone

Участник
32
3
Можете скинуть ссылку на тему, где можно посмотреть как делать скрипт с рабочим окном?
 

Pashyka

Участник
220
17
Парни молю, помогите, ваще ничего не понимаю... тут пересечение переменных идет, из-за чего скрипт не скачивается, но как бы я переменные не менял нифига не работает

Lua:
    local downloadState = 'process'
    downloadUrlToFile(checkUrl, checkPath, function(id, status, p1, p2)
        if status == dlstatus.STATUS_DOWNLOADINGDATA then
            downloadState = 'process'
        elseif status == dlstatus.STATUS_ENDDOWNLOADDATA then
            downloadState = 'succ'
        elseif status == 64 then
            downloadState = 'failed'
        end
    end)
    while downloadState == 'process' do wait(0) end
 
    if downloadState == 'failed' then
        sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Ошибка проверки обновления.', -1)
    else
        local updateIni = inicfg.load(nil, checkPath)
        if updateIni then
            if updateIni.info.vers_text ~= version then
                sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Есть обновление скрипта. Версия: ' .. updateIni.info.vers_text, -1)
                local downloadState = 'process'
                downloadUrlToFile(scriptUrl, thisScript().path, function(id, status, p1, p2)
                    print("called callback with status: " .. status)
                    if status == dlstatus.STATUS_DOWNLOADINGDATA then
                        downloadState = 'process'
                        print(string.format('Загружено %d из %d.', p1, p2))
                    elseif status == dlstatus.STATUS_ENDDOWNLOADDATA then
                        downloadState = 'succ'
                    elseif status == 64 then
                        downloadState = 'failed'
                    end
                end)
                while downloadState == 'process' do wait(0) end
                print(downloadState)
                if downloadState == 'failed' then
                    sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Ошибка загрузки обновления.', -1)
                else
                    sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Скрипт успешно обновлен! Перезагружаю...', -1)
                    wait(150)
                    thisScript():reload()
                    return
                end
            else
                sampAddChatMessage("{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Обновления для скрипта не требуются.", -1)
            end
            os.remove(checkPath)
        end
    end
 

Smeruxa

Известный
1,297
681
Парни молю, помогите, ваще ничего не понимаю... тут пересечение переменных идет, из-за чего скрипт не скачивается, но как бы я переменные не менял нифига не работает

Lua:
    local downloadState = 'process'
    downloadUrlToFile(checkUrl, checkPath, function(id, status, p1, p2)
        if status == dlstatus.STATUS_DOWNLOADINGDATA then
            downloadState = 'process'
        elseif status == dlstatus.STATUS_ENDDOWNLOADDATA then
            downloadState = 'succ'
        elseif status == 64 then
            downloadState = 'failed'
        end
    end)
    while downloadState == 'process' do wait(0) end

    if downloadState == 'failed' then
        sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Ошибка проверки обновления.', -1)
    else
        local updateIni = inicfg.load(nil, checkPath)
        if updateIni then
            if updateIni.info.vers_text ~= version then
                sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Есть обновление скрипта. Версия: ' .. updateIni.info.vers_text, -1)
                local downloadState = 'process'
                downloadUrlToFile(scriptUrl, thisScript().path, function(id, status, p1, p2)
                    print("called callback with status: " .. status)
                    if status == dlstatus.STATUS_DOWNLOADINGDATA then
                        downloadState = 'process'
                        print(string.format('Загружено %d из %d.', p1, p2))
                    elseif status == dlstatus.STATUS_ENDDOWNLOADDATA then
                        downloadState = 'succ'
                    elseif status == 64 then
                        downloadState = 'failed'
                    end
                end)
                while downloadState == 'process' do wait(0) end
                print(downloadState)
                if downloadState == 'failed' then
                    sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Ошибка загрузки обновления.', -1)
                else
                    sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Скрипт успешно обновлен! Перезагружаю...', -1)
                    wait(150)
                    thisScript():reload()
                    return
                end
            else
                sampAddChatMessage("{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Обновления для скрипта не требуются.", -1)
            end
            os.remove(checkPath)
        end
    end
может потому что при скачке идут разные статусы?
 

chapo

🫡 В армии с 17.10.2023. В ЛС НЕ ОТВЕЧАЮ
Друг
8,768
11,210
как реализовать эти функции в луа?
 

Pashyka

Участник
220
17
может потому что при скачке идут разные статусы?


Дело в том, что я менял какие-то переменные, но ля у меня чет багнулся githab и заменил исходник, который я запустил случайно, я теперь не помню что тут менять надо, но если ты понял, нацель пожалуйста чуть подробнее, я уже часа 3 не могу с этой херней разобраться

1627075350098.png
 

Стэнфорд

Потрачен
1,058
540
Обратите внимание, пользователь заблокирован на форуме. Не рекомендуется проводить сделки.
Привет всем, вопрос по чтению файлов. Скрипту дается большой текст, нужно что бы он вывел этот текст в чат по строкам. Как можно это сделать?
 

Стэнфорд

Потрачен
1,058
540
Обратите внимание, пользователь заблокирован на форуме. Не рекомендуется проводить сделки.
Вам поступил перевод на ваш счет в размере $1 от жителя Lek_Bartov(111)
Переделать в регуляторные выражения, хелп (нужно из этой строки узнать сумму и ник игрока
 

paulohardy

вы еще постите говно? тогда я иду к вам
Всефорумный модератор
1,891
1,254
Вам поступил перевод на ваш счет в размере $1 от жителя Lek_Bartov(111)
Переделать в регуляторные выражения, хелп (нужно из этой строки узнать сумму и ник игрока
sum, nick = text:match("в размере $(%d+) от жителя (.-).%d+.")
 
  • Нравится
Реакции: Стэнфорд

Pashyka

Участник
220
17
sum, nick = text:match("в размере $(%d+) от жителя (.-).%d+.")
Пожалуйста помоги с этим, никак не могу понять почему авто-обновление работает плохо, а иногда вообще не работает, что за дичь?



Lua:
    local downloadState = 'process'
    downloadUrlToFile(checkUrl, checkPath, function(id, status, p1, p2)
        if status == dlstatus.STATUS_DOWNLOADINGDATA then
            downloadState = 'process'
        elseif status == dlstatus.STATUS_ENDDOWNLOADDATA then
            downloadState = 'succ'
        elseif status == 64 then
            downloadState = 'failed'
        end
    end)
    while downloadState == 'process' do wait(0) end
 
    if downloadState == 'failed' then
        sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Ошибка проверки обновления.', -1)
    else
        local updateIni = inicfg.load(nil, checkPath)
        if updateIni then
            if updateIni.info.vers_text ~= version then
                sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Есть обновление скрипта. Версия: ' .. updateIni.info.vers_text, -1)
                local downloadState = 'process'
                downloadUrlToFile(scriptUrl, thisScript().path, function(id, status, p1, p2)
                    print("called callback with status: " .. status)
                    if status == dlstatus.STATUS_DOWNLOADINGDATA then
                        downloadState = 'process'
                        print(string.format('Загружено %d из %d.', p1, p2))
                    elseif status == dlstatus.STATUS_ENDDOWNLOADDATA then
                        downloadState = 'succ'
                    elseif status == 64 then
                        downloadState = 'failed'
                    end
                end)
                while downloadState == 'process' do wait(0) end
                print(downloadState)
                if downloadState == 'failed' then
                    sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Ошибка загрузки обновления.', -1)
                else
                    sampAddChatMessage('{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Скрипт успешно обновлен! Перезагружаю...', -1)
                    wait(150)
                    thisScript():reload()
                    return
                end
            else
                sampAddChatMessage("{FFFFFF}[{2EF007}SMI Helper{FFFFFF}] Обновления для скрипта не требуются.", -1)
            end
            os.remove(checkPath)
        end
    end
 

Вложения

  • downloadfile.png
    downloadfile.png
    32.2 KB · Просмотры: 18

HpP

Известный
368
117
Как сделать так в mimgui, чтобы при нажатии на BeginChild, переменная Child_1 меняла значение с 0 на 1?
 

k1sher

Известный
201
4
Скажите пожалуйста как написать простой скрипт работающей по команде.
Чтобы при вводе команды писался бесконечный текст в чат с промежутки. И главное что бы это все работало при свернутой игре.