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

McLore

Известный
559
279
#imring из за чего не работает ? что я не так вставил ? делал из этой темы https://www.blast.hk/threads/21076/post-197574
31: '<eof>' expected near 'end'

код:
require "lib.moonloader"
local sf = require 'sampfuncs'
local encoding = require 'encoding'
local sampev = require 'lib.samp.events'

site = 'https://pastebin.com/raw/ZiJ9Xi3e'

function main()
    while not isSampAvailable() do wait(0) end
    while sampGetCurrentServerName() == 'SA-MP' do wait(0) end
    local users = getTableUsersByUrl(site) -- узнаём таблицу списка.
    local _, myid = sampGetPlayerIdByCharHandle(playerPed) -- Узнаём свой ид.
    if not isAvailableUser(users, sampGetPlayerNickname(myid)) then -- Если срок уже прошёл или в списке нету моего ника, то..
        print('The term is ended or your name is not in the list.')
        thisScript():unload() -- Выгружаем скрипт.
    end
    wait(-1)
end

-- место под команды

     while true do
      wait(0)

  
     end
end


function getTableUsersByUrl(url)
    local n_file, bool, users = os.getenv('TEMP')..os.time(), false, {}
    downloadUrlToFile(url, n_file, function(id, status)
        if status == 6 then bool = true end
    end)
    while not doesFileExist(n_file) do wait(0) end
    if bool then
        local file = io.open(n_file, 'r')
        for w in file:lines() do
            local n, d = w:match('(.*): (.*)')
            users[#users+1] = { name = n, date = d }
        end
        file:close()
        os.remove(n_file)
    end
    return bool, users
end

function isAvailableUser(users, name)
    for i, k in pairs(users) do
        if k.name == name then
            local d, m, y = k.date:match('(%d+)%.(%d+)%.(%d+)')
            local time = {
                day = tonumber(d),
                isdst = true,
                wday = 0,
                yday = 0,
                year = tonumber(y),
                month = tonumber(m),
                hour = 0
            }
            if os.time(time) >= os.time() then return true end
        end
    end
    return false
end
Цикл в мейн перенеси или удали его
 
  • Нравится
Реакции: sep

sep

Известный
673
76
что теперь не так ?

:42: bad argument #1 to 'pairs' (table expected, got boolean)
stack traceback:
[C]: in function 'pairs'
42: in function 'isAvailableUser'
14: in function <D:\games\GTA\moonloader\ïïïïïïïï.lua:8>
Script died due to an error. (3FAEF00C)

код:
site = 'https://pastebin.com/raw/ZiJ9Xi3e'
function main()
    while not isSampAvailable() do wait(0) end
    while sampGetCurrentServerName() == 'SA-MP' do wait(0) end
    local users = getTableUsersByUrl(site) -- узнаём таблицу списка.
    local _, myid = sampGetPlayerIdByCharHandle(playerPed) -- Узнаём свой ид.
    if not isAvailableUser(users, sampGetPlayerNickname(myid)) then -- Если срок уже прошёл или в списке нету моего ника, то..
        sampAddChatMessage("bb", 0x00ff00)
        thisScript():unload() -- Выгружаем скрипт.
    end
    wait(-1)
end


function getTableUsersByUrl(url)
    local n_file, bool, users = os.getenv('TEMP')..os.time(), false, {}
    downloadUrlToFile(url, n_file, function(id, status)
        if status == 6 then bool = true end
    end)
    while not doesFileExist(n_file) do wait(0) end
    if bool then
        local file = io.open(n_file, 'r')
        for w in file:lines() do
            local n, d = w:match('(.*): (.*)')
            users[#users+1] = { name = n, date = d }
        end
        file:close()
        os.remove(n_file)
    end
    return bool, users
end

function isAvailableUser(users, name)
    for i, k in pairs(users) do
        if k.name == name then
            local d, m, y = k.date:match('(%d+)%.(%d+)%.(%d+)')
            local time = {
                day = tonumber(d),
                isdst = true,
                wday = 0,
                yday = 0,
                year = tonumber(y),
                month = tonumber(m),
                hour = 0
            }
            if os.time(time) >= os.time() then return true end
        end
    end
    return false
end
 

paulohardy

вы еще постите говно? тогда я иду к вам
Всефорумный модератор
1,891
1,254
что теперь не так ?

:42: bad argument #1 to 'pairs' (table expected, got boolean)
stack traceback:
[C]: in function 'pairs'
42: in function 'isAvailableUser'
14: in function <D:\games\GTA\moonloader\ïïïïïïïï.lua:8>
Script died due to an error. (3FAEF00C)

код:
site = 'https://pastebin.com/raw/ZiJ9Xi3e'
function main()
    while not isSampAvailable() do wait(0) end
    while sampGetCurrentServerName() == 'SA-MP' do wait(0) end
    local users = getTableUsersByUrl(site) -- узнаём таблицу списка.
    local _, myid = sampGetPlayerIdByCharHandle(playerPed) -- Узнаём свой ид.
    if not isAvailableUser(users, sampGetPlayerNickname(myid)) then -- Если срок уже прошёл или в списке нету моего ника, то..
        sampAddChatMessage("bb", 0x00ff00)
        thisScript():unload() -- Выгружаем скрипт.
    end
    wait(-1)
end


function getTableUsersByUrl(url)
    local n_file, bool, users = os.getenv('TEMP')..os.time(), false, {}
    downloadUrlToFile(url, n_file, function(id, status)
        if status == 6 then bool = true end
    end)
    while not doesFileExist(n_file) do wait(0) end
    if bool then
        local file = io.open(n_file, 'r')
        for w in file:lines() do
            local n, d = w:match('(.*): (.*)')
            users[#users+1] = { name = n, date = d }
        end
        file:close()
        os.remove(n_file)
    end
    return bool, users
end

function isAvailableUser(users, name)
    for i, k in pairs(users) do
        if k.name == name then
            local d, m, y = k.date:match('(%d+)%.(%d+)%.(%d+)')
            local time = {
                day = tonumber(d),
                isdst = true,
                wday = 0,
                yday = 0,
                year = tonumber(y),
                month = tonumber(m),
                hour = 0
            }
            if os.time(time) >= os.time() then return true end
        end
    end
    return false
end
передаешь в функцию isAvailableUser переменную users, которая является булевой, а не таблицей
на 5 строке замени local users = getTableUsersByUrl(site) на local bool, users = getTableUsersByUrl(site)
 
  • Нравится
Реакции: sep

bratik026

Новичок
13
0
Как называются эти три поля со значениями в imgui?
 

Вложения

  • BF23F0AE-BCC6-4004-B9DD-0FC66F6D1F7C.jpeg
    BF23F0AE-BCC6-4004-B9DD-0FC66F6D1F7C.jpeg
    47 KB · Просмотры: 23

Domino

Участник
326
15
Приветствую. Подскажите, пожалуйста, как сделать задержку, чтобы кидало инвайт раз в 500мс. Везде ставил wait, не робит(
 

Вложения

  • rfam_1 (1).lua
    1,012 байт · Просмотры: 8

Pelmeska

Известный
925
234
Приветствую. Подскажите, пожалуйста, как сделать задержку, чтобы кидало инвайт раз в 500мс. Везде ставил wait, не робит(
Lua:
script_name('Версия 0.1')
script_author('Автор Adam_Karleone')
script_description('rfam')
require "lib.moonloader"

function main()
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(2000) end
    sampAddChatMessage('Автор скрипта Adam_Karleone',0xFFFF00)
    sampAddChatMessage('Активация скрипта /rfam',0xFFFF00)
    sampRegisterChatCommand('rfam', rfam)
    while true do
        wait(0)
    end
end

function rfam()

    local peds = getAllChars()
    for _, v in pairs(peds) do
  
        local result, myid = sampGetPlayerIdByCharHandle(playerPed)
        local mx, my, mz = getCharCoordinates(playerPed)
        local x, y, z = getCharCoordinates(v)
        local distance = getDistanceBetweenCoords3d(mx, my, mz, x, y, z)
        local result, id = sampGetPlayerIdByCharHandle(v)
        if result and id ~= sampGetPlayerIdByCharHandle(PLAYER_PED) and distance < 12.0 then
      
            if id ~= myid then
          
            lua_thread.create(function() -- создаем скриптовой поток чтоб использовать wait() вне функции main
            while true do -- делаем бесконечный цикл
            sampSendChat('/faminvite '..tonumber(id)) -- функция которая будет выполнятся в бесконечном цикле
            wait(500) -- задержка для бесконечного цикла в миллисекундах.
                    end
                end)   
            end
        end
    end
end
 
  • Нравится
Реакции: Domino

dinky

Участник
67
20
1631793036826.png
как можно скрыть значек north , не скрывая радар и метки с него
 

chapo

🫡 В армии с 17.10.2023. В ЛС НЕ ОТВЕЧАЮ
Друг
8,775
11,218
Почему consume window message с true в обоих параметрах не блокирует закрытие диалога и чата через esc?
 

barjik

Известный
464
190
Как вывести подсчет?
Например: считает до 150, прибавляя 1+1.
Нужно чтоб каждый раз при подсчете выводил в чат: "1+1=2, 2+1=3 и т.д" все в строчку
code:
function aye()
lua_thread.create(function()
   for i = 1, 150, 1 do
      wait(500)
      sampAddChatMessage(i, -1)
        end
    end)
end
 

EclipsedFlow

Известный
Проверенный
1,038
462
Как записать раз в 3 секунды координаты своего персонажа?
 

chapo

🫡 В армии с 17.10.2023. В ЛС НЕ ОТВЕЧАЮ
Друг
8,775
11,218
Как записать раз в 3 секунды координаты своего персонажа?
Lua:
local coords = {x = 0, y = 0, z = 0}

function main()
    while not isSampAvailable() do wait(0) end
    saveCoords()
    while true do
        wait(0)
        
    end
end

function saveCoords()
    lua_thread.create(function()
        while true do
            wait(3000)
            coords.x, coords.y, coords.z = getCharCoordinates(PLAYER_PED)
        end
    end)
end
 
  • Нравится
Реакции: EclipsedFlow

Dmitriy Makarov

25.05.2021
Проверенный
2,478
1,113
Как вывести подсчет?
Например: считает до 150, прибавляя 1+1.
Нужно чтоб каждый раз при подсчете выводил в чат: "1+1=2, 2+1=3 и т.д" все в строчку
code:
function aye()
lua_thread.create(function()
   for i = 1, 150, 1 do
      wait(500)
      sampAddChatMessage(i, -1)
        end
    end)
end
Записать в переменную, наверное.
Lua:
a, b, c = 1, 10, 1
for i = a, b, c do
    if i < b then
        print(string.format("%d + %d = %d", i, c, i+1))
    end
end
1631813595687.png


Как называются эти три поля со значениями в imgui?
ColorEdit.
Lua:
local color = imgui.ImFloat4(0, 0, 0, 255) -- В начало. R, G, B, A (Вроде.)

-- OnDrawFrame
imgui.ColorEdit4("Color edit", color)

Screenshot_3.png
 
Последнее редактирование:
  • Нравится
Реакции: barjik

barjik

Известный
464
190
Записать в переменную, наверное.
Lua:
a, b, c = 1, 10, 1
for i = a, b, c do
    if i < b then
        print(string.format("%d + %d = %d", i, c, i+1))
    end
end
Посмотреть вложение 114470


ColorEdit.
Lua:
local color = imgui.ImFloat4(0, 0, 0, 255) -- В начало. R, G, B, A (Вроде.)

-- OnDrawFrame
imgui.ColorEdit4("Color edit", color)

Посмотреть вложение 114473
В отрицательную сторону считать можно?
Чет пробовал, не палучилос
 
Последнее редактирование:
  • Bug
Реакции: scroll.