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

himoore

Новичок
5
0
приветствую, требуется помощь в данном баге, не выводит репорт в панельке

1720950149303.png


приветствую, требуется помощь в данном баге, не выводит репорт в панельке

1720950149303.png
при необходимости код скину
 
Последнее редактирование:
  • Bug
Реакции: MLycoris, XRLM и minxty

himoore

Новичок
5
0
приветствую, требуется помощь в данном баге, не выводит репорт в панельке

Посмотреть вложение 246372


при необходимости код скину
Lua:
--1 отвечающая--
function samp.onServerMessage(color, text)
    if not HLcfg.config.invAdmin then
        if text:find('%[.*] '..getMyNick()..'%['..getMyId()..'%] авторизовался как администратор %d+ уровня') then
            HLcfg.config.invAdmin = true
            save()
        end
    end
    if text:find("Не флуди!") then
        return false
    end
    -- Администратор Loqie_Punk[0] ответил Loqie_Punk[0]: 11
    --if text:find('Администратор '..getMyNick()..'%['..getMyId()..'%] ответил:') then
    if text:find("Администратор ".. getMyNick() .."%[".. getMyId() .. "%] ответил .*%: .*") then
        HLcfg.config.dayReports = HLcfg.config.dayReports + 1
        LsessionReport = LsessionReport + 1
        save()
    end
    if text:find("Игрок не вступил в игру!") or text:find('Администратор следит за (.*)%[(%d+)%]') then
        rInfo.id = -1
    end
   if text:find('%(%/arep%) (%S+)%[(%d+)%]%:.*{......}(.*)') then
      local Rnickname, Rid, RtextP = text:match('%(%/arep%) (%S+)%[(%d+)%]%:.*{......}(.*)')
       reports[#reports + 1] = {nickname = Rnickname, id = Rid, textP = RtextP}
   end
 if #reports > 0 then
        if color == -6732289 then
            for k, v in pairs(reports) do
                if k == 1 then
                    if not tableOfNew.AutoReport.v then
                        if text:find('%[.%] (.*)%[(%d+)%] для '..reports[1].nickname..'%['..reports[1].id..'%]: (.*)') then
                            refresh_current_report()
                        end
                    end
                
                
                  elseif #reports > 1 then
                    if text:find('%[.%] (.*)%[(%d+)%] для '..reports[k].nickname..'%['..reports[k].id..'%]: (.*)') then
                        table.remove(reports, k)
                    end
                end
            end
        end
    end
    if elements.checkbox.enableAutoReport.v then
        if text:find('(.*)%[(%d+)%]: %{FFCD00%}'..u8:decode(elements.input.textFindAutoReport.v)) then
            if elements.input.textFindAutoReport.v ~= '' and elements.input.answerAutoReport.v ~= '' then
                local nickRep, idRep = text:match('(%w+_?%w+)%[(%d+)%]: %{FFCD00%}'..u8:decode(elements.input.textFindAutoReport.v))
                answer_flets[#answer_flets + 1] = ('/pm '..idRep..' '..u8:decode(elements.input.answerAutoReport.v))
            else
                sampAddChatMessage('{FF0000}[Ошибка] {FF8C00}Вы не указали ответ/поисковой текст в авто-ответчике', stColor)
            end
        end
    end
    if elements.checkbox.formsEnabled.v then
        for k,v in ipairs(allForms) do
            -- {87CEFA}[A] Ст. Модератор[3] {FFF8DC}Loqie_Punk[0]: 1
            local oth, oth2 = '%{......%}%[A%].+%[%d%] %{......%}', '%{......%}%[A%]%{......%}.+%[%d%] %{......%}'
            local myOth, anoOth = getMyNick()..'%['..getMyId()..'%]: /'..v, '(%w+_%w+)%[(%d+)%]: /'..v..'%s(.*)'
            if (text:find(oth) and text:find(myOth)) or (text:find(oth2) and text:find(myOth)) then
                return true
            elseif (text:find(oth) and text:find(anoOth)) or (text:find(oth2) and text:find(anoOth)) then
                local admin_nick, admin_id, admin_other = text:match(anoOth)
                sampAddChatMessage('{FF0000}[AdminTools] {FFFFFF}Пришла форма, чтобы принять ее нажмите >> K <<', stColor)
                sampAddChatMessage('{FF0000}[AdminTools] {FFFFFF}Чтобы отклонить ее нажмите >> P <<', stColor)
                active_forma = true
                lua_thread.create(function()
                    lasttime = os.time()
                    lasttimes = 0
                    time_out = elements.int.timeOutForma.v
                    while lasttimes < time_out do
                        lasttimes = os.time() - lasttime
                        wait(0)
                        printStyledString("ADMIN FORM " .. time_out - lasttimes .. " WAIT", 1000, 4)
                        if stop_forma then
                            printStyledString('Form already accepted', 1000, 4)
                            stop_forma = false
                            break
                        end
                        if lasttimes == time_out then
                            printStyledString("Forma skipped", 1000, 4)
                        end
                        if isKeyJustPressed(VK_K) and not sampIsChatInputActive() and not sampIsDialogActive() then
                            printStyledString("Admin form accepted", 1000, 4)
                            sampSendChat("/"..v.." "..admin_other.." | "..string.format("%s.%s", admin_nick:match("."), admin_nick:match(".*_(%S+)")))
                            wait(2000)
                            sampSendChat('/a [Форма] Больше не приму')
                            LsessionForma = LsessionForma + 1
                            HLcfg.config.dayForms = HLcfg.config.dayForms + 1
                            save()
                            active_forma = false
                            break
                        elseif isKeyJustPressed(VK_P) and not sampIsChatInputActive() and not sampIsDialogActive() then
                            printStyledString('You missed the form', 1000, 4)
                            active_forma = false
                            break
                        end
                    end
                end)
            end
        end
    end
    if active_forma then
        if text:find('%[.*%] (%w+_?%w+)%[(%d+)%]%: %[Forma%] +') then
            active_forma = false
            stop_forma = true
        end
    end
    chatlog = io.open(getFileName(), "r+")
    chatlog:seek("end", 0);
    chatTime = "[" .. os.date("*t").hour .. ":" .. os.date("*t").min .. ":" .. os.date("*t").sec .. "]"
    chatlog:write(chatTime .. text .. "\n")
    chatlog:flush()
    chatlog:close()
    return {color, text}
end
-- 2 отвечающая --
 if elements.checkbox.areportclick.v then
            if isKeyJustPressed(VK_U) and is_key_check_available() then
                imgui.ShowCursor = not imgui.ShowCursor
            end
        else
            imgui.ShowCursor = true
        end
        imgui.SetNextWindowPos(imgui.ImVec2(imgui.GetIO().DisplaySize.x / 2, imgui.GetIO().DisplaySize.y / 2), imgui.Cond.FirstUseEver, imgui.ImVec2(0.5, 0.5))
        imgui.SetNextWindowSize(imgui.ImVec2(540, 320), imgui.Cond.FirstUseEver)   
        imgui.Begin(u8'Репорт', tableOfNew.AutoReport, imgui.WindowFlags.NoResize + imgui.WindowFlags.NoCollapse)
        imgui.BeginChild('##i_report', imgui.ImVec2(520, 30), true)       
        if #reports > 0 then
            imgui.PushTextWrapPos(500)
            imgui.TextUnformatted(u8(reports[1].nickname..'['..reports[1].id..']: '..reports[1].textP))
            imgui.PopTextWrapPos()
        end
        imgui.EndChild()
        imgui.Separator()
        imgui.PushItemWidth(520)
        if imgui.InputText(u8'##answer_input_report', tableOfNew.answer_report, imgui.InputTextFlags.EnterReturnsTrue) then
            local tempField = string.gsub(tableOfNew.answer_report.v, "", "")
            if #tempField == 1 then
                sampAddChatMessage('{FF0000}[Ошибка] {FFFFFF}Введите корректный ответ.', stColor)
            else
                giveAnswer(u8:decode(tableOfNew.answer_report.v))
                tableOfNew.answer_report.v = ''
            end
        end

Код, естественно, необходим
приложил
 

KyRDa

Участник
89
46
if text:find('%(%/arep%) (%S+)%[(%d+)%]%:.*{......}(.*)') then local Rnickname, Rid, RtextP = text:match('%(%/arep%) (%S+)%[(%d+)%]%:.*{......}(.*)') reports[#reports + 1] = {nickname = Rnickname, id = Rid, textP = RtextP} end
Если это получение информации с репорта, то вот тебе паттерн
Lua:
local Rnickname, Rid, RtextP = text:match('/arep (%S+)%[(%d+)%]: (.*)')
Убирай весь не касающейся к вопросу код. Я мало что понимаю в админских сообщениях, и понять, для меня, где репорт трудно
 

himoore

Новичок
5
0
Если это получение информации с репорта, то вот тебе паттерн
Lua:
local Rnickname, Rid, RtextP = text:match('/arep (%S+)%[(%d+)%]: (.*)')
Убирай весь не касающейся к вопросу код. Я мало что понимаю в админских сообщениях, и понять, для меня, где репорт трудно
no work
 

Oreshka23

Известный
341
166
В таком случае скинь примеры репортов, как они выглядят.
Попробуй вывести результат проверки в чат, также выведи в чат значение переменных Rnickname, Rid, RtextP. Совпадают ли они с ожидаемым результатом?
Lua:
if text:find('%(%/arep%) (%S+)%[(%d+)%]%:.*{......}(.*)') then

Если да, то выведи куда-нибудь (в чат или консоль) таблицу репортов, в которую ты сохраняешь эти значения.
Lua:
reports[#reports + 1] = {nickname = Rnickname, id = Rid, textP = RtextP}
Также проверь, что функция refresh_current_report() не стирает ничего лишнего из этой таблицы. Попробуй вывести таблицу до и после вызова этой функции.

И попробуй вывести значения отсюда в чат:
Lua:
imgui.TextUnformatted(u8(reports[1].nickname..'['..reports[1].id..']: '..reports[1].textP))
 
Последнее редактирование:
  • Нравится
Реакции: KyRDa

himoore

Новичок
5
0
В таком случае скинь примеры репортов, как они выглядят.
Попробуй вывести результат проверки в чат, также выведи в чат значение переменных Rnickname, Rid, RtextP. Совпадают ли они с ожидаемым результатом?


Если да, то выведи куда-нибудь (в чат или консоль) таблицу репортов, в которую ты сохраняешь эти значения.

Также проверь, что функция refresh_current_report() не стирает ничего лишнего из этой таблицы. Попробуй вывести таблицу до и после вызова этой функции.

И попробуй вывести значения отсюда в чат:

Grand Theft Auto  San Andreas Screenshot 2024.07.15 - 10.38.10.97.png

вот как выглядит
 

tomate kudasai/

Участник
96
18
setTrainSpeed(Vehicle train, float speed)
вот код

Код:
local st = false

function main()
    repeat wait(0) until isSampAvailable()
    sampRegisterChatCommand('set.speed',function ()
        setTrainSpeed(storeCarCharIsInNoSave(PLAYER_PED), 20.0)
    end)
    while true do
        wait(0)
    end
end
сразу крашит когда ввожу /set.speed

вот код

Код:
local st = false

function main()
    repeat wait(0) until isSampAvailable()
    sampRegisterChatCommand('set.speed',function ()
        setTrainSpeed(storeCarCharIsInNoSave(PLAYER_PED), 20.0)
    end)
    while true do
        wait(0)
    end
end
сразу крашит когда ввожу /set.speed
я если что в трамвае сижу
 

KyRDa

Участник
89
46
local Rnickname, Rid, RtextP = text:match('%(/arep%) (.+)%[(%d+)%]: (.*)')
Screenshot_2024-07-15-12-17-41-48_df198e732186825c8df26e3c5a10d7cd.jpg
Теперь точно работает
Заменил %S+ на .+ Так будут браться вообще все ники

вот код

Код:
local st = false

function main()
    repeat wait(0) until isSampAvailable
    sampRegisterChatCommand('set.speed',function ()
        setTrainSpeed(storeCarCharIsInNoSave(PLAYER_PED), 20.0)
    end)
    while true do
        wait(0)
    end
end
сразу крашит когда ввожу /set.speed


я если что в трамвае сижу
Самп ещё не загрузился. Ты написал цикл, который будет работать пока самп работает, а не наоборот.
repeat wait(0) until not isSampAvailable()
 
Последнее редактирование:

himoore

Новичок
5
0
local Rnickname, Rid, RtextP = text:match('%(/arep%) (.+)%[(%d+)%]: (.*)')
Посмотреть вложение 246481
Теперь точно работает
Заменил %S+ на .+ Так будут браться вообще все ники


Самп ещё не загрузился. Ты написал цикл, который будет работать пока самп работает, а не наоборот.
repeat wait(0) until not isSampAvailable()
нашел суть бага, не тот код дал
Lua:
function samp.onShowDialog(id, style, title, button1, button2, text)
    if id == 384 then
        OpenDialog = true
        local rNick, rId = title:match("(%S+)%[(%d+)%]")
        local rText = text:match("(.*)")
        reports = {}
         reports[#reports + 1] = {nickname = string.gsub(rNick, "{......}", ""), id = rId, textP = string.gsub(rText, "{......}", "")}
        tableOfNew.AutoReport.v = true
        return false
    end
Grand Theft Auto  San Andreas Screenshot 2024.07.15 - 13.17.10.41.png

Grand Theft Auto  San Andreas Screenshot 2024.07.15 - 13.17.07.57.png

надо избавиться от ник нейма с айди вначале, а остальное оставить