Calendar.cpp 33 KB

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