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

ChаtGPT

Активный
396
96
Помогите решить проблему. Почему-то вместо минимальной дистанции, принт выводит все дистанции. Но я же обращаюсь по индексу "Выведи мне дистанцию по индексу 1" dist[1]
Код:
local zone = {
    --[номер ганг зоны] = координаты вышек делирия
    [20] = "-2840.9284667969, 425.32653808594, 4.5",
    [18] = "-2914.7434082031, 852.33026123047, 6.226879119873",
    [74] = "-2839.6564941406, -22.551847457886, 8.7249555587769",
}

function setNearestMarker(el_id)
    local elements = el_id
    local playerX, playerY, playerZ = getCharCoordinates(PLAYER_PED)

    -- Создаем таблицу для хранения всех расстояний
    local distances = {}

    for _, id in ipairs(elements) do
        for zid, coord in pairs(zone) do
            if zid == id then
                local x, y, z = coord:match("([^,]+), ([^,]+), ([^,]+)")
                local distance = getDistanceBetweenCoords3d(tonumber(x), tonumber(y), tonumber(z), playerX, playerY, playerZ)
                table.insert(distances, distance)
            end
        end
    end
 
 
--Я делал вот так:

    local minDistance = math.huge -- начальное значение минимальной дистанции
    local distan = {}
    for _, dist in pairs(distances) do
      if dist < minDistance then
        minDistance = dist -- обновление минимальной дистанции
        table.insert(distan, dist)
      end
    end

    table.sort(distan)
    print("Наименьшая дистанция:", distan[1])

    Результат:
        Наименьшая дистанция: 33.07654
        Наименьшая дистанция: 57.98767
        Наименьшая дистанция: 40.89786
    Выдаёт все дистанции, почему не по индексу?
end
 
Последнее редактирование:

MrRazrab

Известный
483
240
Помогите решить проблему. Почему-то вместо минимальной дистанции, принт выводит все дистанции. Но я же обращаюсь по индексу "Выведи мне дистанцию по индексу 1" dist[1]
Код:
local zone = {
    --[номер ганг зоны] = координаты вышек делирия
    [20] = "-2840.9284667969, 425.32653808594, 4.5",
    [18] = "-2914.7434082031, 852.33026123047, 6.226879119873",
    [74] = "-2839.6564941406, -22.551847457886, 8.7249555587769",
}

function setNearestMarker(el_id)
    local elements = el_id
    local playerX, playerY, playerZ = getCharCoordinates(PLAYER_PED)

    -- Создаем таблицу для хранения всех расстояний
    local distances = {}

    for _, id in ipairs(elements) do
        for zid, coord in pairs(zone) do
            if zid == id then
                local x, y, z = coord:match("([^,]+), ([^,]+), ([^,]+)")
                local distance = getDistanceBetweenCoords3d(tonumber(x), tonumber(y), tonumber(z), playerX, playerY, playerZ)
                table.insert(distances, distance)
            end
        end
    end
 
 
--Я делал вот так:

    local minDistance = math.huge -- начальное значение минимальной дистанции
 
    for _, dist in pairs(distances) do
      if dist < minDistance then
        minDistance = dist -- обновление минимальной дистанции
        print("Наименьшая дистанция:", dist[1])
      end
    end

    Результат:
        Наименьшая дистанция: 33.07654
        Наименьшая дистанция: 57.98767
        Наименьшая дистанция: 40.89786
    Выдаёт все дистанции, почему не по индексу?
end
Проблема заключается в том, что вы пытаетесь обратиться к элементу списка `dist` по индексу `[1]`, хотя `dist` - это число, а не список. Вместо этого, вам нужно обновить вывод, чтобы выводить `minDistance` вместо `dist[1]`. Вот исправленный код:

Lua:
local minDistance = math.huge -- начальное значение минимальной дистанции

local minIndex = 0 -- индекс элемента с минимальной дистанцией



for index, dist in ipairs(distances) do

  if dist < minDistance then

    minDistance = dist -- обновление минимальной дистанции

    minIndex = index -- обновление индекса минимальной дистанции

  end

end



print("Наименьшая дистанция:", minDistance)

print("Индекс наименьшей дистанции:", minIndex)

Теперь код будет выводить только минимальную дистанцию и ее индекс.
 

tsunamiqq

Участник
433
17
Не ищет:
Lua:
    if id == 156 then
        for line in text:gmatch("%[$(%d+)%]") do
            if line:find("%{......%}Банк бизнеса: %{......%}$(%d+)") then
                sum = line:match("Банк бизнеса: (%d+)")
                sampAddChatMessage(sum, -1)
            end
        end
    end
Lua:
[FFFFFF]Банк бизнеса: [9ACD32]$74575
 

Dmitriy Makarov

25.05.2021
Проверенный
2,505
1,134
Не ищет:
Lua:
    if id == 156 then
        for line in text:gmatch("%[$(%d+)%]") do
            if line:find("%{......%}Банк бизнеса: %{......%}$(%d+)") then
                sum = line:match("Банк бизнеса: (%d+)")
                sampAddChatMessage(sum, -1)
            end
        end
    end
Lua:
[FFFFFF]Банк бизнеса: [9ACD32]$74575
Знак доллара экранировать надо. Поставь перед ним знак %.
Lua:
if id == 156 then
    for line in text:gmatch("[^\r\n]+") do
        if line:find("%{......%}Банк бизнеса: %{......%}%$(%d+)") then
            sum = line:match("%{......%}Банк бизнеса: %{......%}%$(%d+)")
            sampAddChatMessage(sum, -1)
        end
    end
end
У тебя ещё во второй строке какая-то странная штука в gmatch. Не видел подобного..
 
  • Нравится
Реакции: tsunamiqq

tsunamiqq

Участник
433
17
Знак доллара экранировать надо. Поставь перед ним знак %.
Lua:
if id == 156 then
    for line in text:gmatch("[^\r\n]+") do
        if line:find("%{......%}Банк бизнеса: %{......%}%$(%d+)") then
            sum = line:match("%{......%}Банк бизнеса: %{......%}%$(%d+)")
            sampAddChatMessage(sum, -1)
        end
    end
end
У тебя ещё во второй строке какая-то странная штука в gmatch. Не видел подобного..
Спасибо, теперь еще вопрос как сделать тип.., смотри, нужно что-бы эти цыфры суммировались в каждом бизнесе вместе, и к примеру в imgui.Text() писалась сумма общая, и так-же, что-бы не нужно было самому заходить в 156 диалог, а что-бы оно само это делало
 

Dmitriy Makarov

25.05.2021
Проверенный
2,505
1,134
Спасибо, теперь еще вопрос как сделать тип.., смотри, нужно что-бы эти цыфры суммировались в каждом бизнесе вместе, и к примеру в imgui.Text() писалась сумма общая, и так-же, что-бы не нужно было самому заходить в 156 диалог, а что-бы оно само это делало
1. Не особо понял. Нужно понять, как вообще эта система работает с банками бизнесов. А суммирование можешь сделать примерно так:
Lua:
local bank = 0 -- Где-то в начале

-- onShowDialog
local sum = line:match(...) -- Тут твоя строка с line:match.
bank = bank + sum
2. Можешь просто вывести так же, как и в чат – imgui.Text(u8"В банке: $".. tostring(bank)).
3. Опять же, нужно понять эту систему. Ты можешь в цикле каждые N секунд прописывать команду, которая открывает этот диалог и скрывать его через return false, чтобы на экране он не появлялся (по желанию).
 
  • Нравится
Реакции: tsunamiqq

tsunamiqq

Участник
433
17
1. Не особо понял. Нужно понять, как вообще эта система работает с банками бизнесов. А суммирование можешь сделать примерно так:
Lua:
local bank = 0 -- Где-то в начале

-- onShowDialog
local sum = line:match(...) -- Тут твоя строка с line:match.
bank = bank + sum
2. Можешь просто вывести так же, как и в чат – imgui.Text(u8"В банке: $".. tostring(bank)).
3. Опять же, нужно понять эту систему. Ты можешь в цикле каждые N секунд прописывать команду, которая открывает этот диалог и скрывать его через return false, чтобы на экране он не появлялся (по желанию).
Как тут правильно заполнить?
Lua:
return {id, style, title, b1, b2, text.. "  \n- {FFFF00}1: "..tostring(bank)"\n- {FFFF00}2:"}

Я сделал так но ошибка:

Lua:
attempt to call a string value stack traceback:
 

Dmitriy Makarov

25.05.2021
Проверенный
2,505
1,134
Как тут правильно заполнить?
Lua:
return {id, style, title, b1, b2, text.. "  \n- {FFFF00}1: "..tostring(bank)"\n- {FFFF00}2:"}

Я сделал так но ошибка:

Lua:
attempt to call a string value stack traceback:
Lua:
return {id, style, title, b1, b2, text.. "  \n- {FFFF00}1: "..tostring(bank).."\n- {FFFF00}2:"}
 
  • Нравится
Реакции: tsunamiqq

InsaneNooby

Новичок
4
0
Всем привет! Можно ли как то сделать проверку версии на raw pastebin'а? Если да то можете скинуть код?
 

F1stachka

Новичок
4
1
Добрый день помогите пожалуйста решить проблему, кнопка H не нажимается хотя 1 в консоль выводится, так же есть вопрос как сделать чтобы скрипт работал в свернутом режиме.



Lua:
require "lib.moonloader"
local huy = require("samp.events")
local piska = 0


local oX = 250
local oY = 430


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

    while true do
        sampTextdrawCreate(221, "Server_time:", oX, oY)
        sampTextdrawSetLetterSizeAndColor(221, 0.3, 1.7, 0xFFe1e1e1)
        sampTextdrawSetOutlineColor(221, 0.5, 0xFF000000)
        sampTextdrawSetAlign(221, 1)
        sampTextdrawSetStyle(221, 2)

        timer = os.time() + piska
        min = os.date("%M", timer)
        sec = os.date("%S", timer)
        if (min == "00" or min == "22") and sec == "30" then
            emulateKeyPressH(5, 50)
        end
        sampTextdrawCreate(222, os.date("%H:%M:%S", timer), oX + 90, oY)
        sampTextdrawSetLetterSizeAndColor(222, 0.3, 1.7, 0xFFff6347)
        sampTextdrawSetOutlineColor(222, 0.5, 0xFF000000)
        sampTextdrawSetAlign(222, 1)
        sampTextdrawSetStyle(222, 2)
        wait(500)
    end
end
function emulateKeyPressH(repeats, delay)
    for i = 1, repeats do
        setVirtualKeyDown(VK_H, true)
        wait(delay)
        setVirtualKeyDown(VK_H,false)
        print(1)
    end
end
 

Corrygan228

Участник
132
9
хочу сделать биндер, в котором пользователь сможет сам настраивать бинды, кол-во строк и т.д., но так как я всё делаю через массивы, появилась проблема,
для каждого бинда сделать отдельную менюшку в правой части окошка, ведь я не смогу для каждого отдельно создавать массив с инпутами и т.п. и делать проверку есть ли этот элемент в binds_arr, как быть?
Lua:
local binds_arr = {
    u8'Приветствие'
}

--фрейм
imgui.Columns(2, '##main_part_4_select_binder')
imgui.SetColumnWidth(-1, 200)

imgui.BeginChild('##binds_first_column', imgui.ImVec2(186, 250), false)

for k, v in ipairs(binds_arr) do
    if imgui.ButtonClickable(true, v, imgui.ImVec2(170, 20)) then
        select_binds = k
    end
end

imgui.SetCursorPosX(7.3)
if imgui.Button(u8'Добавить', imgui.ImVec2(74, 24)) then
    table.insert(binds_arr, u8'Бинд ' .. #binds_arr + 1)
end

imgui.SameLine()

imgui.SetCursorPosX(88.6)
if imgui.Button(u8'Удалить', imgui.ImVec2(74, 24)) then
    table.remove(binds_arr)
end

imgui.EndChild()

imgui.NextColumn()

imgui.BeginChild('##binds_second_column', imgui.ImVec2(348, 250), false)

for k, v in ipairs(binds_arr) do
    if select_binds == k then
        imgui.Text(u8'Страница №' .. k)
    end
end

imgui.EndChild()

imgui.Columns(1)
1689665880910.png
 

Дядя Энрик.

Активный
338
81
как получить Y \ N от игрока?
Lua:
        keys["onfoot"]["N"] = (bit.band(data.keysData, 131072) == 131072) or nil
        keys["onfoot"]["Y"] = (bit.band(data.keysData, 65536) == 65536) or nil
брал инфу отсюда https://pawn.wiki/index.php?/topic/174-id-klavish/#entry234 , но когда игрок нажимает что-то из этих клавиш, то ничего не отображает, а если брать для теста 1 - 1 это таб, то всё работает.
 

Julimba

Участник
108
10
Lua:
    --==[iniCfg]==--
local cfg = inicfg.load({
    Config = {
        active = 'true',
        times = 'false',
        emoji = 'false',
        vip = 'false',
        rekl = 'false',
        admin = 'false',
        prozA = '0.30',
        prozN = '0.00',
        shir = '1',
        visot = '5',
        reklM = 'true'
    }
}, 'VRManager')

-- main
if not doesFileExist(getWorkingDirectory()..'/config/VRManager.ini') then
    inicfg.save(cfg, 'VRManager')
end

    --==[Toggle's and Slider's]==--
local toggle1 = new.bool(cfg.Config.active)
local toggle2 = new.bool(cfg.Config.times)
local toggle3 = new.bool(cfg.Config.emoji)
local toggle4 = new.bool(cfg.Config.vip)
local toggle5 = new.bool(cfg.Config.rekl)
local toggle6 = new.bool(cfg.Config.admin)
local toggle7 = new.bool(cfg.Config.reklM)
local Slider1 = new.int(cfg.Config.shir)
local Slider2 = new.int(cfg.Config.visot)
local Slider3 = new.float(cfg.Config.prozA)
local Slider4 = new.float(cfg.Config.prozN)

Ошибку выдает на первую строчку тоглов:
[ML] (error) !VR VR VR.lua: C:\Users\User\Desktop\kichiro\moonloader\!VR VR VR.lua:86: cannot convert 'string' to 'bool [1]'
в чем моя ошибка? Это если файл ини удалить, то почему то он не создает в его отсутствие новый и крашит скрипт просто
 

MLycoris

На вид оружие массового семяизвержения
Проверенный
1,992
2,187
Lua:
    --==[iniCfg]==--
local cfg = inicfg.load({
    Config = {
        active = 'true',
        times = 'false',
        emoji = 'false',
        vip = 'false',
        rekl = 'false',
        admin = 'false',
        prozA = '0.30',
        prozN = '0.00',
        shir = '1',
        visot = '5',
        reklM = 'true'
    }
}, 'VRManager')

-- main
if not doesFileExist(getWorkingDirectory()..'/config/VRManager.ini') then
    inicfg.save(cfg, 'VRManager')
end

    --==[Toggle's and Slider's]==--
local toggle1 = new.bool(cfg.Config.active)
local toggle2 = new.bool(cfg.Config.times)
local toggle3 = new.bool(cfg.Config.emoji)
local toggle4 = new.bool(cfg.Config.vip)
local toggle5 = new.bool(cfg.Config.rekl)
local toggle6 = new.bool(cfg.Config.admin)
local toggle7 = new.bool(cfg.Config.reklM)
local Slider1 = new.int(cfg.Config.shir)
local Slider2 = new.int(cfg.Config.visot)
local Slider3 = new.float(cfg.Config.prozA)
local Slider4 = new.float(cfg.Config.prozN)

Ошибку выдает на первую строчку тоглов:
[ML] (error) !VR VR VR.lua: C:\Users\User\Desktop\kichiro\moonloader\!VR VR VR.lua:86: cannot convert 'string' to 'bool [1]'
в чем моя ошибка? Это если файл ини удалить, то почему то он не создает в его отсутствие новый и крашит скрипт просто
Без кавычек пиши true false в первых строчках
 
  • Влюблен
Реакции: Julimba