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 · Просмотры: 114,107
Последнее редактирование:

bab0n

Известный
95
10
Можете показать как вставлять изображения в окна имгуи, а изображения, в свою очередь, брать по ссылке из инета, и можно ли это изображение кинуть на задний фон окна?
 

bab0n

Известный
95
10
@FYP

Кто может помочь решить проблему. Сделал функцию проверки Регов, пишу /getip ID в чате скрипт видит строку > Включает функцию проверки,все проверяется хорошо. Аналогичное делаю через sampSendChat('/getip '..rInfo.id) по нажатию клавиши на клавиатуре , все также проверяется хорошо и никаких проблем нет. Но когда это же делаю через imgui , функция так же выполняется,реги выводятся,но через 5-10 секунд после вывода происходит краш,не понимаю в чем проблема,в логе ничего нет,просто выгружаются все скрипты и гта закрывается.
Кнопка
Lua:
    imgui.SetCursorPos(imgui.ImVec2(133, 6))
        if imgui.Button(u8'Реги') then
            sampSendChat('/getip '..rInfo.id)
        end
Последующее выполнение функции когда найден нужный ответ после команды выше
Lua:
function  SE.onServerMessage(collor,msg)
if msg:find('Nik %[(.*)%]  R%-IP %[(.*)%]  IP %[(.*)%]')  then
local reginick,regip,reglast =    msg:match('Nik %[(.*)%]  R%-IP %[(.*)%]  IP %[(.*)%]')
if statusproverkiregov == false then
        lua_thread.create(function()
            regdanny = {}
checkregi(regip,reglast)
while not regdanny[1] or not regdanny[4] do    wait(0) if statuserrorregi == true then statuserrorregi = false return end end
sampAddChatMessage(reginick, -1)
sampAddChatMessage(regdanny[1]..'  '..regdanny[4], -1)
sampAddChatMessage(regdanny[2]..'   '..regdanny[5], -1)
sampAddChatMessage(regdanny[3]..'   '..regdanny[6], -1)
end)
else
    sampAddChatMessage('[Ошибка] Я не могу работать так быстро!', 0xFF44FF)
end
end
end

Сама функция
Lua:
  function checkregi(ipregdanny,iplastregdanny)
        statusproverkiregov=true
                    regdanny = {}
                    regdanny2 = {}
                  lua_thread.create(function()
        async_http_request("GET", 'http://ip-api.com/json/'..u8(ipregdanny)..'?lang=ru' , nil,
        function(response)
        local regvrem = u8:decode(response.text)
    local city4,country3,isp2 = regvrem:match('city%":%"(.*)%"%,%"country%":%"(.*)%"%,%"countryCode.*%,%"isp%":%"(.*)%"%,%"lat"')
regdanny = {country3,city4,isp2}
        end,
        function(err)
        sampAddChatMessage('[ERROR] Не удалось загрузить информацию о IP REG',0xFF0000)
            regdanny[1] = 'ERROR'
        end)
        async_http_request("GET", 'http://ip-api.com/json/'..u8(iplastregdanny)..'?lang=ru' , nil,
        function(response)
        local regvrem = u8:decode(response.text)
        local city1,country1,isp1 = regvrem:match('city%":%"(.*)%"%,%"country%":%"(.*)%"%,%"countryCode.*%,%"isp%":%"(.*)%"%,%"lat"')
    regdanny2 = {country1,city1,isp1}
        end,
        function(err)
        sampAddChatMessage('[ERROR] Не удалось загрузить информацию о IP LAST',0xFF0000)
        regdanny2[1] = 'ERROR'
        end)
while not regdanny2[1] or not regdanny[1] do wait(0) end
if not regdanny2[1]:find('ERROR') or not regdanny[1]:find('ERROR') then
regdanny = {regdanny[1],regdanny[2],regdanny[3],regdanny2[1],regdanny2[2],regdanny2[3]}
wait(500)
statusproverkiregov=false
else
        sampAddChatMessage('[Ошибка] Произошла ошибка проверки. Повторите попытку.', -1)
        statuserrorregi = true
end
end)
end
Попробуй убивать в конце каждый поток (return)
 

bab0n

Известный
95
10
Как использовать Combo в MImGui?
Lua:
if imgui.Combo(u8'', selected_item, {u8'Заголовок','1, '2', '3', '4'}, 5) then
if selected_item.v == 0 then
    -- Функция заголовка. Также их функции можно убрать и тогда это пустой текст
  end
if selected_item.v == 1 then
    sampAddChatMessage('1', -1)
  end
    if selected_item.v == 2 then
        sampAddChatMessage('2', -1)
    end
    if selected_item.v == 3 then
        sampAddChatMessage('3', -1)
    end
    if selected_item.v == 4 then
        sampAddChatMessage('4', -1)
    end
end

Также перед Комбо добавь
Lua:
selected_item = imgui.ImInt(0)
0 - это пункт который будет выбран изначально
 

Musaigen

abobusnik
Проверенный
1,583
1,302
Lua:
if imgui.Combo(u8'', selected_item, {u8'Заголовок','1, '2', '3', '4'}, 5) then
if selected_item.v == 0 then
    -- Функция заголовка. Также их функции можно убрать и тогда это пустой текст
  end
if selected_item.v == 1 then
    sampAddChatMessage('1', -1)
  end
    if selected_item.v == 2 then
        sampAddChatMessage('2', -1)
    end
    if selected_item.v == 3 then
        sampAddChatMessage('3', -1)
    end
    if selected_item.v == 4 then
        sampAddChatMessage('4', -1)
    end
end

Также перед Комбо добавь
Lua:
selected_item = imgui.ImInt(0)
0 - это пункт который будет выбран изначально
Видимо ты не понимаешь что такое MImGui
 

AnWu

Guardian of Order
Всефорумный модератор
4,690
5,192
@FYP

Кто может помочь решить проблему. Сделал функцию проверки Регов, пишу /getip ID в чате скрипт видит строку > Включает функцию проверки,все проверяется хорошо. Аналогичное делаю через sampSendChat('/getip '..rInfo.id) по нажатию клавиши на клавиатуре , все также проверяется хорошо и никаких проблем нет. Но когда это же делаю через imgui , функция так же выполняется,реги выводятся,но через 5-10 секунд после вывода происходит краш,не понимаю в чем проблема,в логе ничего нет,просто выгружаются все скрипты и гта закрывается.
Кнопка
Lua:
    imgui.SetCursorPos(imgui.ImVec2(133, 6))
        if imgui.Button(u8'Реги') then
            sampSendChat('/getip '..rInfo.id)
        end
Последующее выполнение функции когда найден нужный ответ после команды выше
Lua:
function  SE.onServerMessage(collor,msg)
if msg:find('Nik %[(.*)%]  R%-IP %[(.*)%]  IP %[(.*)%]')  then
local reginick,regip,reglast =    msg:match('Nik %[(.*)%]  R%-IP %[(.*)%]  IP %[(.*)%]')
if statusproverkiregov == false then
        lua_thread.create(function()
            regdanny = {}
checkregi(regip,reglast)
while not regdanny[1] or not regdanny[4] do    wait(0) if statuserrorregi == true then statuserrorregi = false return end end
sampAddChatMessage(reginick, -1)
sampAddChatMessage(regdanny[1]..'  '..regdanny[4], -1)
sampAddChatMessage(regdanny[2]..'   '..regdanny[5], -1)
sampAddChatMessage(regdanny[3]..'   '..regdanny[6], -1)
end)
else
    sampAddChatMessage('[Ошибка] Я не могу работать так быстро!', 0xFF44FF)
end
end
end

Сама функция
Lua:
  function checkregi(ipregdanny,iplastregdanny)
        statusproverkiregov=true
                    regdanny = {}
                    regdanny2 = {}
                  lua_thread.create(function()
        async_http_request("GET", 'http://ip-api.com/json/'..u8(ipregdanny)..'?lang=ru' , nil,
        function(response)
        local regvrem = u8:decode(response.text)
    local city4,country3,isp2 = regvrem:match('city%":%"(.*)%"%,%"country%":%"(.*)%"%,%"countryCode.*%,%"isp%":%"(.*)%"%,%"lat"')
regdanny = {country3,city4,isp2}
        end,
        function(err)
        sampAddChatMessage('[ERROR] Не удалось загрузить информацию о IP REG',0xFF0000)
            regdanny[1] = 'ERROR'
        end)
        async_http_request("GET", 'http://ip-api.com/json/'..u8(iplastregdanny)..'?lang=ru' , nil,
        function(response)
        local regvrem = u8:decode(response.text)
        local city1,country1,isp1 = regvrem:match('city%":%"(.*)%"%,%"country%":%"(.*)%"%,%"countryCode.*%,%"isp%":%"(.*)%"%,%"lat"')
    regdanny2 = {country1,city1,isp1}
        end,
        function(err)
        sampAddChatMessage('[ERROR] Не удалось загрузить информацию о IP LAST',0xFF0000)
        regdanny2[1] = 'ERROR'
        end)
while not regdanny2[1] or not regdanny[1] do wait(0) end
if not regdanny2[1]:find('ERROR') or not regdanny[1]:find('ERROR') then
regdanny = {regdanny[1],regdanny[2],regdanny[3],regdanny2[1],regdanny2[2],regdanny2[3]}
wait(500)
statusproverkiregov=false
else
        sampAddChatMessage('[Ошибка] Произошла ошибка проверки. Повторите попытку.', -1)
        statuserrorregi = true
end
end)
end
Нельзя использовать потоки в событии. Баг. Переносит на другой кадр через создание другого потока (как бы странно это не звучало)
 

FBenz

Активный
328
40
Как завершить imgui.Columns()? В файле imgui поискал - не нашел.
У меня вот такой код:
Lua:
imgui.Columns(2, 1, true)

 imgui.Button(u8'Один')
 imgui.NextColumn()

 imgui.Button(u8'Два')
imgui.NextColumn()

   imgui.Text(u8'Проверочный текст')
Так вот "проверочный текст" почему-то отображается в первой колонне, а не под всеми колоннами. То есть как будто я написал:
Lua:
imgui.Columns(2, 1, true)

  imgui.Button(u8'Один')
  imgui.Text(u8'Проверочный текст')
  imgui.NextColumn()

  imgui.Button(u8'Два')
imgui.NextColumn()
Как вылечить? Как сделать текст ПОД колонной?
UPD: Разобрался. Может, кому поможет:
В конце добавить imgui.Columns(1)
 
Последнее редактирование:

Adventurer

Известный
151
69
upload_2019-2-5_3-21-4.png

Было зафиксировано моим тестером при проверке одного из моего проектов..
Появляется только в случае, если было открыто imgui окно. С ошибками, касающихся imgui, сталкиваюсь впервые. Хотелось бы узнать, что это вообще такое, и как с этим бороться.
 

Shamanhcik

Известный
33
7
Почему при добавлении картинки в imgui, она загружается белой.
Lua:
logo = nil

logo = imgui.CreateTextureFromFile(getGameDirectory() .. '\\moonloader\\MJHelper\\images\\logo.png')

imgui.Image(logo, imgui.ImVec2(300, 160))
 

Leatington

Известный
258
71
Почему при добавлении картинки в imgui, она загружается белой.
Lua:
logo = nil

logo = imgui.CreateTextureFromFile(getGameDirectory() .. '\\moonloader\\MJHelper\\images\\logo.png')

imgui.Image(logo, imgui.ImVec2(300, 160))
Возьми любое другое название кроме "logo", из-за этого проблема.

Посмотреть вложение 24783
Было зафиксировано моим тестером при проверке одного из моего проектов..
Появляется только в случае, если было открыто imgui окно. С ошибками, касающихся imgui, сталкиваюсь впервые. Хотелось бы узнать, что это вообще такое, и как с этим бороться.
Код imgui окна, при открытии которого возникает краш дай
 

kotov

Участник
128
12
Посмотреть вложение 24783
Было зафиксировано моим тестером при проверке одного из моего проектов..
Появляется только в случае, если было открыто imgui окно. С ошибками, касающихся imgui, сталкиваюсь впервые. Хотелось бы узнать, что это вообще такое, и как с этим бороться.
Проверь везде ли закрывается окно imgui.End(), imgui.EndChild() и т.д. У меня только из за этого такая ошибка бывала