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

вайега52

Налуашил состояние
Модератор
3,000
3,138

а 34 пакет и 25 рпс это одно и то же? просто они одновременно отправляются
если есть разница, то какая?
Если правильно помню, 25 рпц отправляется тобой в том случае, когда ты пробуешь подключиться, а 34 пакет, когда сервер уже одобрил твое подключение

не совсем понял
то есть просто проверять sampIsLocalPlayerSpawned()?
Да, эта функция проверяет, заспавнен или нет, локальный игрок (ты). Но она зависит от сампфункса и вроде бы плохо работает на р3

Как получить текст который ввёл игрок в поле ввода в диалоге, без использования samp events?
Lua:
function onSendRpc(id, bs, priority, reliability, orderingChannel, shiftTs)
    if (id == 62) then
        local dialogId = raknetBitStreamReadInt16(bs)
        local button = raknetBitStreamReadInt8(bs)
        local listboxId = raknetBitStreamReadInt16(bs)
        local input = raknetBitStreamReadString(bs, raknetBitStreamReadInt8(bs))
        print("Текст:", input)
    end
end

кто может убрать команду и поставить на кнопку сбив?
Lua:
local hitler = false


function main()
    while not isSampAvailable() do wait(0) end
    while true do wait(0)
        if hitler then
            taskPlayAnim(PLAYER_PED, "Pointup_loop", "ON_LOOKERS", 12, true, false, false, false, -1)          
        end    
    end
end


addEventHandler("onWindowMessage", function(msg, wparam)
    if not isPauseMenuActive() then
        if (msg == 0x100) or (msg == 0x101) then
            if (wparam == 113) then
                consumeWindowMessage(true, false)
                if (msg == 0x101) then
                    hitler = not hitler
                end
            elseif (wparam == 46) and not sampIsCursorActive() then
                consumeWindowMessage(true, false)
                if (msg == 0x101) then
                    hitler = false
                    clearCharTasks(PLAYER_PED)
                end              
            end
        end
    end
end)
 
Последнее редактирование:

goodflex

Активный
278
59
Как в чекбоксе сделать цикл с проверкой?

Lua:
            if imgui.Checkbox(u8'test', test) then
                runToPoint:run(2480, -1668)
                local tx, ty = 2480, -1668
                if (getDistanceBetweenCoords2d(tx, ty, getCharCoordinates(PLAYER_PED)) < 1) then
                    runToPoint:run(2452, -1667)
                end
            end
 

вайега52

Налуашил состояние
Модератор
3,000
3,138
Как в чекбоксе сделать цикл с проверкой?

Lua:
            if imgui.Checkbox(u8'test', test) then
                runToPoint:run(2480, -1668)
                local tx, ty = 2480, -1668
                if (getDistanceBetweenCoords2d(tx, ty, getCharCoordinates(PLAYER_PED)) < 1) then
                    runToPoint:run(2452, -1667)
                end
            end
можно так, но я подозреваю, что в твоем случае лучше использовать другое
Lua:
if imgui.Checkbox(u8'test', test) then
    lua_thread.create(function()
        local rand = 0
        while (rand ~= 4) do
            rand = math.random(0, 10)
        end
        print("end")
    )
end
 
  • Нравится
Реакции: goodflex

XRLM

Против ветра рождённый
Модератор
1,656
1,335
Как получить текст который ввёл игрок в поле ввода в диалоге, без использования samp events?
хук onSendDialogResponse

можно так, но я подозреваю, что в твоем случае лучше использовать другое
Lua:
if imgui.Checkbox(u8'test', test) then
    lua_thread.create(function()
        local rand = 0
        while (rand ~= 4) do
            rand = math.random(0, 10)
        end
        print("end")
    )
end
вроде как создавать треад не обязательно во фрейме
 
  • Вау
Реакции: вайега52

chromiusj

Известный
Модератор
6,017
4,339
как получить смещение от координат на функции getCharCoordinates()?
возможно долбаебский вопрос,допустим если X отклонилась на 1 координату,то вызвался бы там синий экран смерти,или просто написало,что отклонилось на 1 координату
 

XRLM

Против ветра рождённый
Модератор
1,656
1,335
как получить смещение от координат на функции getCharCoordinates()?
возможно долбаебский вопрос,допустим если X отклонилась на 1 координату,то вызвался бы там синий экран смерти,или просто написало,что отклонилось на 1 координату
типо расстояние от первой точки до точки смещения или прям сколько сместилось по x, по y, по z?
 

вайега52

Налуашил состояние
Модератор
3,000
3,138
как получить смещение от координат на функции getCharCoordinates()?
возможно долбаебский вопрос,допустим если X отклонилась на 1 координату,то вызвался бы там синий экран смерти,или просто написало,что отклонилось на 1 координату
 

chromiusj

Известный
Модератор
6,017
4,339
типо расстояние от первой точки до точки смещения или прям сколько сместилось по x, по y, по z?
Код:
        local X, Y, Z = getCharCoordinates(PLAYER_PED)
        if X == math.floor(X + 1) or X == math.floor(X - 1) then
        sampAddChatMessage('блядь я отошел', -1)
вот так типо
не то
 
  • Вау
  • Bug
Реакции: DZONE и qdIbp

вайега52

Налуашил состояние
Модератор
3,000
3,138
Он зависим от библиотеки samp events
Мне кажется нету смысла самостоятельно хукать то, что есть в самп евентсе, это стоит делать только для практики и понимания того, как устроена эта либа.
 

Макс | Lycorn

Участник
166
13
Подскажите, как делается биндер? Мне нужно, что-бы текст который написали в InputTextMultiline отправлялся в чат по кмд/кнопке которую указали, и с задержкой так же.
Lua:
local bindersettings = {
    commands = {
        bindercmd1 = imgui.ImBuffer(256)
    },
    hotkeys = {
        binderhotkey1 = imgui.ImBuffer(256)
    },
    delay = {
        binderdelay1 = imgui.ImInt(1000)
    },
    text = {
        bindertext1 = imgui.ImBuffer(256)
    }
}
               


imgui.BeginChild('##bindertext', imgui.ImVec2(620, 387), true)
            imgui.Text(u8'Введите команду:')
            imgui.SameLine()
            imgui.PushItemWidth(150)
            imgui.InputText(u8'##bindercmd', bindersettings.commands.bindercmd1)
            imgui.PopItemWidth()
            imgui.Text(u8'Введите кнопку:')
            imgui.SameLine()
            imgui.PushItemWidth(150)
            imgui.InputText(u8'##binderhotkey', bindersettings.hotkeys.binderhotkey1)
            imgui.PopItemWidth()
            imgui.Text(u8'Введите задержку в секундах')
            imgui.SameLine()
            imgui.PushItemWidth(150)
            imgui.InputInt(u8'##binderdelay', bindersettings.delay.binderdelay1)
            imgui.PopItemWidth()
            imgui.BeginChild('##textbinder', imgui.ImVec2(605,30), true)
            imgui.CenterText(u8'Введите текст')
            imgui.EndChild()
            imgui.InputTextMultiline(u8"##", bindersettings.text.bindertext1, imgui.ImVec2(605,265))
            imgui.EndChild()
 

вайега52

Налуашил состояние
Модератор
3,000
3,138
Подскажите, как делается биндер? Мне нужно, что-бы текст который написали в InputTextMultiline отправлялся в чат по кмд/кнопке которую указали, и с задержкой так же.
Lua:
local bindersettings = {
    commands = {
        bindercmd1 = imgui.ImBuffer(256)
    },
    hotkeys = {
        binderhotkey1 = imgui.ImBuffer(256)
    },
    delay = {
        binderdelay1 = imgui.ImInt(1000)
    },
    text = {
        bindertext1 = imgui.ImBuffer(256)
    }
}
              


imgui.BeginChild('##bindertext', imgui.ImVec2(620, 387), true)
            imgui.Text(u8'Введите команду:')
            imgui.SameLine()
            imgui.PushItemWidth(150)
            imgui.InputText(u8'##bindercmd', bindersettings.commands.bindercmd1)
            imgui.PopItemWidth()
            imgui.Text(u8'Введите кнопку:')
            imgui.SameLine()
            imgui.PushItemWidth(150)
            imgui.InputText(u8'##binderhotkey', bindersettings.hotkeys.binderhotkey1)
            imgui.PopItemWidth()
            imgui.Text(u8'Введите задержку в секундах')
            imgui.SameLine()
            imgui.PushItemWidth(150)
            imgui.InputInt(u8'##binderdelay', bindersettings.delay.binderdelay1)
            imgui.PopItemWidth()
            imgui.BeginChild('##textbinder', imgui.ImVec2(605,30), true)
            imgui.CenterText(u8'Введите текст')
            imgui.EndChild()
            imgui.InputTextMultiline(u8"##", bindersettings.text.bindertext1, imgui.ImVec2(605,265))
            imgui.EndChild()
Lua:
if imgui.Button("Send") then
    sampSendChat(u8:decode(bindersettings.text.bindertext1.v))
end
 

kyrtion

Известный
1,404
544
Подскажите, как делается биндер? Мне нужно, что-бы текст который написали в InputTextMultiline отправлялся в чат по кмд/кнопке которую указали, и с задержкой так же.
Lua:
local bindersettings = {
    commands = {
        bindercmd1 = imgui.ImBuffer(256)
    },
    hotkeys = {
        binderhotkey1 = imgui.ImBuffer(256)
    },
    delay = {
        binderdelay1 = imgui.ImInt(1000)
    },
    text = {
        bindertext1 = imgui.ImBuffer(256)
    }
}
              


imgui.BeginChild('##bindertext', imgui.ImVec2(620, 387), true)
            imgui.Text(u8'Введите команду:')
            imgui.SameLine()
            imgui.PushItemWidth(150)
            imgui.InputText(u8'##bindercmd', bindersettings.commands.bindercmd1)
            imgui.PopItemWidth()
            imgui.Text(u8'Введите кнопку:')
            imgui.SameLine()
            imgui.PushItemWidth(150)
            imgui.InputText(u8'##binderhotkey', bindersettings.hotkeys.binderhotkey1)
            imgui.PopItemWidth()
            imgui.Text(u8'Введите задержку в секундах')
            imgui.SameLine()
            imgui.PushItemWidth(150)
            imgui.InputInt(u8'##binderdelay', bindersettings.delay.binderdelay1)
            imgui.PopItemWidth()
            imgui.BeginChild('##textbinder', imgui.ImVec2(605,30), true)
            imgui.CenterText(u8'Введите текст')
            imgui.EndChild()
            imgui.InputTextMultiline(u8"##", bindersettings.text.bindertext1, imgui.ImVec2(605,265))
            imgui.EndChild()
вам следует поиграть с значением. Почему именно .v? а почему его нельзя убрать, как наоборот сделать, и так далее