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

chapo

tg/inst: @moujeek
Всефорумный модератор
9,238
12,672
как открыть определенный пункт из меню паузы?
 

Shepi

Активный
177
36
Lua:
local rkeys = require 'rkeys'
imgui.HotKey = require('imgui_addons').HotKey

local tLastKeys = {}
local bindText = {v = {VK_R}} -- Тут клавиша наша. Буква R.

-- main
cmdbind = rkeys.registerHotKey(bindText.v, true, function()
    sampAddChatMessage("Тест", -1) -- Этот код выполнится при нажатии на забинженную кнопку в ХотКее.
end)

-- OnDrawFrame
if imgui.HotKey("##", bindText, tLastKeys, 100) then -- Сам ХотКей. Первый аргумент - это не текст рядом, а текст внутри ХотКея.
    rkeys.changeHotKey(cmdbind, bindText.v)
end
imgui.SameLine()
imgui.Text("- HotKey")
Посмотреть вложение 114283


По этому примеру сделай, там работает всё.
Окей да, спасибо, работает, но такой вопрос. Если я захочу сделать в одном меню допустим CollapsingHeader, в нем к примеру кнопку. Если я так сделаю, то кнопка будет отображаться под чайлдом, как сделать так, что-бы была под хеадером?
 

Dmitriy Makarov

25.05.2021
Проверенный
2,512
1,139
Окей да, спасибо, работает, но такой вопрос. Если я захочу сделать в одном меню допустим CollapsingHeader, в нем к примеру кнопку. Если я так сделаю, то кнопка будет отображаться под чайлдом, как сделать так, что-бы была под хеадером?
Будет отображаться или отображается? ;0
В таких случаях imgui.SameLine() должен помочь. До проверки menu пропиши imgui.SameLine(). Но по идее не должно такого быть.
 

sep

Известный
714
79
#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
 

McLore

Известный
564
284
#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

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

: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,997
1,339
что теперь не так ?

: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 · Просмотры: 54

Domino

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

Вложения

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

Pelmeska

Известный
934
229
Приветствую. Подскажите, пожалуйста, как сделать задержку, чтобы кидало инвайт раз в 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

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