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

Youngever

Новичок
22
1
Хотелось бы что бы скрипт выполнял эту функцию с моим актёром. Как это сделать? Я глупый.
Код:
local car = storeCarCharIsInNoSave(PLAYER_PED)
    result, s = sampGetVehicleIdByCarHandle(car)
     BS = raknetNewBitStream()
     raknetBitStreamWriteInt32(BS, s)
     raknetBitStreamWriteInt32(BS, -1)
     raknetBitStreamWriteInt32(BS, -1)
     raknetBitStreamWriteInt32(BS, 4)
     raknetSendRpc(96, BS)
     raknetDeleteBitStream(BS)
 

Anton Nixon

Активный
474
48
Вот рабочий варик, но как по мне лучше в диалог записывать админов, а не в чат...
Lua:
sampRegisterChatCommand("checker", function()
    local admins = {} -- создание пустого массива
    file = io.open(getGameDirectory() .. "\\moonloader\\config\\checker\\checker.ini", "r+")
    for line in file:lines() do
        local nick, lvl = line:match('(.+) (%d+)')
        admins[#admins+1] = {nick = nick, lvl = lvl} -- заполнение массива
    end
    file:close()
    table.sort(admins, function(a, b) return tonumber(a.lvl) > tonumber(b.lvl) end) -- сортировка списка по лвл админки
    for k, v in pairs(admins) do
        sampAddChatMessage(v.nick..' ('..v.lvl..')', color_checker.admin[tonumber(v.lvl)]) -- вывод в чат
    end
end)
Переделал код под рендер, сейчас выводит только последнюю строчку из файла, как сделать чтобы все выводилось?
Содержимое ini файла:
Код:
Anton_Nixon 6
Petya_Vasechkin 5
Ivan_Ivanov 4
Sasha_Petrov 3
Pavel_Tereshkin 2
Dasha_Kirushina 1
Итог, как оно выходит:
sa-mp-002.png

Сам код:
Lua:
require "lib.moonloader"
require "lib.sampfuncs"
local inicfg = require 'inicfg'
local encoding = require 'encoding'
encoding.default = 'CP1251'
u8 = encoding.UTF8
local key = require 'vkeys'

local show_check = false

local color_checker = {
    admin = {
        [1]="2dbae0",
        [2]="2dbae0",
        [3]="8612b8",
        [4]="2e4ebf",
        [5]="34eb5b",
        [6]="009122",
    },
    leader = {
        [1]="0xe5ff00",
        [2]="0x0011ff",
        [3]="0x943915",
        [4]="0xff54f1",
        [5]="0xff7817",
        [6]="0x007a0e",
        [7]="0xd80fdb",
        [8]="0xf2c200",
        [9]="0x8f2df7",
        [10]="0x02cccc",
        [11]="0x2e825c",
        [12]="0xc4007c",
        [13]="0xcf0000",
    }
}

function main()
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end
    initializeRender()
    sampAddChatMessage("чекер загружен", -1)
    if not doesDirectoryExist('moonloader\\config\\checker') then createDirectory('moonloader\\config\\checker') end
    sampRegisterChatCommand("checker", function()
       show_check = not show_check
    end)

    while true do
        
        if show_check then
            local admins = {} -- создание пустого массива
            file = io.open(getGameDirectory() .. "\\moonloader\\config\\checker\\checker.ini", "r+")
            for line in file:lines() do
                local nick, lvl = line:match('(.+) (%d+)')
                admins[#admins+1] = {nick = nick, lvl = lvl} -- заполнение массива
            end
            file:close()
            table.sort(admins, function(a, b) return tonumber(a.lvl) > tonumber(b.lvl) end) -- сортировка списка по лвл админки
            for k, v in pairs(admins) do
                --sampAddChatMessage(v.nick..' ('..v.lvl..')', color_checker.admin[tonumber(v.lvl)]) -- вывод в чат
                AdminText = "Администраторы в сети:\n"
                for b = 0, 1001 do
                    if sampIsPlayerConnected(b) then
                        name = sampGetPlayerNickname(b)
                    end
                    adminStreamed = sampGetCharHandleBySampPlayerId(b)
                    adminPaused = sampIsPlayerPaused(b)
                    for i = 1, #admins do
                        if adminPaused then
                            adminList = string.format("{%s}%d. %s[%d] {FF0000}AFK", color_checker.admin[tonumber(v.lvl)], i, v.nick, b) 
                        end
                        if adminStreamed then
                            adminList = string.format("{%s}%d. %s[%d] #", color_checker.admin[tonumber(v.lvl)], i, v.nick, b)
                        else
                            adminList = string.format("{%s}%d. %s[%d]", color_checker.admin[tonumber(v.lvl)], i, v.nick, b)
                        end
                        AdminText = string.format("%s %s\n", AdminText, adminList)
                    end
                    break
                end
            end
            renderFontDrawText(sfont, AdminText, 1120, 400, 0xffffffff)
        end
        wait(0)
    end
end

function initializeRender()
    sfont = renderCreateFont("Arial", 9, 13)
end
 
Последнее редактирование:

Mr_Incognito

Активный
331
50
Привет всем!
Как сделать вкладки на имгуи (типа как в хроме вкладки, если есть такое. вроде видел где то)?
 

G W

Участник
141
5
Как с серверного текстдрава получить информацию и вывести его на imgui ?
 

Anton Nixon

Активный
474
48
Почему не хочет отображать, что игрок в афк и почему ник выводит множество раз, а не один
sa-mp-003.png
код:
Lua:
function main()
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end
    initializeRender()
    sampAddChatMessage("чекер загружен", -1)
    if not doesDirectoryExist('moonloader\\config\\checker') then createDirectory('moonloader\\config\\checker') end
    sampRegisterChatCommand("checker", function()
       show_check = not show_check
    end)

    while true do
        wait(0)
        if show_check then
            local admins = {} -- создание пустого массива
            file = io.open(getGameDirectory() .. "\\moonloader\\config\\checker\\checker.ini", "r+")
            for line in file:lines() do
                local nick, lvl = line:match('(.+) (%d+)')
                admins[#admins+1] = {nick = nick, lvl = lvl} -- заполнение массива
            end
            file:close()
            table.sort(admins, function(a, b) return tonumber(a.lvl) > tonumber(b.lvl) end) -- сортировка списка по лвл админки
            for k, v in pairs(admins) do
                AdminText = "Администраторы в сети:\n"
                for b = 0, 1001 do
                    if sampIsPlayerConnected(b) then
                        name = sampGetPlayerNickname(b)
                        adminStreamed = sampGetCharHandleBySampPlayerId(b)
                        adminPaused = sampIsPlayerPaused(b)
                        if name == v.nick then
                            for i = 1, #admins do
                                if adminPaused and adminStreamed then
                                    adminList = string.format("{%s}%d. %s [%d] {FF0000}[AFK] {737373}[#]", color_checker.admin[tonumber(v.lvl)], i, v.nick, b)
                                else
                                    adminList = string.format("{%s}%d. %s [%d]", color_checker.admin[tonumber(v.lvl)], i, v.nick, b)  
                                end
                                if adminStreamed then
                                    adminList = string.format("{%s}%d. %s [%d] {737373}[#]", color_checker.admin[tonumber(v.lvl)], i, v.nick, b)
                                end
                                AdminText = string.format("%s %s\n", AdminText, adminList)
                            end
                            break
                        end
                    end
                end
                renderFontDrawText(sfont, AdminText, 1120, 400, 0xffffffff)
            end
        end
    end
end
 

Mr_Incognito

Активный
331
50
Где найти цвета в таком формате imgui.ImVec4(1.000, 0.992, 0.992, 0.670)?
Может ресурс есть какой то по подбору?
 

CaJlaT

Овощ
Модератор
2,808
2,619
Почему не хочет отображать, что игрок в афк и почему ник выводит множество раз, а не один
код:
Lua:
function main()
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end
    initializeRender()
    sampAddChatMessage("чекер загружен", -1)
    if not doesDirectoryExist('moonloader\\config\\checker') then createDirectory('moonloader\\config\\checker') end
    sampRegisterChatCommand("checker", function()
       show_check = not show_check
    end)

    while true do
        wait(0)
        if show_check then
            local admins = {} -- создание пустого массива
            file = io.open(getGameDirectory() .. "\\moonloader\\config\\checker\\checker.ini", "r+")
            for line in file:lines() do
                local nick, lvl = line:match('(.+) (%d+)')
                admins[#admins+1] = {nick = nick, lvl = lvl} -- заполнение массива
            end
            file:close()
            table.sort(admins, function(a, b) return tonumber(a.lvl) > tonumber(b.lvl) end) -- сортировка списка по лвл админки
            for k, v in pairs(admins) do
                AdminText = "Администраторы в сети:\n"
                for b = 0, 1001 do
                    if sampIsPlayerConnected(b) then
                        name = sampGetPlayerNickname(b)
                        adminStreamed = sampGetCharHandleBySampPlayerId(b)
                        adminPaused = sampIsPlayerPaused(b)
                        if name == v.nick then
                            for i = 1, #admins do
                                if adminPaused and adminStreamed then
                                    adminList = string.format("{%s}%d. %s [%d] {FF0000}[AFK] {737373}[#]", color_checker.admin[tonumber(v.lvl)], i, v.nick, b)
                                else
                                    adminList = string.format("{%s}%d. %s [%d]", color_checker.admin[tonumber(v.lvl)], i, v.nick, b) 
                                end
                                if adminStreamed then
                                    adminList = string.format("{%s}%d. %s [%d] {737373}[#]", color_checker.admin[tonumber(v.lvl)], i, v.nick, b)
                                end
                                AdminText = string.format("%s %s\n", AdminText, adminList)
                            end
                            break
                        end
                    end
                end
                renderFontDrawText(sfont, AdminText, 1120, 400, 0xffffffff)
            end
        end
    end
end
проблема в том, что ты делаешь цикл в цикле+ в 1 цикле у тебя постоянная перезапись текста:AdminText = "Администраторы в сети:\n"
Lua:
function main()
    if not isSampLoaded() and not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end
    sampRegisterChatCommand("checker", function()
       show_check = not show_check
    end)
    local admins = {} -- создание пустого массива
    file = io.open(getGameDirectory() .. "\\moonloader\\config\\checker\\checker.ini", "r+")
    for line in file:lines() do
        local nick, lvl = line:match('(.+) (%d+)')
        admins[#admins+1] = {nick = nick, lvl = lvl} -- заполнение массива
    end
    file:close()
    table.sort(admins, function(a, b) return tonumber(a.lvl) > tonumber(b.lvl) end) -- сортировка списка по лвл админки
    --лучше заполнять массив админов не в вечном цикле, а при инициализации скрипта
    while true do
        wait(0)
        if show_check then
            AdminText = "Администраторы в сети:\n"
            for k, v in pairs(admins) do
                if sampIsPlayerConnected(idbynick(v.nick)) then
                    adminStreamed = sampGetCharHandleBySampPlayerId(b)
                    adminPaused = sampIsPlayerPaused(b)
                    if adminStreamed then
                        if adminPaused then
                            adminList = string.format("{%s}%d. %s [%d] {FF0000}[AFK] {737373}[#]", color_checker.admin[tonumber(v.lvl)], k, v.nick, idbynick(v.nick))
                        else
                            adminList = string.format("{%s}%d. %s [%d] {737373}[#]", color_checker.admin[tonumber(v.lvl)], k, v.nick, idbynick(v.nick))
                        end
                    else
                        adminList = string.format("{%s}%d. %s [%d]", color_checker.admin[tonumber(v.lvl)], k, v.nick, idbynick(v.nick)) 
                    end
                    AdminText = string.format("%s %s\n", AdminText, adminList)
                end
            end
            renderFontDrawText(sfont, AdminText, 200, 400, 0xffffffff)
        end
    end
end
function idbynick(nick)
    for i = 0, sampGetMaxPlayerId() do
        if sampIsPlayerConnected(i) then
            if sampGetPlayerNickname(i) == nick then
                id = i
            end
        end
    end
    local i = id
    id = nil
    return i == nil and -1 or i
end
1592511564976.png
Взял себя рандомных игроков с сервера, дал рандомный лвл админки
Код:
Ladyboy 5
bluemoon 4
CaJlaT 6
 
  • Нравится
Реакции: McLore и Anton Nixon

Anton Nixon

Активный
474
48
проблема в том, что ты делаешь цикл в цикле+ в 1 цикле у тебя постоянная перезапись текста:AdminText = "Администраторы в сети:\n"
Lua:
function main()
    if not isSampLoaded() and not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end
    sampRegisterChatCommand("checker", function()
       show_check = not show_check
    end)
    local admins = {} -- создание пустого массива
    file = io.open(getGameDirectory() .. "\\moonloader\\config\\checker\\checker.ini", "r+")
    for line in file:lines() do
        local nick, lvl = line:match('(.+) (%d+)')
        admins[#admins+1] = {nick = nick, lvl = lvl} -- заполнение массива
    end
    file:close()
    table.sort(admins, function(a, b) return tonumber(a.lvl) > tonumber(b.lvl) end) -- сортировка списка по лвл админки
    --лучше заполнять массив админов не в вечном цикле, а при инициализации скрипта
    while true do
        wait(0)
        if show_check then
            AdminText = "Администраторы в сети:\n"
            for k, v in pairs(admins) do
                if sampIsPlayerConnected(idbynick(v.nick)) then
                    adminStreamed = sampGetCharHandleBySampPlayerId(b)
                    adminPaused = sampIsPlayerPaused(b)
                    if adminStreamed then
                        if adminPaused then
                            adminList = string.format("{%s}%d. %s [%d] {FF0000}[AFK] {737373}[#]", color_checker.admin[tonumber(v.lvl)], k, v.nick, idbynick(v.nick))
                        else
                            adminList = string.format("{%s}%d. %s [%d] {737373}[#]", color_checker.admin[tonumber(v.lvl)], k, v.nick, idbynick(v.nick))
                        end
                    else
                        adminList = string.format("{%s}%d. %s [%d]", color_checker.admin[tonumber(v.lvl)], k, v.nick, idbynick(v.nick))
                    end
                    AdminText = string.format("%s %s\n", AdminText, adminList)
                end
            end
            renderFontDrawText(sfont, AdminText, 200, 400, 0xffffffff)
        end
    end
end
function idbynick(nick)
    for i = 0, sampGetMaxPlayerId() do
        if sampIsPlayerConnected(i) then
            if sampGetPlayerNickname(i) == nick then
                id = i
            end
        end
    end
    local i = id
    id = nil
    return i == nil and -1 or i
end
Взял себя рандомных игроков с сервера, дал рандомный лвл админки
Код:
Ladyboy 5
bluemoon 4
CaJlaT 6
А как исправить тогда порядковый номер в списке
sa-mp-004.png
 

axxne

Новичок
2
0
Подскажите, как мне на луа закрыть игру при выходе в esc(афк) на 10 минут+ ?
 

trefa

Известный
Всефорумный модератор
2,097
1,233
Где найти цвета в таком формате imgui.ImVec4(1.000, 0.992, 0.992, 0.670)?
Может ресурс есть какой то по подбору?
rgb color picker + надо делить каждое значение на 256

проблема в том, что ты делаешь цикл в цикле+ в 1 цикле у тебя постоянная перезапись текста:AdminText = "Администраторы в сети:\n"
Lua:
function main()
    if not isSampLoaded() and not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end
    sampRegisterChatCommand("checker", function()
       show_check = not show_check
    end)
    local admins = {} -- создание пустого массива
    file = io.open(getGameDirectory() .. "\\moonloader\\config\\checker\\checker.ini", "r+")
    for line in file:lines() do
        local nick, lvl = line:match('(.+) (%d+)')
        admins[#admins+1] = {nick = nick, lvl = lvl} -- заполнение массива
    end
    file:close()
    table.sort(admins, function(a, b) return tonumber(a.lvl) > tonumber(b.lvl) end) -- сортировка списка по лвл админки
    --лучше заполнять массив админов не в вечном цикле, а при инициализации скрипта
    while true do
        wait(0)
        if show_check then
            AdminText = "Администраторы в сети:\n"
            for k, v in pairs(admins) do
                if sampIsPlayerConnected(idbynick(v.nick)) then
                    adminStreamed = sampGetCharHandleBySampPlayerId(b)
                    adminPaused = sampIsPlayerPaused(b)
                    if adminStreamed then
                        if adminPaused then
                            adminList = string.format("{%s}%d. %s [%d] {FF0000}[AFK] {737373}[#]", color_checker.admin[tonumber(v.lvl)], k, v.nick, idbynick(v.nick))
                        else
                            adminList = string.format("{%s}%d. %s [%d] {737373}[#]", color_checker.admin[tonumber(v.lvl)], k, v.nick, idbynick(v.nick))
                        end
                    else
                        adminList = string.format("{%s}%d. %s [%d]", color_checker.admin[tonumber(v.lvl)], k, v.nick, idbynick(v.nick))
                    end
                    AdminText = string.format("%s %s\n", AdminText, adminList)
                end
            end
            renderFontDrawText(sfont, AdminText, 200, 400, 0xffffffff)
        end
    end
end
function idbynick(nick)
    for i = 0, sampGetMaxPlayerId() do
        if sampIsPlayerConnected(i) then
            if sampGetPlayerNickname(i) == nick then
                id = i
            end
        end
    end
    local i = id
    id = nil
    return i == nil and -1 or i
end
Взял себя рандомных игроков с сервера, дал рандомный лвл админки
Код:
Ladyboy 5
bluemoon 4
CaJlaT 6
Ты бы подумал над оптимизацией, знаешь как у тебя ФПС просаживаться будут. Нормальные люди через хуки делают
 
Последнее редактирование: