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

#Northn

Pears Project — уже запущен!
Всефорумный модератор
2,657
2,549
text = 'Склад FBI: 15000/50000'
не прокатит ибо вместо FBI может быть LSPD а вместо 15000 любое другое число до 50000
Я так понимаю, что тебе нужно получать эти строки из чата? Если да - кинь эти строки ПРЯМО ИЗ ЧАТЛОГА сюда.
----------------------------------------
Lua:
function main()
  repeat wait(0) until isSampAvailable()
wait(2000)
sampRegisterChatCommand('whois', whois)
while true do
  wait(0)
end
end

function whois(id)
  if id == "" then
    sampAddChatMessage("{FFCD0A}[WhoIs]: {FFFFFF}Используйте:{FFCD0A}/whois [PlayerId]", -1)
local result = sampIsPlayerConnected(id)
if not result then
sampAddChatMessage("{FFCD0A}[WhoIs]: {FFFFFF}Игрока нет на сервере!", -1)
else
  local nick = sampGetPlayerNickname(id)
  local result = sampIsPlayerNpc(id)
  if result then
    npc = 'Да'
  else
    npc = 'Нет'
  end
  local result = sampIsPlayerPaused(id)
if result then
  afk = 'Да'
else
  afk = 'Нет'
end
sampAddChatMessage("{FFCD0A}[WhoIs]: {FFFFFF}Ник: :{FFCD0A}"..nick..'{FFFFFF}, ID: {FFCD0A}'..id..'{FFFFFF}, NPC: {FFCD0A}'..npc..'{FFFFFF}, AFK: {FFCD0A}' .. afk ..
'{FFFFFF}.', -1)
while true do
wait(40)
if isKeyDown(17) and isKeyDown(82) then -- CTRL+R
  while isKeyDown(17) and isKeyDown(82) do wait(80) end
  reloadScripts()
end
end
end
end
end
Почему когда ввожу /whois без id просто /whois оно выдаёт инфу по 0 id
А когда ввожу допустим /whois 1 оно вообще не реагирует, вроде всё правильно написано. В чем проблема?

Lua:
function main()
  repeat wait(0) until isSampAvailable()
    wait(2000)
    sampRegisterChatCommand('whois', whois)
    while true do
        wait(0)
    end
end

function whois(id)
    local id = string.match(id, '%s*(.+)')
    if id == nil then
        sampAddChatMessage("{FFCD0A}[WhoIs]: {FFFFFF}Используйте:{FFCD0A}/whois [PlayerId]", -1)
    else
        if not sampIsPlayerConnected(id) then
            sampAddChatMessage("{FFCD0A}[WhoIs]: {FFFFFF}Игрока нет на сервере!", -1)
        else
            local nick = sampGetPlayerNickname(id)
            if sampIsPlayerNpc(id) then
                npc = 'Да'
            else
                npc = 'Нет'
            end
            if sampIsPlayerPaused(id) then
                afk = 'Да'
            else
                afk = 'Нет'
            end
            sampAddChatMessage("{FFCD0A}[WhoIs]: {FFFFFF}Ник: :{FFCD0A}"..nick..'{FFFFFF}, ID: {FFCD0A}'..id..'{FFFFFF}, NPC: {FFCD0A}'..npc..'{FFFFFF}, AFK: {FFCD0A}' .. afk ..'{FFFFFF}.', -1)
        end
    end
end
 

dmitri4

Известный
452
79
Я так понимаю, что тебе нужно получать эти строки из чата? Если да - кинь эти строки ПРЯМО ИЗ ЧАТЛОГА сюда.
----------------------------------------

Код:
[21:40:49] Встаньте на красный маркер, чтобы заполнить грузовик материалами!
[21:40:51] Материалы успешно загружены! Материалов: 5000/5000
[21:40:51] Введите: /carm, чтобы начать развозку материалов
[21:41:02] Метка на карте указана в виде красного маркера
[21:41:06] Материалы успешно доставлены! Материалов: 0/5000
[21:41:06] Склад FBI: 25000/50000
Нужны значения FBI и 25000 ибо место FBI может быть любая другая фракция так же и склад
 

#Northn

Pears Project — уже запущен!
Всефорумный модератор
2,657
2,549
Код:
[21:40:49] Встаньте на красный маркер, чтобы заполнить грузовик материалами!
[21:40:51] Материалы успешно загружены! Материалов: 5000/5000
[21:40:51] Введите: /carm, чтобы начать развозку материалов
[21:41:02] Метка на карте указана в виде красного маркера
[21:41:06] Материалы успешно доставлены! Материалов: 0/5000
[21:41:06] Склад FBI: 25000/50000
Нужны значения FBI и 25000 ибо место FBI может быть любая другая фракция так же и склад
Lua:
local sampev = require 'lib.samp.events'

function main() wait(-1) end

function sampev.onServerMessage(color, text)
    if text:find('Склад .+: %d+/%d+') then
        local fraction, summ = text:match('Склад (.+): (%d+)/%d+')
        -- Далее ты можешь делать всё, что захочешь, в fraction - название фракции, в summ - количество бокприпасов в складе.
    end
end

Для работы требуется SAMP.Lua(https://blast.hk/threads/14624/)
 
  • Нравится
Реакции: dmitri4 и asocial demon

asocial demon

Потрачен
175
14
Обратите внимание, пользователь заблокирован на форуме. Не рекомендуется проводить сделки.
@#Northn прошу у тебя как более прошаренного, так как тут никто не смог дать ответ.
Есть какие-либо примеры работы с синхрой и отправкой RPC, так как ничего найти не смог(
 

#Northn

Pears Project — уже запущен!
Всефорумный модератор
2,657
2,549
@#Northn прошу у тебя как более прошаренного, так как тут никто не смог дать ответ.
Есть какие-либо примеры работы с синхрой и отправкой RPC, так как ничего найти не смог(
Первый вариант - работать с SAMP.Lua(https://blast.hk/threads/14624/)
Второе - изучить работу RakNet'a и создавать свои функции. Информация - Гайд - Работа с RakNet хуками с помощью SAMPFUNCS(https://blast.hk/threads/17440/)
 

wD.D159

Известный
Друг
446
480
Помогите составить регулярку на LUA для сообщения
Код:
** Жалоба от Daniel_Raven™(115) на plevo4ek(61): текст

+ нужно узнать ID на которого подана жалоба.
 

#Northn

Pears Project — уже запущен!
Всефорумный модератор
2,657
2,549
Помогите составить регулярку на LUA для сообщения
Код:
** Жалоба от Daniel_Raven™(115) на plevo4ek(61): текст

+ нужно узнать ID на которого подана жалоба.
Lua:
local sampev = require 'lib.samp.events'

function main() wait(-1) end

function sampev.onServerMessage(color, text)
    if text:find('%*%* Жалоба от .+%(%d+%) на .+%(%d+%): .+') then
        local ID1, ID2, text = text:match('%*%* Жалоба от .+%((%d+)%) на .+%((%d+)%): (.+)')
        --code
    end
end
Первый - ID подавшего, второй - ID того, на кого подали, третий - текст.
 
  • Нравится
Реакции: wD.D159

DeMoN3D

Известный
366
77
Приветствую всех, есть простенький код, который считает, сколько раз в чате появилась указанная фраза, но у меня возникли некоторые сложности с ним. При вводе /fly или /fly qwe, скрипт должен писать, что формат не верный, но этого не происходит и он пишет "Активирован". Но даже при вводе /fly 1 или /fly 2 ничего не происходит. Он так же пишет, что активирован, но не считает. Ошибок в логе нет, кроме как насчет param >=3 , насчет этого ошибка следующая: [ML] (error) fly.lua: E:\GTA San Andreas\moonloader\fly.lua:21: attempt to compare string with number. Я знаю, что она означает, но не знаю, как она решается, поэтому оставил для наглядности задумки. Буду благодарен, если кто поможет. Вот код:
Lua:
-- Библиотеки
require "lib.moonloader"
require "lib.sampfuncs"
local hook = require('lib.samp.events')
    local r = 0
    local k = 0
    local z = 0
    local ms = nil
function main()
    while not isSampfuncsLoaded() or not isSampLoaded() do wait(1000) end
    while not isSampAvailable() do wait(5000) end
    sampRegisterChatCommand('fly', elactive)
    sampRegisterChatCommand('cfly', cmd_cfly)
    while true do wait(-1)
    end
end



function elactive(param)
    if param == nil or param >=3 then
        sampAddChatMessage("введи 1 или 2", -1)
        return
    end
    ms = param
    sampAddChatMessage("FlyMat: "..ms, -1)
    elactiv = not elactiv
    sampAddChatMessage("FlyMat: ".. (elactiv and "Активирован." or "Деактивирован."), - 1)
end

function hook.onServerMessage(color,text)
    if text:find('Высадка пассажиров') and elactiv and ms == 1 then
        r = r+1 --  action
        k = math.floor(r/2)
        if ms == 1 then
           z = r*1449
     elseif ms == 2 then
           z = math.floor(r*1165.5)
        end
        sampAddChatMessage("Количество рейсов: {35bf4d}"..r, 0xFFFFFF)
        sampAddChatMessage("Количество кругов: {35bf4d}"..k, 0xFFFFFF)
        sampAddChatMessage("Примерная зарплата: {35bf4d}"..z.."$", 0xFFFFFF)
    end
end

function cmd_cfly()
    r = 0
    k = 0
    z = 0
    printStringNow("Clear!", 3000)
end
 

#Northn

Pears Project — уже запущен!
Всефорумный модератор
2,657
2,549
Приветствую всех, есть простенький код, который считает, сколько раз в чате появилась указанная фраза, но у меня возникли некоторые сложности с ним. При вводе /fly или /fly qwe, скрипт должен писать, что формат не верный, но этого не происходит и он пишет "Активирован". Но даже при вводе /fly 1 или /fly 2 ничего не происходит. Он так же пишет, что активирован, но не считает. Ошибок в логе нет, кроме как насчет param >=3 , насчет этого ошибка следующая: [ML] (error) fly.lua: E:\GTA San Andreas\moonloader\fly.lua:21: attempt to compare string with number. Я знаю, что она означает, но не знаю, как она решается, поэтому оставил для наглядности задумки. Буду благодарен, если кто поможет. Вот код:
Lua:
-- Библиотеки
require "lib.moonloader"
require "lib.sampfuncs"
local hook = require('lib.samp.events')
    local r = 0
    local k = 0
    local z = 0
    local ms = nil
function main()
    while not isSampfuncsLoaded() or not isSampLoaded() do wait(1000) end
    while not isSampAvailable() do wait(5000) end
    sampRegisterChatCommand('fly', elactive)
    sampRegisterChatCommand('cfly', cmd_cfly)
    while true do wait(-1)
    end
end



function elactive(param)
    if param == nil or param >=3 then
        sampAddChatMessage("введи 1 или 2", -1)
        return
    end
    ms = param
    sampAddChatMessage("FlyMat: "..ms, -1)
    elactiv = not elactiv
    sampAddChatMessage("FlyMat: ".. (elactiv and "Активирован." or "Деактивирован."), - 1)
end

function hook.onServerMessage(color,text)
    if text:find('Высадка пассажиров') and elactiv and ms == 1 then
        r = r+1 --  action
        k = math.floor(r/2)
        if ms == 1 then
           z = r*1449
     elseif ms == 2 then
           z = math.floor(r*1165.5)
        end
        sampAddChatMessage("Количество рейсов: {35bf4d}"..r, 0xFFFFFF)
        sampAddChatMessage("Количество кругов: {35bf4d}"..k, 0xFFFFFF)
        sampAddChatMessage("Примерная зарплата: {35bf4d}"..z.."$", 0xFFFFFF)
    end
end

function cmd_cfly()
    r = 0
    k = 0
    z = 0
    printStringNow("Clear!", 3000)
end
Lua:
-- Библиотеки
require "lib.moonloader"
require "lib.sampfuncs"
local hook = require('lib.samp.events')
local r = 0
local k = 0
local z = 0
local ms = nil
function main()
    while not isSampfuncsLoaded() or not isSampLoaded() do wait(1000) end
    while not isSampAvailable() do wait(50) end
    sampRegisterChatCommand('fly', elactive)
    sampRegisterChatCommand('cfly', cmd_cfly)
    wait(-1)
end

function elactive(param)
    local param = string.match(param, '(%d+)')
    if param == nil or param >=3 or param <= 0 then
        sampAddChatMessage("введи 1 или 2", -1)
        return
    end
    ms = param
    sampAddChatMessage("FlyMat: "..ms, -1)
    elactiv = not elactiv
    sampAddChatMessage("FlyMat: ".. (elactiv and "Активирован." or "Деактивирован."), - 1)
end

function hook.onServerMessage(color,text)
    if text:find('Высадка пассажиров') and elactiv and ms == 1 then
        r = r+1 --  action
        k = math.floor(r/2)
        if ms == 1 then
            z = r*1449
        elseif ms == 2 then
            z = math.floor(r*1165.5)
        end
        sampAddChatMessage("Количество рейсов: {35bf4d}"..r, 0xFFFFFF)
        sampAddChatMessage("Количество кругов: {35bf4d}"..k, 0xFFFFFF)
        sampAddChatMessage("Примерная зарплата: {35bf4d}"..z.."$", 0xFFFFFF)
    end
end

function cmd_cfly()
    r = 0
    k = 0
    z = 0
    printStringNow("Clear!", 3000)
end
 

DeMoN3D

Известный
366
77
Lua:
-- Библиотеки
require "lib.moonloader"
require "lib.sampfuncs"
local hook = require('lib.samp.events')
local r = 0
local k = 0
local z = 0
local ms = nil
function main()
    while not isSampfuncsLoaded() or not isSampLoaded() do wait(1000) end
    while not isSampAvailable() do wait(50) end
    sampRegisterChatCommand('fly', elactive)
    sampRegisterChatCommand('cfly', cmd_cfly)
    wait(-1)
end

function elactive(param)
    local param = string.match(param, '(%d+)')
    if param == nil or param >=3 or param <= 0 then
        sampAddChatMessage("введи 1 или 2", -1)
        return
    end
    ms = param
    sampAddChatMessage("FlyMat: "..ms, -1)
    elactiv = not elactiv
    sampAddChatMessage("FlyMat: ".. (elactiv and "Активирован." or "Деактивирован."), - 1)
end

function hook.onServerMessage(color,text)
    if text:find('Высадка пассажиров') and elactiv and ms == 1 then
        r = r+1 --  action
        k = math.floor(r/2)
        if ms == 1 then
            z = r*1449
        elseif ms == 2 then
            z = math.floor(r*1165.5)
        end
        sampAddChatMessage("Количество рейсов: {35bf4d}"..r, 0xFFFFFF)
        sampAddChatMessage("Количество кругов: {35bf4d}"..k, 0xFFFFFF)
        sampAddChatMessage("Примерная зарплата: {35bf4d}"..z.."$", 0xFFFFFF)
    end
end

function cmd_cfly()
    r = 0
    k = 0
    z = 0
    printStringNow("Clear!", 3000)
end
Когда я ничего не пишу после /fly он мне пишет, мол введи 1 или 2, это работает.
но если я пишу /fly 1 или /fly 2, то случается следующая ошибка:

[ML] (error) fly.lua: E:\GTA San Andreas\moonloader\fly.lua:19: attempt to compare string with number
stack traceback:
E:\GTA San Andreas\moonloader\fly.lua:19: in function <E:\GTA San Andreas\moonloader\fly.lua:17>
[ML] (error) fly.lua: Script died due to an error. (0F38CFEC)
 

TheGood11

Участник
173
10
Есть скрипт активация которого на CapsLock, делаю вот так, но он не активируется (кнопка должна быть зажата)
Lua:
local sampev = require 'lib.samp.events'

function main()
    while not isSampAvailable() do wait(100) end
    sampRegisterChatCommand('capslock', capslock)
    while true do wait(0)
        if act then setGameKeyState(20, -128) end
    end
end

function capslock() act = not act end
 

Ken Block

Известный
432
31
Когда я ничего не пишу после /fly он мне пишет, мол введи 1 или 2, это работает.
но если я пишу /fly 1 или /fly 2, то случается следующая ошибка:

[ML] (error) fly.lua: E:\GTA San Andreas\moonloader\fly.lua:19: attempt to compare string with number
stack traceback:
E:\GTA San Andreas\moonloader\fly.lua:19: in function <E:\GTA San Andreas\moonloader\fly.lua:17>
[ML] (error) fly.lua: Script died due to an error. (0F38CFEC)
Lua:
require "lib.moonloader"
require "lib.sampfuncs"
local hook = require('lib.samp.events')
    local r = 0
    local k = 0
    local z = 0
    local ms = 0
elactiv = false
function main()
    while not isSampfuncsLoaded() or not isSampLoaded() do wait(1000) end
    while not isSampAvailable() do wait(5000) end
    sampRegisterChatCommand('fly', elactive)
    sampRegisterChatCommand('cfly', cmd_cfly)
    while true do wait(-1)
    end
end



function elactive(param)
    if param == '' or param == nil then
        if param == 1 or param == 2 then ms = param
sampAddChatMessage("FlyMat: "..ms, -1)
end
    else
    if elactiv then elactiv = false else elactiv = true end
end
end

function hook.onServerMessage(color,text)
    if text:find('Высадка пассажиров') and elactiv then
        r = r+1 --  action
        k = math.floor(r/2)
        if ms == 1 then
           z = r*1449
     elseif ms == 2 then
           z = math.floor(r*1165.5)
        end
        sampAddChatMessage("Количество рейсов: {35bf4d}"..r, 0xFFFFFF)
        sampAddChatMessage("Количество кругов: {35bf4d}"..k, 0xFFFFFF)
        sampAddChatMessage("Примерная зарплата: {35bf4d}"..z.."$", 0xFFFFFF)
    end
end

function cmd_cfly()
    r = 0
    k = 0
    z = 0
    printStringNow("Clear!", 3000)
end
Мейби сработает. Писал с телефона так шо код кривой. Если сработает, то ошибка возможно была в том, что ты не дописал elactiv после загрузки библиотек

Ну да, после того как я пропишу команду /capslock, то должен зажаться CapsLock, он может и зажимается, но действие другого скрипта, не происходит.
Ну так делай ахк скрипт. На луа ты такого не сделаешь (но я хз)
 

TheGood11

Участник
173
10
Lua:
require "lib.moonloader"
require "lib.sampfuncs"
local hook = require('lib.samp.events')
    local r = 0
    local k = 0
    local z = 0
    local ms = 0
elactiv = false
function main()
    while not isSampfuncsLoaded() or not isSampLoaded() do wait(1000) end
    while not isSampAvailable() do wait(5000) end
    sampRegisterChatCommand('fly', elactive)
    sampRegisterChatCommand('cfly', cmd_cfly)
    while true do wait(-1)
    end
end



function elactive(param)
    if param == '' or param == nil then
        if param == 1 or param == 2 then ms = param
sampAddChatMessage("FlyMat: "..ms, -1)
end
    else
    if elactiv then elactiv = false else elactiv = true end
end
end

function hook.onServerMessage(color,text)
    if text:find('Высадка пассажиров') and elactiv then
        r = r+1 --  action
        k = math.floor(r/2)
        if ms == 1 then
           z = r*1449
     elseif ms == 2 then
           z = math.floor(r*1165.5)
        end
        sampAddChatMessage("Количество рейсов: {35bf4d}"..r, 0xFFFFFF)
        sampAddChatMessage("Количество кругов: {35bf4d}"..k, 0xFFFFFF)
        sampAddChatMessage("Примерная зарплата: {35bf4d}"..z.."$", 0xFFFFFF)
    end
end

function cmd_cfly()
    r = 0
    k = 0
    z = 0
    printStringNow("Clear!", 3000)
end
Мейби сработает. Писал с телефона так шо код кривой. Если сработает, то ошибка возможно была в том, что ты не дописал elactiv после загрузки библиотек


Ну так делай ахк скрипт. На луа ты такого не сделаешь (но я хз)

У меня нету АХК, по этому и хочу на ЛУА)