2021-11-21 11:25:26 +01:00
|
|
|
#include "Bar.hpp"
|
|
|
|
|
|
|
|
#include <codecvt>
|
|
|
|
#include <locale>
|
|
|
|
|
|
|
|
#include "../windowManager.hpp"
|
|
|
|
|
|
|
|
void CStatusBar::setup(Vector2D origin, Vector2D size) {
|
2021-11-21 12:40:03 +01:00
|
|
|
if (origin.x != -1 && origin.y != -1) {
|
|
|
|
m_vecPosition = origin;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size.x != -1 && size.y != -1) {
|
|
|
|
m_vecSize = size;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t values[4];
|
|
|
|
|
|
|
|
// window
|
|
|
|
m_iWindowID = (xcb_generate_id(g_pWindowManager->DisplayConnection));
|
|
|
|
|
|
|
|
values[0] = XCB_EVENT_MASK_EXPOSURE | XCB_EVENT_MASK_SUBSTRUCTURE_REDIRECT | XCB_EVENT_MASK_SUBSTRUCTURE_NOTIFY | XCB_EVENT_MASK_PROPERTY_CHANGE;
|
|
|
|
|
|
|
|
xcb_create_window(g_pWindowManager->DisplayConnection, g_pWindowManager->Depth, m_iWindowID,
|
|
|
|
g_pWindowManager->Screen->root, m_vecPosition.x, m_vecPosition.y, m_vecSize.x, m_vecSize.y,
|
|
|
|
0, XCB_WINDOW_CLASS_INPUT_OUTPUT, g_pWindowManager->Screen->root_visual,
|
|
|
|
XCB_CW_EVENT_MASK, values);
|
|
|
|
|
|
|
|
// map
|
|
|
|
xcb_map_window(g_pWindowManager->DisplayConnection, m_iWindowID);
|
2021-11-21 11:25:26 +01:00
|
|
|
|
|
|
|
// Create a pixmap for writing to.
|
|
|
|
m_iPixmap = xcb_generate_id(g_pWindowManager->DisplayConnection);
|
|
|
|
xcb_create_pixmap(g_pWindowManager->DisplayConnection, g_pWindowManager->Depth, m_iPixmap, m_iWindowID, m_vecSize.x, m_vecSize.y);
|
|
|
|
|
2021-11-21 12:40:03 +01:00
|
|
|
// setup contexts.. ugh..
|
2021-11-21 11:25:26 +01:00
|
|
|
|
|
|
|
auto contextBG = &m_mContexts["BG"];
|
|
|
|
contextBG->GContext = xcb_generate_id(g_pWindowManager->DisplayConnection);
|
|
|
|
|
|
|
|
values[0] = 0x111111;
|
|
|
|
values[1] = 0x111111;
|
|
|
|
xcb_create_gc(g_pWindowManager->DisplayConnection, contextBG->GContext, m_iPixmap, XCB_GC_BACKGROUND | XCB_GC_FOREGROUND, values);
|
|
|
|
|
|
|
|
//
|
|
|
|
//
|
|
|
|
|
|
|
|
auto contextBASETEXT = &m_mContexts["BASETEXT"];
|
|
|
|
|
|
|
|
contextBASETEXT->GContext = xcb_generate_id(g_pWindowManager->DisplayConnection);
|
|
|
|
contextBASETEXT->Font = xcb_generate_id(g_pWindowManager->DisplayConnection);
|
|
|
|
xcb_open_font(g_pWindowManager->DisplayConnection, contextBASETEXT->Font, 5, "fixed");
|
|
|
|
values[0] = 0xFFFFFF;
|
|
|
|
values[1] = 0x111111;
|
|
|
|
values[2] = contextBASETEXT->Font;
|
|
|
|
|
|
|
|
xcb_create_gc(g_pWindowManager->DisplayConnection, contextBASETEXT->GContext, m_iPixmap, XCB_GC_BACKGROUND | XCB_GC_FOREGROUND | XCB_GC_FONT, values);
|
|
|
|
|
|
|
|
//
|
|
|
|
//
|
|
|
|
|
|
|
|
auto contextHITEXT = &m_mContexts["HITEXT"];
|
|
|
|
contextHITEXT->GContext = xcb_generate_id(g_pWindowManager->DisplayConnection);
|
|
|
|
contextHITEXT->Font = contextBASETEXT->Font;
|
|
|
|
values[0] = 0x000000;
|
|
|
|
values[1] = 0xFF3333;
|
|
|
|
values[2] = contextHITEXT->Font;
|
|
|
|
|
|
|
|
xcb_create_gc(g_pWindowManager->DisplayConnection, contextHITEXT->GContext, m_iPixmap, XCB_GC_BACKGROUND | XCB_GC_FOREGROUND | XCB_GC_FONT, values);
|
|
|
|
|
|
|
|
//
|
|
|
|
//
|
|
|
|
|
|
|
|
auto contextMEDBG = &m_mContexts["MEDBG"];
|
|
|
|
contextMEDBG->GContext = xcb_generate_id(g_pWindowManager->DisplayConnection);
|
|
|
|
|
|
|
|
values[0] = 0xFF3333;
|
|
|
|
values[1] = 0x111111;
|
|
|
|
xcb_create_gc(g_pWindowManager->DisplayConnection, contextMEDBG->GContext, m_iPixmap, XCB_GC_BACKGROUND | XCB_GC_FOREGROUND, values);
|
|
|
|
|
|
|
|
// don't, i use it later
|
|
|
|
//xcb_close_font(g_pWindowManager->DisplayConnection, contextBASETEXT->Font);
|
|
|
|
}
|
|
|
|
|
2021-11-21 12:40:03 +01:00
|
|
|
void CStatusBar::destroy() {
|
|
|
|
xcb_close_font(g_pWindowManager->DisplayConnection, m_mContexts["HITEXT"].Font);
|
|
|
|
xcb_destroy_window(g_pWindowManager->DisplayConnection, m_iWindowID);
|
|
|
|
xcb_destroy_window(g_pWindowManager->DisplayConnection, m_iPixmap);
|
|
|
|
|
|
|
|
xcb_free_gc(g_pWindowManager->DisplayConnection, m_mContexts["BG"].GContext);
|
|
|
|
xcb_free_gc(g_pWindowManager->DisplayConnection, m_mContexts["MEDBG"].GContext);
|
|
|
|
xcb_free_gc(g_pWindowManager->DisplayConnection, m_mContexts["TEXT"].GContext);
|
|
|
|
xcb_free_gc(g_pWindowManager->DisplayConnection, m_mContexts["HITEXT"].GContext);
|
|
|
|
}
|
|
|
|
|
2021-11-21 11:25:26 +01:00
|
|
|
int getTextWidth(std::string text, xcb_font_t font) {
|
|
|
|
|
|
|
|
// conv from utf8 to UCS-2 (what the fuck Xorg why)
|
|
|
|
std::wstring_convert<std::codecvt_utf8<wchar_t>> strCnv;
|
|
|
|
std::wstring wideString = strCnv.from_bytes(text);
|
|
|
|
|
|
|
|
// create a xcb string
|
|
|
|
xcb_char2b_t bytes[wideString.length()];
|
|
|
|
|
|
|
|
for (int i = 0; i < wideString.length(); ++i) {
|
|
|
|
bytes[i].byte1 = 0x0; // Only ASCII support. TODO: Maybe more?
|
|
|
|
bytes[i].byte2 = wideString[i] & 0xFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
xcb_generic_error_t* error;
|
|
|
|
const auto COOKIE = xcb_query_text_extents(g_pWindowManager->DisplayConnection, font, wideString.length() - 1, (xcb_char2b_t*)bytes);
|
|
|
|
xcb_query_text_extents_reply_t* reply = xcb_query_text_extents_reply(g_pWindowManager->DisplayConnection, COOKIE, &error);
|
|
|
|
if (!reply) {
|
|
|
|
Debug::log(ERR, "Text extent failed, code " + std::to_string(error->error_code));
|
|
|
|
free(error);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto WIDTH = reply->overall_width;
|
|
|
|
free(reply);
|
2021-11-21 20:17:49 +01:00
|
|
|
return WIDTH + 5;
|
2021-11-21 11:25:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CStatusBar::draw() {
|
|
|
|
|
2021-11-21 19:59:59 +01:00
|
|
|
const auto WORKSPACE = g_pWindowManager->getWorkspaceByID(g_pWindowManager->activeWorkspaces[m_iMonitorID]);
|
|
|
|
|
|
|
|
if (WORKSPACE->getHasFullscreenWindow())
|
2021-11-21 11:42:44 +01:00
|
|
|
return; // Do not draw a bar on a fullscreen window.
|
|
|
|
|
2021-11-21 11:25:26 +01:00
|
|
|
xcb_rectangle_t rectangles[] = {{m_vecPosition.x, m_vecPosition.y, m_vecSize.x + m_vecPosition.x, m_vecPosition.y + m_vecSize.y}};
|
|
|
|
xcb_poly_fill_rectangle(g_pWindowManager->DisplayConnection, m_iPixmap, m_mContexts["BG"].GContext, 1, rectangles);
|
|
|
|
|
|
|
|
// Draw workspaces
|
|
|
|
int drawnWorkspaces = 0;
|
|
|
|
for (int i = 0; i <= g_pWindowManager->getHighestWorkspaceID(); ++i) {
|
|
|
|
|
|
|
|
const auto WORKSPACE = g_pWindowManager->getWorkspaceByID(i);
|
|
|
|
|
|
|
|
if (!WORKSPACE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
std::string workspaceName = std::to_string(i);
|
|
|
|
|
2021-11-21 19:59:59 +01:00
|
|
|
if (WORKSPACE->getID() == g_pWindowManager->activeWorkspaces[m_iMonitorID]) {
|
2021-11-21 11:25:26 +01:00
|
|
|
xcb_rectangle_t rectangleActive[] = { { m_vecSize.y * drawnWorkspaces, 0, m_vecSize.y, m_vecSize.y } };
|
|
|
|
xcb_poly_fill_rectangle(g_pWindowManager->DisplayConnection, m_iPixmap, m_mContexts["MEDBG"].GContext, 1, rectangleActive);
|
|
|
|
}
|
|
|
|
|
|
|
|
xcb_image_text_8(g_pWindowManager->DisplayConnection, workspaceName.length(), m_iPixmap,
|
2021-11-21 19:59:59 +01:00
|
|
|
WORKSPACE->getID() == g_pWindowManager->activeWorkspaces[m_iMonitorID] ? m_mContexts["HITEXT"].GContext : m_mContexts["BASETEXT"].GContext,
|
2021-11-21 11:25:26 +01:00
|
|
|
m_vecSize.y * drawnWorkspaces + m_vecSize.y / 2.f - 2, m_vecSize.y - (m_vecSize.y - 10) / 2, workspaceName.c_str());
|
|
|
|
|
|
|
|
drawnWorkspaces++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Draw time to the right
|
2021-11-21 20:17:49 +01:00
|
|
|
std::string TIME = exec("date +%I:%M\\ %p");
|
|
|
|
TIME = TIME.substr(0, TIME.length() - 1);
|
2021-11-21 11:25:26 +01:00
|
|
|
xcb_image_text_8(g_pWindowManager->DisplayConnection, TIME.length(), m_iPixmap,
|
2021-11-21 20:17:49 +01:00
|
|
|
m_mContexts["BASETEXT"].GContext, m_vecSize.x - getTextWidth(TIME, m_mContexts["BASETEXT"].Font), (m_vecSize.y - (m_vecSize.y - 10) / 2),
|
2021-11-21 11:25:26 +01:00
|
|
|
TIME.c_str());
|
|
|
|
|
|
|
|
xcb_flush(g_pWindowManager->DisplayConnection);
|
|
|
|
|
|
|
|
xcb_copy_area(g_pWindowManager->DisplayConnection, m_iPixmap, m_iWindowID, m_mContexts["BG"].GContext,
|
|
|
|
0, 0, 0, 0, m_vecSize.x, m_vecSize.y);
|
|
|
|
}
|