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

tlwsn

Известный
537
85
INCOMING_RPCS[RPC.SERVERQUIT] = {'onPlayerQuit', {playerId = 'int16'}, {reason = 'int8'}}
reason возвращает цифру, что эти цифры значат?
 

#Northn

Pears Project — уже запущен!
Всефорумный модератор
2,657
2,551
INCOMING_RPCS[RPC.SERVERQUIT] = {'onPlayerQuit', {playerId = 'int16'}, {reason = 'int8'}}
reason возвращает цифру, что эти цифры значат?
Причины выхода из сервера. (/q, краш, зависло), насчёт порядка не уверен.
не получается, вот отрывок с кода:
Lua:
local phv = os.getenv('TEMP') .. '\\testqVERSION.json'
local dlstatus = require('moonloader').download_status
local fpath = os.getenv('TEMP') .. '\\testq.lua'
local key = require "vkeys"
local vkeys = require 'vkeys'
local wm = require 'lib.windows.message'
local imgui = require 'imgui'
local encoding = require 'encoding'
local sampev = require 'lib.samp.events'
local rkeys = require 'rkeys'
imgui.ShowCursor = false
local style = imgui.GetStyle()
local colors = style.Colors
local clr = imgui.Col
local ImVec4 = imgui.ImVec4
local dlstatus = require('moonloader').download_status
imgui.ToggleButton = require('imgui_addons').ToggleButton
imgui.HotKey = require('imgui_addons').HotKey
imgui.Spinner = require('imgui_addons').Spinner
imgui.BufferingBar = require('imgui_addons').BufferingBar
encoding.default = 'CP1251'
u8 = encoding.UTF8

function main()
   if not isSampLoaded() then
      return
   end
   while not isSampAvailable() do
      wait(0)
    end
    sampRegisterChatCommand("test", function ()
        window.v = true
    end)
    sampRegisterChatCommand("inv", function ()
      lua_thread.create(function ()
        sampSendChat ('xyila')
      end)
    end)
    bindID = rkeys.registerHotKey(ActiveMenu.v, true, function ()
        window.v = not window.v
    end)
    while true do
        local valid, ped = getCharPlayerIsTargeting(PLAYER_HANDLE)
        if valid and doesCharExist(ped) then
          local result, id = sampGetPlayerIdByCharHandle(ped)
          if result then
            if valid and doesCharExist(ped) and isKeyJustPressed(88) then
               nickname = sampGetPlayerNickname(id)
            end
          end
        end
        if wasKeyPressed(key.VK_X) then
            bool, player = getCharPlayerIsTargeting(PLAYER_HANDLE)
            if bool then
                _, id = sampGetPlayerIdByCharHandle(player)
            end
        end
        wait(0)
        imgui.Process = window.v
    end
        update2()
    print('Update starts.')
    while update2 ~= false do wait(20) end
    print('Update check successful.')
end

function update2()
    local fpath = os.getenv('TEMP') .. '\\pHelpUpdaterVERSION.json'
    downloadUrlToFile('https://drive.google.com/open?id=1pMOCTfLNnC2V2fuGIQ7pa5zqgNnnEwYA', fpath, function(id, status, p1, p2)
        if p1 > 1 and p2 == 0 then
            print("Превышено время ожидания загрузки. (Нет доступа к интернету)")
        elseif status == dlstatus.STATUS_ENDDOWNLOADDATA then
            local f = io.open(fpath, 'r')
            if f then
                local info = decodeJson(f:read('*a'))
                updatelink = info.updateurl
                if info and info.latest then
                    version = tonumber(info.latest)
                    if version > tonumber(thisScript().version) then
                        lua_thread.create(goupdate2)
                    else
                        update2 = false
                    end
                end
            end
        end
    end)
end

--скачивание актуальной версии
function goupdate2()
    downloadUrlToFile(updatelink, thisScript().path, function(id3, status1, p13, p23) lua_thread.create(function()
            if status1 == dlstatus.STATUS_ENDDOWNLOADDATA then
                sampAddChatMessage(("[ PHELP ]{ffffff}: Обновление завершено!"), 0x0088ff)
                wait(200)
                reloadScripts()
            end
        end)
    end)
end

Lua:
local phv = os.getenv('TEMP') .. '\\testqVERSION.json'
local dlstatus = require('moonloader').download_status
local fpath = os.getenv('TEMP') .. '\\testq.lua'
local key = require "vkeys"
local vkeys = require 'vkeys'
local wm = require 'lib.windows.message'
local imgui = require 'imgui'
local encoding = require 'encoding'
local sampev = require 'lib.samp.events'
local rkeys = require 'rkeys'
imgui.ShowCursor = false
local style = imgui.GetStyle()
local colors = style.Colors
local clr = imgui.Col
local ImVec4 = imgui.ImVec4
local dlstatus = require('moonloader').download_status
imgui.ToggleButton = require('imgui_addons').ToggleButton
imgui.HotKey = require('imgui_addons').HotKey
imgui.Spinner = require('imgui_addons').Spinner
imgui.BufferingBar = require('imgui_addons').BufferingBar
encoding.default = 'CP1251'
u8 = encoding.UTF8

function main()
   if not isSampLoaded() then
      return
   end
   while not isSampAvailable() do
      wait(0)
    end
    update2()
    print('Update starts.')
    while update2 ~= false do wait(20) end
    print('Update check successful.')
    sampRegisterChatCommand("test", function ()
        window.v = true
    end)
    sampRegisterChatCommand("inv", function ()
      lua_thread.create(function ()
        sampSendChat ('xyila')
      end)
    end)
    bindID = rkeys.registerHotKey(ActiveMenu.v, true, function ()
        window.v = not window.v
    end)
    while true do
        local valid, ped = getCharPlayerIsTargeting(PLAYER_HANDLE)
        if valid and doesCharExist(ped) then
          local result, id = sampGetPlayerIdByCharHandle(ped)
          if result then
            if valid and doesCharExist(ped) and isKeyJustPressed(88) then
               nickname = sampGetPlayerNickname(id)
            end
          end
        end
        if wasKeyPressed(key.VK_X) then
            bool, player = getCharPlayerIsTargeting(PLAYER_HANDLE)
            if bool then
                _, id = sampGetPlayerIdByCharHandle(player)
            end
        end
        wait(0)
        imgui.Process = window.v
    end
end

function update2()
    local fpath = os.getenv('TEMP') .. '\\pHelpUpdaterVERSION.json'
    downloadUrlToFile('https://drive.google.com/open?id=1pMOCTfLNnC2V2fuGIQ7pa5zqgNnnEwYA', fpath, function(id, status, p1, p2)
        if p1 > 1 and p2 == 0 then
            print("Превышено время ожидания загрузки. (Нет доступа к интернету)")
        elseif status == dlstatus.STATUS_ENDDOWNLOADDATA then
            local f = io.open(fpath, 'r')
            if f then
                local info = decodeJson(f:read('*a'))
                updatelink = info.updateurl
                if info and info.latest then
                    version = tonumber(info.latest)
                    if version > tonumber(thisScript().version) then
                        lua_thread.create(goupdate2)
                    else
                        update2 = false
                    end
                end
            end
        end
    end)
end

--скачивание актуальной версии
function goupdate2()
    downloadUrlToFile(updatelink, thisScript().path, function(id3, status1, p13, p23) lua_thread.create(function()
            if status1 == dlstatus.STATUS_ENDDOWNLOADDATA then
                sampAddChatMessage(("[ PHELP ]{ffffff}: Обновление завершено!"), 0x0088ff)
                wait(200)
                reloadScripts()
            end
        end)
    end)
end

А вообще, у тебя это работать не будет, нужна прямая ссылка на файл, а не на сайт.
 

sdfaw

Активный
717
150
Причины выхода из сервера. (/q, краш, зависло), насчёт порядка не уверен.


Lua:
local phv = os.getenv('TEMP') .. '\\testqVERSION.json'
local dlstatus = require('moonloader').download_status
local fpath = os.getenv('TEMP') .. '\\testq.lua'
local key = require "vkeys"
local vkeys = require 'vkeys'
local wm = require 'lib.windows.message'
local imgui = require 'imgui'
local encoding = require 'encoding'
local sampev = require 'lib.samp.events'
local rkeys = require 'rkeys'
imgui.ShowCursor = false
local style = imgui.GetStyle()
local colors = style.Colors
local clr = imgui.Col
local ImVec4 = imgui.ImVec4
local dlstatus = require('moonloader').download_status
imgui.ToggleButton = require('imgui_addons').ToggleButton
imgui.HotKey = require('imgui_addons').HotKey
imgui.Spinner = require('imgui_addons').Spinner
imgui.BufferingBar = require('imgui_addons').BufferingBar
encoding.default = 'CP1251'
u8 = encoding.UTF8

function main()
   if not isSampLoaded() then
      return
   end
   while not isSampAvailable() do
      wait(0)
    end
    update2()
    print('Update starts.')
    while update2 ~= false do wait(20) end
    print('Update check successful.')
    sampRegisterChatCommand("test", function ()
        window.v = true
    end)
    sampRegisterChatCommand("inv", function ()
      lua_thread.create(function ()
        sampSendChat ('xyila')
      end)
    end)
    bindID = rkeys.registerHotKey(ActiveMenu.v, true, function ()
        window.v = not window.v
    end)
    while true do
        local valid, ped = getCharPlayerIsTargeting(PLAYER_HANDLE)
        if valid and doesCharExist(ped) then
          local result, id = sampGetPlayerIdByCharHandle(ped)
          if result then
            if valid and doesCharExist(ped) and isKeyJustPressed(88) then
               nickname = sampGetPlayerNickname(id)
            end
          end
        end
        if wasKeyPressed(key.VK_X) then
            bool, player = getCharPlayerIsTargeting(PLAYER_HANDLE)
            if bool then
                _, id = sampGetPlayerIdByCharHandle(player)
            end
        end
        wait(0)
        imgui.Process = window.v
    end
end

function update2()
    local fpath = os.getenv('TEMP') .. '\\pHelpUpdaterVERSION.json'
    downloadUrlToFile('https://drive.google.com/open?id=1pMOCTfLNnC2V2fuGIQ7pa5zqgNnnEwYA', fpath, function(id, status, p1, p2)
        if p1 > 1 and p2 == 0 then
            print("Превышено время ожидания загрузки. (Нет доступа к интернету)")
        elseif status == dlstatus.STATUS_ENDDOWNLOADDATA then
            local f = io.open(fpath, 'r')
            if f then
                local info = decodeJson(f:read('*a'))
                updatelink = info.updateurl
                if info and info.latest then
                    version = tonumber(info.latest)
                    if version > tonumber(thisScript().version) then
                        lua_thread.create(goupdate2)
                    else
                        update2 = false
                    end
                end
            end
        end
    end)
end

--скачивание актуальной версии
function goupdate2()
    downloadUrlToFile(updatelink, thisScript().path, function(id3, status1, p13, p23) lua_thread.create(function()
            if status1 == dlstatus.STATUS_ENDDOWNLOADDATA then
                sampAddChatMessage(("[ PHELP ]{ffffff}: Обновление завершено!"), 0x0088ff)
                wait(200)
                reloadScripts()
            end
        end)
    end)
end

А вообще, у тебя это работать не будет, нужна прямая ссылка на файл, а не на сайт.
Meet Google Drive – One place for all your files(https://doc-04-00-docs.googleusercontent.com/docs/securesc/b6335lkejs98i8a8v8cbihlke3oolu9r/hqgntuoi08faqloocjhmge6mrp7mo3v6/1547992800000/01793882389258385998/01793882389258385998/1pMOCTfLNnC2V2fuGIQ7pa5zqgNnnEwYA?e=download) такая сыллка?
 

tlwsn

Известный
537
85
из-за чего может быть такой трабл?
Lua:
[18:30:50.244483] (exception) script: CJSON: Expected value but found T_END at character 1
[18:30:50.244483] (error) script: H:\Games\GTA San Andreas1\moonloader\script.lua:3763: attempt to index upvalue 'config_keys' (a nil value)
 

ImStery

Новичок
1
0
Когда я пытался сделать бинд на NUM1 столкнулся с проблемой:в игре эта клавиша разворачивает камеру на 180 градусов. Вопрос: Как убрать значение какой-либо клавиши во время работы скрипта?
 

Natami

Участник
375
26
Когда я пытался сделать бинд на NUM1 столкнулся с проблемой:в игре эта клавиша разворачивает камеру на 180 градусов. Вопрос: Как убрать значение какой-либо клавиши во время работы скрипта?
поменять на другую клавишу,все просто
 

kotov

Участник
128
12
Как этот textdraw "клик" отсканировать с помощью "find?" Типа если он находит 12:22 then
Хз как достать значение из текстдрава, но если просто из переменной то:

Lua:
str = "12:22"
if str.find("12:22") then
    --code
end

как мне добавить это х, чтобы закрыть грушу во всплывающем окне? внимание, нет кнопки, функция по умолчанию видна на этом скриншоте Imgur(https://imgur.com/7rEE2yZ)
Lua:
local main_window_state = imgui.ImBool(false) -- перед кодом

imgui.Process = main_window_state.v -- в main

imgui.Begin("Окно", main_window_state) -- в imgui.OnDrawFrame()

Советую почитать https://blast.hk/threads/19292/ + то что люди спрашивают в теме, полезно.
 

MrYurkoo

Известный
101
8
Как зашифровать/закриптовать код, чтобы его не мог никто скопипиздить?
 

Akionka

akionka.lua
Проверенный
740
502
Как зашифровать/закриптовать код, чтобы его не мог никто скопипиздить?
https://blast.hk/threads/17276

Как этот textdraw "клик" отсканировать с помощью "find?" Типа если он находит 12:22 then
Повесить хук и чекать, типа так.
Lua:
local sampev = require 'lib.samp.events'
function sampev.onShowTextDraw(id, td)
    if td.text:find("12:22") then
        -- do smthg
    end
end