Calendar.cpp 30 KB

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