Calendar.cpp 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737
  1. /*
  2. * Copyright (c) 2019-2020, Ryan Grieb <ryan.m.grieb@gmail.com>
  3. * Copyright (c) 2020-2022, the SerenityOS developers.
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <AK/DateConstants.h>
  8. #include <LibCore/DateTime.h>
  9. #include <LibGUI/Calendar.h>
  10. #include <LibGUI/Painter.h>
  11. #include <LibGUI/Window.h>
  12. #include <LibGfx/Font/FontDatabase.h>
  13. #include <LibGfx/Palette.h>
  14. REGISTER_WIDGET(GUI, Calendar);
  15. namespace GUI {
  16. static auto const extra_large_font = Gfx::BitmapFont::load_from_file("/res/fonts/MarietaRegular36.font");
  17. static auto const large_font = Gfx::BitmapFont::load_from_file("/res/fonts/MarietaRegular24.font");
  18. static auto const medium_font = Gfx::BitmapFont::load_from_file("/res/fonts/PebbletonRegular14.font");
  19. static auto const small_font = Gfx::BitmapFont::load_from_file("/res/fonts/KaticaRegular10.font");
  20. Calendar::Calendar(Core::DateTime date_time, Mode mode)
  21. : m_selected_date(date_time)
  22. , m_mode(mode)
  23. {
  24. set_fill_with_background_color(true);
  25. for (int i = 0; i < 7; i++) {
  26. Day day;
  27. m_days.append(move(day));
  28. }
  29. for (int i = 0; i < 12; i++) {
  30. MonthTile month;
  31. m_months.append(move(month));
  32. for (int j = 0; j < 42; j++) {
  33. Tile tile;
  34. m_tiles[i].append(move(tile));
  35. }
  36. }
  37. update_tiles(m_selected_date.year(), m_selected_date.month());
  38. }
  39. void Calendar::set_grid(bool show)
  40. {
  41. if (m_grid == show)
  42. return;
  43. m_grid = show;
  44. }
  45. void Calendar::toggle_mode()
  46. {
  47. m_mode == Month ? m_mode = Year : m_mode = Month;
  48. set_show_days_of_the_week(!m_show_days);
  49. set_show_year(!m_show_year);
  50. set_show_month_and_year(!m_show_month_year);
  51. update_tiles(this->view_year(), this->view_month());
  52. this->resize(this->height(), this->width());
  53. invalidate_layout();
  54. }
  55. void Calendar::resize_event(GUI::ResizeEvent& event)
  56. {
  57. m_event_size.set_width(event.size().width() - (frame_thickness() * 2));
  58. m_event_size.set_height(event.size().height() - (frame_thickness() * 2));
  59. if (mode() == Month) {
  60. if (m_event_size.width() < 160 || m_event_size.height() < 130)
  61. set_show_month_and_year(false);
  62. else if (m_event_size.width() >= 160 && m_event_size.height() >= 130)
  63. set_show_month_and_year(true);
  64. set_show_year(false);
  65. int const GRID_LINES = 6;
  66. int tile_width = (m_event_size.width() - GRID_LINES) / 7;
  67. int width_remainder = (m_event_size.width() - GRID_LINES) % 7;
  68. int y_offset = is_showing_days_of_the_week() ? 16 : 0;
  69. y_offset += is_showing_month_and_year() ? 24 : 0;
  70. int tile_height = (m_event_size.height() - y_offset - GRID_LINES) / 6;
  71. int height_remainder = (m_event_size.height() - y_offset - GRID_LINES) % 6;
  72. set_unadjusted_tile_size(tile_width, tile_height);
  73. tile_width < 30 || tile_height < 30 ? set_grid(false) : set_grid(true);
  74. for (int i = 0; i < 42; i++) {
  75. m_tiles[0][i].width = tile_width;
  76. m_tiles[0][i].height = tile_height;
  77. }
  78. for (auto& day : m_days)
  79. day.width = tile_width;
  80. for (int i = 0; i < width_remainder; i++) {
  81. m_days[i].width = (tile_width + 1);
  82. for (int j = i; j < i + 36; j += 7) {
  83. m_tiles[0][j].width = tile_width + 1;
  84. }
  85. }
  86. for (int j = 0; j < height_remainder * 7; j++)
  87. m_tiles[0][j].height = tile_height + 1;
  88. if (is_showing_days_of_the_week()) {
  89. for (int i = 0; i < 7; i++) {
  90. if (m_event_size.width() < 138)
  91. m_days[i].name = micro_day_names[i];
  92. else if (m_event_size.width() < 200)
  93. m_days[i].name = mini_day_names[i];
  94. else if (m_event_size.width() < 480)
  95. m_days[i].name = short_day_names[i];
  96. else
  97. m_days[i].name = long_day_names[i];
  98. }
  99. }
  100. } else {
  101. if (m_event_size.width() < 140 && m_event_size.height() < 120)
  102. set_show_year(false);
  103. else if (m_event_size.width() >= 140 && m_event_size.height() >= 120)
  104. set_show_year(true);
  105. set_show_month_and_year(false);
  106. int const VERT_GRID_LINES = 27;
  107. int const HORI_GRID_LINES = 15;
  108. int const THREADING = 3;
  109. int const MONTH_TITLE = 19;
  110. int tile_width = (m_event_size.width() - VERT_GRID_LINES) / 28;
  111. int width_remainder = (m_event_size.width() - VERT_GRID_LINES) % 28;
  112. int y_offset = is_showing_year() ? 22 : 0;
  113. y_offset += (MONTH_TITLE * 3) + (THREADING * 3);
  114. int tile_height = (m_event_size.height() - y_offset - HORI_GRID_LINES) / 18;
  115. int height_remainder = (m_event_size.height() - y_offset - HORI_GRID_LINES) % 18;
  116. set_grid(false);
  117. set_unadjusted_tile_size(tile_width, tile_height);
  118. if (unadjusted_tile_size().width() < 17 || unadjusted_tile_size().height() < 13)
  119. m_show_month_tiles = true;
  120. else
  121. m_show_month_tiles = false;
  122. if (m_show_month_tiles) {
  123. int month_tile_width = m_event_size.width() / 4;
  124. int width_remainder = m_event_size.width() % 4;
  125. int y_offset = is_showing_year() ? 23 : 0;
  126. int month_tile_height = (m_event_size.height() - y_offset) / 3;
  127. int height_remainder = (m_event_size.height() - y_offset) % 3;
  128. for (int i = 0; i < 12; i++) {
  129. m_months[i].width = month_tile_width;
  130. m_months[i].height = month_tile_height;
  131. if (m_event_size.width() < 250)
  132. m_months[i].name = short_month_names[i];
  133. else
  134. m_months[i].name = long_month_names[i];
  135. }
  136. if (width_remainder) {
  137. for (int i = 0; i < width_remainder; i++) {
  138. for (int j = i; j < 12; j += 4) {
  139. m_months[j].width = month_tile_width + 1;
  140. }
  141. }
  142. }
  143. if (height_remainder) {
  144. for (int i = 0; i < height_remainder * 4; i++) {
  145. m_months[i].height = month_tile_height + 1;
  146. }
  147. }
  148. return;
  149. }
  150. for (int i = 0; i < 12; i++) {
  151. int remainder = 0;
  152. if (i == 0 || i == 4 || i == 8)
  153. remainder = min(width_remainder, 7);
  154. if (i == 1 || i == 5 || i == 9)
  155. width_remainder > 7 ? remainder = min(width_remainder - 7, 7) : remainder = 0;
  156. if (i == 2 || i == 6 || i == 10)
  157. width_remainder > 14 ? remainder = min(width_remainder - 14, 7) : remainder = 0;
  158. if (i == 3 || i == 7 || i == 11)
  159. width_remainder > 21 ? remainder = width_remainder - 21 : remainder = 0;
  160. m_month_size[i].set_width(remainder + 6 + tile_width * 7);
  161. if (i >= 0 && i <= 3)
  162. remainder = min(height_remainder, 6);
  163. if (i >= 4 && i <= 7)
  164. height_remainder > 6 ? remainder = min(height_remainder - 6, 6) : remainder = 0;
  165. if (i >= 8 && i <= 12)
  166. height_remainder > 12 ? remainder = height_remainder - 12 : remainder = 0;
  167. m_month_size[i].set_height(remainder + 5 + tile_height * 6);
  168. for (int j = 0; j < 42; j++) {
  169. m_tiles[i][j].width = tile_width;
  170. m_tiles[i][j].height = tile_height;
  171. }
  172. }
  173. if (width_remainder) {
  174. for (int i = 0; i < 12; i += 4) {
  175. for (int j = 0; j < min(width_remainder, 7); j++) {
  176. for (int k = j; k < j + 36; k += 7) {
  177. m_tiles[i][k].width = tile_width + 1;
  178. }
  179. }
  180. }
  181. }
  182. if (width_remainder > 7) {
  183. for (int i = 1; i < 12; i += 4) {
  184. for (int j = 0; j < min(width_remainder - 7, 7); j++) {
  185. for (int k = j; k < j + 36; k += 7) {
  186. m_tiles[i][k].width = tile_width + 1;
  187. }
  188. }
  189. }
  190. }
  191. if (width_remainder > 14) {
  192. for (int i = 2; i < 12; i += 4) {
  193. for (int j = 0; j < min(width_remainder - 14, 7); j++) {
  194. for (int k = j; k < j + 36; k += 7) {
  195. m_tiles[i][k].width = tile_width + 1;
  196. }
  197. }
  198. }
  199. }
  200. if (width_remainder > 21) {
  201. for (int i = 3; i < 12; i += 4) {
  202. for (int j = 0; j < width_remainder - 21; j++) {
  203. for (int k = j; k < j + 36; k += 7) {
  204. m_tiles[i][k].width = tile_width + 1;
  205. }
  206. }
  207. }
  208. }
  209. if (height_remainder) {
  210. for (int i = 0; i < 4; i++) {
  211. for (int j = 0; j < min(height_remainder, 6) * 7; j++) {
  212. m_tiles[i][j].height = tile_height + 1;
  213. }
  214. }
  215. }
  216. if (height_remainder > 6) {
  217. for (int i = 4; i < 8; i++) {
  218. for (int j = 0; j < min(height_remainder - 6, 6) * 7; j++) {
  219. m_tiles[i][j].height = tile_height + 1;
  220. }
  221. }
  222. }
  223. if (height_remainder > 12) {
  224. for (int i = 8; i < 12; i++) {
  225. for (int j = 0; j < (height_remainder - 12) * 7; j++) {
  226. m_tiles[i][j].height = tile_height + 1;
  227. }
  228. }
  229. }
  230. }
  231. }
  232. void Calendar::update_tiles(unsigned view_year, unsigned view_month)
  233. {
  234. set_view_date(view_year, view_month);
  235. unsigned months;
  236. mode() == Month ? months = 1 : months = 12;
  237. for (unsigned i = 0; i < months; i++) {
  238. if (mode() == Year)
  239. view_month = i + 1;
  240. for (unsigned j = 0; j < 42; j++) {
  241. auto date_time = Core::DateTime::create(view_year, view_month, 1);
  242. unsigned start_of_month = date_time.weekday();
  243. unsigned year;
  244. unsigned month;
  245. unsigned day;
  246. if (start_of_month == 0 && mode() != Year) {
  247. month = (view_month - 1 == 0) ? 12 : view_month - 1;
  248. year = (month == 12) ? view_year - 1 : view_year;
  249. date_time.set_time(year, month, 1);
  250. day = (date_time.days_in_month() - 6 + j);
  251. } else if (start_of_month > j) {
  252. month = (view_month - 1 == 0) ? 12 : view_month - 1;
  253. year = (month == 12) ? view_year - 1 : view_year;
  254. date_time.set_time(year, month, 1);
  255. day = (date_time.days_in_month() - (start_of_month) + j) + 1;
  256. } else if ((j - start_of_month) + 1 > date_time.days_in_month()) {
  257. month = (view_month + 1) > 12 ? 1 : view_month + 1;
  258. year = (month == 1) ? view_year + 1 : view_year;
  259. day = ((j - start_of_month) + 1) - date_time.days_in_month();
  260. } else {
  261. month = view_month;
  262. year = view_year;
  263. day = (j - start_of_month) + 1;
  264. }
  265. date_time.set_time(year, month, day);
  266. m_tiles[i][j].date_time = date_time;
  267. m_tiles[i][j].is_outside_selected_month = (date_time.month() != view_month
  268. || date_time.year() != view_year);
  269. m_tiles[i][j].is_selected = (date_time.year() == m_selected_date.year()
  270. && date_time.month() == m_selected_date.month()
  271. && date_time.day() == m_selected_date.day()
  272. && (mode() == Year ? !m_tiles[i][j].is_outside_selected_month : true));
  273. m_tiles[i][j].is_today = (date_time.day() == Core::DateTime::now().day()
  274. && date_time.month() == Core::DateTime::now().month()
  275. && date_time.year() == Core::DateTime::now().year());
  276. }
  277. }
  278. update();
  279. }
  280. String Calendar::formatted_date(Format format)
  281. {
  282. switch (format) {
  283. case ShortMonthYear:
  284. return String::formatted("{} {}", short_month_names[view_month() - 1], view_year());
  285. case LongMonthYear:
  286. return String::formatted("{} {}", long_month_names[view_month() - 1], view_year());
  287. case MonthOnly:
  288. return String::formatted("{}", long_month_names[view_month() - 1]);
  289. case YearOnly:
  290. return String::number(view_year());
  291. default:
  292. VERIFY_NOT_REACHED();
  293. }
  294. }
  295. void Calendar::paint_event(GUI::PaintEvent& event)
  296. {
  297. GUI::Frame::paint_event(event);
  298. GUI::Painter painter(*this);
  299. painter.add_clip_rect(frame_inner_rect());
  300. painter.add_clip_rect(event.rect());
  301. if (has_grid())
  302. painter.fill_rect(frame_inner_rect(), palette().threed_shadow2());
  303. else
  304. painter.fill_rect(frame_inner_rect(), palette().base());
  305. painter.translate(frame_thickness(), frame_thickness());
  306. int width = unadjusted_tile_size().width();
  307. int height = unadjusted_tile_size().height();
  308. int x_offset = 0;
  309. int y_offset = 0;
  310. if (is_showing_year()) {
  311. auto year_only_rect = Gfx::IntRect(
  312. 0,
  313. 0,
  314. frame_inner_rect().width(),
  315. 22);
  316. y_offset += year_only_rect.height();
  317. painter.fill_rect(year_only_rect, palette().hover_highlight());
  318. painter.draw_text(year_only_rect, formatted_date(YearOnly), medium_font->bold_variant(), Gfx::TextAlignment::Center, palette().base_text());
  319. painter.draw_line({ 0, y_offset }, { frame_inner_rect().width(), y_offset }, (!m_show_month_tiles ? palette().threed_shadow1() : palette().threed_shadow2()), 1);
  320. y_offset += 1;
  321. if (!m_show_month_tiles) {
  322. painter.draw_line({ 0, y_offset }, { frame_inner_rect().width(), y_offset }, palette().threed_highlight(), 1);
  323. y_offset += 1;
  324. }
  325. } else if (is_showing_month_and_year()) {
  326. auto month_year_rect = Gfx::IntRect(
  327. 0,
  328. 0,
  329. frame_inner_rect().width(),
  330. 22);
  331. painter.fill_rect(month_year_rect, palette().hover_highlight());
  332. month_year_rect.set_width(frame_inner_rect().width() / 2);
  333. painter.draw_text(month_year_rect, formatted_date(MonthOnly), medium_font->bold_variant(), Gfx::TextAlignment::Center, palette().base_text());
  334. month_year_rect.set_x(month_year_rect.width() + (frame_inner_rect().width() % 2 ? 1 : 0));
  335. painter.draw_text(month_year_rect, formatted_date(YearOnly), medium_font->bold_variant(), Gfx::TextAlignment::Center, palette().base_text());
  336. y_offset += 22;
  337. painter.draw_line({ 0, y_offset }, { frame_inner_rect().width(), y_offset }, palette().threed_shadow1(), 1);
  338. y_offset += 1;
  339. painter.draw_line({ 0, y_offset }, { frame_inner_rect().width(), y_offset }, palette().threed_highlight(), 1);
  340. y_offset += 1;
  341. }
  342. if (mode() == Year && m_show_month_tiles) {
  343. int i = 0;
  344. for (int j = 0; j < 3; j++) {
  345. x_offset = 0;
  346. for (int k = 0; k < 4; k++) {
  347. if (k > 0)
  348. x_offset += m_months[i - 1].width;
  349. auto month_tile_rect = Gfx::IntRect(
  350. x_offset,
  351. y_offset,
  352. m_months[i].width,
  353. m_months[i].height);
  354. m_months[i].rect = month_tile_rect.translated(frame_thickness(), frame_thickness());
  355. Gfx::StylePainter::paint_button(
  356. painter, month_tile_rect, palette(),
  357. Gfx::ButtonStyle::Normal,
  358. m_months[i].is_being_pressed,
  359. m_months[i].is_hovered,
  360. false, true, false);
  361. set_font(small_font);
  362. painter.draw_text(month_tile_rect, m_months[i].name, font(), Gfx::TextAlignment::Center, palette().base_text());
  363. i++;
  364. }
  365. y_offset += m_months[i - 1].height;
  366. }
  367. return;
  368. }
  369. if (is_showing_days_of_the_week()) {
  370. auto days_of_the_week_rect = Gfx::IntRect(
  371. 0,
  372. y_offset,
  373. frame_inner_rect().width(),
  374. 16);
  375. painter.fill_rect(days_of_the_week_rect, palette().hover_highlight());
  376. for (int i = 0; i < 7; i++) {
  377. if (i > 0)
  378. x_offset += m_days[i - 1].width + 1;
  379. Gfx::IntRect day_rect = Gfx::IntRect(
  380. x_offset,
  381. y_offset,
  382. m_days[i].width,
  383. 16);
  384. painter.draw_text(day_rect, m_days[i].name, small_font->bold_variant(), Gfx::TextAlignment::Center, palette().base_text());
  385. }
  386. y_offset += days_of_the_week_rect.height();
  387. painter.draw_line({ 0, y_offset }, { frame_inner_rect().width(), y_offset }, palette().threed_shadow2(), 1);
  388. y_offset += 1;
  389. }
  390. if (mode() == Month) {
  391. int i = 0;
  392. for (int j = 0; j < 6; j++) {
  393. x_offset = 0;
  394. if (j > 0)
  395. y_offset += m_tiles[0][(j - 1) * 7].height + 1;
  396. for (int k = 0; k < 7; k++) {
  397. if (k > 0)
  398. x_offset += m_tiles[0][k - 1].width + 1;
  399. auto tile_rect = Gfx::IntRect(
  400. x_offset,
  401. y_offset,
  402. m_tiles[0][i].width,
  403. m_tiles[0][i].height);
  404. m_tiles[0][i].rect = tile_rect.translated(frame_thickness(), frame_thickness());
  405. if (m_tiles[0][i].is_hovered || m_tiles[0][i].is_selected)
  406. painter.fill_rect(tile_rect, palette().hover_highlight());
  407. else
  408. painter.fill_rect(tile_rect, palette().base());
  409. auto text_alignment = Gfx::TextAlignment::TopRight;
  410. auto text_rect = Gfx::IntRect(
  411. x_offset,
  412. y_offset + 4,
  413. m_tiles[0][i].width - 4,
  414. font().glyph_height() + 4);
  415. if (width > 150 && height > 150) {
  416. set_font(extra_large_font);
  417. } else if (width > 100 && height > 100) {
  418. set_font(large_font);
  419. } else if (width > 50 && height > 50) {
  420. set_font(medium_font);
  421. } else if (width >= 30 && height >= 30) {
  422. set_font(small_font);
  423. } else {
  424. set_font(small_font);
  425. text_alignment = Gfx::TextAlignment::Center;
  426. text_rect = Gfx::IntRect(tile_rect);
  427. }
  428. auto display_date = String::number(m_tiles[0][i].date_time.day());
  429. if (m_tiles[0][i].is_selected && (width < 30 || height < 30))
  430. painter.draw_rect(tile_rect, palette().base_text());
  431. if (m_tiles[0][i].is_today && !m_tiles[0][i].is_outside_selected_month) {
  432. painter.draw_text(text_rect, display_date, font().bold_variant(), text_alignment, palette().base_text());
  433. } else if (m_tiles[0][i].is_outside_selected_month) {
  434. painter.draw_text(text_rect, display_date, m_tiles[0][i].is_today ? font().bold_variant() : font(), text_alignment, Color::LightGray);
  435. } else {
  436. painter.draw_text(text_rect, display_date, font(), text_alignment, palette().base_text());
  437. }
  438. i++;
  439. }
  440. }
  441. } else {
  442. for (int i = 0; i < 4; i++) {
  443. static int x_month_offset;
  444. x_month_offset += (i > 0 ? m_month_size[i - 1].width() + 1 : 0);
  445. auto month_rect = Gfx::IntRect(
  446. x_month_offset,
  447. y_offset,
  448. m_month_size[i].width(),
  449. 19);
  450. painter.fill_rect(month_rect, palette().hover_highlight());
  451. painter.draw_text(month_rect, long_month_names[i], medium_font->bold_variant(), Gfx::TextAlignment::Center, palette().base_text());
  452. if (i > 0 && i < 4) {
  453. painter.draw_line({ x_month_offset - 1, y_offset - 1 }, { x_month_offset - 1, y_offset + 18 }, palette().threed_shadow2(), 1);
  454. painter.draw_line({ x_month_offset, y_offset - 1 }, { x_month_offset, y_offset + 18 }, palette().threed_highlight(), 1);
  455. }
  456. if (i == 3)
  457. x_month_offset = 0;
  458. }
  459. y_offset += 19;
  460. painter.draw_line({ 0, y_offset }, { frame_inner_rect().width(), y_offset }, palette().threed_shadow2(), 1);
  461. y_offset += 1;
  462. int x_translation = 0;
  463. int y_translation = y_offset;
  464. for (int l = 0; l < 12; l++) {
  465. if ((l > 0 && l < 4) || (l > 4 && l < 8) || (l > 8)) {
  466. x_translation += m_month_size[l - 1].width() + 1;
  467. } else if (l % 4 == 0) {
  468. x_translation = 0;
  469. }
  470. if (l < 4 || (l > 4 && l < 8) || l > 8) {
  471. y_offset = y_translation;
  472. } else if (l == 4 || l == 8) {
  473. y_translation += m_month_size[l - 1].height();
  474. painter.draw_line({ 0, y_translation }, { frame_inner_rect().width(), y_translation }, palette().threed_shadow1(), 1);
  475. y_translation += 1;
  476. painter.draw_line({ 0, y_translation }, { frame_inner_rect().width(), y_translation }, palette().threed_highlight(), 1);
  477. y_translation += 1;
  478. y_offset = y_translation;
  479. for (int i = l; i < (l == 4 ? 8 : 12); i++) {
  480. static int x_month_offset;
  481. x_month_offset += (i > (l == 4 ? 4 : 8) ? m_month_size[i - 1].width() + 1 : 0);
  482. auto month_rect = Gfx::IntRect(
  483. x_month_offset,
  484. y_offset,
  485. m_month_size[i].width(),
  486. 19);
  487. painter.fill_rect(month_rect, palette().hover_highlight());
  488. painter.draw_text(month_rect, long_month_names[i], medium_font->bold_variant(), Gfx::TextAlignment::Center, palette().base_text());
  489. if (i > (l == 4 ? 4 : 8) && i < (l == 4 ? 8 : 12)) {
  490. painter.draw_line({ x_month_offset - 1, y_offset - 1 }, { x_month_offset - 1, y_offset + 18 }, palette().threed_shadow2(), 1);
  491. painter.draw_line({ x_month_offset, y_offset - 1 }, { x_month_offset, y_offset + 18 }, palette().threed_highlight(), 1);
  492. }
  493. if (i == 7 || i == 11)
  494. x_month_offset = 0;
  495. }
  496. y_translation += 19;
  497. painter.draw_line({ 0, y_translation }, { frame_inner_rect().width(), y_translation }, palette().threed_shadow2(), 1);
  498. y_translation += 1;
  499. y_offset = y_translation;
  500. }
  501. int i = 0;
  502. for (int j = 0; j < 6; j++) {
  503. x_offset = 0;
  504. if (j > 0)
  505. y_offset += m_tiles[l][(j - 1) * 7].height + (j < 6 ? 1 : 0);
  506. if (j == 0 && l != 3 && l != 7 && l != 11) {
  507. painter.draw_line(
  508. { m_month_size[l].width() + x_translation, y_offset },
  509. { m_month_size[l].width() + x_translation, y_offset + m_month_size[l].height() },
  510. palette().threed_shadow2(),
  511. 1);
  512. }
  513. for (int k = 0; k < 7; k++) {
  514. if (k > 0)
  515. x_offset += m_tiles[l][k - 1].width + 1;
  516. auto tile_rect = Gfx::IntRect(
  517. x_offset + x_translation,
  518. y_offset,
  519. m_tiles[l][i].width,
  520. m_tiles[l][i].height);
  521. m_tiles[l][i].rect = tile_rect.translated(frame_thickness(), frame_thickness());
  522. if (m_tiles[l][i].is_hovered || m_tiles[l][i].is_selected)
  523. painter.fill_rect(tile_rect, palette().hover_highlight());
  524. else
  525. painter.fill_rect(tile_rect, palette().base());
  526. if (width > 50 && height > 50) {
  527. set_font(medium_font);
  528. } else {
  529. set_font(small_font);
  530. }
  531. auto display_date = String::number(m_tiles[l][i].date_time.day());
  532. if (m_tiles[l][i].is_selected)
  533. painter.draw_rect(tile_rect, palette().base_text());
  534. if (m_tiles[l][i].is_today && !m_tiles[l][i].is_outside_selected_month) {
  535. painter.draw_text(tile_rect, display_date, font().bold_variant(), Gfx::TextAlignment::Center, palette().base_text());
  536. } else if (!m_tiles[l][i].is_outside_selected_month) {
  537. painter.draw_text(tile_rect, display_date, font(), Gfx::TextAlignment::Center, palette().base_text());
  538. }
  539. i++;
  540. }
  541. }
  542. }
  543. }
  544. }
  545. void Calendar::leave_event(Core::Event&)
  546. {
  547. int months;
  548. mode() == Month ? months = 1 : months = 12;
  549. for (int i = 0; i < months; i++) {
  550. if (mode() == Year && m_show_month_tiles) {
  551. m_months[i].is_hovered = false;
  552. continue;
  553. } else {
  554. for (int j = 0; j < 42; j++) {
  555. m_tiles[i][j].is_hovered = false;
  556. }
  557. }
  558. }
  559. update();
  560. }
  561. void Calendar::mousemove_event(GUI::MouseEvent& event)
  562. {
  563. static int last_index_i;
  564. static int last_index_j;
  565. if (mode() == Year && m_show_month_tiles) {
  566. if (m_months[last_index_i].rect.contains(event.position()) && (m_months[last_index_i].is_hovered || m_months[last_index_i].is_being_pressed)) {
  567. return;
  568. } else {
  569. m_months[last_index_i].is_hovered = false;
  570. m_months[last_index_i].is_being_pressed = false;
  571. update(m_months[last_index_i].rect);
  572. }
  573. } else {
  574. if (m_tiles[last_index_i][last_index_j].rect.contains(event.position()) && m_tiles[last_index_i][last_index_j].is_hovered) {
  575. return;
  576. } else {
  577. m_tiles[last_index_i][last_index_j].is_hovered = false;
  578. update(m_tiles[last_index_i][last_index_j].rect);
  579. }
  580. }
  581. int months;
  582. mode() == Month ? months = 1 : months = 12;
  583. for (int i = 0; i < months; i++) {
  584. if (mode() == Year && m_show_month_tiles) {
  585. if (m_months[i].rect.contains(event.position())) {
  586. if (m_currently_pressed_index == -1 || m_currently_pressed_index == i)
  587. m_months[i].is_hovered = true;
  588. if (m_currently_pressed_index == i)
  589. m_months[i].is_being_pressed = true;
  590. update(m_months[last_index_i].rect);
  591. if (m_months[i].is_being_pressed == true)
  592. m_currently_pressed_index = i;
  593. last_index_i = i;
  594. update(m_months[i].rect);
  595. break;
  596. }
  597. } else {
  598. for (int j = 0; j < 42; j++) {
  599. if (mode() == Year && m_tiles[i][j].is_outside_selected_month)
  600. continue;
  601. if (m_tiles[i][j].rect.contains(event.position())) {
  602. m_tiles[i][j].is_hovered = true;
  603. update(m_tiles[last_index_i][last_index_j].rect);
  604. last_index_i = i;
  605. last_index_j = j;
  606. update(m_tiles[i][j].rect);
  607. break;
  608. }
  609. }
  610. }
  611. }
  612. }
  613. void Calendar::mouseup_event(GUI::MouseEvent& event)
  614. {
  615. int months;
  616. mode() == Month ? months = 1 : months = 12;
  617. for (int i = 0; i < months; i++) {
  618. if (mode() == Year && m_show_month_tiles) {
  619. if (m_months[i].rect.contains(event.position()) && m_months[i].is_being_pressed) {
  620. set_view_date(view_year(), (unsigned)i + 1);
  621. toggle_mode();
  622. if (on_month_click)
  623. on_month_click();
  624. }
  625. } else {
  626. for (int j = 0; j < 42; j++) {
  627. if (mode() == Year && m_tiles[i][j].is_outside_selected_month)
  628. continue;
  629. if (m_tiles[i][j].rect.contains(event.position())) {
  630. m_previous_selected_date = m_selected_date;
  631. m_selected_date = m_tiles[i][j].date_time;
  632. update_tiles(m_selected_date.year(), m_selected_date.month());
  633. if (on_tile_click)
  634. on_tile_click();
  635. }
  636. }
  637. }
  638. if (months == 12) {
  639. m_months[i].is_being_pressed = false;
  640. m_months[i].is_hovered = false;
  641. }
  642. }
  643. m_currently_pressed_index = -1;
  644. update();
  645. }
  646. void Calendar::mousedown_event(GUI::MouseEvent& event)
  647. {
  648. if (mode() == Year && m_show_month_tiles) {
  649. for (int i = 0; i < 12; i++) {
  650. if (m_months[i].rect.contains(event.position())) {
  651. m_months[i].is_being_pressed = true;
  652. m_currently_pressed_index = i;
  653. update(m_months[i].rect);
  654. break;
  655. }
  656. }
  657. }
  658. }
  659. void Calendar::doubleclick_event(GUI::MouseEvent& event)
  660. {
  661. int months;
  662. mode() == Month ? months = 1 : months = 12;
  663. for (int i = 0; i < months; i++) {
  664. for (int j = 0; j < 42; j++) {
  665. if (m_tiles[i][j].date_time.day() != m_previous_selected_date.day())
  666. continue;
  667. if (mode() == Year && m_tiles[i][j].is_outside_selected_month)
  668. continue;
  669. if (m_tiles[i][j].rect.contains(event.position())) {
  670. if (on_tile_doubleclick)
  671. on_tile_doubleclick();
  672. }
  673. }
  674. }
  675. }
  676. }