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

imring

Ride the Lightning
Всефорумный модератор
2,362
2,545
Lua:
        if isKeyJustPressed(VK_I) and not sampIsChatInputActive() and not sampIsDialogActive() and not isPauseMenuActive() then
            if zakaz == true then
                thisid = id
                nick = sampGetPlayerNickname(id)
                moneyy = money
                local distance = sampGetDistanceLocalPlayerToPlayerByPlayerId(id)
                local color = string.format("%06X", ARGBtoRGB(sampGetPlayerColor(id)))
                local score = sampGetPlayerScore(id)
                local nick = nick:gsub('_', ' ')
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Ник: {"..color.."}"..nick.." {ffffff}[ {800000}"..id.." {ffffff}], LVL: {800000}"..score.."{ffffff}, расстояние: {800000}"..distance, -1)
                wait(300)
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Чтобы попросить выдачу Вам контракта нажмите {800000}P{ffffff}.", -1)
                wait(300)
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Чтобы отказаться от контракта нажмите {800000}O{ffffff}.", -1)
            else
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Новых заказов не поступало.", -1)
            end
        end

[16:36:15.130430] (error) Hitman Help: C:\Games\KEON GTA BY LONSY\moonloader\Hitman Help.lua:225: attempt to concatenate local 'bool' (a boolean value)
[16:36:15.131430] (error) Hitman Help: Script died due to error. (1009A6DC)
посмотри на пример, потом на свой код.
 

ZKelo

Известный
82
25
Нужно определить что игрок ввёл адрес сайта в диалог. Проверку, является ли введённая строка валидным адресом сайта, делаю с помощью регулярки.
1525172729.png

Вот сама регулярка работает, но в Lua всегда возвращается nil. Я что-то забыл?
Lua:
local url = "https://mysite.ru/"
local regex = "^(https?:%/%/(%w+%.)+%w+)%/?$"

print(url:match(regex)) -- nil
 

imring

Ride the Lightning
Всефорумный модератор
2,362
2,545
Нужно определить что игрок ввёл адрес сайта в диалог. Проверку, является ли введённая строка валидным адресом сайта, делаю с помощью регулярки.

Вот сама регулярка работает, но в Lua всегда возвращается nil. Я что-то забыл?
Lua:
local url = "https://mysite.ru/"
local regex = "^(https?:%/%/(%w+%.)+%w+)%/?$"

print(url:match(regex)) -- nil
Lua:
local url = 'https://mysite.ru/http'
local regex = "^http.%:%/%/(.+%..+)%/.-$"

print(url:match(regex)) -- mysite.ru
 
  • Нравится
Реакции: ZKelo

штейн

Известный
Проверенный
1,003
688
Lua:
function sampGetDistanceLocalPlayerToPlayerByPlayerId(playerId)
  local playerId = tonumber(playerId, 10)
  if not playerId then return end
  local res, han = sampGetCharHandleBySampPlayerId(playerId)
  if res then
    local x, y, z = getCharCoordinates(playerPed)
    local xx, yy, zz = getCharCoordinates(han)
    return true, getDistanceBetweenCoords3d(x, y, z, xx, yy, zz)
  end
  return false
end
Lua:
-- пример.
local id = math.random(0, 1000)
local bool, distance = sampGetDistanceLocalPlayerToPlayerByPlayerId(id)
assert(bool, 'Player '..id..' not in sight.')
print('Distance player '..id..': '..distance)

[18:31:17.714829] (error) Hitman Help: C:\Games\KEON GTA BY LONSY\moonloader\Hitman Help.lua:225: Player 184 not in sight.
[18:31:17.716330] (error) Hitman Help: Script died due to error. (0C6C3B74)

Lua:
        if isKeyJustPressed(VK_I) and not sampIsChatInputActive() and not sampIsDialogActive() and not isPauseMenuActive() then
            if zakaz == true then
                thisid = id
                nick = sampGetPlayerNickname(id)
                moneyy = money
                local color = string.format("%06X", ARGBtoRGB(sampGetPlayerColor(id)))
                local score = sampGetPlayerScore(id)
                local nick = nick:gsub('_', ' ')
                local bool, distance = sampGetDistanceLocalPlayerToPlayerByPlayerId(id)
                assert(bool, 'Player '..id..' not in sight.')
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Ник: {"..color.."}"..nick.." {ffffff}[ {800000}"..id.." {ffffff}], LVL: {800000}"..score.."{ffffff}, расстояние: {800000}"..distance, -1)
                wait(300)
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Чтобы попросить выдачу Вам контракта нажмите {800000}P{ffffff}.", -1)
                wait(300)
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Чтобы отказаться от контракта нажмите {800000}O{ffffff}.", -1)
            else
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Новых заказов не поступало.", -1)
            end
        end
 
1,417
1,032
[18:31:17.714829] (error) Hitman Help: C:\Games\KEON GTA BY LONSY\moonloader\Hitman Help.lua:225: Player 184 not in sight.
[18:31:17.716330] (error) Hitman Help: Script died due to error. (0C6C3B74)

Lua:
        if isKeyJustPressed(VK_I) and not sampIsChatInputActive() and not sampIsDialogActive() and not isPauseMenuActive() then
            if zakaz == true then
                thisid = id
                nick = sampGetPlayerNickname(id)
                moneyy = money
                local color = string.format("%06X", ARGBtoRGB(sampGetPlayerColor(id)))
                local score = sampGetPlayerScore(id)
                local nick = nick:gsub('_', ' ')
                local bool, distance = sampGetDistanceLocalPlayerToPlayerByPlayerId(id)
                assert(bool, 'Player '..id..' not in sight.')
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Ник: {"..color.."}"..nick.." {ffffff}[ {800000}"..id.." {ffffff}], LVL: {800000}"..score.."{ffffff}, расстояние: {800000}"..distance, -1)
                wait(300)
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Чтобы попросить выдачу Вам контракта нажмите {800000}P{ffffff}.", -1)
                wait(300)
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Чтобы отказаться от контракта нажмите {800000}O{ffffff}.", -1)
            else
                sampAddChatMessage("[ {800000}HitMan {ffffff}]: Новых заказов не поступало.", -1)
            end
        end
assert замени на проверку if then end
 

lanini

Известный
35
9
Пишу автозаказ продуктов дял DRP. Все хорошо работает, кроме взятия цифр из диалога. Пишет что значение nil. (в 28 строке уже ошибка)
Lua:
script_name('AutoProds for DRP')
script_version_number(1)
script_version('1.0')
script_author('Sergey Lani')
sctipt_description('Thanks for Fardin :3')
local sampev = require 'lib.samp.events'

require "lib.moonloader"
require "lib.sampfuncs"

function main()
    if not isSampfuncsLoaded() or not isSampLoaded() then return end
    while not isSampAvailable() do wait(100) end
    active = 1
    sampRegisterChatCommand("prod", cmd_prod)
    while true do
        wait(0)
    end
end

function sampev.onShowDialog(dialogId, style, title, button1, button2, text)
    if pact ~= 0 then
        if dialogId == 512 then
            pact = 0
        end
        if dialogId == 495 then
            sklad = string.match(text, "Кол%-во товара%: %{FFFFFF%}%$(%d+)")
            print(sklad)
            zakaz = string.match(text, "Заказано%: %{FFFFFF%}%$(%d+)")
            print(zakaz)
            local sklad = tonumber(sklad)
            print(sklad)
            local zakaz = tonumber(zakaz)
            print(zakaz)
            local total = sklad + zakaz
            print(total)
            prods = 10000 - total
            print(prods)
        end
        if dialogId == 512 or dialogId == 511 or dialogId == 495 then
              lua_thread.create(function ()
              wait(100)
              sampSendDialogResponse(dialogId, 0, "", "")
              end)
              return false
        end
    end
end




function cmd_prod()
    if not isPauseMenuActive() and isPlayerPlaying(playerHandle) then
        if active == 1 then
            pact = 1
            if pact == 1 then
            sampSendChat("/business")
            sampSendDialogResponse(511, 1, 1, "")
            sampSendDialogResponse(495, 0, "", "")
            sampSendDialogResponse(511, 1, 0, "")
            sampSendDialogResponse(512, 1, "", prods)
        end
        end
    end
end
Содержимое диалога ID 495:
Код:
{4682B4}Название: {FFFFFF}Автосалон
{4682B4}Владелец: {FFFFFF}Sebastiano_Corrigan
{4682B4}Кол-во дней: {FFFFFF}30
{4682B4}Гос. цена {FFFFFF}$3000000
{4682B4}Касса: {FFFFFF}$68329
{4682B4}Цена товара: {FFFFFF}$500
{4682B4}К-во товара: {FFFFFF}7840 ед.
{4682B4}Заказано: {FFFFFF}2100 ед.
{4682B4}Оплата: {FFFFFF}$3750/день
{4682B4}Налогообложение: {FFFFFF}6
{4682B4}Состояние: {FFFFFF}{33AA33}Открыт
 

itsLegend

Фонд борьбы за жуков 🐞
Администратор
2,696
1,468
Пишу автозаказ продуктов дял DRP. Все хорошо работает, кроме взятия цифр из диалога. Пишет что значение nil. (в 28 строке уже ошибка)
Lua:
script_name('AutoProds for DRP')
script_version_number(1)
script_version('1.0')
script_author('Sergey Lani')
sctipt_description('Thanks for Fardin :3')
local sampev = require 'lib.samp.events'

require "lib.moonloader"
require "lib.sampfuncs"

function main()
    if not isSampfuncsLoaded() or not isSampLoaded() then return end
    while not isSampAvailable() do wait(100) end
    active = 1
    sampRegisterChatCommand("prod", cmd_prod)
    while true do
        wait(0)
    end
end

function sampev.onShowDialog(dialogId, style, title, button1, button2, text)
    if pact ~= 0 then
        if dialogId == 512 then
            pact = 0
        end
        if dialogId == 495 then
            sklad = string.match(text, "Кол%-во товара%: %{FFFFFF%}%$(%d+)")
            print(sklad)
            zakaz = string.match(text, "Заказано%: %{FFFFFF%}%$(%d+)")
            print(zakaz)
            local sklad = tonumber(sklad)
            print(sklad)
            local zakaz = tonumber(zakaz)
            print(zakaz)
            local total = sklad + zakaz
            print(total)
            prods = 10000 - total
            print(prods)
        end
        if dialogId == 512 or dialogId == 511 or dialogId == 495 then
              lua_thread.create(function ()
              wait(100)
              sampSendDialogResponse(dialogId, 0, "", "")
              end)
              return false
        end
    end
end




function cmd_prod()
    if not isPauseMenuActive() and isPlayerPlaying(playerHandle) then
        if active == 1 then
            pact = 1
            if pact == 1 then
            sampSendChat("/business")
            sampSendDialogResponse(511, 1, 1, "")
            sampSendDialogResponse(495, 0, "", "")
            sampSendDialogResponse(511, 1, 0, "")
            sampSendDialogResponse(512, 1, "", prods)
        end
        end
    end
end
Содержимое диалога ID 495:
Код:
{4682B4}Название: {FFFFFF}Автосалон
{4682B4}Владелец: {FFFFFF}Sebastiano_Corrigan
{4682B4}Кол-во дней: {FFFFFF}30
{4682B4}Гос. цена {FFFFFF}$3000000
{4682B4}Касса: {FFFFFF}$68329
{4682B4}Цена товара: {FFFFFF}$500
{4682B4}К-во товара: {FFFFFF}7840 ед.
{4682B4}Заказано: {FFFFFF}2100 ед.
{4682B4}Оплата: {FFFFFF}$3750/день
{4682B4}Налогообложение: {FFFFFF}6
{4682B4}Состояние: {FFFFFF}{33AA33}Открыт
А ты точно по своему примеру "содержимое диалога 495" делал?
 

Rowland

Новичок
114
2
Можно ли как-то изменить значки игроков на миникарте, типа вместо треугольников и квадратов, кружочки
 

ШPEK

Известный
1,474
525
Lua:
require 'lib.moonloader'

function main()
while not isSampAvailable() do wait(100) end
local id = sampGetPlayerIdByCharHandle(PLAYER_HANDLE)
local name = sampGetPlayerNickname(id)
local nam, surnam = name:match("(.*)_(.*)")
local nick = nam.." "..surnam
while true do
    wait(0)
    if isKeyJustPressed(VK_1) then
        sampSendChat("Здравствуйте! я ваш лечащий врач "..nick)
        wait(1500)
        sampSendChat("Чем я могу вам помочь?")
    end
    end
    end
Почему id = 0??? Хотя id другой.
 

Вложения

  • upload_2018-5-1_18-12-21.png
    upload_2018-5-1_18-12-21.png
    11.6 KB · Просмотры: 250

RTD

Потужно
Модератор
399
470
Lua:
require 'lib.moonloader'

function main()
while not isSampAvailable() do wait(100) end
local id = sampGetPlayerIdByCharHandle(PLAYER_HANDLE)
local name = sampGetPlayerNickname(id)
local nam, surnam = name:match("(.*)_(.*)")
local nick = nam.." "..surnam
while true do
    wait(0)
    if isKeyJustPressed(VK_1) then
        sampSendChat("Здравствуйте! я ваш лечащий врач "..nick)
        wait(1500)
        sampSendChat("Чем я могу вам помочь?")
    end
    end
    end
Почему id = 0??? Хотя id другой.
Ты хотя бы немного документацию читаешь???
moonloader - globals | BlastHack — DEV_WIKI(https://blast.hk/wiki/moonloader:globals)
Lua:
local _,id = sampGetPlayerIdByCharHandle(PLAYER_PED)
 

imring

Ride the Lightning
Всефорумный модератор
2,362
2,545
Lua:
require 'lib.moonloader'

function main()
while not isSampAvailable() do wait(100) end
local id = sampGetPlayerIdByCharHandle(PLAYER_HANDLE)
local name = sampGetPlayerNickname(id)
local nam, surnam = name:match("(.*)_(.*)")
local nick = nam.." "..surnam
while true do
    wait(0)
    if isKeyJustPressed(VK_1) then
        sampSendChat("Здравствуйте! я ваш лечащий врач "..nick)
        wait(1500)
        sampSendChat("Чем я могу вам помочь?")
    end
    end
    end
Почему id = 0??? Хотя id другой.
1-ое: в функции sampGetPlayerIdByCharHandle надо использовать ped игрока, а не handle.
2-ое: функция sampGetPlayerIdByCharHandle возвращает result и id, а не просто id.
3-ее: ты получаешь id игрока, когда ты не подключён к серверу.
 
  • Нравится
Реакции: RTD

#Northn

Pears Project — уже запущен!
Всефорумный модератор
2,650
2,535
Получение серийника логического диска.
Lua:
local ffi = require("ffi")
ffi.cdef[[
int __stdcall GetVolumeInformationA(
    const char* lpRootPathName,
    char* lpVolumeNameBuffer,
    uint32_t nVolumeNameSize,
    uint32_t* lpVolumeSerialNumber,
    uint32_t* lpMaximumComponentLength,
    uint32_t* lpFileSystemFlags,
    char* lpFileSystemNameBuffer,
    uint32_t nFileSystemNameSize
);
]]
local serial = ffi.new("unsigned long[1]", 0)
ffi.C.GetVolumeInformationA(nil, nil, 0, serial, nil, nil, nil, 0)
serial = serial[0]

Как это юзать? Дайте пример.