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

Firus20016

Участник
148
15
Привет всем. Пожалуйста помогите, скрипт умирает я хз почему.

Лог:
[ML] (error) medicalmedic.lua: cannot resume non-suspended coroutine
stack traceback:
[C]: in function 'getCharPlayerIsTargeting'
C:\Games\1223\moonloader\medicalmedic.lua:9: in function <C:\Games\1223\moonloader\medicalmedic.lua:5>
[ML] (error) medicalmedic.lua: Script died due to an error. (1C8704B4)

Код:
Lua:
local sampev = require 'lib.samp.events'
local vk = require('vkeys')
local stape = 0

function main()
    while not isSampAvailable() do wait(0) end
    while true do wait(0)
        if isKeyDown(2) and isKeyJustPressed(49) then
            result, ped = getCharPlayerIsTargeting(PLAYER_PED)
            if result then
                r, i = sampGetPlayerIdByCharHandle(ped)
                if r then
                    sampSendChat(i)
                end
            end
        end
    end
end
 

Musaigen

abobusnik
Проверенный
1,619
1,402
Как зарендерить пули?
Сохранять координаты старта и конца пули, переводить в 2D координаты, рисовать.
Координаты можно получить хуком onBulletSync(если нужны свои то и onSendBulletSync) или игровым хуком.
Как сделать? - Думай сам.
 

astynk

Известный
Проверенный
741
532
Привет всем. Пожалуйста помогите, скрипт умирает я хз почему.

Лог:
[ML] (error) medicalmedic.lua: cannot resume non-suspended coroutine
stack traceback:
[C]: in function 'getCharPlayerIsTargeting'
C:\Games\1223\moonloader\medicalmedic.lua:9: in function <C:\Games\1223\moonloader\medicalmedic.lua:5>
[ML] (error) medicalmedic.lua: Script died due to an error. (1C8704B4)

Код:
Lua:
local sampev = require 'lib.samp.events'
local vk = require('vkeys')
local stape = 0

function main()
    while not isSampAvailable() do wait(0) end
    while true do wait(0)
        if isKeyDown(2) and isKeyJustPressed(49) then
            result, ped = getCharPlayerIsTargeting(PLAYER_PED)
            if result then
                r, i = sampGetPlayerIdByCharHandle(ped)
                if r then
                    sampSendChat(i)
                end
            end
        end
    end
end
PLAYER_HANDLE, а не PLAYER_PED
 
  • Нравится
Реакции: Firus20016

Firus20016

Участник
148
15
Пожалуйста, помогите. Как остановить выполнение этого скрипта нажатием кнопки PageDown?
Код:
Lua:
function main()
    if not isSampfuncsLoaded() or not isSampLoaded() then return end
    while not isSampAvailable() do wait(100) end
    while true do
        wait(1)
        if isKeyDown(2) and isKeyJustPressed(49) then
                result, ped = getCharPlayerIsTargeting()
                if result then
                        r, i = sampGetPlayerIdByCharHandle(ped)
                        if r then
                            sampSendChat("Здравствуйте. Я, Флип Сайз, сотрудник данного медицинского центра, что вас беспокоит?")
                            sampAddChatMessage("Нажмите <ENTER>, чтобы продолжить.", -50000)
                            stage = 1
                        end
                end
        end
        if stage == 1 and isKeyDown(13) then
            sampSendChat("/me нырнув правой рукой в карман, вытянул оттуда блокнот и ручку")
            wait(1250)
            sampSendChat("/todo Хорошо, понял, ничего страшного*записывая в блокнот, все сказанное пациентом.")
            wait(1250)
            sampSendChat("/do Открытая сумка весит на плече правой руки.")
            wait(1250)
            sampSendChat("/me несколькими движениями нащупал лекарство")
            wait(1250)
            sampSendChat("/do Лекарство в левой руке.")
          wait(1250)
            sampSendChat("/todo Вот, держите*передавая лекарство человеку напротив.")
            wait(1250)
            sampSendChat("Принимайте эти таблетки,и через некоторое время вам станет лучше")
            wait(100)
            sampSendChat('/heal '..i)
            stage = 0
        end

        if isKeyDown(2) and isKeyJustPressed(50) then
                result, ped = getCharPlayerIsTargeting()
                if result then
                        r, i = sampGetPlayerIdByCharHandle(ped)
                        if r then
                            sampSendChat("Здравствуйте, как я понял, Вам нужна мед.карта.")
                            wait(1250)
                            sampSendChat("Предоставьте пожалуйста Ваш паспорт.")
                            sampAddChatMessage("Нажмите <1>, чтобы выдать новую мед.карту.", -50000)
                            sampAddChatMessage("Нажмите <2>, чтобы обновить мед.карту.", -50000)
                            stape = 3
                        end
                end
        end
        if stape == 3 and isKeyDown(50) then
            stape = 0
            if p < 6 then
            t = 8000
            end
            if p > 5 and p < 11 then
            t = 24000
            end
            if p > 10 and p < 16 then
            t = 32000
            end
            if p > 15 and p < 21 then
            t = 40000
            end
            if p > 20 and p < 30 then
            t = 48000
            end
            if p > 29 and t < 56000 then
            t = 56000
          end
            sampSendChat("Хорошо, я Вас понял. Вам нужно обновить данные в мед.карте.")
            wait(1250)
            sampSendChat(('Для обновления карты необходимо заплатить гос.пошлину в размере %s$, после чего мы продолжим.'):format(t))
            wait(1250)
            sampSendChat("/b Оплатите с помощью команды /pay или /trade")
            sampAddChatMessage("Нажмите <ENTER>, чтобы продолжить.", -50000)
            p = 0
            t = 0
            step = 1
        end
        if step == 1 and isKeyDown(13) then
            sampSendChat("/todo Благодарю, Вас*взяв паспорт в левую руку.")
            wait(1250)
            sampSendChat("/do Паспорт в левой руке.")
            wait(1250)
            sampSendChat("Не волнуйтесь, скоро я его Вам отдам!")
            wait(1250)
            sampSendChat("/me положил паспорт на стол")
            wait(1250)
            sampSendChat("/do Паспорт лежит на столе.")
            wait(1250)
            sampSendChat("/me подошел к столу и сел на стул")
            wait(1250)
            sampSendChat("/me пододвинул правой рукой паспорт к себе и открыл его")
            wait(1250)
            sampSendChat("/do Паспорт открыт.")
            wait(1250)
            sampSendChat("/do Ручка лежит в правом кармане.")
            wait(1250)
            sampSendChat("/me плавным движением левой руки вытащил ручку из кармана")
            wait(1250)
            sampSendChat("/do Ручка в левой руке.")
            wait(1250)
            sampSendChat("/do Чистые бланки для заполнения лежат на столе.")
            wait(1250)
            sampSendChat("/me плавным движением правой руки пододвинул чистые бланки к себе")
            wait(1250)
            sampSendChat("/me начал переписывать данные с паспорта на бланки")
            wait(1250)
            sampSendChat("/do Данные с паспорта переписаны на бланки.")
            wait(1250)
            sampSendChat("Жалобы на здоровье есть?")
            sampAddChatMessage("Нажмите <ENTER>, чтобы продолжить.", -50000)
            step = 2
        end
        if step == 2 and isKeyDown(13) then
            sampSendChat("/me записал в бланк")
            wait(1250)
            sampSendChat("Хорошо, теперь маленький тест на психику.")
            wait(1250)
            if z > 3 then
                z = 0
            end
            if z == 0 then
            sampSendChat("Вы видите как горит лес, что Вы будете делать?")
          end
          if z == 1 then
          sampSendChat("Вы видите, как человек упал и схватился за живот, что Вы будете делать?")
         end
          if z == 3 then
          sampSendChat("Вы видите, как несколько людей врываются в банк с оружием, что Вы будете делать?")
         end
            z = z + 1
            sampAddChatMessage("Нажмите <1>, чтобы поставить диагноз 'Полноcтью здоров(ая)'", -50000)
            sampAddChatMessage("Нажмите <2>, чтобы поставить диагноз 'Есть отклонения'", -50000)
            sampAddChatMessage("Нажмите <3>, чтобы поставить диагноз 'Психически не здоров(а)'", -50000)
            step = 3
        end
        if step == 3 and isKeyDown(49) then
            sampSendChat("/me поставил печать 'Полноcтью здоров(ая)'")
            wait(1250)
            sampSendChat("/me взял в левую руку мед. карту, а паспорт в правую")
            wait(1250)
            sampSendChat("/do Мед.карта и паспорт в руках.")
            wait(1250)
            sampSendChat("/todo Не болейте, всего доброго*передавая мед.карту и паспорт.")
            wait(1250)
            sampSendChat(("/medcard %s"):format(i.." "..3))
            step = 0
        end
        if step == 3 and isKeyDown(50) then
            sampSendChat("/me поставил печать 'Есть отклонения'")
            wait(1250)
            sampSendChat("/me взял в левую руку мед. карту, а паспорт в правую")
            wait(1250)
            sampSendChat("/do Мед.карта и паспорт в руках.")
            wait(1250)
            sampSendChat("/todo Не болейте, всего доброго*передавая мед.карту и паспорт.")
            wait(1250)
            sampSendChat(("/medcard %s"):format(i.." "..2))
            step = 0
        end
        if step == 3 and isKeyDown(51) then
            sampSendChat("/me поставил печать 'Психически не здоров(а)'")
            wait(1250)
            sampSendChat("/me взял в левую руку мед. карту, а паспорт в правую")
            wait(1250)
            sampSendChat("/do Мед.карта и паспорт в руках.")
          wait(1250)
            sampSendChat("/todo Не болейте, всего доброго*передавая мед.карту и паспорт.")
            wait(1250)
            sampSendChat(("/medcard %s"):format(i.." "..1))
            step = 0
        end
        if stape == 3 and isKeyDown(49) then
            sampSendChat("Хорошо, я Вас понял. Вам нужно оформить новую мед.карту.")
            wait(1250)
            sampSendChat("Для оформления карты необходимо заплатить гос.пошлину в размере 8000$, после чего мы продолжим.")
            wait(1250)
            sampSendChat("/b Оплатите с помощью команды /pay или /trade")
            sampAddChatMessage("Нажмите <ENTER>, чтобы продолжить.", -50000)
            stape = 4
        end
        if stape == 4 and isKeyDown(13) then
            sampSendChat("/todo Благодарю, Вас*взяв паспорт в левую руку.")
            wait(1250)
            sampSendChat("/do Паспорт в левой руке.")
            wait(1250)
            sampSendChat("Не волнуйтесь, скоро я его Вам отдам!")
            wait(1250)
            sampSendChat("/me положил паспорт на стол")
            wait(1250)
            sampSendChat("/do Паспорт лежит на столе.")
            wait(1250)
            sampSendChat("/me подошел к столу и сел на стул")
            wait(1250)
            sampSendChat("/me пододвинул правой рукой паспорт к себе и открыл его")
            wait(1250)
            sampSendChat("/do Паспорт открыт.")
            wait(1250)
            sampSendChat("/do Ручка лежит в правом кармане.")
            wait(1250)
            sampSendChat("/me плавным движением левой руки вытащил ручку из кармана")
            wait(1250)
            sampSendChat("/do Ручка в левой руке.")
            wait(1250)
            sampSendChat("/do Чистые бланки для заполнения лежат на столе.")
            wait(1250)
            sampSendChat("/me плавным движением правой руки пододвинул чистые бланки к себе")
            wait(1250)
            sampSendChat("/me начал переписывать данные с паспорта на бланки")
            wait(1250)
            sampSendChat("/do Данные с паспорта переписаны на бланки.")
            wait(1250)
            sampSendChat("Жалобы на здоровье есть?")
            sampAddChatMessage("Нажмите <ENTER>, чтобы продолжить.", -50000)
            stape = 5
        end
        if stape == 5 and isKeyDown(13) then
            sampSendChat("/me записал в бланк")
            wait(1250)
            sampSendChat("Хорошо, теперь маленький тест на психику.")
            wait(1250)
            if z > 3 then
                z = 0
            end
            if z == 0 then
            sampSendChat("Вы видите как горит лес, что Вы будете делать?")
          end
          if z == 1 then
          sampSendChat("Вы видите, как человек упал и схватился за живот, что Вы будете делать?")
         end
          if z == 3 then
          sampSendChat("Вы видите, как несколько людей врываются в банк с оружием, что Вы будете делать?")
         end
            z = z + 1
            sampAddChatMessage("Нажмите <1>, чтобы поставить диагноз 'Полноcтью здоров(ая)'", -50000)
            sampAddChatMessage("Нажмите <2>, чтобы поставить диагноз 'Есть отклонения'", -50000)
            sampAddChatMessage("Нажмите <3>, чтобы поставить диагноз 'Психически не здоров(а)'", -50000)
            stape = 6
        end
        if stape == 6 and isKeyDown(49) then
            sampSendChat("/me поставил печать 'Полноcтью здоров(ая)'")
            wait(1250)
            sampSendChat("/me взял в левую руку мед. карту, а паспорт в правую")
            wait(1250)
            sampSendChat("/do Мед.карта и паспорт в руках.")
            wait(1250)
            sampSendChat("/todo Не болейте, всего доброго*передавая мед.карту и паспорт.")
            wait(1250)
            sampSendChat(("/medcard %s"):format(i.." "..3))
            stape = 0
        end
        if stape == 6 and isKeyDown(50) then
            sampSendChat("/me поставил печать 'Есть отклонения'")
            wait(1250)
            sampSendChat("/me взял в левую руку мед. карту, а паспорт в правую")
            wait(1250)
            sampSendChat("/do Мед.карта и паспорт в руках.")
            wait(1250)
            sampSendChat("/todo Не болейте, всего доброго*передавая мед.карту и паспорт.")
            wait(1250)
            sampSendChat(("/medcard %s"):format(i.." "..2))
            stape = 0
        end
        if stape == 6 and isKeyDown(51) then
            sampSendChat("/me поставил печать 'Психически не здоров(a)'")
            wait(1250)
            sampSendChat("/me взял в левую руку мед. карту, а паспорт в правую")
            wait(1250)
            sampSendChat("/do Мед.карта и паспорт в руках.")
            wait(1250)
            sampSendChat("/todo Не болейте, всего доброго*передавая мед.карту и паспорт.")
            wait(1250)
            sampSendChat(("/medcard %s"):format(i.." "..1))
            stape = 0
        end
    end
end
 

UnikalnoeName

Известный
20
1
почему крашит? Иногда пашет норм
AutoLogin.Server.Pass_1 = String_Imgui.v

[22:13:08.090383] (error) Helper: E:\GTA San Andreas\moonloader\new.lua:318: attempt to index upvalue 'AutoLogin' (a nil value)
stack traceback:
E:\GTA San Andreas\moonloader\new.lua:318: in function 'OnDrawFrame'
E:\GTA San Andreas\moonloader\lib\imgui.lua:1377: in function <E:\GTA San Andreas\moonloader\lib\imgui.lua:1366>
[22:13:08.090383] (error) Helper: Script died due to an error. (0EF8D554)
 

Pheonixxx

Потрачен
263
46
Обратите внимание, пользователь заблокирован на форуме. Не рекомендуется проводить сделки.

UnikalnoeName

Известный
20
1
  • Влюблен
Реакции: Pheonixxx

Pheonixxx

Потрачен
263
46
Обратите внимание, пользователь заблокирован на форуме. Не рекомендуется проводить сделки.
попробуй
Код:
local you_gay = require('lib.samp.events')

function you_gay.onCreate3DText(id, color, position, distance, testLOS, attachedPlayerId, attachedVehicleId, text)
    if string.match(text, 'Механик') then
    -- here code
     end
end
Ошибок нету, но при появление текста " Механие " не жмет H
Lua:
local you_gay = require('lib.samp.events')

function you_gay.onCreate3DText(id, color, position, distance, testLOS, attachedPlayerId, attachedVehicleId, text)
    if string.match(text, 'Механик') then
    setVirtualKeyDown(VK_H, false)
     end
end
 

BIG BOG

Известный
653
142
Ошибок нету, но при появление текста " Механие " не жмет H
Lua:
local you_gay = require('lib.samp.events')

function you_gay.onCreate3DText(id, color, position, distance, testLOS, attachedPlayerId, attachedVehicleId, text)
    if string.match(text, 'Механик') then
    setVirtualKeyDown(VK_H, false)
     end
end
о хоспади, кто тебя
setVirtualKeyDown(VK_H, false)
научил?там как минимум true надо...
и лучше используй setGameKeyState(int key, int state)
 
  • Нравится
Реакции: Pheonixxx

EmilkaL

Известный
85
5
У меня проблема, когда использую имгуи с командами, имгуи отказывается вызываться без цикла на команде, мб я что-то не так делаю или что?
Lua:
script_name("Save gun")

local imgui = require 'imgui'

imgui.Process = false
local main_window_state = imgui.ImBool(false)

function imgui.OnDrawFrame()
    if main_window_state.v then
        imgui.SetNextWindowSize(imgui.ImVec2(770, 100), imgui.Cond.FirstUseEver) -- size
        imgui.Begin('1234', main_window_state)
        imgui.DragFloat("Size", sz_ref, 0.2, 2.0, 72.0, "%.0f");
        imgui.End()
    end
end

function main()
    imgui.Process = main_window_state.v
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end
   
    sampRegisterChatCommand('gunsave', cmd_save)
   
   
end

function gunsave()
main_window_state.v = not main_window_state.v
end
 

Dmitriy Makarov

25.05.2021
Проверенный
2,505
1,134
У меня проблема, когда использую имгуи с командами, имгуи отказывается вызываться без цикла на команде, мб я что-то не так делаю или что?
Lua:
script_name("Save gun")

local imgui = require 'imgui'

imgui.Process = false
local main_window_state = imgui.ImBool(false)

function imgui.OnDrawFrame()
    if main_window_state.v then
        imgui.SetNextWindowSize(imgui.ImVec2(770, 100), imgui.Cond.FirstUseEver) -- size
        imgui.Begin('1234', main_window_state)
        imgui.DragFloat("Size", sz_ref, 0.2, 2.0, 72.0, "%.0f");
        imgui.End()
    end
end

function main()
    imgui.Process = main_window_state.v
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end
  
    sampRegisterChatCommand('gunsave', cmd_save)
  
  
end

function gunsave()
main_window_state.v = not main_window_state.v
end
Lua:
script_name("Save gun")
local imgui = require 'imgui'
local main_window_state = imgui.ImBool(false)

function main()
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end
    sampRegisterChatCommand('gunsave', function() main_window_state.v = not main_window_state.v end)
    while true do
        wait(0)
        imgui.Process = main_window_state.v
    end
end

function imgui.OnDrawFrame()
    if main_window_state.v then
        imgui.SetNextWindowSize(imgui.ImVec2(770, 100), imgui.Cond.FirstUseEver) -- size
        imgui.Begin('1234', main_window_state)
        imgui.DragFloat("Size", sz_ref, 0.2, 2.0, 72.0, "%.0f");
    end
    imgui.End()
end
 
  • Нравится
Реакции: EmilkaL

castlefamily

Участник
104
3
Как пофиксить пропадание ников после nameTagOff()?
Часто бывает что ники вообще пропадают, над всеми игроками

Lua:
function nameTagOff()
local pStSet = sampGetServerSettingsPtr();
mem.setfloat(pStSet + 39, NTdist)
mem.setint8(pStSet + 47, NTwalls)
mem.setint8(pStSet + 56, NTshow)
nameTag = false
 

EmilkaL

Известный
85
5
Lua:
script_name("Save gun")
local imgui = require 'imgui'
local main_window_state = imgui.ImBool(false)

function main()
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end
    sampRegisterChatCommand('gunsave', function() main_window_state.v = not main_window_state.v end)
    while true do
        wait(0)
        imgui.Process = main_window_state.v
    end
end

function imgui.OnDrawFrame()
    if main_window_state.v then
        imgui.SetNextWindowSize(imgui.ImVec2(770, 100), imgui.Cond.FirstUseEver) -- size
        imgui.Begin('1234', main_window_state)
        imgui.DragFloat("Size", sz_ref, 0.2, 2.0, 72.0, "%.0f");
    end
    imgui.End()
end
а типо то, что imgui под main назодится, это ничего? Просто до этого мне сказали держать ее над main