Moon ImGui — Dear ImGui for MoonLoader

29072770-aef614e2-7c7b-11e7-8861-212c2c416439.png


Это ImGui - самый функциональный и одновременно самый простой GUI-фреймворк из всех, что мне известны.
И теперь каждый из вас сможет использовать его в своих Lua скриптах для MoonLoader.

Представляю вам Moon ImGui - Lua биндинг ImGui для MoonLoader. Он включает все основные возможности фреймворка, а API максимально приближен к оригинальному по мере возможного.
Тут не будут подробно рассматриваться возможности и особенности ImGui, в этой теме будет рассказано и показано как использовать Moon ImGui в Lua. За информацией по самому ImGui сюда:

Начнём с самого элементарного:
Lua:
local imgui = require 'imgui' -- загружаем библиотеку

-- в этой функции осуществляется вся работа с ImGui
-- она вызывается каждый кадр, но только если imgui.Process равен true
function imgui.OnDrawFrame()
  imgui.Begin('My window') -- новое окно с заголовком 'My window'
  imgui.Text('Hello world') -- простой текст внутри этого окна
  imgui.End() -- конец окна
end

function main()
  imgui.Process = true -- ImGui будет обрабатываться, пока imgui.Process равен true
  -- в этом примере мы просто активируем ImGui сразу же после загрузки игры
end
Результат:
upload_2017-11-25_18-29-2.png


Это работает, но что-то не впечатляет, согласитесь. Окно изначально маленькое, показывается сразу при старте и его нельзя закрыть.
Сделаем его побольше, добавим активацию и какое-нибудь действие:
Lua:
local imgui = require 'imgui'
local key = require 'vkeys'

-- одно из основных отличий от оригинального апи
-- все переменные, значения которых записываются в ImGui по указателю, могут использоваться только через специальные типы
local main_window_state = imgui.ImBool(false)
function imgui.OnDrawFrame()
  if main_window_state.v then -- чтение и запись значения такой переменной осуществляется через поле v (или Value)
    imgui.SetNextWindowSize(imgui.ImVec2(150, 200), imgui.Cond.FirstUseEver) -- меняем размер
    -- но для передачи значения по указателю - обязательно напрямую
    -- тут main_window_state передаётся функции imgui.Begin, чтобы можно было отследить закрытие окна нажатием на крестик
    imgui.Begin('My window', main_window_state)
    imgui.Text('Hello world')
    if imgui.Button('Press me') then -- а вот и кнопка с действием
      -- условие будет выполнено при нажатии на неё
      printStringNow('Button pressed!', 1000)
    end
    imgui.End()
  end
end

function main()
  while true do
    wait(0)
    if wasKeyPressed(key.VK_X) then -- активация по нажатию клавиши X
        main_window_state.v = not main_window_state.v -- переключаем статус активности окна, не забываем про .v
    end
    imgui.Process = main_window_state.v -- теперь значение imgui.Process всегда будет задаваться в зависимости от активности основного окна
  end
end
Теперь окно выглядит так и его можно скрыть:
upload_2017-11-25_18-29-17.png


В коде была использована переменная типа ImBool - это один из новых типов, добавленных в Moon ImGui. Такой подход неизбежен, поскольку в луа невозможна передача базовых типов по ссылке. Это не единственное изменение, есть и другие, вам потребуется о них знать. Вот их полный список:

Разница между C++ API и Lua API
ОписаниеВ C++В Lua
Все функции из пространства имён ImGui, как и все типы, и все перечисления находятся в таблице, возвращаемой модулемImGui::Text("text");
ImVec2(0.1f, 2.3f);
imgui.Text("text");
imgui.ImVec2(0.1, 2.3);
Названия перечислений (enum) и их значений лишились префиксов и символа "_" в концеImGuiWindowFlags_NoTitleBarimgui.WindowFlags.NoTitleBar
Значения базовых типов, которые в ImGui записываются по указателю, должны быть использованы через специальные типы: ImBool для bool, ImFloat для float, ImInt для int и unsigned int, ImFloat2-4 для float[2-4], ImInt2-4 для int[2-4]static bool win = false; ImGui::Begin("window", &win);
win = false;
local win = imgui.ImBool(false) imgui.Begin("window", win)
win.v = false
Функции с переменным количеством аргументов для форматирования текста не поддерживают форматирование, используйте string.formatImGui::Text("hey, %s", name)imgui.Text(string.format('hey, %s', name))
Функции InputText и InputTextMultiline принимают ImBuffer вместо char* buf + size_t buf_sizechar buf[256]{};
ImGui::InputText('input', buf, sizeof(buf))
local buf = imgui.ImBuffer(256);
imgui.InputText('input', buf)
Динамические массивы в виде массива указателей + количество элементов заменены таблицамиconst char* items[] = {"1", "2", "3"}; ImGui::ListBox("list", &lb_cur, items, 3)imgui.ListBox('list', lb_cur, {'1', '2', '3'})
Функции с аргументами const char* str_start, const char* str_end, идущими подряд, принимают обычную строкуImGui::TextUnformatted(some_str, some_str + 24)imgui.TextUnformatted(some_str)
Все функции, принимающие калбэк + user_data, принимают ImCallbackvoid swszCb(ImGuiSizeConstraintCallbackData*) {};
ImGui::SetNextWindowSizeConstraints(size_min, size_max, &swszCb, (void*)&my_data)
local swszCb = imgui.ImCallback(function(data) end)
imgui.SetNextWindowSizeConstraints(size_min, size_max, swszCb)
ImFont::CalcTextSizeA,
ImFontAtlas::CustomRect::CalcCustomRectUV,
ImFontAtlas::GetTexDataAsRGBA32,
ImFontAtlas::GetTexDataAsAlpha8,
ImFontAtlas::GlyphRangesBuilder::BuildRanges,
ImGui::ColorConvertRGBtoHSV и
ImGui::ColorConvertHSVtoRGB
возвращают значения вместо изменения по ссылке
float r, g, b;
ImGui::ColorConvertHSVtoRGB(h, s, v, r, g, b);
local r, g, b = imgui.ColorConvertHSVtoRGB(h, s, v)
ImGuiIO::IniFilename и ImGuiIO::LogFilename принимают ImBuffer вместо указателя на строкуconst char ini_path[] = "my/path.ini";
ImGui::GetIO().IniFilename = ini_path;
local ini_path = imgui.ImBuffer('my/path.ini')
imgui.GetIO().IniFilename = ini_path
Изменение ImGuiTextEditCallbackData::Buf автоматически обновляет длину и задаёт значение BufDirtys.copy(data.Buf, data.BufSize);
data.BufTextLen = s.length();
data.BufDirty = true;
data.Buf = 'text'

Остальные изменения
  • Добавлено несколько дополнительных функций
    • ImColor::ToU32 - преобразование ImColor в целое
    • ImGui::GetStyleColorU32 вместо ImGui::GetColorU32 для цвета стиля
    • ImGui::PlotLinesEx вместо callback-варианта PlotLines
    • ImGui::PlotHistogramEx вместо callback-варианта PlotHistogram
    • ImDrawList::AddTextEx вместо AddText с дополнительными аргументами
  • Отсутствуют функции
    • Функции, принимающие и возвращающие void*-идентификаторы (например, PushID(void*) и GetID(void*))
    • Функции с va_list-форматированием текста (например, TextV)
Теперь, зная всё это, вы уже можете начать работать с Moon ImGui. За списком функций ImGui и примерами на C++ обращайтесь на официальную страницу.

Но это ещё не всё. Вся работа с текстом в ImGui основана на UTF-8, т.е. текст не ограничен лишь стандартным набором символов. Но т.к. GTA, SAMP и MoonLoader не поддерживают юникод, кодировки необходимо конвертировать.

Работа с другими языками на примере русского
В MoonLoader v.025 были добавлены библиотеки lua-iconv и encoding, они призваны помочь в работе с разными кодировками текста.
Следующий пример показывает как использовать текст на русском в ImGui:
Скрипт должен быть сохранён в кодировке Windows-1251
Lua:
local imgui = require 'imgui'
local encoding = require 'encoding' -- загружаем библиотеку
encoding.default = 'CP1251' -- указываем кодировку по умолчанию, она должна совпадать с кодировкой файла. CP1251 - это Windows-1251
u8 = encoding.UTF8 -- и создаём короткий псевдоним для кодировщика UTF-8

local test_text_buffer = imgui.ImBuffer(256)
function imgui.OnDrawFrame()
  imgui.Begin(u8'Основное окно') -- обратите внимание на u8 перед текстом, это и есть преобразование кодировки
  if imgui.InputText(u8'Вводить текст сюда', test_text_buffer) then -- условие будет срабатывать при изменении текста
    -- здесь первая строка передаётся по-обычному, без u8
    -- но введённый текст при выводе преобразуется обратно из UTF-8 в кодировку по умолчанию, т.е. в Windows-1251
    print('Введённый текст:', u8:decode(test_text_buffer.v)) -- при работе с ImBuffer тоже не забывайте о .v
  end
  imgui.Text(u8'Введённый текст: ' .. test_text_buffer.v) -- но тут обратное преобразование введённого текста не требуется, т.к. текст буфера и так в UTF-8
  imgui.Text(u8(string.format('Текущая дата: %s', os.date()))) -- u8 - это функция, её можно использовать и с неконстантными строками
  imgui.End()
end

function main()
  imgui.Process = true
end
Результат:
upload_2017-11-25_18-29-37.png


Не так уж и сложно, верно? Текст, передаваемый ImGui - кодируем, текст, получаемый из ImGui - декодируем. Если в вашем скрипте много текста на русском для ImGui, но мало текста, выводимого через MoonLoader (в лог или чат SA:MP, к примеру), то можно сделать наоборот - сохранить скрипт в кодировке UTF-8 и не кодировать текст, передаваемый ImGui, а вместо этого кодировать текст при работе с функциями мунлоадера.

Остальные особенности
В Moon ImGui есть несколько дополнительных возможностей. В частности, они реализуют взаимодействие с игрой и управление интерфейсом.
Код:
Параметры
imgui.BeforeDrawFrame = nil  -  опциональный калбэк. Если он задан, будет вызываться каждый кадр перед OnDrawFrame и NewFrame самого ImGui. Его можно использовать для загрузки шрифтов и текстур (будьте осторожны, он вызывается каждый кадр)
imgui.OnDrawFrame = nil  -  основной калбэк для рендера, о нём вы уже знаете
imgui.Process = false  -  только если задано true, имгуи будет обрабатываться и выводиться
imgui.RenderInMenu = false  -  показывать интерфейс в меню паузы, по умолчанию отключено
imgui.ShowCursor = true  -  показывать курсор, по умолчанию включен. Отключение может пригодиться, если нужно рисовать только какой-то оверлей без взаимодействия с ним
imgui.LockPlayer = false  -  отключить управление игроком, пока ImGui активен. Если не задано, то управление игроком будет отключаться только когда ImGui требуется обработать ввод с клавиатуры

Функции
после загрузки новых шрифтов и обязательно вне OnDrawFrame
function imgui.CreateTextureFromFile(path)  -  загрузить текстуру из файла. Возвращает загруженную текстуру или nil в случае неудачи
function imgui.CreateTextureFromMemory(address, size)  -  загрузить текстуру из изображения, хранящегося по указанному адресу в памяти. Возвращает загруженную текстуру или nil в случае неудачи
function imgui.GetTextureFromAddress(address)  -  получить ImGui-совместимую текстуру по адресу памяти
function imgui.ReleaseTexture(texture)  -  выгрузить текстуру, загруженную с помощью CreateTextureFromFile или CreateTextureFromMemory\
function imgui.RebuildFonts()  -  пересобрать внутреннюю текстуру шрифтов. Необходимо использовать
function imgui.SwitchContext()  -  переключить ImGui-контекст на принадлежащий скрипту. В BeforeDrawFrame и OnDrawFrame контекст переключается автоматически, так что в них эту функцию использовать нет нужды.

На этом всё. Во вложениях есть пример с демонстрацией использования всех этих фич, рекомендую посмотреть код и пощупать его в игре. Вот скриншот:

upload_2017-11-25_18-28-12.png

Полезные штуки для разработчиков
FontAwesome 4: https://blast.hk/threads/19292/post-168990
FontAwesome 5: https://blast.hk/threads/19292/post-335148
ImGui Pie: https://blast.hk/threads/19648/post-226145
Внутриигровая песочница: https://blast.hk/threads/19292/post-219453
ImGui Custom (хоткеи и ещё чет): https://blast.hk/threads/22080/
Global notification: https://blast.hk/threads/21619/
Стили:
https://blast.hk/threads/19292/post-260462

Ссылки
Скачать Moon ImGui (Download)
Установка:
распаковать содержимое архива в корневую папку игры.
Требуется MoonLoader v.026 или выше
Официальная страница ImGui
Пример на C++ с применением большинства возможностей (imgui_demo.cpp)
Все функции ImGui (imgui.h. Список поддерживаемых также есть в imgui.lua)
 

Вложения

  • moon imgui demo.lua
    9.2 KB · Просмотры: 123,087
Последнее редактирование:

vbg

Известный
41
17
как в imgui.Combo можно сделать так что бы при перезагрузке скрипта по умолчанию стоял тот вариант что был выбран до этого...


тоесть ранее был выбран пункт Фиолетовый а после перезагрузки сбрасывается на первый.
 
1,417
1,033
как в imgui.Combo можно сделать так что бы при перезагрузке скрипта по умолчанию стоял тот вариант что был выбран до этого...


тоесть ранее был выбран пункт Фиолетовый а после перезагрузки сбрасывается на первый.
сохраняй значение в ини файл и при загрузке считывай
 
  • Нравится
Реакции: vbg

Frapsy

Известный
Проверенный
393
227
Как вставить картинку в окно и центрировать ее? Вставить пытался, но вместо картинки белый квадрат :D А центрировать еще не допер как.. Как бы я не гуглил - застрелите меня, примеров не так уж и густо, если вообще есть :(

Код:
mlogo = imgui.CreateTextureFromFile(getWorkingDirectory() .. '\\MoD-Helper\\images\\logo.png')
imgui.Image(mlogo, imgui.ImVec2(250, 200))
 
  • Нравится
Реакции: wojciech?

AnWu

Известный
Всефорумный модератор
4,778
5,405
Как вставить картинку в окно и центрировать ее? Вставить пытался, но вместо картинки белый квадрат :D А центрировать еще не допер как.. Как бы я не гуглил - застрелите меня, примеров не так уж и густо, если вообще есть :(

Код:
mlogo = imgui.CreateTextureFromFile(getWorkingDirectory() .. '\\MoD-Helper\\images\\logo.png')
imgui.Image(mlogo, imgui.ImVec2(250, 200))
mlogo нужно загружать вне onDrameDraw, при показе проверять mlogo на nil значение.
По поводу белого квадрата хз. Вроде верно всё.
Отцентровать можно с помощью imgui.SetCursorPos(imgui.ImVec2(x, y))
Скинь саму картинку чтоли.
можешь глянуть у меня в темах ImGui SA:MP Scoreboard, там целая система тем сделана и загрузка изображений есть.
 

Frapsy

Известный
Проверенный
393
227
mlogo нужно загружать вне onDrameDraw, при показе проверять mlogo на nil значение.
По поводу белого квадрата хз. Вроде верно всё.
Отцентровать можно с помощью imgui.SetCursorPos(imgui.ImVec2(x, y))
Скинь саму картинку чтоли.
можешь глянуть у меня в темах ImGui SA:MP Scoreboard, там целая система тем сделана и загрузка изображений есть.

Вынес mlogo перед onFrameDraw, при показе, уже в onDrawFrame, проверку на nil значение сделал в этом виде:
Код:
if mlogo then 
        imgui.Image(mlogo, imgui.ImVec2(250, 200))
else
        imgui.Text("Error with image loading")
end

Текста нет, но есть белый квадрат :D Проверку на то, видит ли скрипт картинку или нет сделал в виде:
Код:
if doesFileExist("moonloader\\MoD-Helper\\images\\logo.png") then
    mlogo = imgui.CreateTextureFromFile('moonloader\\MoD-Helper\\images\\logo.png')
    sampAddChatMessage("exist", -1)
else
    sampAddChatMessage("not exist", -1)
    mlogo = nil
end

И хоть убейте, белый квадрат... Сам кусок кода из onDrawFrame:

Код:
if win_state['info'].v then
        imgui.SetNextWindowPos(imgui.ImVec2(sw/2, sh/2), imgui.Cond.FirstUseEver, imgui.ImVec2(0.5, 0.5))
        imgui.SetNextWindowSize(imgui.ImVec2(915, 400), imgui.Cond.FirstUseEver)
        imgui.Begin(u8('Информация'), win_state['info'], imgui.WindowFlags.NoResize)
        imgui.BeginChild('left pane', imgui.ImVec2(200, 0), true)
        if not selected then selected = 1 end
        if imgui.Selectable(u8('один'), false) then selected = 1 end
        imgui.Separator()
        if imgui.Selectable(u8('два'), false) then selected = 2 end
        imgui.Separator()
        if imgui.Selectable(u8('три'), false) then selected = 3 end
        imgui.Separator()
        if imgui.Selectable(u8('четыре'), false) then selected = 4 end
        imgui.Separator()
        if imgui.Selectable(u8('пять'), false) then selected = 5 end
        imgui.EndChild()
        imgui.SameLine()
        imgui.BeginGroup()
        if selected == 1 then
            --код какой нить
        end
        if selected == 2 then
            if mlogo then
                --imgui.SetCursorPos(imgui.ImVec2(250, 200))
                imgui.Image(mlogo, imgui.ImVec2(250, 200))
            else
                imgui.Text("Error with image loading")
            end
        end
        -- ...
        imgui.EndGroup()
        imgui.End()
    end
 
  • Нравится
Реакции: FBenz

Alkasch226

Известный
43
4
Можно ли как-то отцентровать текст ровно посередине(допустим, по горизонтали) окна imgui?
 

AnWu

Известный
Всефорумный модератор
4,778
5,405
Можно ли как-то отцентровать текст ровно посередине(допустим, по горизонтали) окна imgui?
local size = imgui.CalcTextSize("ТРАЛАЛА")
size.x - ширина, size.y - высота. Дели высоту пополам, отнимай от половины высоты окна.

Вынес mlogo перед onFrameDraw, при показе, уже в onDrawFrame, проверку на nil значение сделал в этом виде:
Код:
if mlogo then
        imgui.Image(mlogo, imgui.ImVec2(250, 200))
else
        imgui.Text("Error with image loading")
end

Текста нет, но есть белый квадрат :D Проверку на то, видит ли скрипт картинку или нет сделал в виде:
Код:
if doesFileExist("moonloader\\MoD-Helper\\images\\logo.png") then
    mlogo = imgui.CreateTextureFromFile('moonloader\\MoD-Helper\\images\\logo.png')
    sampAddChatMessage("exist", -1)
else
    sampAddChatMessage("not exist", -1)
    mlogo = nil
end

И хоть убейте, белый квадрат... Сам кусок кода из onDrawFrame:

Код:
if win_state['info'].v then
        imgui.SetNextWindowPos(imgui.ImVec2(sw/2, sh/2), imgui.Cond.FirstUseEver, imgui.ImVec2(0.5, 0.5))
        imgui.SetNextWindowSize(imgui.ImVec2(915, 400), imgui.Cond.FirstUseEver)
        imgui.Begin(u8('Информация'), win_state['info'], imgui.WindowFlags.NoResize)
        imgui.BeginChild('left pane', imgui.ImVec2(200, 0), true)
        if not selected then selected = 1 end
        if imgui.Selectable(u8('один'), false) then selected = 1 end
        imgui.Separator()
        if imgui.Selectable(u8('два'), false) then selected = 2 end
        imgui.Separator()
        if imgui.Selectable(u8('три'), false) then selected = 3 end
        imgui.Separator()
        if imgui.Selectable(u8('четыре'), false) then selected = 4 end
        imgui.Separator()
        if imgui.Selectable(u8('пять'), false) then selected = 5 end
        imgui.EndChild()
        imgui.SameLine()
        imgui.BeginGroup()
        if selected == 1 then
            --код какой нить
        end
        if selected == 2 then
            if mlogo then
                --imgui.SetCursorPos(imgui.ImVec2(250, 200))
                imgui.Image(mlogo, imgui.ImVec2(250, 200))
            else
                imgui.Text("Error with image loading")
            end
        end
        -- ...
        imgui.EndGroup()
        imgui.End()
    end
Картинку покажи
 
  • Нравится
Реакции: Alkasch226

Frapsy

Известный
Проверенный
393
227
Полный код дай.
в spoiler + code.
Держи.

Код:
script_name('Moon ImGui Example')
script_author('FYP')
script_description('Demonstrates Moon ImGui features')

local key = require 'vkeys'
local imgui = require 'imgui'
local pie = require 'imgui_piemenu'
local encoding = require 'encoding'
encoding.default = 'CP1251'
u8 = encoding.UTF8

local mlogo = nil

function apply_custom_style()
    imgui.SwitchContext()
    local style = imgui.GetStyle()
    local colors = style.Colors
    local clr = imgui.Col
    local ImVec4 = imgui.ImVec4

    style.WindowRounding = 2.0
    style.WindowTitleAlign = imgui.ImVec2(0.5, 0.84)
    style.ChildWindowRounding = 2.0
    style.FrameRounding = 2.0
    style.ItemSpacing = imgui.ImVec2(5.0, 4.0)
    style.ScrollbarSize = 13.0
    style.ScrollbarRounding = 0
    style.GrabMinSize = 8.0
    style.GrabRounding = 1.0
    -- style.Alpha =
    -- style.WindowPadding =
    -- style.WindowMinSize =
    -- style.FramePadding =
    -- style.ItemInnerSpacing =
    -- style.TouchExtraPadding =
    -- style.IndentSpacing =
    -- style.ColumnsMinSpacing = ?
    -- style.ButtonTextAlign =
    -- style.DisplayWindowPadding =
    -- style.DisplaySafeAreaPadding =
    -- style.AntiAliasedLines =
    -- style.AntiAliasedShapes =
    -- style.CurveTessellationTol =

    colors[clr.Text]                   = ImVec4(1.00, 1.00, 1.00, 1.00)
    colors[clr.TextDisabled]           = ImVec4(0.50, 0.50, 0.50, 1.00)
    colors[clr.WindowBg]               = ImVec4(0.06, 0.06, 0.06, 0.94)
    colors[clr.ChildWindowBg]          = ImVec4(1.00, 1.00, 1.00, 0.00)
    colors[clr.PopupBg]                = ImVec4(0.08, 0.08, 0.08, 0.94)
    colors[clr.ComboBg]                = colors[clr.PopupBg]
    colors[clr.Border]                 = ImVec4(0.43, 0.43, 0.50, 0.50)
    colors[clr.BorderShadow]           = ImVec4(0.00, 0.00, 0.00, 0.00)
    colors[clr.FrameBg]                = ImVec4(0.16, 0.29, 0.48, 0.54)
    colors[clr.FrameBgHovered]         = ImVec4(0.26, 0.59, 0.98, 0.40)
    colors[clr.FrameBgActive]          = ImVec4(0.26, 0.59, 0.98, 0.67)
    colors[clr.TitleBg]                = ImVec4(0.04, 0.04, 0.04, 1.00)
    colors[clr.TitleBgActive]          = ImVec4(0.16, 0.29, 0.48, 1.00)
    colors[clr.TitleBgCollapsed]       = ImVec4(0.00, 0.00, 0.00, 0.51)
    colors[clr.MenuBarBg]              = ImVec4(0.14, 0.14, 0.14, 1.00)
    colors[clr.ScrollbarBg]            = ImVec4(0.02, 0.02, 0.02, 0.53)
    colors[clr.ScrollbarGrab]          = ImVec4(0.31, 0.31, 0.31, 1.00)
    colors[clr.ScrollbarGrabHovered]   = ImVec4(0.41, 0.41, 0.41, 1.00)
    colors[clr.ScrollbarGrabActive]    = ImVec4(0.51, 0.51, 0.51, 1.00)
    colors[clr.CheckMark]              = ImVec4(0.26, 0.59, 0.98, 1.00)
    colors[clr.SliderGrab]             = ImVec4(0.24, 0.52, 0.88, 1.00)
    colors[clr.SliderGrabActive]       = ImVec4(0.26, 0.59, 0.98, 1.00)
    colors[clr.Button]                 = ImVec4(0.26, 0.59, 0.98, 0.40)
    colors[clr.ButtonHovered]          = ImVec4(0.26, 0.59, 0.98, 1.00)
    colors[clr.ButtonActive]           = ImVec4(0.06, 0.53, 0.98, 1.00)
    colors[clr.Header]                 = ImVec4(0.26, 0.59, 0.98, 0.31)
    colors[clr.HeaderHovered]          = ImVec4(0.26, 0.59, 0.98, 0.80)
    colors[clr.HeaderActive]           = ImVec4(0.26, 0.59, 0.98, 1.00)
    colors[clr.Separator]              = colors[clr.Border]
    colors[clr.SeparatorHovered]       = ImVec4(0.26, 0.59, 0.98, 0.78)
    colors[clr.SeparatorActive]        = ImVec4(0.26, 0.59, 0.98, 1.00)
    colors[clr.ResizeGrip]             = ImVec4(0.26, 0.59, 0.98, 0.25)
    colors[clr.ResizeGripHovered]      = ImVec4(0.26, 0.59, 0.98, 0.67)
    colors[clr.ResizeGripActive]       = ImVec4(0.26, 0.59, 0.98, 0.95)
    colors[clr.CloseButton]            = ImVec4(0.41, 0.41, 0.41, 0.50)
    colors[clr.CloseButtonHovered]     = ImVec4(0.98, 0.39, 0.36, 1.00)
    colors[clr.CloseButtonActive]      = ImVec4(0.98, 0.39, 0.36, 1.00)
    colors[clr.PlotLines]              = ImVec4(0.61, 0.61, 0.61, 1.00)
    colors[clr.PlotLinesHovered]       = ImVec4(1.00, 0.43, 0.35, 1.00)
    colors[clr.PlotHistogram]          = ImVec4(0.90, 0.70, 0.00, 1.00)
    colors[clr.PlotHistogramHovered]   = ImVec4(1.00, 0.60, 0.00, 1.00)
    colors[clr.TextSelectedBg]         = ImVec4(0.26, 0.59, 0.98, 0.35)
    colors[clr.ModalWindowDarkening]   = ImVec4(0.80, 0.80, 0.80, 0.35)
end
apply_custom_style()


local win_state = {}
win_state['main'] = imgui.ImBool(false)
win_state['info'] = imgui.ImBool(false)
win_state['help'] = imgui.ImBool(false)


cb_render_in_menu = imgui.ImBool(imgui.RenderInMenu)
cb_lock_player = imgui.ImBool(imgui.LockPlayer)
cb_show_cursor = imgui.ImBool(imgui.ShowCursor)


if doesFileExist(getGameDirectory() .. '\\moonloader\\MoD-Helper\\images\\logo.png') then
        mlogo = imgui.CreateTextureFromFile(getGameDirectory() .. '\\moonloader\\MoD-Helper\\images\\logo.png')
        --mlogo = imgui.CreateTextureFromFile(getGameDirectory() .. '\\sampgui.png')
        sampAddChatMessage("exist", -1)
    else
        sampAddChatMessage("not exist", -1)
        mlogo = nil
    end




function imgui.OnDrawFrame()
    local sw, sh = getScreenResolution()
    local btn_size = imgui.ImVec2(-0.1, 0)
 
    if win_state['main'].v then
     
        imgui.SetNextWindowPos(imgui.ImVec2(sw / 2, sh / 2), imgui.Cond.FirstUseEver, imgui.ImVec2(0.5, 0.5))
        imgui.SetNextWindowSize(imgui.ImVec2(300, 300), imgui.Cond.FirstUseEver)
        imgui.Begin(u8'Ministry of Defence', win_state['main'], imgui.WindowFlags.NoResize)
        if imgui.Button(u8'Информация', btn_size) then
            win_state['info'].v = not win_state['info'].v
        end
        if imgui.Button(u8'Помощь', btn_size) then
            win_state['help'].v = not win_state['help'].v
        end

        if imgui.Button(u8'Кружочек', btn_size) then
            imgui.OpenPopup('PieMenu')
        end
     
        if pie.BeginPiePopup('PieMenu', 1) then
            if pie.PieMenuItem('TestSub 228') then
                sampAddChatMessage('Test',-1)
                print("TestSub")
            end
            if pie.PieMenuItem('TestSub 1337') then
                sampAddChatMessage('Test2',-1)
                print("TestSub2")
            end
            if pie.BeginPieMenu(u8'Другое') then
                print("Другое")
                if pie.BeginPieMenu(u8'Еще другое') then
                    if pie.PieMenuItem('SubSub') then end
                    if pie.PieMenuItem('SubSub2') then end
                    pie.EndPieMenu()
                end
                pie.EndPieMenu()
            end
            pie.EndPiePopup()
        end

        if imgui.CollapsingHeader('Options') then
            if imgui.Checkbox('Render in menu', cb_render_in_menu) then
                imgui.RenderInMenu = cb_render_in_menu.v
            end
            if imgui.Checkbox('Lock player controls', cb_lock_player) then
                imgui.LockPlayer = cb_lock_player.v
            end
            if imgui.Checkbox('Show cursor', cb_show_cursor) then
                imgui.ShowCursor = cb_show_cursor.v
            end
        end
        imgui.End()
    end

 
    if win_state['info'].v then
        imgui.SetNextWindowPos(imgui.ImVec2(sw/2, sh/2), imgui.Cond.FirstUseEver, imgui.ImVec2(0.5, 0.5))
        imgui.SetNextWindowSize(imgui.ImVec2(915, 400), imgui.Cond.FirstUseEver)
        imgui.Begin(u8('Информация'), win_state['info'], imgui.WindowFlags.NoResize)
        imgui.BeginChild('left pane', imgui.ImVec2(200, 0), true)
        if not selected then selected = 1 end
        if imgui.Selectable(u8('один'), false) then selected = 1 end
        imgui.Separator()
        if imgui.Selectable(u8('два'), false) then selected = 2 end
        imgui.Separator()
        if imgui.Selectable(u8('три'), false) then selected = 3 end
        imgui.Separator()
        if imgui.Selectable(u8('четыре'), false) then selected = 4 end
        imgui.Separator()
        if imgui.Selectable(u8('пять'), false) then selected = 5 end
        imgui.EndChild()
        imgui.SameLine()
        imgui.BeginGroup()
        if selected == 1 then
            imgui.Text("text 228")
        end
        if selected == 2 then
            if mlogo then
                imgui.Image(mlogo, imgui.ImVec2(250, 200))
            else
                imgui.Text("Error with image loading")
            end
        end
        -- ...
        imgui.EndGroup()
        imgui.End()
    end
    if win_state['help'].v then
        imgui.SetNextWindowPos(imgui.ImVec2(650, 20), imgui.Cond.FirstUseEver)
        imgui.ShowTestWindow(show_test_window)
    end

end


function main()
    if not isSampLoaded() or not isSampfuncsLoaded() then return end
    while not isSampAvailable() do wait(100) end
 
 
    while true do
        wait(0)
        if wasKeyPressed(key.VK_B) and not sampIsChatInputActive() and not sampIsDialogActive() then
            win_state['main'].v = not win_state['main'].v
        end
        imgui.Process = win_state['main'].v
    end
end
 
Последнее редактирование:

Frapsy

Известный
Проверенный
393
227
Не вижу особых проблем. Попробуй пустой скрипт с показом изображения. Может что-то мешает.
запустив только этот скрипт, убрав все остальные, сняв блокировку от винды 10 с картинки - как было... Чертовый белый квадратик.
Imgur(https://imgur.com/a/4g5NJ69)

Что мешать то может..
upd: cкинул знакомому - тоже белый квадрат ;D
 
Последнее редактирование: