mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 22:22:45 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			169 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			169 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
 | |
|  * All rights reserved.
 | |
|  *
 | |
|  * Redistribution and use in source and binary forms, with or without
 | |
|  * modification, are permitted provided that the following conditions are met:
 | |
|  *
 | |
|  * 1. Redistributions of source code must retain the above copyright notice, this
 | |
|  *    list of conditions and the following disclaimer.
 | |
|  *
 | |
|  * 2. Redistributions in binary form must reproduce the above copyright notice,
 | |
|  *    this list of conditions and the following disclaimer in the documentation
 | |
|  *    and/or other materials provided with the distribution.
 | |
|  *
 | |
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 | |
|  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | |
|  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 | |
|  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
 | |
|  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 | |
|  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 | |
|  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
 | |
|  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 | |
|  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 | |
|  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | |
|  */
 | |
| 
 | |
| #include <LibCore/ArgsParser.h>
 | |
| #include <LibCore/DateTime.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <time.h>
 | |
| 
 | |
| const int line_width = 70;
 | |
| const int line_count = 8;
 | |
| const int column_width = 22;
 | |
| 
 | |
| char print_buffer[line_width * line_count];
 | |
| char temp_buffer[line_width * 8];
 | |
| 
 | |
| int target_year;
 | |
| int target_month;
 | |
| int target_day;
 | |
| 
 | |
| int current_year;
 | |
| int current_month;
 | |
| 
 | |
| static void append_to_print(char* buffer, int row, int column, char* text)
 | |
| {
 | |
|     int starting_point = (line_width * row) + (column * column_width);
 | |
|     for (int i = 0; text[i] != '\0'; i++) {
 | |
|         buffer[starting_point + i] = text[i];
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void insert_month_to_print(int column, int month, int year)
 | |
| {
 | |
|     int printing_column = column;
 | |
|     int printing_row = 0;
 | |
| 
 | |
|     // FIXME: Both the month name and month header text should be provided by a locale
 | |
|     sprintf(temp_buffer, "     %02u - %04u    ", month, year);
 | |
|     append_to_print(print_buffer, printing_row, printing_column, temp_buffer);
 | |
|     printing_row++;
 | |
| 
 | |
|     sprintf(temp_buffer, "Su Mo Tu We Th Fr Sa");
 | |
|     append_to_print(print_buffer, printing_row, printing_column, temp_buffer);
 | |
|     printing_row++;
 | |
|     int day_to_print = 1;
 | |
|     auto date_time = Core::DateTime::create(year, month, 1);
 | |
|     int first_day_of_week_for_month = date_time.weekday();
 | |
|     int days_in_the_month = date_time.days_in_month();
 | |
|     int last_written_chars = 0;
 | |
|     for (int i = 1; day_to_print <= days_in_the_month; ++i) {
 | |
|         if (i - 1 < first_day_of_week_for_month) {
 | |
|             last_written_chars += sprintf(temp_buffer + last_written_chars, "   ");
 | |
|         } else {
 | |
|             if (year == current_year && month == current_month && target_day == day_to_print) {
 | |
|                 // FIXME: To replicate Unix cal it would be better to use "\x1b[30;47m%2d\x1b[0m " in here instead of *.
 | |
|                 //        However, doing that messes up the layout.
 | |
|                 last_written_chars += sprintf(temp_buffer + last_written_chars, "%2d*", day_to_print);
 | |
|             } else {
 | |
|                 last_written_chars += sprintf(temp_buffer + last_written_chars, "%2d ", day_to_print);
 | |
|             }
 | |
|             day_to_print++;
 | |
|         }
 | |
| 
 | |
|         append_to_print(print_buffer, printing_row, printing_column, temp_buffer);
 | |
| 
 | |
|         if (i % 7 == 0) {
 | |
|             printing_row++;
 | |
|             memset(temp_buffer, ' ', line_width * 8);
 | |
|             temp_buffer[line_width * 8 - 1] = '\0';
 | |
|             last_written_chars = 0;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void clean_buffers()
 | |
| {
 | |
|     for (int i = 1; i < line_width * line_count; ++i) {
 | |
|         print_buffer[i - 1] = i % line_width == 0 ? '\n' : ' ';
 | |
|     }
 | |
|     print_buffer[line_width * line_count - 1] = '\0';
 | |
| 
 | |
|     for (int i = 0; i < line_width; ++i) {
 | |
|         temp_buffer[i] = ' ';
 | |
|     }
 | |
|     temp_buffer[line_width - 1] = '\0';
 | |
| }
 | |
| 
 | |
| int main(int argc, char** argv)
 | |
| {
 | |
|     int day = 0;
 | |
|     int month = 0;
 | |
|     int year = 0;
 | |
| 
 | |
|     Core::ArgsParser args_parser;
 | |
|     // FIXME: This should ensure two values get parsed as month + year
 | |
|     args_parser.add_positional_argument(day, "Day of year", "day", Core::ArgsParser::Required::No);
 | |
|     args_parser.add_positional_argument(month, "Month", "month", Core::ArgsParser::Required::No);
 | |
|     args_parser.add_positional_argument(year, "Year", "year", Core::ArgsParser::Required::No);
 | |
|     args_parser.parse(argc, argv);
 | |
| 
 | |
|     time_t now = time(nullptr);
 | |
|     auto* tm = localtime(&now);
 | |
| 
 | |
|     // Hack: workaround two values parsing as day + month.
 | |
|     if (day && month && !year) {
 | |
|         year = month;
 | |
|         month = day;
 | |
|         day = 0;
 | |
|     }
 | |
| 
 | |
|     bool year_mode = !day && !month && year;
 | |
| 
 | |
|     if (!year)
 | |
|         year = tm->tm_year + 1900;
 | |
|     if (!month)
 | |
|         month = tm->tm_mon + 1;
 | |
|     if (!day)
 | |
|         day = tm->tm_mday;
 | |
| 
 | |
|     current_year = year;
 | |
|     current_month = month;
 | |
| 
 | |
|     clean_buffers();
 | |
| 
 | |
|     if (year_mode) {
 | |
|         printf("                             ");
 | |
|         printf("Year %4d", year);
 | |
|         printf("                             \n\n");
 | |
| 
 | |
|         for (int i = 1; i < 12; ++i) {
 | |
|             insert_month_to_print(0, i++, year);
 | |
|             insert_month_to_print(1, i++, year);
 | |
|             insert_month_to_print(2, i, year);
 | |
|             printf(print_buffer);
 | |
|             printf("\n");
 | |
|             clean_buffers();
 | |
|         }
 | |
|     } else {
 | |
|         insert_month_to_print(0, month, year);
 | |
|         printf(print_buffer);
 | |
|         printf("\n\n");
 | |
|         clean_buffers();
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | 
