Mercurial > minori
comparison dep/animone/src/win/x11.cc @ 258:862d0d8619f6
*: HUUUGE changes
animia has been renamed to animone, so instead of thinking of a
health condition, you think of a beautiful flower :)
I've also edited some of the code for animone, but I have no idea
if it even works or not because I don't have a mac or windows
machine lying around. whoops!
... anyway, all of the changes divergent from Anisthesia are now
licensed under BSD. it's possible that I could even rewrite most
of the code to where I don't even have to keep the MIT license,
but that's thinking too far into the future
I've been slacking off on implementing the anime seasons page,
mostly out of laziness. I think I'd have to create another db file
specifically for the seasons
anyway, this code is being pushed *primarily* because the hard drive
it's on is failing! yay :)
| author | Paper <paper@paper.us.eu.org> |
|---|---|
| date | Mon, 01 Apr 2024 02:43:44 -0400 |
| parents | |
| children | 1a6a5d3a94cd |
comparison
equal
deleted
inserted
replaced
| 257:699a20c57dc8 | 258:862d0d8619f6 |
|---|---|
| 1 #include "animone/win/x11.h" | |
| 2 #include "animone.h" | |
| 3 #include "animone/fd.h" /* GetProcessName() */ | |
| 4 #include "animone/win.h" | |
| 5 | |
| 6 #include <xcb/res.h> | |
| 7 #include <xcb/xcb.h> | |
| 8 | |
| 9 #include <climits> | |
| 10 #include <cstdint> | |
| 11 #include <cstring> | |
| 12 #include <set> | |
| 13 #include <string> | |
| 14 | |
| 15 #include <chrono> | |
| 16 | |
| 17 #include <iostream> | |
| 18 | |
| 19 /* This uses XCB (and it uses it *right*), so it should be plenty fast */ | |
| 20 | |
| 21 static size_t str_nlen(const char* s, size_t len) { | |
| 22 size_t i = 0; | |
| 23 for (; i < len && s[i]; i++) | |
| 24 ; | |
| 25 return i; | |
| 26 } | |
| 27 | |
| 28 namespace animone::internal::x11 { | |
| 29 | |
| 30 static bool GetAllTopLevelWindowsEWMH(xcb_connection_t* connection, const std::vector<xcb_window_t>& roots, | |
| 31 std::set<xcb_window_t>& result) { | |
| 32 const xcb_atom_t Atom__NET_CLIENT_LIST = [connection] { | |
| 33 static constexpr std::string_view name = "_NET_CLIENT_LIST"; | |
| 34 xcb_intern_atom_cookie_t cookie = ::xcb_intern_atom(connection, true, name.size(), name.data()); | |
| 35 std::unique_ptr<xcb_intern_atom_reply_t> reply(::xcb_intern_atom_reply(connection, cookie, NULL)); | |
| 36 | |
| 37 xcb_atom_t atom = reply->atom; | |
| 38 | |
| 39 return atom; | |
| 40 }(); | |
| 41 if (Atom__NET_CLIENT_LIST == XCB_ATOM_NONE) | |
| 42 return false; // BTFO | |
| 43 | |
| 44 bool success = false; | |
| 45 | |
| 46 std::vector<xcb_get_property_cookie_t> cookies; | |
| 47 cookies.reserve(roots.size()); | |
| 48 | |
| 49 for (const auto& root : roots) | |
| 50 cookies.push_back(::xcb_get_property(connection, 0, root, Atom__NET_CLIENT_LIST, XCB_ATOM_ANY, 0L, UINT_MAX)); | |
| 51 | |
| 52 for (const auto& cookie : cookies) { | |
| 53 std::unique_ptr<xcb_get_property_reply_t> reply(::xcb_get_property_reply(connection, cookie, NULL)); | |
| 54 | |
| 55 if (reply) { | |
| 56 xcb_window_t* value = reinterpret_cast<xcb_window_t*>(::xcb_get_property_value(reply)); | |
| 57 int len = ::xcb_get_property_value_length(reply); | |
| 58 | |
| 59 for (size_t i = 0; i < len; i++) | |
| 60 result.insert(value[i]); | |
| 61 | |
| 62 success |= true; | |
| 63 } | |
| 64 } | |
| 65 | |
| 66 return success; | |
| 67 } | |
| 68 | |
| 69 /* This is called on every window. What this does is: | |
| 70 * 1. Gets the tree of children | |
| 71 * 2. Searches all children recursively for a WM_STATE property | |
| 72 * 3. If that failed... return the original window | |
| 73 */ | |
| 74 static bool WalkWindows(xcb_connection_t* connection, int depth, xcb_atom_t Atom_WM_STATE, const xcb_window_t* windows, | |
| 75 int windows_len, std::set<xcb_window_t>& result) { | |
| 76 /* The depth we should start returning at. */ | |
| 77 static constexpr int CUTOFF = 2; | |
| 78 | |
| 79 bool success = false; | |
| 80 | |
| 81 std::vector<xcb_query_tree_cookie_t> cookies; | |
| 82 cookies.reserve(windows_len); | |
| 83 | |
| 84 for (int i = 0; i < windows_len; i++) | |
| 85 cookies.push_back(::xcb_query_tree(connection, windows[i])); | |
| 86 | |
| 87 for (const auto& cookie : cookies) { | |
| 88 std::unique_ptr<xcb_query_tree_reply_t> query_tree_reply(::xcb_query_tree_reply(connection, cookie, NULL)); | |
| 89 | |
| 90 xcb_window_t* tree_children = ::xcb_query_tree_children(query_tree_reply); | |
| 91 int tree_children_len = ::xcb_query_tree_children_length(query_tree_reply); | |
| 92 | |
| 93 std::vector<xcb_get_property_cookie_t> state_property_cookies; | |
| 94 state_property_cookies.reserve(tree_children_len); | |
| 95 | |
| 96 for (int i = 0; i < tree_children_len; i++) | |
| 97 state_property_cookies.push_back( | |
| 98 ::xcb_get_property(connection, 0, tree_children[i], Atom_WM_STATE, Atom_WM_STATE, 0, 0)); | |
| 99 | |
| 100 for (int i = 0; i < tree_children_len; i++) { | |
| 101 std::unique_ptr<xcb_get_property_reply_t> get_property_reply( | |
| 102 ::xcb_get_property_reply(connection, state_property_cookies[i], NULL)); | |
| 103 | |
| 104 /* X11 is unfriendly here. what this means is "did the property exist?" */ | |
| 105 if (get_property_reply->format || get_property_reply->type || get_property_reply->length) { | |
| 106 result.insert(tree_children[i]); | |
| 107 if (depth >= CUTOFF) | |
| 108 return true; | |
| 109 | |
| 110 success |= true; | |
| 111 continue; | |
| 112 } | |
| 113 } | |
| 114 | |
| 115 if (WalkWindows(connection, depth + 1, Atom_WM_STATE, tree_children, tree_children_len, result)) { | |
| 116 success |= true; | |
| 117 if (depth >= CUTOFF) | |
| 118 return true; | |
| 119 continue; | |
| 120 } | |
| 121 } | |
| 122 | |
| 123 return success; | |
| 124 } | |
| 125 | |
| 126 static bool GetAllTopLevelWindowsICCCM(xcb_connection_t* connection, const std::vector<xcb_window_t>& roots, | |
| 127 std::set<xcb_window_t>& result) { | |
| 128 bool success = false; | |
| 129 | |
| 130 xcb_atom_t Atom_WM_STATE = [connection] { | |
| 131 static constexpr std::string_view name = "WM_STATE"; | |
| 132 xcb_intern_atom_cookie_t cookie = ::xcb_intern_atom(connection, true, name.size(), name.data()); | |
| 133 xcb_intern_atom_reply_t* reply = ::xcb_intern_atom_reply(connection, cookie, NULL); | |
| 134 | |
| 135 xcb_atom_t atom = reply->atom; | |
| 136 free(reply); | |
| 137 return atom; | |
| 138 }(); | |
| 139 if (Atom_WM_STATE == XCB_ATOM_NONE) | |
| 140 return success; | |
| 141 | |
| 142 std::vector<xcb_query_tree_cookie_t> cookies; | |
| 143 cookies.reserve(roots.size()); | |
| 144 | |
| 145 for (const auto& root : roots) | |
| 146 cookies.push_back(::xcb_query_tree(connection, root)); | |
| 147 | |
| 148 for (const auto& cookie : cookies) | |
| 149 success |= WalkWindows(connection, 0, Atom_WM_STATE, roots.data(), roots.size(), result); | |
| 150 | |
| 151 return success; | |
| 152 } | |
| 153 | |
| 154 bool EnumerateWindows(window_proc_t window_proc) { | |
| 155 if (!window_proc) | |
| 156 return false; | |
| 157 | |
| 158 xcb_connection_t* connection = ::xcb_connect(NULL, NULL); | |
| 159 if (!connection) | |
| 160 return false; | |
| 161 | |
| 162 std::set<xcb_window_t> windows; | |
| 163 { | |
| 164 std::vector<xcb_window_t> roots; | |
| 165 { | |
| 166 xcb_screen_iterator_t iter = ::xcb_setup_roots_iterator(xcb_get_setup(connection)); | |
| 167 for (; iter.rem; ::xcb_screen_next(&iter)) | |
| 168 roots.push_back(iter.data->root); | |
| 169 } | |
| 170 | |
| 171 if (!GetAllTopLevelWindowsEWMH(connection, roots, windows)) | |
| 172 GetAllTopLevelWindowsICCCM(connection, roots, windows); | |
| 173 } | |
| 174 | |
| 175 struct WindowCookies { | |
| 176 xcb_window_t window; | |
| 177 xcb_get_property_cookie_t class_property_cookie; | |
| 178 xcb_get_property_cookie_t name_property_cookie; | |
| 179 xcb_res_query_client_ids_cookie_t pid_property_cookie; | |
| 180 }; | |
| 181 | |
| 182 std::vector<WindowCookies> window_cookies; | |
| 183 window_cookies.reserve(windows.size()); | |
| 184 | |
| 185 for (const auto& window : windows) { | |
| 186 xcb_res_client_id_spec_t spec = {.client = window, .mask = XCB_RES_CLIENT_ID_MASK_LOCAL_CLIENT_PID}; | |
| 187 | |
| 188 WindowCookies window_cookie = { | |
| 189 window, ::xcb_get_property(connection, 0, window, XCB_ATOM_WM_CLASS, XCB_ATOM_STRING, 0L, 2048L), | |
| 190 ::xcb_get_property(connection, 0, window, XCB_ATOM_WM_NAME, XCB_ATOM_STRING, 0L, UINT_MAX), | |
| 191 ::xcb_res_query_client_ids(connection, 1, &spec)}; | |
| 192 | |
| 193 window_cookies.push_back(window_cookie); | |
| 194 } | |
| 195 | |
| 196 for (const auto& window_cookie : window_cookies) { | |
| 197 Window win = {0}; | |
| 198 win.id = window_cookie.window; | |
| 199 { | |
| 200 /* Class name */ | |
| 201 std::unique_ptr<xcb_get_property_reply_t> reply( | |
| 202 ::xcb_get_property_reply(connection, window_cookie.class_property_cookie, NULL)); | |
| 203 | |
| 204 if (reply && reply->format == 8) { | |
| 205 const char* data = reinterpret_cast<char*>(::xcb_get_property_value(reply.get())); | |
| 206 const int data_len = ::xcb_get_property_value_length(reply.get()); | |
| 207 | |
| 208 int instance_len = str_nlen(data, data_len); | |
| 209 const char* class_name = data + instance_len + 1; | |
| 210 | |
| 211 win.class_name = std::string(class_name, str_nlen(class_name, data_len - (instance_len + 1))); | |
| 212 } | |
| 213 } | |
| 214 { | |
| 215 /* Title text */ | |
| 216 std::unique_ptr<xcb_get_property_reply_t> reply( | |
| 217 ::xcb_get_property_reply(connection, window_cookie.name_property_cookie, NULL)); | |
| 218 | |
| 219 if (reply) { | |
| 220 const char* data = reinterpret_cast<char*>(::xcb_get_property_value(reply.get())); | |
| 221 int len = ::xcb_get_property_value_length(reply.get()); | |
| 222 | |
| 223 win.text = std::string(data, len); | |
| 224 } | |
| 225 } | |
| 226 Process proc = {0}; | |
| 227 { | |
| 228 /* PID */ | |
| 229 std::unique_ptr<xcb_res_query_client_ids_reply_t> reply( | |
| 230 ::xcb_res_query_client_ids_reply(connection, window_cookie.pid_property_cookie, NULL)); | |
| 231 | |
| 232 if (reply) { | |
| 233 xcb_res_client_id_value_iterator_t it = ::xcb_res_query_client_ids_ids_iterator(reply); | |
| 234 for (; it.rem; ::xcb_res_client_id_value_next(&it)) { | |
| 235 if (it.data->spec.mask & XCB_RES_CLIENT_ID_MASK_LOCAL_CLIENT_PID) { | |
| 236 proc.pid = *reinterpret_cast<uint32_t*>(::xcb_res_client_id_value_value(it.data)); | |
| 237 fd::GetProcessName(proc.pid, proc.name); /* fill this in if we can */ | |
| 238 break; | |
| 239 } | |
| 240 } | |
| 241 } | |
| 242 } | |
| 243 | |
| 244 if (!window_proc(proc, win)) { | |
| 245 ::xcb_disconnect(connection); | |
| 246 return false; | |
| 247 } | |
| 248 } | |
| 249 | |
| 250 ::xcb_disconnect(connection); | |
| 251 | |
| 252 return true; | |
| 253 } | |
| 254 | |
| 255 } // namespace animone::internal::x11 |
