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

Pashyka

Участник
220
17
у тебя скорее всего стоит autoreboot.lua


Да, стоял там, но все равно, он типо написал ошибка загрузки, потом я релогаю скрипты сам, он не релогает автоматом и он пишет уже с новой версией

1627045525863.png


1627045544366.png



Lua:
while downloadState == 'process' do wait(0) end
    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

Может в этой части кода ошибочка, найти никак не могу
 
Последнее редактирование:

meowprd

Тот самый Котовский
Проверенный
1,278
718
Да, стоял там, но все равно, он типо написал ошибка загрузки, потом я релогаю скрипты сам, он не релогает автоматом и он пишет уже с новой версией

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

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


Lua:
while downloadState == 'process' do wait(0) end
    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

Может в этой части кода ошибочка, найти никак не могу
кстати, я забыл удаление файла сделать, мб проблема в этом
 

Pashyka

Участник
220
17
кстати, я забыл удаление файла сделать, мб проблема в этом

Удаление файла добавил перед всем этим сам, вот весь код обновления


Lua:
function main() -- главная функция
    if not isSampLoaded() or not isSampfuncsLoaded() then error(script_name..' needs SA:MP and SAMPFUNCS!') end
    while not isSampAvailable() do wait(100) end
    repeat
        wait(0)
    until sampIsLocalPlayerSpawned()
    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)
                    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
                    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
end
 

meowprd

Тот самый Котовский
Проверенный
1,278
718
Удаление файла добавил перед всем этим сам, вот весь код обновления


Lua:
function main() -- главная функция
    if not isSampLoaded() or not isSampfuncsLoaded() then error(script_name..' needs SA:MP and SAMPFUNCS!') end
    while not isSampAvailable() do wait(100) end
    repeat
        wait(0)
    until sampIsLocalPlayerSpawned()
    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)
                    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
                    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
end
он пишет что ошибка загрузки обновления и все? или дальше выполняется код, если есть такой? а то не вижу даже wait(-1) или беск. цикл
 

Pashyka

Участник
220
17
он пишет что ошибка загрузки обновления и все? или дальше выполняется код, если есть такой? а то не вижу даже wait(-1) или беск. цикл


Выполняется код, который до самого авто-обновления после спавна, ну и остальной тоже, то есть скрипт сам по себе не загружается, но если релогнуть его то запустит новую версию, сам скрипт работает дальше, код выполняется

1627046683843.png


Lua:
repeat
    wait(0)
until sampIsLocalPlayerSpawned()
lua_thread.create(function()
    wait(3000)
    sampSendChat('/stats')
end)
 

meowprd

Тот самый Котовский
Проверенный
1,278
718
Выполняется код, который до самого авто-обновления после спавна, ну и остальной тоже, то есть скрипт сам по себе не загружается, но если релогнуть его то запустит новую версию, сам скрипт работает дальше, код выполняется

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

Lua:
repeat
    wait(0)
until sampIsLocalPlayerSpawned()
lua_thread.create(function()
    wait(3000)
    sampSendChat('/stats')
end)
напиши в вк мне
 

Sanchez.

Известный
706
188
Я пытаюсь достать из статса текст, и вот что получается :(
1627047901460.png


Пытаюсь достать: "Статус: [Платина]"
1627047920727.png


Вот код:
Lua:
require 'lib.moonloader'

local hook = require 'lib.samp.events'

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

function hook.onShowDialog(dialogId, style, title, button1, button2, text)
    if dialogId == 235 then
        if text:find('Статус: {......}(.+)') then
            status = text:match('Статус: {......}(.+)')
            sampAddChatMessage("Ваш статус: {FFF409}" .. status,-1)
        end
    end
end

В чем проблема?
 

Вложения

  • 1627047887069.png
    1627047887069.png
    6.9 KB · Просмотры: 13

meowprd

Тот самый Котовский
Проверенный
1,278
718
Я пытаюсь достать из статса текст, и вот что получается :(
Посмотреть вложение 106638

Пытаюсь достать: "Статус: [Платина]"
Посмотреть вложение 106639

Вот код:
Lua:
require 'lib.moonloader'

local hook = require 'lib.samp.events'

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

function hook.onShowDialog(dialogId, style, title, button1, button2, text)
    if dialogId == 235 then
        if text:find('Статус: {......}(.+)') then
            status = text:match('Статус: {......}(.+)')
            sampAddChatMessage("Ваш статус: {FFF409}" .. status,-1)
        end
    end
end

В чем проблема?
дай фулл текст диалога
 

Sanchez.

Известный
706
188
дай фулл текст диалога
Код:
{FFFFFF}Имя: {B83434}[Family_Babetape]
{FFFFFF}Пол: {B83434}[Мужчина]
{FFFFFF}Уровень: {B83434}[4]
{FFFFFF}Уважение: {B83434}[7/20]
{FFFFFF}Деньги: {B83434}[$637357]
{FFFFFF}Евро: {B83434}[0]
{FFFFFF}BTC: {B83434}[0]
{FFFFFF}Номер телефона: {B83434}[4167202]
{FFFFFF}Деньги в банке: {B83434}[$2400]
{FFFFFF}Состояние личного счета: {B83434}[$0]
{FFFFFF}Деньги на депозите: {B83434}[$0]
{FFFFFF}Работа: {B83434}[Таксист]
{FFFFFF}Организация: {B83434}[Не имеется]
{FFFFFF}Уровень розыска: {B83434}[0]
{FFFFFF}Законопослушность: {B83434}31/100{FFFFFF}

Защита: {B83434}[-0% урона]{FFFFFF}
Регенерация: {B83434}[0 HP в мин.]{FFFFFF}
Урон: {B83434}[+0 урона]{FFFFFF}
Удача: {B83434}[шанс 0% крит.урона]{FFFFFF}

{FFFFFF}Предупреждения: {B83434}[0]
{FFFFFF}Наркозависимость: {B83434}0     {529020}[Нет зависимости]
{FFFFFF}Банковская карта: {B83434}[ Имеется ]
{FFFFFF}Статус: {B83434}[Платина]
{FFFFFF}Семья: {B83434}Нет
{FFFFFF}Возможность владеть 5-ю бизнесами: {B83434}[НЕТ]{FFFFFF}
{FFFFFF}Возможность владеть 4-я домами: {B83434}[НЕТ]{FFFFFF}
 

meowprd

Тот самый Котовский
Проверенный
1,278
718
Код:
{FFFFFF}Имя: {B83434}[Family_Babetape]
{FFFFFF}Пол: {B83434}[Мужчина]
{FFFFFF}Уровень: {B83434}[4]
{FFFFFF}Уважение: {B83434}[7/20]
{FFFFFF}Деньги: {B83434}[$637357]
{FFFFFF}Евро: {B83434}[0]
{FFFFFF}BTC: {B83434}[0]
{FFFFFF}Номер телефона: {B83434}[4167202]
{FFFFFF}Деньги в банке: {B83434}[$2400]
{FFFFFF}Состояние личного счета: {B83434}[$0]
{FFFFFF}Деньги на депозите: {B83434}[$0]
{FFFFFF}Работа: {B83434}[Таксист]
{FFFFFF}Организация: {B83434}[Не имеется]
{FFFFFF}Уровень розыска: {B83434}[0]
{FFFFFF}Законопослушность: {B83434}31/100{FFFFFF}

Защита: {B83434}[-0% урона]{FFFFFF}
Регенерация: {B83434}[0 HP в мин.]{FFFFFF}
Урон: {B83434}[+0 урона]{FFFFFF}
Удача: {B83434}[шанс 0% крит.урона]{FFFFFF}

{FFFFFF}Предупреждения: {B83434}[0]
{FFFFFF}Наркозависимость: {B83434}0     {529020}[Нет зависимости]
{FFFFFF}Банковская карта: {B83434}[ Имеется ]
{FFFFFF}Статус: {B83434}[Платина]
{FFFFFF}Семья: {B83434}Нет
{FFFFFF}Возможность владеть 5-ю бизнесами: {B83434}[НЕТ]{FFFFFF}
{FFFFFF}Возможность владеть 4-я домами: {B83434}[НЕТ]{FFFFFF}
Lua:
if text:find('Статус: {......}(.+)') then
    status = text:match('Статус: {......}(%S+)')
    print(status) -- output: [Платина]
end

Если я правильно понял, то вот.
 
  • Нравится
Реакции: Pashyka и Sanchez.

Sanchez.

Известный
706
188
Lua:
if text:find('Статус: {......}(.+)') then
    status = text:match('Статус: {......}(%S+)')
    print(status) -- output: [Платина]
end

Если я правильно понял, то вот.
Спасибо, только еще вопрос... Как сделать, чтобы каждые 5 секунд в чате выводился мой статус, не открывая /stats?
 

meowprd

Тот самый Котовский
Проверенный
1,278
718
Спасибо, только еще вопрос... Как сделать, чтобы каждые 5 секунд в чате выводился мой статус, не открывая /stats?
Ты можешь выводить только старый статус, который ранее был записан.
А потом, я так думаю, в чате явно есть оповещение о смене статуса, можно хукать оттуда.
Чтобы выводить раз в 5 секунд:
Lua:
local lastSend = os.clock()

while true do
    wait(0)
    if os.clock() - lastSend > 5 then
        sampAddChatMessage(status, -1)
        lastSend = os.clock()
    end
end

код не проверял, должно работать
 
  • Нравится
Реакции: Pashyka

Sanchez.

Известный
706
188
Ты можешь выводить только старый статус, который ранее был записан.
А потом, я так думаю, в чате явно есть оповещение о смене статуса, можно хукать оттуда.
Чтобы выводить раз в 5 секунд:
Lua:
local lastSend = os.clock()

while true do
    wait(0)
    if os.clock() - lastSend > 5 then
        sampAddChatMessage(status, -1)
        lastSend = os.clock()
    end
end

код не проверял, должно работать
Хорошо, спасибо. Просто мне нужно, чтобы статус автоматически отправлялся в чат, не открывая /stats. Типо где-то нужно написать чтобы типо бесконечно писался /stats с return false. Вот это я хочу реализовать
 

Curtis

Участник
282
10
Помогите сделать простой таймер с текстом, который будет работать даже при афк