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

Andrinall

Известный
679
532
Кодировка у меня 1251, в консоль у меня вообще ничего не пишет, я немного тогда поменял текст, и оно писало в консоль "nil"
Ну вообще должно работать -_-
1685280866368.png
 

Andrinall

Известный
679
532
Может я скину код скрипта, сделаешь что-бы правильно было?)
Ну для начала с правильной кодировкой попробуй)
Да и я тестить не смогу что поправлю(именно самповский скрипт) ибо я не игрок подобных серверов)
 

Макс | Lycorn

Участник
156
13
Ну для начала с правильной кодировкой попробуй)
Да и я тестить не смогу что поправлю(именно самповский скрипт) ибо я не игрок подобных серверов)
Ну ты мне будешь прост скидывать тогда, только скрывай текст)
С кодировкой я щас разбираюсь, нормальные буквы делаю
 

Andrinall

Известный
679
532
Ну ты мне будешь прост скидывать тогда, только скрывай текст)
С кодировкой я щас разбираюсь, нормальные буквы делаю
Обладатели Visual Studio Code и Sublime Text могут конвертить кодировку без кривых символов, как я помню. (reopen with encoding или set encoding для sublime)
За остальные редакторы не шарю(просто к сведению). Просто кидать код могу, да.
 

Макс | Lycorn

Участник
156
13
Обладатели Visual Studio Code и Sublime Text могут конвертить кодировку без кривых символов, как я помню. (reopen with encoding или set encoding для sublime)
За остальные редакторы не шарю(просто к сведению). Просто кидать код могу, да.
О все, я сделал, выводит, спасибо :)
UPD: Выводит только число, а слово нет
 

Макс | Lycorn

Участник
156
13
Последнее редактирование:
  • Эм
Реакции: YarikVL
D

deleted-user-422095

Гость
  • Нравится
Реакции: YarikVL

Макс | Lycorn

Участник
156
13
По той же схеме как тебе описали выше, либо https://regex101.com/ либо https://www.blast.hk/forums/16/
Та я вот и попробовал по той же схеме, мне бы еще хотя-бы одну штучку такую помогли бы сделать, дальше бы уже понял

А что это за прога вообще?
 

chapo

🫡 В армии с 17.10.2023. В ЛС НЕ ОТВЕЧАЮ
Друг
8,771
11,214
Теперь вопрос, как сделать для других пунктов так же?) На лвл "Лет в штате"; Ник "Имя:", попытался, не выходит..
выведи в принт текст диалога, скопируй текст из мунлога (именно из мунлога, не из консоли) и кинь сюда
 

Макс | Lycorn

Участник
156
13
выведи в принт текст диалога, скопируй текст из мунлога (именно из мунлога, не из консоли) и кинь сюда
Lua:
[FFFFFF]Организация: [FFD700]TV студия
[FFFFFF]Лет в штате: [FFD700]135
{ffffff}[FFFFFF][FFFFFF]Имя: [FFD700]Tsunami_Nakamura
[FFFFFF]Пол: [FFD700]Мужской
[CEAD2A]Наркозависимость: 0.0[FFFFFF]
То что мне нужно, айди диалога 1234
Вот то что ранее сделали:
Lua:
function sampev.onShowDialog(id, style, title, b1, b2, text)
    if id == 1234 then
        text = text:gsub("{%x+}", "")
        for line in text:gmatch("[^\n]+") do
            if line:find("^Законопослушность:%s(%d+)/%d+$") then
                sum = line:match("^Законопослушность:%s(%d+)/%d+$")
                print('\nЗаконопослушность: '..sum)
            end
        end
    end
end
 

Andrinall

Известный
679
532
Та я вот и попробовал по той же схеме, мне бы еще хотя-бы одну штучку такую помогли бы сделать, дальше бы уже понял


А что это за прога вообще?
Прога - Sublime Text 3 с настроенным Build System на luajit.exe
Я бы для всех пунктов статы сделал вот так вот(либо stats_pattern всунуть в stats чтоб обращаться через stats.pattern)
upd: по такому принципу любые строчки диалога вписать можно

Lua:
local stats_pattern = {
    name = "^Имя: (%w+_%w+)$",
    gender = "^Пол: (.*)$",
    years = "^Лет в штате: (%d+)$",
    birthday = "^Дата рождения: (%d+%.%d+%.%d+)$",
    pass_serial = "^Серия: (.*)$",
    pass_number = "^Номер: (.*)$",
    law_obedience = "^Законопослушность: (%d+)/%d+$",
    marital_name = "^Семейное положение: [ Женат(а) на (%w+_%w+) ]$",
    fraction = "^Организация: (.*)$",
    rank = "^Должность: (.*)$",
    job = "^Работа: (.*)$",
    millitary_ticket = "^Военный билет: [ (.*) ]$",
    psychiatric_hospital = "^Лечился в Психиатрической больнице: (%d+) раз .*$"
}

local stats = {
    name = nil,
    gender = nil,
    years = 0,
    birthday = "01.01.1999",
    pass_serial = 0000,
    pass_number = 0000,
    law_obedience = 0,
    marital_name = nil,
    fraction = nil,
    rank = nil,
    job = nil,
    millitary_ticket = nil,
    psychiatric_hospital = 0
}

function sampev.onShowDialog(id, style, title, b1, b2, text)
    if id == 1234 then
        text = text:gsub("{%x+}", "")
        for line in text:gmatch("[^\n]+") do
            for k, v in pairs(stats_pattern) do
                if line:find(v) then
                    stats[k] = line:match(v)
                end
            end
        end
    end
end

use:
Lua:
print(([[
Ник: %s
Пол: %s
Лет: %d
Дата рождения: %s
Паспорт: %s %s
Законопослушность: %d
Семейное положение: %s
Организация: %s
Должность: %s
Работа: %s
Военный билет: %s
Лечение в Псих.больнице: %s
]]):format(stats.name or "Неизвестно",
    stats.gender or "Неизвестно",
    stats.years or 0,
    stats.birthday or "01.01.1999",
    stats.pass_serial or "0000", stats.pass_number or "0000",
    stats.law_obedience or 0,
    stats.marital_name and ("Женат(а) на " .. stats.marital_name) or "Не женат(а)",
    stats.fraction or "Отсутствует",
    stats.rank or "Отсутствует",
    stats.job or "Отсутствует",
    stats.millitary_ticket or "Нет",
    stats.psychiatric_hospital or "Не проходил(а)"
))

demo
1685284860166.png
 
Последнее редактирование:

Макс | Lycorn

Участник
156
13
Прога - Sublime Text 3 с настроенным Build System на luajit.exe
Я бы для всех пунктов статы сделал вот так вот(либо stats_pattern всунуть в stats чтоб обращаться через stats.pattern)
upd: по такому принципу любые строчки диалога вписать можно

Lua:
local stats_pattern = {
    name = "^Имя: (%w+_%w+)$",
    gender = "^Пол: (.*)$",
    years = "^Лет в штате: (%d+)$",
    birthday = "^Дата рождения: (%d+%.%d+%.%d+)$",
    pass_serial = "^Серия: (.*)$",
    pass_number = "^Номер: (.*)$",
    law_obedience = "^Законопослушность: (%d+)/%d+$",
    marital_name = "^Семейное положение: [ Женат(а) на (%w+_%w+) ]$",
    fraction = "^Организация: (.*)$",
    rank = "^Должность: (.*)$",
    job = "^Работа: (.*)$",
    millitary_ticket = "^Военный билет: [ (.*) ]$",
    psychiatric_hospital = "^Лечился в Психиатрической больнице: (%d+) раз .*$"
}

local stats = {
    name = nil,
    gender = nil,
    years = 0,
    birthday = "01.01.1999",
    pass_serial = 0000,
    pass_number = 0000,
    law_obedience = 0,
    marital_name = nil,
    fraction = nil,
    rank = nil,
    job = nil,
    millitary_ticket = nil,
    psychiatric_hospital = 0
}

function sampev.onShowDialog(id, style, title, b1, b2, text)
    if id == 1234 then
        text = text:gsub("{%x+}", "")
        for line in text:gmatch("[^\n]+") do
            for k, v in pairs(stats_pattern) do
                if line:find(v) then
                    stats[k] = line:match(v)
                end
            end
        end
    end
end

use:
Lua:
print(([[
Ник: %s
Пол: %s
Лет: %d
Дата рождения: %s
Паспорт: %s %s
Законопослушность: %d
Семейное положение: %s
Организация: %s
Должность: %s
Работа: %s
Военный билет: %s
Лечение в Псих.больнице: %s
]]):format(stats.name or "Неизвестно",
    stats.gender or "Неизвестно",
    stats.years or 0,
    stats.birthday or "01.01.1999",
    stats.pass_serial or "0000", stats.pass_number or "0000",
    stats.law_obedience or 0,
    stats.marital_name and ("Женат(а) на " .. stats.marital_name) or "Не женат(а)",
    stats.fraction or "Отсутствует",
    stats.rank or "Отсутствует",
    stats.job or "Отсутствует",
    stats.millitary_ticket or "Нет",
    stats.psychiatric_hospital or "Не проходил(а)"
))

demo
Посмотреть вложение 202953
Спасибо, попробую щас сделать

Прога - Sublime Text 3 с настроенным Build System на luajit.exe
Я бы для всех пунктов статы сделал вот так вот(либо stats_pattern всунуть в stats чтоб обращаться через stats.pattern)
upd: по такому принципу любые строчки диалога вписать можно

Lua:
local stats_pattern = {
    name = "^Имя: (%w+_%w+)$",
    gender = "^Пол: (.*)$",
    years = "^Лет в штате: (%d+)$",
    birthday = "^Дата рождения: (%d+%.%d+%.%d+)$",
    pass_serial = "^Серия: (.*)$",
    pass_number = "^Номер: (.*)$",
    law_obedience = "^Законопослушность: (%d+)/%d+$",
    marital_name = "^Семейное положение: [ Женат(а) на (%w+_%w+) ]$",
    fraction = "^Организация: (.*)$",
    rank = "^Должность: (.*)$",
    job = "^Работа: (.*)$",
    millitary_ticket = "^Военный билет: [ (.*) ]$",
    psychiatric_hospital = "^Лечился в Психиатрической больнице: (%d+) раз .*$"
}

local stats = {
    name = nil,
    gender = nil,
    years = 0,
    birthday = "01.01.1999",
    pass_serial = 0000,
    pass_number = 0000,
    law_obedience = 0,
    marital_name = nil,
    fraction = nil,
    rank = nil,
    job = nil,
    millitary_ticket = nil,
    psychiatric_hospital = 0
}

function sampev.onShowDialog(id, style, title, b1, b2, text)
    if id == 1234 then
        text = text:gsub("{%x+}", "")
        for line in text:gmatch("[^\n]+") do
            for k, v in pairs(stats_pattern) do
                if line:find(v) then
                    stats[k] = line:match(v)
                end
            end
        end
    end
end

use:
Lua:
print(([[
Ник: %s
Пол: %s
Лет: %d
Дата рождения: %s
Паспорт: %s %s
Законопослушность: %d
Семейное положение: %s
Организация: %s
Должность: %s
Работа: %s
Военный билет: %s
Лечение в Псих.больнице: %s
]]):format(stats.name or "Неизвестно",
    stats.gender or "Неизвестно",
    stats.years or 0,
    stats.birthday or "01.01.1999",
    stats.pass_serial or "0000", stats.pass_number or "0000",
    stats.law_obedience or 0,
    stats.marital_name and ("Женат(а) на " .. stats.marital_name) or "Не женат(а)",
    stats.fraction or "Отсутствует",
    stats.rank or "Отсутствует",
    stats.job or "Отсутствует",
    stats.millitary_ticket or "Нет",
    stats.psychiatric_hospital or "Не проходил(а)"
))

demo
Посмотреть вложение 202953
Вопрос, а как это все сделать что-бы писало в imgui.Text?) Пока еще не шарю сильно в lua языке, учусь только :)
 
Последнее редактирование:

chapo

🫡 В армии с 17.10.2023. В ЛС НЕ ОТВЕЧАЮ
Друг
8,771
11,214
Lua:
[FFFFFF]Организация: [FFD700]TV студия
[FFFFFF]Лет в штате: [FFD700]135
{ffffff}[FFFFFF][FFFFFF]Имя: [FFD700]Tsunami_Nakamura
[FFFFFF]Пол: [FFD700]Мужской
[CEAD2A]Наркозависимость: 0.0[FFFFFF]
То что мне нужно, айди диалога 1234
Вот то что ранее сделали:
Lua:
function sampev.onShowDialog(id, style, title, b1, b2, text)
    if id == 1234 then
        text = text:gsub("{%x+}", "")
        for line in text:gmatch("[^\n]+") do
            if line:find("^Законопослушность:%s(%d+)/%d+$") then
                sum = line:match("^Законопослушность:%s(%d+)/%d+$")
                print('\nЗаконопослушность: '..sum)
            end
        end
    end
end
кароче на и не еби никому мозги (сделано через колхоз так как иначе выдает too many captures)
Lua:
local groups = {
    donate = { value = 'none', pattern = 'Текущее состояние счета:         (%d+) az coins'},
    name = { value = 'none', pattern = '{FFFFFF}Имя: {B83434}%[(.+)%] '},
    gender = { value = 'none', pattern = '{FFFFFF}Пол: {B83434}%[(.+)%] '},
    hp = { value = 'none', pattern = '{FFFFFF}Здоровье: {B83434}%[(%d+)%/%d+%]'},
    lvl = { value = 'none', pattern = '{FFFFFF}Уровень: {B83434}%[(%d+)%] '},
    xp = { value = 'none', pattern = '{FFFFFF}Уважение: {B83434}%[(%d+)%/%d+%] '},
    cashSA = { value = 'none', pattern = '{FFFFFF}Наличные деньги %(SA%$%): {B83434}%[%$(%d+)%]'},
    cashVC = { value = 'none', pattern = '{FFFFFF}Наличные деньги %(VC%$%): {B83434}%[%$(%d+)%]'},
    euro = { value = 'none', pattern = '{FFFFFF}Евро: {B83434}%[(%d+)%] '},
    btc = { value = 'none', pattern = '{FFFFFF}BTC: {B83434}%[(%d+)%]'},
    number = { value = 'none', pattern = '{FFFFFF}Номер телефона: {B83434}%[(%d+)%]'},
    bank = { value = 'none', pattern = '{FFFFFF}Деньги в банке: {B83434}%[%$(%d+)%]'},
    deposit = { value = 'none', pattern = '{FFFFFF}Деньги на депозите: {B83434}%[%$(%d+)%]'},
    job = { value = 'none', pattern = '{FFFFFF}Работа: {B83434}%[(.+)%]'},
    org = { value = 'none', pattern = '{FFFFFF}Организация: {B83434}%[(.+)%] '},
    wanted = { value = 'none', pattern = '{FFFFFF}Уровень розыска: {B83434}%[(%d+)%] '},
    zakonka = { value = 'none', pattern = '{FFFFFF}Законопослушность: {B83434}(%d+)%/100{FFFFFF}'},
    e_protection = { value = 'none', pattern = 'Защита: {B83434}%[(.+)%]{FFFFFF}'},
    e_regen = { value = 'none', pattern = 'Регенерация: {B83434}%[(.+)%]{FFFFFF}'},
    e_luck = { value = 'none', pattern = 'Урон: {B83434}%[(.+)%]{FFFFFF}'},
    e_maxhp = { value = 'none', pattern = 'Удача: {B83434}%[(.+)%]{FFFFFF}'},
    e_maxarmour = { value = 'none', pattern = 'Макс. HP: {B83434}%[(.+)%]{FFFFFF}'},
    e_bashchance = { value = 'none', pattern = 'Макс. Брони: {B83434}%[(.+)%]{FFFFFF}'},
    e_bashchance2 = { value = 'none', pattern = 'Шанс оглушения: {B83434}%[(.+)%]{FFFFFF}'},
    warns = { value = 'none', pattern = 'Шанс оглушения %(оглушающий плод%): {FF6347}(.+)'},
    drugsaddict = { value = 'none', pattern = '{FFFFFF}Предупреждения: {B83434}%[(%d+)%]'},
    drugsaddictText = { value = 'none', pattern = '{FFFFFF}Наркозависимость: {B83434}(.+)%s+{274B09}%[(.+)%]'},
    bankCardStatus = { value = 'none', pattern = '{FFFFFF}Банковская карта: {B83434}%[(.+)%]'},
    vipStatus = { value = 'none', pattern = '{FFFFFF}Статус: {B83434}%[(.+)%]'},
    fam = { value = 'none', pattern = '{FFFFFF}Семья: {B83434}(.+)'},
    upgrade_biz = { value = 'none', pattern = '{FFFFFF}Возможность владеть 5-ю бизнесами: {B83434}%[(.+)%]{FFFFFF}'},
    upgrade_house = { value = 'none', pattern = '{FFFFFF}Возможность владеть 4-я домами: {B83434}%[(.+)%]{FFFFFF}'},
    upgrade_farm = { value = 'none', pattern = '{FFFFFF}Возможность владеть 2-я фермами: {B83434}%[(.+)%]{FFFFFF}'},
    upgrade_sklad = { value = 'none', pattern = '{FFFFFF}Возможность владеть 2-я складскими помещениями: {B83434}%[(.+)%]{FFFFFF}'},
}

function readStats(text)
    local result = groups
    for line in text:gmatch('[^\n]+') do
        for fieldName, field in pairs(result) do
            local value = line:match(field.pattern)
            if value then
                result[fieldName].value = value
            end
        end
    end
    return result
end

function sampev.onShowDialog(dialogId, style, title, button1, button2, text)
    if title:find('статистика') then
        local stats = readStats(text)
        print('бабки на банке:', stats.bank)
    end
end
1685285876910.png



упс, я там некоторые строки перепутал, но я думаю что ты и сам сможешь поменять паттерны местами
1685286112565.png
 
  • Нравится
Реакции: Andrinall и YarikVL