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

Frapsy

Известный
Проверенный
393
226
Что не так? [00:16:38.315727] (system) perebor.lua: Script terminated. (03F02CCC)
В твоем отрывке ошибки нет. Кидай полную ошибку из лога, а не то, что скрипт ушел в аут.
Есть только на клео
Исходников с луашным реконнектом полно.
Как научить сестру-чайника программировать на Lua? :D
Ровным счетом так же, как и научить ставить чайник на разогрев.
 

komnatq

Известный
203
90
Подскажите, пожалуйста, как можно удалить строчку с деньгами, при этом оставив остальную часть дефолтного худа
KWX4GkU.png

 

BlackGoblin

Известный
519
215
Шо не так? Почему он ругается когда задержку добавляю?
Код:
local sampev = require "lib.samp.events"

function sampev.onServerMessage(color, msg)
    if msg:find('123') then
        setVirtualKeyDown(78, true)
                wait(150)
                setVirtualKeyDown(78, false)
                return false
    end
end
 

Frapsy

Известный
Проверенный
393
226
Шо не так? Почему он ругается когда задержку добавляю?
Код:
local sampev = require "lib.samp.events"

function sampev.onServerMessage(color, msg)
    if msg:find('123') then
        setVirtualKeyDown(78, true)
                wait(150)
                setVirtualKeyDown(78, false)
                return false
    end
end
Задержки можно использовать только в main() или же в потоке lua.thread_create()
 
  • Нравится
Реакции: BlackGoblin

serhiyrubin

Известный
396
106
Подскажите, появилась ошибка
- cannot resume non-suspended coroutine
Грешит на функцию которая вызывается одновременно из нескольких потоков
Это может быть причиной?
 

SERGEEV

Известный
6
1
Подскажите как можно показать свои TextDraw'ы
Функция:
Код:
sampTextdrawCreate(1, "hello", 1176, 740)
Каких либо результатов не дает

 

Quasper

Известный
834
354
Ребзя, возник вопрос, я получаю строку из диалога через sampGetListboxItemText, как мне получить цвет этой строки?
 

Glockersik

Участник
86
3
1)подскажите пожалуйста код , который ищет 3д объекты , которые находятся в зоне стрима
2) Для чего использают data.text и какие ее функции?
 
Последнее редактирование:

DolgorukovGTA

Известный
Проверенный
652
345
Lua:
local main_color = 0x34C924

function main()
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(0) end
    while true do wait(0) end
    sampRegisterChatCommand(gpsbiz, submenus_show)
        if isPlayerPlaying(PLAYER_HANDLE) then
             if result then
                submenus_show(mainmenu, "Меню", "Выбрать", "Закрыть", main_color)
            end
        end
    end
      

        local mainmenu = {
{
    title = 'Бизнес #1',
    onclick = function()
        sampSendChat('/gps b 1')
        end
    }
    {
    title = 'Бизнес #2',
    onclick = function()
        sampSendChat('/gps b 2')
        end
    }
}

function submenus_show(menu, caption, select_button, close_button, back_button)
   select_button, close_button, back_button = select_button or 'Выбрать', close_button or 'Закрыть', back_button or 'Назад'
   prev_menus = {}
   function display(menu, id, caption)
       local string_list = {}
       for i, v in ipairs(menu) do
           table.insert(string_list, type(v.submenu) == 'table' and v.title .. '' or v.title)
       end
       sampShowDialog(id, caption, table.concat(string_list, '\n'), select_button, (#prev_menus > 0) and back_button or close_button, sf.DIALOG_STYLE_LIST)
       repeat
           wait(0)
           local result, button, list = sampHasDialogRespond(id)
           if result then
               if button == 1 and list ~= -1 then
                   local item = menu[list + 1]
                   if type(item.submenu) == 'table' then -- submenu
                       table.insert(prev_menus, {menu = menu, caption = caption})
                       if type(item.onclick) == 'function' then
                           item.onclick(menu, list + 1, item.submenu)
                       end
                       return display(item.submenu, id + 1, item.submenu.title and item.submenu.title or item.title)
                   elseif type(item.onclick) == 'function' then
                       local result = item.onclick(menu, list + 1)
                       if not result then return result end
                       return display(menu, id, caption)
                   end
               else -- if button == 0
                   if #prev_menus > 0 then
                       local prev_menu = prev_menus[#prev_menus]
                       prev_menus[#prev_menus] = nil
                       return display(prev_menu.menu, id - 1, prev_menu.caption)
                   end
                   return false
               end
           end
       until result
   end
   return display(menu, 31337, caption or menu.title)
end

Наверное, я делаю что-то не так, мой скрипт должен показывать мне диалог со строками "Бизнес #1" и "Бизнес #2" по команде /gpsbiz, а после клика по нужной строке в диалоге он должен выписывать соответствующие команды в чат. Помогите мне разобраться, что не так.
 

Petr_Sergeevich

Известный
Проверенный
707
296
Lua:
local main_color = 0x34C924

function main()
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(0) end
    while true do wait(0) end
    sampRegisterChatCommand(gpsbiz, submenus_show)
        if isPlayerPlaying(PLAYER_HANDLE) then
             if result then
                submenus_show(mainmenu, "Меню", "Выбрать", "Закрыть", main_color)
            end
        end
    end
   

        local mainmenu = {
{
    title = 'Бизнес #1',
    onclick = function()
        sampSendChat('/gps b 1')
        end
    }
    {
    title = 'Бизнес #2',
    onclick = function()
        sampSendChat('/gps b 2')
        end
    }
}

function submenus_show(menu, caption, select_button, close_button, back_button)
   select_button, close_button, back_button = select_button or 'Выбрать', close_button or 'Закрыть', back_button or 'Назад'
   prev_menus = {}
   function display(menu, id, caption)
       local string_list = {}
       for i, v in ipairs(menu) do
           table.insert(string_list, type(v.submenu) == 'table' and v.title .. '' or v.title)
       end
       sampShowDialog(id, caption, table.concat(string_list, '\n'), select_button, (#prev_menus > 0) and back_button or close_button, sf.DIALOG_STYLE_LIST)
       repeat
           wait(0)
           local result, button, list = sampHasDialogRespond(id)
           if result then
               if button == 1 and list ~= -1 then
                   local item = menu[list + 1]
                   if type(item.submenu) == 'table' then -- submenu
                       table.insert(prev_menus, {menu = menu, caption = caption})
                       if type(item.onclick) == 'function' then
                           item.onclick(menu, list + 1, item.submenu)
                       end
                       return display(item.submenu, id + 1, item.submenu.title and item.submenu.title or item.title)
                   elseif type(item.onclick) == 'function' then
                       local result = item.onclick(menu, list + 1)
                       if not result then return result end
                       return display(menu, id, caption)
                   end
               else -- if button == 0
                   if #prev_menus > 0 then
                       local prev_menu = prev_menus[#prev_menus]
                       prev_menus[#prev_menus] = nil
                       return display(prev_menu.menu, id - 1, prev_menu.caption)
                   end
                   return false
               end
           end
       until result
   end
   return display(menu, 31337, caption or menu.title)
end

Наверное, я делаю что-то не так, мой скрипт должен показывать мне диалог со строками "Бизнес #1" и "Бизнес #2" по команде /gpsbiz, а после клика по нужной строке в диалоге он должен выписывать соответствующие команды в чат. Помогите мне разобраться, что не так.

Lua:
sf = require "sampfuncs"
isMenu = false

dialogMenu = {
    {
        title = 'Бизнес #1',
        onclick = function()
            sampSendChat('/gps b 1')
        end
    },
    {
        title = 'Бизнес #2',
        onclick = function()
            sampSendChat('/gps b 2')
        end
    }
}

function main()
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(0) end

    sampRegisterChatCommand("gpsbiz", function()
        isMenu = not isMenu
    end)

    while true do
        wait(0)
        if isMenu then
            submenus_show(dialogMenu, "Меню", "Выбрать", "Закрыть", "Назад")
            isMenu = false
        end
    end
end

function submenus_show(menu, caption, select_button, close_button, back_button)
   select_button, close_button, back_button = select_button or 'Выбрать', close_button or 'Закрыть', back_button or 'Назад'
   prev_menus = {}
   function display(menu, id, caption)
       local string_list = {}
       for i, v in ipairs(menu) do
           table.insert(string_list, type(v.submenu) == 'table' and v.title .. '' or v.title)
       end
       sampShowDialog(id, caption, table.concat(string_list, '\n'), select_button, (#prev_menus > 0) and back_button or close_button, sf.DIALOG_STYLE_LIST)
       repeat
           wait(0)
           local result, button, list = sampHasDialogRespond(id)
           if result then
               if button == 1 and list ~= -1 then
                   local item = menu[list + 1]
                   if type(item.submenu) == 'table' then -- submenu
                       table.insert(prev_menus, {menu = menu, caption = caption})
                       if type(item.onclick) == 'function' then
                           item.onclick(menu, list + 1, item.submenu)
                       end
                       return display(item.submenu, id + 1, item.submenu.title and item.submenu.title or item.title)
                   elseif type(item.onclick) == 'function' then
                       local result = item.onclick(menu, list + 1)
                       if not result then return result end
                       return display(menu, id, caption)
                   end
               else -- if button == 0
                   if #prev_menus > 0 then
                       local prev_menu = prev_menus[#prev_menus]
                       prev_menus[#prev_menus] = nil
                       return display(prev_menu.menu, id - 1, prev_menu.caption)
                   end
                   return false
               end
           end
       until result
   end
   return display(menu, 31337, caption or menu.title)
end