127 lines
3.1 KiB
C++
127 lines
3.1 KiB
C++
#include "Window.h"
|
|
|
|
#include <BAN/Debug.h>
|
|
#include <BAN/ScopeGuard.h>
|
|
|
|
#include <LibGUI/Window.h>
|
|
|
|
#include <sys/banan-os.h>
|
|
#include <sys/mman.h>
|
|
#include <sys/socket.h>
|
|
#include <unistd.h>
|
|
|
|
Window::~Window()
|
|
{
|
|
munmap(m_fb_addr, client_width() * client_height() * 4);
|
|
smo_delete(m_smo_key);
|
|
|
|
LibGUI::EventPacket::DestroyWindowEvent packet;
|
|
|
|
BAN::Vector<uint8_t> buffer;
|
|
if (!buffer.resize(packet.serialized_size()).is_error())
|
|
{
|
|
packet.serialize(buffer.span());
|
|
|
|
size_t total_sent = 0;
|
|
while (total_sent < buffer.size())
|
|
{
|
|
const ssize_t nsend = send(m_client_fd, buffer.data() + total_sent, buffer.size() - total_sent, 0);
|
|
if (nsend <= 0)
|
|
break;
|
|
total_sent += nsend;
|
|
}
|
|
}
|
|
|
|
close(m_client_fd);
|
|
}
|
|
|
|
BAN::ErrorOr<void> Window::initialize(BAN::StringView title, uint32_t width, uint32_t height)
|
|
{
|
|
m_title.clear();
|
|
TRY(m_title.append(title));
|
|
TRY(resize(width, height));
|
|
return {};
|
|
}
|
|
|
|
BAN::ErrorOr<void> Window::resize(uint32_t width, uint32_t height)
|
|
{
|
|
const size_t fb_bytes = width * height * 4;
|
|
|
|
long smo_key = smo_create(fb_bytes, PROT_READ | PROT_WRITE);
|
|
if (smo_key == -1)
|
|
return BAN::Error::from_errno(errno);
|
|
BAN::ScopeGuard smo_deleter([&]() { smo_delete(smo_key); });
|
|
|
|
uint32_t* fb_addr = static_cast<uint32_t*>(smo_map(smo_key));
|
|
if (fb_addr == nullptr)
|
|
return BAN::Error::from_errno(errno);
|
|
memset(fb_addr, 0xFF, fb_bytes);
|
|
BAN::ScopeGuard smo_unmapper([&]() { munmap(fb_addr, fb_bytes); });
|
|
|
|
{
|
|
const auto old_area = m_client_area;
|
|
|
|
m_client_area.max_x = m_client_area.min_x + width;
|
|
m_client_area.max_y = m_client_area.min_y + height;
|
|
auto title_bar_ret = prepare_title_bar();
|
|
m_client_area = old_area;
|
|
|
|
if (title_bar_ret.is_error())
|
|
return title_bar_ret.release_error();
|
|
}
|
|
|
|
smo_deleter.disable();
|
|
smo_unmapper.disable();
|
|
|
|
if (m_fb_addr)
|
|
munmap(m_fb_addr, client_width() * client_height() * 4);
|
|
if (m_smo_key)
|
|
smo_delete(m_smo_key);
|
|
|
|
m_fb_addr = fb_addr;
|
|
m_smo_key = smo_key;
|
|
|
|
m_client_area.max_x = m_client_area.min_x + width;
|
|
m_client_area.max_y = m_client_area.min_y + height;
|
|
|
|
return {};
|
|
}
|
|
|
|
BAN::ErrorOr<void> Window::prepare_title_bar()
|
|
{
|
|
const uint32_t font_w = m_font.width();
|
|
const uint32_t font_h = m_font.height();
|
|
const uint32_t font_p = m_font.pitch();
|
|
|
|
TRY(m_title_bar_data.resize(title_bar_width() * title_bar_height()));
|
|
for (auto& pixel : m_title_bar_data)
|
|
pixel = 0xFFFFFFFF;
|
|
|
|
const auto text_area = title_text_area();
|
|
|
|
for (size_t i = 0; i < m_title.size() && (i + 1) * font_w < static_cast<uint32_t>(text_area.width()); i++)
|
|
{
|
|
const auto* glyph = m_font.glyph(m_title[i]);
|
|
if (glyph == nullptr)
|
|
continue;
|
|
|
|
const int32_t y_off = (font_h < (uint32_t)title_bar_height()) ? (title_bar_height() - font_h) / 2 : 0;
|
|
const int32_t x_off = y_off + i * font_w;
|
|
for (int32_t y = 0; (uint32_t)y < font_h; y++)
|
|
{
|
|
if (y + y_off >= title_bar_height())
|
|
break;
|
|
for (int32_t x = 0; (uint32_t)x < font_w; x++)
|
|
{
|
|
if (x + x_off >= text_area.width())
|
|
break;
|
|
const uint8_t bitmask = 1 << (font_w - x - 1);
|
|
if (glyph[y * font_p] & bitmask)
|
|
m_title_bar_data[(y_off + y) * title_bar_width() + (x_off + x)] = 0xFF000000;
|
|
}
|
|
}
|
|
}
|
|
|
|
return {};
|
|
}
|