Compare commits
20 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 22333b6593 | |||
| 096b1d1eb6 | |||
| dfa0774242 | |||
| 24aa979a69 | |||
| 64b4797908 | |||
| 522b6fa517 | |||
| bee49ee7dc | |||
| accc2a79bc | |||
| 6e573ab992 | |||
| 941bbcdb5b | |||
| 925b42ceb7 | |||
| 4c0abe5464 | |||
| e4670f1b26 | |||
| 58e9428299 | |||
| 5fc24393c5 | |||
| 04f0ef76ee | |||
| 17e850e6d7 | |||
| 01cf082351 | |||
| 5088b273b6 | |||
| 17f4316947 |
@@ -1,8 +1,7 @@
|
|||||||
# Compiler and flags
|
# Compiler and flags
|
||||||
CPPC = g++
|
CPPC = g++
|
||||||
CPPC_FLAGS = -std=c++23 -s -O3 -lncursesw -lcurl -lmenu -lpanel -Wall -Wextra -Wno-write-strings
|
CPPC_FLAGS = -std=c++23 -s -O3 -lncursesw -lcurl -lmenu -lpanel -Wall -Wextra -Wno-write-strings
|
||||||
# Debug flags:
|
DEBUG_FLAGS = -ggdb -std=c++23 -lncursesw -lcurl -lmenu -lpanel -Wall -Wextra -Wno-write-strings
|
||||||
# CPPC_FLAGS = -ggdb -std=c++23 -lncursesw -lcurl -lmenu -lpanel -Wall -Wextra -Wno-write-strings
|
|
||||||
|
|
||||||
|
|
||||||
SRC_PATH := src
|
SRC_PATH := src
|
||||||
@@ -11,16 +10,18 @@ BIN_PATH := build/bin
|
|||||||
|
|
||||||
|
|
||||||
SRC_FILES := $(shell find $(SRC_PATH) -name '*.cpp')
|
SRC_FILES := $(shell find $(SRC_PATH) -name '*.cpp')
|
||||||
# Generate corresponding object file paths by replacing src/ with build/obj/
|
|
||||||
OBJ_FILES := $(patsubst $(SRC_PATH)/%.cpp,$(OBJ_PATH)/%.o,$(SRC_FILES))
|
OBJ_FILES := $(patsubst $(SRC_PATH)/%.cpp,$(OBJ_PATH)/%.o,$(SRC_FILES))
|
||||||
|
|
||||||
|
|
||||||
all: make-build-dir $(BIN_PATH)/bakatui
|
all: make-build-dir $(BIN_PATH)/bakatui
|
||||||
|
|
||||||
|
|
||||||
|
debug: CPPC_FLAGS = $(DEBUG_FLAGS)
|
||||||
|
debug: make-build-dir $(BIN_PATH)/bakatui
|
||||||
|
|
||||||
|
|
||||||
make-build-dir:
|
make-build-dir:
|
||||||
mkdir -p $(OBJ_PATH)
|
mkdir -p $(OBJ_PATH)
|
||||||
mkdir -p $(OBJ_PATH)/marks
|
|
||||||
mkdir -p $(BIN_PATH)
|
mkdir -p $(BIN_PATH)
|
||||||
|
|
||||||
|
|
||||||
@@ -32,7 +33,10 @@ $(OBJ_PATH)/%.o: $(SRC_PATH)/%.cpp
|
|||||||
$(CPPC) $(CPPC_FLAGS) -c $< -o $@
|
$(CPPC) $(CPPC_FLAGS) -c $< -o $@
|
||||||
|
|
||||||
|
|
||||||
|
install:
|
||||||
|
@install -vpm 755 -o root -g root $(BIN_PATH)/bakatui /usr/bin/
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
rm -fr build
|
rm -fr build
|
||||||
|
|
||||||
.PHONY: all clean install
|
.PHONY: all clean install debug
|
||||||
|
|||||||
@@ -3,6 +3,12 @@
|
|||||||
|
|
||||||
> ### Dependencies:
|
> ### Dependencies:
|
||||||
> [nlohmann-json](https://github.com/nlohmann/json)
|
> [nlohmann-json](https://github.com/nlohmann/json)
|
||||||
|
> ncurses
|
||||||
|
|
||||||
> [!IMPORTANT]
|
> [!IMPORTANT]
|
||||||
> Only marks work right now
|
> Only folowing works:
|
||||||
|
> - [x] Login
|
||||||
|
> - [x] Marks
|
||||||
|
> - [x] Timetable
|
||||||
|
> - [ ] Komens
|
||||||
|
> - [ ] Absence
|
||||||
|
|||||||
+1
-1
@@ -1,7 +1,7 @@
|
|||||||
#include <string_view>
|
#include <string_view>
|
||||||
#ifndef VERSION
|
#ifndef VERSION
|
||||||
|
|
||||||
#define VERSION "0.6"
|
#define VERSION "0.7.1"
|
||||||
#define NAME "bakatui"
|
#define NAME "bakatui"
|
||||||
|
|
||||||
inline constexpr auto hash_djb2a(const std::string_view sv) {
|
inline constexpr auto hash_djb2a(const std::string_view sv) {
|
||||||
|
|||||||
+12
-1
@@ -1,5 +1,7 @@
|
|||||||
#include "helper_funcs.h"
|
#include "helper_funcs.h"
|
||||||
#include "color.h"
|
#include "color.h"
|
||||||
|
#include "main.h"
|
||||||
|
#include "memory.h"
|
||||||
#include "net.h"
|
#include "net.h"
|
||||||
#include <codecvt>
|
#include <codecvt>
|
||||||
#include <csignal>
|
#include <csignal>
|
||||||
@@ -13,6 +15,9 @@
|
|||||||
#include <string>
|
#include <string>
|
||||||
#include <termios.h>
|
#include <termios.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
std::vector<allocation> *current_allocated;
|
||||||
|
|
||||||
void safe_exit(int code) {
|
void safe_exit(int code) {
|
||||||
switch (code) {
|
switch (code) {
|
||||||
@@ -37,6 +42,7 @@ void safe_exit(int code) {
|
|||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
delete_all(current_allocated);
|
||||||
|
|
||||||
curl_easy_cleanup(curl);
|
curl_easy_cleanup(curl);
|
||||||
endwin();
|
endwin();
|
||||||
@@ -150,7 +156,12 @@ void wprint_in_middle(WINDOW *win, int starty, int startx, int width,
|
|||||||
temp = (width - length) / 2;
|
temp = (width - length) / 2;
|
||||||
x = startx + (int)temp;
|
x = startx + (int)temp;
|
||||||
wattron(win, color);
|
wattron(win, color);
|
||||||
mvwaddwstr(win, y, x, string);
|
if (mvwaddwstr(win, y, x, string) == ERR) {
|
||||||
|
if (config.verbose) {
|
||||||
|
std::wcerr << RED "[ERROR]" << RESET " wprint_in_middle failed to print "
|
||||||
|
<< string << "\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
wattroff(win, color);
|
wattroff(win, color);
|
||||||
refresh();
|
refresh();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,14 @@
|
|||||||
|
#include "memory.h"
|
||||||
#include <curses.h>
|
#include <curses.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
// header guard
|
// header guard
|
||||||
#ifndef _ba_hf_hg_
|
#ifndef _ba_hf_hg_
|
||||||
#define _ba_hf_hg_
|
#define _ba_hf_hg_
|
||||||
|
|
||||||
|
extern std::vector<allocation> *current_allocated;
|
||||||
|
|
||||||
#define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
|
#define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
|
||||||
|
|
||||||
void safe_exit(int code);
|
void safe_exit(int code);
|
||||||
|
|||||||
@@ -5,9 +5,7 @@
|
|||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
#include <curses.h>
|
#include <curses.h>
|
||||||
#include <locale>
|
|
||||||
#include <menu.h>
|
#include <menu.h>
|
||||||
#include <string>
|
|
||||||
|
|
||||||
#include "marks.h"
|
#include "marks.h"
|
||||||
|
|
||||||
|
|||||||
+28
-18
@@ -1,5 +1,6 @@
|
|||||||
#include "marks.h"
|
#include "marks.h"
|
||||||
#include "helper_funcs.h"
|
#include "helper_funcs.h"
|
||||||
|
#include "memory.h"
|
||||||
#include "net.h"
|
#include "net.h"
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <cstddef>
|
#include <cstddef>
|
||||||
@@ -8,12 +9,11 @@
|
|||||||
#include <cstring>
|
#include <cstring>
|
||||||
#include <curses.h>
|
#include <curses.h>
|
||||||
#include <format>
|
#include <format>
|
||||||
#include <iostream>
|
|
||||||
#include <locale>
|
|
||||||
#include <menu.h>
|
#include <menu.h>
|
||||||
#include <nlohmann/json.hpp>
|
#include <nlohmann/json.hpp>
|
||||||
#include <panel.h>
|
#include <panel.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
using nlohmann::json;
|
using nlohmann::json;
|
||||||
|
|
||||||
@@ -29,24 +29,28 @@ using nlohmann::json;
|
|||||||
|
|
||||||
#define DEFAULT_PADDING 4
|
#define DEFAULT_PADDING 4
|
||||||
|
|
||||||
|
std::vector<allocation> marks_allocated;
|
||||||
|
|
||||||
void init_wins(WINDOW **wins, int n, json marks_json);
|
void init_wins(WINDOW **wins, int n, json marks_json);
|
||||||
void win_show(WINDOW *win, wchar_t *label, int label_color, int width,
|
void win_show(WINDOW *win, wchar_t *label, int label_color, int width,
|
||||||
int height, json marks_json, int SubjectIndex);
|
int height, json marks_json, int SubjectIndex);
|
||||||
|
|
||||||
void marks_page() {
|
void marks_page() {
|
||||||
// DONT FORGET TO UNCOMMENT
|
current_allocated = &marks_allocated;
|
||||||
json resp_from_api = bakaapi::get_data_from_endpoint("api/3/marks");
|
json resp_from_api;
|
||||||
// std::ifstream f("test-data/marks3.json");
|
{
|
||||||
// json resp_from_api = json::parse(f);
|
std::string endpoint = "api/3/marks";
|
||||||
|
resp_from_api = bakaapi::get_data_from_endpoint(endpoint);
|
||||||
WINDOW **my_wins;
|
}
|
||||||
size_t size_my_wins = resp_from_api["Subjects"].size();
|
size_t size_my_wins = resp_from_api["Subjects"].size();
|
||||||
my_wins = new (std::nothrow) WINDOW *[size_my_wins];
|
WINDOW **my_wins = new (std::nothrow) WINDOW *[size_my_wins];
|
||||||
|
marks_allocated.push_back({WINDOW_ARRAY, my_wins, size_my_wins});
|
||||||
|
|
||||||
PANEL **my_panels;
|
|
||||||
size_t size_my_panels = resp_from_api["Subjects"].size();
|
size_t size_my_panels = resp_from_api["Subjects"].size();
|
||||||
my_panels = new (std::nothrow) PANEL *[size_my_panels];
|
PANEL **my_panels = new (std::nothrow) PANEL *[size_my_panels];
|
||||||
|
marks_allocated.push_back({PANEL_ARRAY, my_panels, size_my_panels});
|
||||||
|
|
||||||
|
// trows compiler warning for some reason but cannot be removed
|
||||||
PANEL *top;
|
PANEL *top;
|
||||||
int ch;
|
int ch;
|
||||||
|
|
||||||
@@ -67,7 +71,9 @@ void marks_page() {
|
|||||||
|
|
||||||
// store all original window position
|
// store all original window position
|
||||||
int *original_y = new int[size_my_wins];
|
int *original_y = new int[size_my_wins];
|
||||||
|
marks_allocated.push_back({GENERIC_ARRAY, original_y, size_my_wins});
|
||||||
int *original_x = new int[size_my_wins];
|
int *original_x = new int[size_my_wins];
|
||||||
|
marks_allocated.push_back({GENERIC_ARRAY, original_x, size_my_wins});
|
||||||
for (size_t i = 0; i < size_my_wins; ++i) {
|
for (size_t i = 0; i < size_my_wins; ++i) {
|
||||||
getbegyx(my_wins[i], original_y[i], original_x[i]);
|
getbegyx(my_wins[i], original_y[i], original_x[i]);
|
||||||
}
|
}
|
||||||
@@ -122,10 +128,7 @@ void marks_page() {
|
|||||||
// Cleanup
|
// Cleanup
|
||||||
endwin();
|
endwin();
|
||||||
clear();
|
clear();
|
||||||
delete[] my_wins;
|
delete_all(&marks_allocated);
|
||||||
delete[] my_panels;
|
|
||||||
delete[] original_y;
|
|
||||||
delete[] original_x;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Put all the windows */
|
/* Put all the windows */
|
||||||
@@ -137,7 +140,7 @@ void init_wins(WINDOW **wins, int n, json marks_json) {
|
|||||||
x = DEFAULT_X_OFFSET;
|
x = DEFAULT_X_OFFSET;
|
||||||
uint8_t curent_color = 0;
|
uint8_t curent_color = 0;
|
||||||
|
|
||||||
int MaxHight = 0;
|
unsigned int MaxHight = 0;
|
||||||
// this loop through subjects
|
// this loop through subjects
|
||||||
for (i = 0; i < n; ++i) {
|
for (i = 0; i < n; ++i) {
|
||||||
|
|
||||||
@@ -154,7 +157,9 @@ void init_wins(WINDOW **wins, int n, json marks_json) {
|
|||||||
wsub_name.c_str(), wsub_avg_s.c_str());
|
wsub_name.c_str(), wsub_avg_s.c_str());
|
||||||
|
|
||||||
size_t max_text_length = wcslen(label);
|
size_t max_text_length = wcslen(label);
|
||||||
for (int j = 0; j < marks_json["Subjects"][i]["Marks"].size(); j++) {
|
for (unsigned int j = 0; j < static_cast<unsigned int>(
|
||||||
|
marks_json["Subjects"][i]["Marks"].size());
|
||||||
|
j++) {
|
||||||
std::string caption =
|
std::string caption =
|
||||||
rm_tr_le_whitespace(marks_json["Subjects"][i]["Marks"][j]["Caption"]);
|
rm_tr_le_whitespace(marks_json["Subjects"][i]["Marks"][j]["Caption"]);
|
||||||
std::string theme =
|
std::string theme =
|
||||||
@@ -179,7 +184,9 @@ void init_wins(WINDOW **wins, int n, json marks_json) {
|
|||||||
MaxHight = 0;
|
MaxHight = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (marks_json["Subjects"][i]["Marks"].size() * 2 + DEFAULT_PADDING >
|
if (static_cast<unsigned int>(marks_json["Subjects"][i]["Marks"].size()) *
|
||||||
|
2 +
|
||||||
|
DEFAULT_PADDING >
|
||||||
MaxHight) {
|
MaxHight) {
|
||||||
MaxHight =
|
MaxHight =
|
||||||
marks_json["Subjects"][i]["Marks"].size() * 2 + DEFAULT_PADDING;
|
marks_json["Subjects"][i]["Marks"].size() * 2 + DEFAULT_PADDING;
|
||||||
@@ -198,6 +205,9 @@ void init_wins(WINDOW **wins, int n, json marks_json) {
|
|||||||
/* Show the window with a border and a label */
|
/* Show the window with a border and a label */
|
||||||
void win_show(WINDOW *win, wchar_t *label, int label_color, int width,
|
void win_show(WINDOW *win, wchar_t *label, int label_color, int width,
|
||||||
int height, json marks_json, int SubjectIndex) {
|
int height, json marks_json, int SubjectIndex) {
|
||||||
|
|
||||||
|
// is the compiler smoking weed or something, why is it thinking starty is not
|
||||||
|
// used ??
|
||||||
int startx, starty;
|
int startx, starty;
|
||||||
|
|
||||||
wresize(win, height, width);
|
wresize(win, height, width);
|
||||||
|
|||||||
@@ -0,0 +1,45 @@
|
|||||||
|
#include "memory.h"
|
||||||
|
#include "color.h"
|
||||||
|
#include <iostream>
|
||||||
|
#include <ncurses.h>
|
||||||
|
#include <panel.h>
|
||||||
|
|
||||||
|
void delete_all(std::vector<allocation> *allocated) {
|
||||||
|
for (long long i = allocated->size() - 1; i >= 0; i--) {
|
||||||
|
switch (allocated->at(i).type) {
|
||||||
|
case WINDOW_ARRAY: {
|
||||||
|
WINDOW **windows = static_cast<WINDOW **>(allocated->at(i).ptr);
|
||||||
|
for (std::size_t j = 0; j < allocated->at(i).size; j++) {
|
||||||
|
delwin(windows[j]);
|
||||||
|
}
|
||||||
|
delete[] windows;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case PANEL_ARRAY: {
|
||||||
|
PANEL **panels = static_cast<PANEL **>(allocated->at(i).ptr);
|
||||||
|
for (std::size_t j = 0; j < allocated->at(i).size; j++) {
|
||||||
|
del_panel(panels[j]);
|
||||||
|
}
|
||||||
|
delete[] panels;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case GENERIC_ARRAY:
|
||||||
|
delete[] static_cast<char *>(allocated->at(i).ptr);
|
||||||
|
break;
|
||||||
|
case WINDOW_TYPE:
|
||||||
|
delwin(static_cast<WINDOW *>(allocated->at(i).ptr));
|
||||||
|
break;
|
||||||
|
case PANEL_TYPE:
|
||||||
|
del_panel(static_cast<PANEL *>(allocated->at(i).ptr));
|
||||||
|
break;
|
||||||
|
case GENERIC_TYPE:
|
||||||
|
delete static_cast<char *>(allocated->at(i).ptr);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
std::cerr << RED "[!!CRITICAL!!]" << RESET " Unknown allocation type"
|
||||||
|
<< "\n";
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
allocated->pop_back();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
// header guard
|
||||||
|
#ifndef _ba_me_hg_
|
||||||
|
#define _ba_me_hg_
|
||||||
|
|
||||||
|
#include <cstddef>
|
||||||
|
#include <vector>
|
||||||
|
enum AllocationType {
|
||||||
|
WINDOW_ARRAY,
|
||||||
|
PANEL_ARRAY,
|
||||||
|
GENERIC_ARRAY,
|
||||||
|
WINDOW_TYPE,
|
||||||
|
PANEL_TYPE,
|
||||||
|
GENERIC_TYPE
|
||||||
|
};
|
||||||
|
|
||||||
|
struct allocation {
|
||||||
|
AllocationType type;
|
||||||
|
void *ptr;
|
||||||
|
std::size_t size;
|
||||||
|
};
|
||||||
|
|
||||||
|
void delete_all(std::vector<allocation> *allocated);
|
||||||
|
#endif
|
||||||
+14
-18
@@ -21,7 +21,7 @@
|
|||||||
using nlohmann::json;
|
using nlohmann::json;
|
||||||
|
|
||||||
// metods
|
// metods
|
||||||
enum {
|
enum metod {
|
||||||
GET,
|
GET,
|
||||||
POST,
|
POST,
|
||||||
};
|
};
|
||||||
@@ -38,8 +38,8 @@ size_t WriteCallback(void *contents, size_t size, size_t nmemb,
|
|||||||
return totalSize;
|
return totalSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::tuple<std::string, int>
|
std::tuple<std::string, int> send_curl_request(std::string endpoint, metod type,
|
||||||
send_curl_request(std::string endpoint, uint8_t type, std::string req_data) {
|
std::string req_data) {
|
||||||
std::string response;
|
std::string response;
|
||||||
std::string url = baka_api_url + endpoint;
|
std::string url = baka_api_url + endpoint;
|
||||||
if (type == GET) {
|
if (type == GET) {
|
||||||
@@ -47,8 +47,6 @@ send_curl_request(std::string endpoint, uint8_t type, std::string req_data) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (curl) {
|
if (curl) {
|
||||||
// DEBUG
|
|
||||||
// std::clog << BLUE"[LOG]" << RESET" sending to endpoint: " << url << "\n";
|
|
||||||
curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
|
curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
|
||||||
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteCallback);
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteCallback);
|
||||||
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &response);
|
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &response);
|
||||||
@@ -123,14 +121,12 @@ void login(std::string username, std::string password) {
|
|||||||
json resp_parsed = json::parse(response);
|
json resp_parsed = json::parse(response);
|
||||||
|
|
||||||
access_token = resp_parsed["access_token"];
|
access_token = resp_parsed["access_token"];
|
||||||
|
|
||||||
// DEBUG
|
|
||||||
std::cout << "access token: " << access_token << std::endl;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void refresh_access_token() {
|
void refresh_access_token() {
|
||||||
// DEBUG
|
if (config.verbose) {
|
||||||
std::clog << "refreshing access token please wait...\n";
|
std::clog << "refreshing access token please wait...\n";
|
||||||
|
}
|
||||||
|
|
||||||
json authfile_parsed = json::parse(SoRAuthFile(false, ""));
|
json authfile_parsed = json::parse(SoRAuthFile(false, ""));
|
||||||
|
|
||||||
@@ -141,9 +137,6 @@ void refresh_access_token() {
|
|||||||
"token&refresh_token={}",
|
"token&refresh_token={}",
|
||||||
refresh_token);
|
refresh_token);
|
||||||
|
|
||||||
// DEBUG
|
|
||||||
std::clog << "calling send_curl_request() with folowing req_data\n"
|
|
||||||
<< req_data << std::endl;
|
|
||||||
auto [response, http_code] = send_curl_request("api/login", POST, req_data);
|
auto [response, http_code] = send_curl_request("api/login", POST, req_data);
|
||||||
if (http_code != 200) {
|
if (http_code != 200) {
|
||||||
std::cerr << RED "[ERROR] " << RESET << http_code
|
std::cerr << RED "[ERROR] " << RESET << http_code
|
||||||
@@ -157,6 +150,7 @@ void refresh_access_token() {
|
|||||||
|
|
||||||
access_token = resp_parsed["access_token"];
|
access_token = resp_parsed["access_token"];
|
||||||
}
|
}
|
||||||
|
|
||||||
void is_access_token_empty() {
|
void is_access_token_empty() {
|
||||||
if (access_token.empty()) {
|
if (access_token.empty()) {
|
||||||
json authfile_parsed = json::parse(SoRAuthFile(false, ""));
|
json authfile_parsed = json::parse(SoRAuthFile(false, ""));
|
||||||
@@ -165,19 +159,21 @@ void is_access_token_empty() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// supports all endpoints that only require access_token
|
// supports all endpoints that only require access_token
|
||||||
json get_data_from_endpoint(std::string endpoint) {
|
json get_data_from_endpoint(std::string &endpoint,
|
||||||
|
std::string additional_data) {
|
||||||
is_access_token_empty();
|
is_access_token_empty();
|
||||||
|
access_token_refreshed:
|
||||||
std::string req_data =
|
std::string req_data =
|
||||||
std::format("Authorization=Bearer&access_token={}", access_token);
|
std::format("Authorization=Bearer&access_token={}", access_token);
|
||||||
|
if (!additional_data.empty()) {
|
||||||
|
req_data.append(std::format("&{}", additional_data));
|
||||||
|
}
|
||||||
|
|
||||||
auto [response, http_code] = send_curl_request(endpoint, GET, req_data);
|
auto [response, http_code] = send_curl_request(endpoint, GET, req_data);
|
||||||
|
|
||||||
if (http_code != 200) {
|
if (http_code != 200) {
|
||||||
// DEBUG
|
|
||||||
std::clog << "Failed geting data from endpoint: " << endpoint
|
|
||||||
<< " code: " << http_code << "\nrequest: " << req_data
|
|
||||||
<< "\nresponse: " << response << std::endl;
|
|
||||||
refresh_access_token();
|
refresh_access_token();
|
||||||
|
goto access_token_refreshed;
|
||||||
}
|
}
|
||||||
|
|
||||||
return json::parse(response);
|
return json::parse(response);
|
||||||
|
|||||||
@@ -11,6 +11,6 @@ extern CURL *curl;
|
|||||||
namespace bakaapi {
|
namespace bakaapi {
|
||||||
void login(std::string username, std::string password);
|
void login(std::string username, std::string password);
|
||||||
void refresh_access_token();
|
void refresh_access_token();
|
||||||
json get_data_from_endpoint(std::string endpoint);
|
json get_data_from_endpoint(std::string &endpoint, std::string data = "");
|
||||||
} // namespace bakaapi
|
} // namespace bakaapi
|
||||||
#endif
|
#endif
|
||||||
+410
-150
@@ -2,29 +2,46 @@
|
|||||||
#include "color.h"
|
#include "color.h"
|
||||||
#include "const.h"
|
#include "const.h"
|
||||||
#include "helper_funcs.h"
|
#include "helper_funcs.h"
|
||||||
|
#include "memory.h"
|
||||||
#include "net.h"
|
#include "net.h"
|
||||||
#include "types.h"
|
#include "types.h"
|
||||||
|
#include <bits/chrono.h>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <ctime>
|
||||||
#include <curses.h>
|
#include <curses.h>
|
||||||
#include <cwchar>
|
#include <cwchar>
|
||||||
#include <fstream>
|
#include <iomanip>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <ncurses.h>
|
#include <ncurses.h>
|
||||||
#include <nlohmann/json.hpp>
|
#include <nlohmann/json.hpp>
|
||||||
#include <panel.h>
|
#include <panel.h>
|
||||||
|
#include <sstream>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <sys/types.h>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
using nlohmann::json;
|
using nlohmann::json;
|
||||||
#define BOTTOM_PADDING 3
|
#define BOTTOM_PADDING 5
|
||||||
|
|
||||||
#define DEFAULT_OFFSET 3
|
#define DEFAULT_OFFSET 3
|
||||||
|
|
||||||
|
std::vector<allocation> timetable_allocated;
|
||||||
|
|
||||||
const wchar_t *day_abriviations[] = {nullptr, L"Mo", L"Tu", L"We",
|
const wchar_t *day_abriviations[] = {nullptr, L"Mo", L"Tu", L"We",
|
||||||
L"Th", L"Fr", L"Sa", L"Su"};
|
L"Th", L"Fr", L"Sa", L"Su"};
|
||||||
|
|
||||||
void draw_grid(const uint8_t num_of_columns, const uint8_t num_of_rows,
|
void draw_days(WINDOW **&day_windows, uint16_t cell_height, uint8_t num_of_days,
|
||||||
const uint16_t cell_width, const uint16_t cell_height);
|
json &resp_from_api);
|
||||||
|
|
||||||
|
void draw_lessons(WINDOW **&lesson_windows, uint8_t num_of_columns,
|
||||||
|
uint16_t cell_width, std::vector<uint8_t> &HourIdLookupTable,
|
||||||
|
json &resp_from_api);
|
||||||
|
|
||||||
|
void draw_cells(uint8_t num_of_columns, uint8_t num_of_days,
|
||||||
|
uint16_t cell_width, uint16_t cell_height,
|
||||||
|
std::vector<std::vector<WINDOW *>> &cells,
|
||||||
|
std::vector<uint8_t> &HourIdLookupTable, json &resp_from_api);
|
||||||
|
|
||||||
uint8_t hour_id_to_index(const std::vector<uint8_t> &HourIdLookupTable,
|
uint8_t hour_id_to_index(const std::vector<uint8_t> &HourIdLookupTable,
|
||||||
uint8_t id) {
|
uint8_t id) {
|
||||||
@@ -57,12 +74,33 @@ std::wstring get_data_for_atom(json &resp_from_api, json *atom,
|
|||||||
.get<std::string>());
|
.get<std::string>());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
json *find_atom_by_indexes(json &resp_from_api, uint8_t day_index,
|
||||||
|
uint8_t hour_index,
|
||||||
|
const std::vector<uint8_t> &HourIdLookupTable) {
|
||||||
|
for (uint8_t k = 0; k < resp_from_api["Days"][day_index]["Atoms"].size();
|
||||||
|
k++) {
|
||||||
|
if (resp_from_api["Days"][day_index]["Atoms"][k]["HourId"].get<uint8_t>() ==
|
||||||
|
HourIdLookupTable[hour_index]) {
|
||||||
|
return &resp_from_api["Days"][day_index]["Atoms"][k];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr; // No matching atom found
|
||||||
|
}
|
||||||
|
|
||||||
void timetable_page() {
|
void timetable_page() {
|
||||||
// DONT FORGET TO UNCOMMENT
|
current_allocated = &timetable_allocated;
|
||||||
// json resp_from_api =
|
auto dateSelected = std::chrono::system_clock::now();
|
||||||
// bakaapi::get_data_from_endpoint("api/3/timetable/actual");
|
reload_for_new_week:
|
||||||
std::ifstream f("test-data/timetable.json");
|
std::time_t date_time_t = std::chrono::system_clock::to_time_t(dateSelected);
|
||||||
json resp_from_api = json::parse(f);
|
std::tm local_time = *std::localtime(&date_time_t);
|
||||||
|
|
||||||
|
std::stringstream date_stringstream;
|
||||||
|
date_stringstream << std::put_time(&local_time, "%Y-%m-%d");
|
||||||
|
|
||||||
|
std::string date_string = "date=" + date_stringstream.str();
|
||||||
|
std::string endpoint = "api/3/timetable/actual";
|
||||||
|
|
||||||
|
json resp_from_api = bakaapi::get_data_from_endpoint(endpoint, date_string);
|
||||||
|
|
||||||
// this may be unnecessary but i dont have enaugh data to test it
|
// this may be unnecessary but i dont have enaugh data to test it
|
||||||
// it sorts the hours by start time
|
// it sorts the hours by start time
|
||||||
@@ -73,6 +111,8 @@ void timetable_page() {
|
|||||||
|
|
||||||
Id_and_Start_time *temp_hour_sorting_array =
|
Id_and_Start_time *temp_hour_sorting_array =
|
||||||
new Id_and_Start_time[resp_from_api["Hours"].size()];
|
new Id_and_Start_time[resp_from_api["Hours"].size()];
|
||||||
|
timetable_allocated.push_back({GENERIC_ARRAY, temp_hour_sorting_array,
|
||||||
|
resp_from_api["Hours"].size()});
|
||||||
|
|
||||||
for (uint8_t i = 0; i < resp_from_api["Hours"].size(); i++) {
|
for (uint8_t i = 0; i < resp_from_api["Hours"].size(); i++) {
|
||||||
temp_hour_sorting_array[i] = std::make_tuple(
|
temp_hour_sorting_array[i] = std::make_tuple(
|
||||||
@@ -117,6 +157,7 @@ void timetable_page() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
delete[] temp_hour_sorting_array;
|
delete[] temp_hour_sorting_array;
|
||||||
|
timetable_allocated.pop_back();
|
||||||
}
|
}
|
||||||
|
|
||||||
// some lambda dark magic
|
// some lambda dark magic
|
||||||
@@ -156,28 +197,360 @@ void timetable_page() {
|
|||||||
const uint16_t cell_height = (LINES - BOTTOM_PADDING) / num_of_days;
|
const uint16_t cell_height = (LINES - BOTTOM_PADDING) / num_of_days;
|
||||||
|
|
||||||
WINDOW **day_windows = new WINDOW *[num_of_days];
|
WINDOW **day_windows = new WINDOW *[num_of_days];
|
||||||
|
timetable_allocated.push_back({WINDOW_ARRAY, day_windows, num_of_days});
|
||||||
|
|
||||||
WINDOW **lesson_windows = new WINDOW *[num_of_columns];
|
WINDOW **lesson_windows = new WINDOW *[num_of_columns];
|
||||||
|
timetable_allocated.push_back({WINDOW_ARRAY, lesson_windows, num_of_columns});
|
||||||
|
|
||||||
std::vector<std::vector<WINDOW *>> cells(
|
std::vector<std::vector<WINDOW *>> cells(
|
||||||
num_of_days, std::vector<WINDOW *>(num_of_columns));
|
num_of_days, std::vector<WINDOW *>(num_of_columns));
|
||||||
|
|
||||||
|
// init day windows
|
||||||
for (uint8_t i = 0; i < num_of_days; i++) {
|
for (uint8_t i = 0; i < num_of_days; i++) {
|
||||||
day_windows[i] = newwin(cell_height, DEFAULT_OFFSET,
|
day_windows[i] = newwin(cell_height, DEFAULT_OFFSET,
|
||||||
i * cell_height + DEFAULT_OFFSET, 0);
|
i * cell_height + DEFAULT_OFFSET, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
// init cell windows
|
||||||
|
for (uint8_t i = 0; i < num_of_columns; i++) {
|
||||||
|
lesson_windows[i] =
|
||||||
|
newwin(DEFAULT_OFFSET, cell_width, 0, i * cell_width + DEFAULT_OFFSET);
|
||||||
|
}
|
||||||
|
draw_lessons(lesson_windows, num_of_columns, cell_width, HourIdLookupTable,
|
||||||
|
resp_from_api);
|
||||||
|
// days have to be drawn after lessons for some reason i actualy have no idea
|
||||||
|
// why
|
||||||
|
draw_days(day_windows, cell_height, num_of_days, resp_from_api);
|
||||||
|
|
||||||
|
// init the cell windows
|
||||||
|
for (uint8_t i = 0; i < num_of_days; i++) {
|
||||||
|
for (uint8_t j = 0; j < num_of_columns; j++) {
|
||||||
|
cells[i][j] =
|
||||||
|
newwin(cell_height, cell_width, i * cell_height + DEFAULT_OFFSET,
|
||||||
|
j * cell_width + DEFAULT_OFFSET);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
draw_cells(num_of_columns, num_of_days, cell_width, cell_height, cells,
|
||||||
|
HourIdLookupTable, resp_from_api);
|
||||||
|
|
||||||
|
refresh();
|
||||||
|
|
||||||
|
SelectorType selected_cell(0, 0, 0, num_of_columns - 1, 0, num_of_days - 1);
|
||||||
|
std::array<WINDOW *, 4> selector_windows;
|
||||||
|
std::array<PANEL *, 4> selector_panels;
|
||||||
|
|
||||||
|
{
|
||||||
|
const chtype corners[] = {
|
||||||
|
ACS_ULCORNER, /* Upper left corner */
|
||||||
|
ACS_URCORNER, /* Upper right corner */
|
||||||
|
ACS_LLCORNER, /* Lower left corner */
|
||||||
|
ACS_LRCORNER /* Lower right corner */
|
||||||
|
};
|
||||||
|
|
||||||
|
unsigned short x_offset, y_offset;
|
||||||
|
for (uint8_t i = 0; i < selector_windows.size(); i++) {
|
||||||
|
|
||||||
|
if (!(i % 2 == 0)) {
|
||||||
|
x_offset = cell_width - 1;
|
||||||
|
} else {
|
||||||
|
x_offset = 0;
|
||||||
|
}
|
||||||
|
if (!(i < 2)) {
|
||||||
|
y_offset = cell_height - 1;
|
||||||
|
} else {
|
||||||
|
y_offset = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
selector_windows[i] =
|
||||||
|
newwin(1, 1, DEFAULT_OFFSET + y_offset, DEFAULT_OFFSET + x_offset);
|
||||||
|
timetable_allocated.push_back({WINDOW_TYPE, selector_windows[i], 1});
|
||||||
|
|
||||||
|
selector_panels[i] = new_panel(selector_windows[i]);
|
||||||
|
timetable_allocated.push_back({PANEL_TYPE, selector_panels[i], 1});
|
||||||
|
|
||||||
|
wattron(selector_windows[i], COLOR_PAIR(COLOR_RED));
|
||||||
|
mvwaddch(selector_windows[i], 0, 0, corners[i]);
|
||||||
|
wattroff(selector_windows[i], COLOR_PAIR(COLOR_RED));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
attron(COLOR_PAIR(COLOR_BLUE));
|
||||||
|
mvprintw(LINES - 2, 0,
|
||||||
|
"Arrows/hjkl to select | ENTER to show info | p/n to select weeks "
|
||||||
|
"|F1 to exit");
|
||||||
|
{
|
||||||
|
std::tm end_week = local_time;
|
||||||
|
std::tm start_week = local_time;
|
||||||
|
|
||||||
|
// Convert tm_wday (0-6) to API day format (1-7)
|
||||||
|
int current_wday = (local_time.tm_wday == 0) ? 7 : local_time.tm_wday;
|
||||||
|
|
||||||
|
// Get days of week from API (1-7 format where Monday is 1)
|
||||||
|
uint8_t start_day = resp_from_api["Days"][0]["DayOfWeek"].get<uint8_t>();
|
||||||
|
uint8_t end_day =
|
||||||
|
resp_from_api["Days"][resp_from_api["Days"].size() - 1]["DayOfWeek"]
|
||||||
|
.get<uint8_t>();
|
||||||
|
|
||||||
|
// Calculate days back to start day (handles week wraparound)
|
||||||
|
int days_back = (current_wday >= start_day)
|
||||||
|
? (current_wday - start_day)
|
||||||
|
: (current_wday + 7 - start_day);
|
||||||
|
|
||||||
|
// Calculate days forward to end day (handles week wraparound)
|
||||||
|
int days_forward = (current_wday <= end_day) ? (end_day - current_wday)
|
||||||
|
: (end_day + 7 - current_wday);
|
||||||
|
|
||||||
|
// Adjust dates
|
||||||
|
start_week.tm_mday -= days_back;
|
||||||
|
end_week.tm_mday += days_forward;
|
||||||
|
|
||||||
|
// Normalize the dates
|
||||||
|
std::mktime(&start_week);
|
||||||
|
std::mktime(&end_week);
|
||||||
|
|
||||||
|
// Format the dates as strings
|
||||||
|
std::stringstream start_week_strstream, end_week_strstream;
|
||||||
|
start_week_strstream << std::put_time(&start_week, "%d.%m.%Y");
|
||||||
|
end_week_strstream << std::put_time(&end_week, "%d.%m.%Y");
|
||||||
|
|
||||||
|
// kern. developer approved ↓↓
|
||||||
|
mvprintw(LINES - 2,
|
||||||
|
COLS - (start_week_strstream.str().length() + 3 +
|
||||||
|
end_week_strstream.str().length()),
|
||||||
|
"%s",
|
||||||
|
(start_week_strstream.str() + " - " + end_week_strstream.str())
|
||||||
|
.c_str());
|
||||||
|
}
|
||||||
|
attroff(COLOR_PAIR(COLOR_BLUE));
|
||||||
|
|
||||||
|
update_panels();
|
||||||
|
doupdate();
|
||||||
|
|
||||||
|
WINDOW *infobox_window;
|
||||||
|
PANEL *infobox_panel;
|
||||||
|
|
||||||
|
bool is_info_box_open = false;
|
||||||
|
int ch;
|
||||||
|
while ((ch = getch()) != KEY_F(1)) {
|
||||||
|
if (is_info_box_open) {
|
||||||
|
|
||||||
|
hide_panel(infobox_panel);
|
||||||
|
del_panel(infobox_panel);
|
||||||
|
|
||||||
|
delwin(infobox_window);
|
||||||
|
|
||||||
|
touchwin(stdscr);
|
||||||
|
refresh();
|
||||||
|
|
||||||
|
// Redraw everithing
|
||||||
|
draw_days(day_windows, cell_height, num_of_days, resp_from_api);
|
||||||
|
draw_lessons(lesson_windows, num_of_columns, cell_width,
|
||||||
|
HourIdLookupTable, resp_from_api);
|
||||||
|
draw_cells(num_of_columns, num_of_days, cell_width, cell_height, cells,
|
||||||
|
HourIdLookupTable, resp_from_api);
|
||||||
|
|
||||||
|
for (uint8_t i = 0; i < selector_panels.size(); i++) {
|
||||||
|
top_panel(selector_panels[i]);
|
||||||
|
}
|
||||||
|
update_panels();
|
||||||
|
doupdate();
|
||||||
|
|
||||||
|
is_info_box_open = false;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
run_loop_again:
|
||||||
|
switch (ch) {
|
||||||
|
case KEY_UP:
|
||||||
|
case 'k':
|
||||||
|
selected_cell.y--;
|
||||||
|
break;
|
||||||
|
case KEY_DOWN:
|
||||||
|
case 'j':
|
||||||
|
selected_cell.y++;
|
||||||
|
break;
|
||||||
|
case KEY_LEFT:
|
||||||
|
case 'h':
|
||||||
|
selected_cell.x--;
|
||||||
|
break;
|
||||||
|
case KEY_RIGHT:
|
||||||
|
case 'l':
|
||||||
|
selected_cell.x++;
|
||||||
|
break;
|
||||||
|
case 'p':
|
||||||
|
case 'n':
|
||||||
|
(ch == 'p') ? dateSelected = dateSelected - std::chrono::days(7)
|
||||||
|
: dateSelected = dateSelected + std::chrono::days(7);
|
||||||
|
delete_all(&timetable_allocated);
|
||||||
|
goto reload_for_new_week;
|
||||||
|
break;
|
||||||
|
case 10: // ENTER
|
||||||
|
json *atom = find_atom_by_indexes(resp_from_api, selected_cell.y,
|
||||||
|
selected_cell.x, HourIdLookupTable);
|
||||||
|
if (atom == nullptr) {
|
||||||
|
std::cerr << RED "[ERROR]" << RESET " Selector at invalid position\n";
|
||||||
|
safe_exit(129);
|
||||||
|
}
|
||||||
|
|
||||||
|
infobox_window = newwin(LINES * 0.6, COLS * 0.6, LINES * 0.2, COLS * 0.2);
|
||||||
|
infobox_panel = new_panel(infobox_window);
|
||||||
|
is_info_box_open = true;
|
||||||
|
|
||||||
|
wattron(infobox_window, COLOR_PAIR(COLOR_MAGENTA));
|
||||||
|
box(infobox_window, 0, 0);
|
||||||
|
mvwaddch(infobox_window, 2, 0, ACS_LTEE);
|
||||||
|
mvwhline(infobox_window, 2, 1, ACS_HLINE, COLS * 0.6 - 2);
|
||||||
|
mvwaddch(infobox_window, 2, COLS * 0.6 - 1, ACS_RTEE);
|
||||||
|
wattroff(infobox_window, COLOR_PAIR(COLOR_MAGENTA));
|
||||||
|
|
||||||
|
std::wstring Caption;
|
||||||
|
if (atom->contains("Change") && !atom->at("Change").is_null()) {
|
||||||
|
if (!atom->at("Change")["TypeName"].is_null()) {
|
||||||
|
Caption = string_to_wstring(
|
||||||
|
atom->at("Change")["TypeName"].get<std::string>());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Caption.empty()) {
|
||||||
|
try {
|
||||||
|
Caption = get_data_for_atom(resp_from_api, atom, "Subjects",
|
||||||
|
"SubjectId", "Name");
|
||||||
|
} catch (...) {
|
||||||
|
__asm__("nop");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::wstring Teacher = L"";
|
||||||
|
try {
|
||||||
|
Teacher = get_data_for_atom(resp_from_api, atom, "Teachers",
|
||||||
|
"TeacherId", "Name");
|
||||||
|
} catch (...) {
|
||||||
|
__asm__("nop");
|
||||||
|
}
|
||||||
|
Teacher.insert(0, L"Teacher: ");
|
||||||
|
|
||||||
|
std::wstring Groups = L"";
|
||||||
|
try {
|
||||||
|
for (uint8_t i = 0; i < atom->at("GroupIds").size(); i++) {
|
||||||
|
for (uint8_t j = 0; j < resp_from_api["Groups"].size(); j++) {
|
||||||
|
if (resp_from_api["Groups"][j]["Id"].get<std::string>() ==
|
||||||
|
atom->at("GroupIds")[i].get<std::string>()) {
|
||||||
|
Groups.append(string_to_wstring(
|
||||||
|
resp_from_api["Groups"][j]["Name"].get<std::string>()));
|
||||||
|
if (static_cast<size_t>(i + 1) < atom->at("GroupIds").size()) {
|
||||||
|
Groups.append(L", ");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (const std::exception &e) {
|
||||||
|
std::cerr << RED "[ERROR]" << RESET " " << e.what() << "\n";
|
||||||
|
}
|
||||||
|
Groups = wrm_tr_le_whitespace(Groups);
|
||||||
|
Groups.insert(0, L"Groups: ");
|
||||||
|
|
||||||
|
std::wstring Room = L"";
|
||||||
|
try {
|
||||||
|
Room =
|
||||||
|
get_data_for_atom(resp_from_api, atom, "Rooms", "RoomId", "Name");
|
||||||
|
if (Room.empty()) {
|
||||||
|
Room = get_data_for_atom(resp_from_api, atom, "Rooms", "RoomId",
|
||||||
|
"Abbrev");
|
||||||
|
;
|
||||||
|
}
|
||||||
|
} catch (...) {
|
||||||
|
__asm__("nop");
|
||||||
|
}
|
||||||
|
Room.insert(0, L"Room: ");
|
||||||
|
|
||||||
|
std::wstring Theme = L"";
|
||||||
|
try {
|
||||||
|
Theme = wrm_tr_le_whitespace(
|
||||||
|
string_to_wstring(atom->at("Theme").get<std::string>()));
|
||||||
|
} catch (...) {
|
||||||
|
__asm__("nop");
|
||||||
|
}
|
||||||
|
Theme.insert(0, L"Theme: ");
|
||||||
|
|
||||||
|
wprint_in_middle(infobox_window, 1, 0, getmaxx(infobox_window),
|
||||||
|
Caption.c_str(), COLOR_PAIR(COLOR_CYAN));
|
||||||
|
|
||||||
|
// printing out of order to reduce wattro* directives
|
||||||
|
wattron(infobox_window, COLOR_PAIR(COLOR_YELLOW));
|
||||||
|
mvwaddwstr(infobox_window, 3, 1, Teacher.c_str());
|
||||||
|
mvwaddwstr(infobox_window, 5, 1, Room.c_str());
|
||||||
|
wattroff(infobox_window, COLOR_PAIR(COLOR_YELLOW));
|
||||||
|
|
||||||
|
wattron(infobox_window, COLOR_PAIR(COLOR_CYAN));
|
||||||
|
mvwaddwstr(infobox_window, 4, 1, Groups.c_str());
|
||||||
|
mvwaddwstr(infobox_window, 6, 1, Theme.c_str());
|
||||||
|
wattroff(infobox_window, COLOR_PAIR(COLOR_CYAN));
|
||||||
|
|
||||||
|
wattron(infobox_window, COLOR_PAIR(COLOR_BLUE));
|
||||||
|
mvwaddstr(infobox_window, getmaxy(infobox_window) - 2, 1,
|
||||||
|
"Press any key to close");
|
||||||
|
wattroff(infobox_window, COLOR_PAIR(COLOR_BLUE));
|
||||||
|
|
||||||
|
top_panel(infobox_panel);
|
||||||
|
update_panels();
|
||||||
|
doupdate();
|
||||||
|
continue;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
{ // print selected indicator
|
||||||
|
chtype top_left_corner =
|
||||||
|
mvwinch(cells[selected_cell.y][selected_cell.x], 0, 0);
|
||||||
|
|
||||||
|
if (!((top_left_corner & A_CHARTEXT) == 32)) {
|
||||||
|
for (uint8_t i = 0; i < selector_panels.size(); i++) {
|
||||||
|
unsigned short x_offset, y_offset;
|
||||||
|
if (!(i % 2 == 0)) {
|
||||||
|
x_offset = cell_width - 1;
|
||||||
|
} else {
|
||||||
|
x_offset = 0;
|
||||||
|
}
|
||||||
|
if (!(i < 2)) {
|
||||||
|
y_offset = cell_height - 1;
|
||||||
|
} else {
|
||||||
|
y_offset = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
move_panel(selector_panels[i],
|
||||||
|
DEFAULT_OFFSET + y_offset + selected_cell.y * cell_height,
|
||||||
|
DEFAULT_OFFSET + x_offset + selected_cell.x * cell_width);
|
||||||
|
}
|
||||||
|
draw_cells(num_of_columns, num_of_days, cell_width, cell_height, cells,
|
||||||
|
HourIdLookupTable, resp_from_api);
|
||||||
|
update_panels();
|
||||||
|
doupdate();
|
||||||
|
|
||||||
|
} else {
|
||||||
|
// skip if the cell is empty
|
||||||
|
goto run_loop_again;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delete_all(&timetable_allocated);
|
||||||
|
clear();
|
||||||
|
endwin();
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw_days(WINDOW **&day_windows, uint16_t cell_height, uint8_t num_of_days,
|
||||||
|
json &resp_from_api) {
|
||||||
|
for (uint8_t i = 0; i < num_of_days; i++) {
|
||||||
// this wont draw left boarder window making it so it looks partially
|
// this wont draw left boarder window making it so it looks partially
|
||||||
// offscreen
|
// offscreen
|
||||||
wborder(day_windows[i], ' ', 0, 0, 0, ACS_HLINE, 0, ACS_HLINE, 0);
|
wborder(day_windows[i], ' ', 0, 0, 0, ACS_HLINE, 0, ACS_HLINE, 0);
|
||||||
const wchar_t *day_abriv =
|
|
||||||
day_abriviations[resp_from_api["Days"][i]["DayOfWeek"].get<uint8_t>()];
|
|
||||||
|
|
||||||
wprint_in_middle(day_windows[i], cell_height / 2, 0, wcslen(day_abriv),
|
mvwaddwstr(
|
||||||
day_abriv, COLOR_PAIR(0));
|
day_windows[i], cell_height / 2, 0,
|
||||||
|
day_abriviations[resp_from_api["Days"][i]["DayOfWeek"].get<uint8_t>()]);
|
||||||
wrefresh(day_windows[i]);
|
wrefresh(day_windows[i]);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw_lessons(WINDOW **&lesson_windows, uint8_t num_of_columns,
|
||||||
|
uint16_t cell_width, std::vector<uint8_t> &HourIdLookupTable,
|
||||||
|
json &resp_from_api) {
|
||||||
for (uint8_t i = 0; i < num_of_columns; i++) {
|
for (uint8_t i = 0; i < num_of_columns; i++) {
|
||||||
|
|
||||||
lesson_windows[i] =
|
|
||||||
newwin(DEFAULT_OFFSET, cell_width, 0, i * cell_width + DEFAULT_OFFSET);
|
|
||||||
wborder(lesson_windows[i], 0, 0, ' ', 0, ACS_VLINE, ACS_VLINE, 0, 0);
|
wborder(lesson_windows[i], 0, 0, ' ', 0, ACS_VLINE, ACS_VLINE, 0, 0);
|
||||||
std::wstring caption;
|
std::wstring caption;
|
||||||
std::wstring start_time;
|
std::wstring start_time;
|
||||||
@@ -186,9 +559,6 @@ void timetable_page() {
|
|||||||
for (uint8_t j = 0; j < resp_from_api["Hours"].size(); j++) {
|
for (uint8_t j = 0; j < resp_from_api["Hours"].size(); j++) {
|
||||||
if (resp_from_api["Hours"][j]["Id"].get<uint8_t>() ==
|
if (resp_from_api["Hours"][j]["Id"].get<uint8_t>() ==
|
||||||
HourIdLookupTable[i]) {
|
HourIdLookupTable[i]) {
|
||||||
// DEBUG
|
|
||||||
// std::clog <<
|
|
||||||
// resp_from_api["Hours"][j]["Caption"].get<std::string>();
|
|
||||||
|
|
||||||
std::string caption_ascii =
|
std::string caption_ascii =
|
||||||
resp_from_api["Hours"][j]["Caption"].get<std::string>();
|
resp_from_api["Hours"][j]["Caption"].get<std::string>();
|
||||||
@@ -210,33 +580,27 @@ void timetable_page() {
|
|||||||
|
|
||||||
hour_id_found:
|
hour_id_found:
|
||||||
|
|
||||||
wprint_in_middle(lesson_windows[i], 0, cell_width / 2, caption.length(),
|
wprint_in_middle(lesson_windows[i], 0, 0, cell_width, caption.c_str(),
|
||||||
caption.c_str(), COLOR_PAIR(0));
|
COLOR_PAIR(0));
|
||||||
wprint_in_middle(lesson_windows[i], 1, 1, start_time.length(),
|
mvwaddwstr(lesson_windows[i], 1, 1, start_time.c_str());
|
||||||
start_time.c_str(), COLOR_PAIR(0));
|
print_in_middle(lesson_windows[i], 1, 0, cell_width, "-", COLOR_PAIR(0));
|
||||||
print_in_middle(lesson_windows[i], 1, cell_width / 2, 1, "-",
|
mvwaddwstr(lesson_windows[i], 1, cell_width - end_time.length() - 1,
|
||||||
COLOR_PAIR(0));
|
end_time.c_str());
|
||||||
wprint_in_middle(lesson_windows[i], 1, cell_width - end_time.length() - 1,
|
|
||||||
end_time.length(), end_time.c_str(), COLOR_PAIR(0));
|
|
||||||
wrefresh(lesson_windows[i]);
|
wrefresh(lesson_windows[i]);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw_cells(uint8_t num_of_columns, uint8_t num_of_days,
|
||||||
|
uint16_t cell_width, uint16_t cell_height,
|
||||||
|
std::vector<std::vector<WINDOW *>> &cells,
|
||||||
|
std::vector<uint8_t> &HourIdLookupTable, json &resp_from_api) {
|
||||||
for (uint8_t i = 0; i < num_of_days; i++) {
|
for (uint8_t i = 0; i < num_of_days; i++) {
|
||||||
for (uint8_t j = 0; j < num_of_columns; j++) {
|
for (uint8_t j = 0; j < num_of_columns; j++) {
|
||||||
cells[i][j] =
|
|
||||||
newwin(cell_height, cell_width, i * cell_height + DEFAULT_OFFSET,
|
|
||||||
j * cell_width + DEFAULT_OFFSET);
|
|
||||||
|
|
||||||
json *atom;
|
json *atom = find_atom_by_indexes(resp_from_api, i, j, HourIdLookupTable);
|
||||||
for (uint8_t k = 0; k < resp_from_api["Days"][i]["Atoms"].size(); k++) {
|
if (atom == nullptr) {
|
||||||
if (resp_from_api["Days"][i]["Atoms"][k]["HourId"].get<uint8_t>() ==
|
continue;
|
||||||
HourIdLookupTable[j]) {
|
|
||||||
atom = &resp_from_api["Days"][i]["Atoms"][k];
|
|
||||||
goto correct_atom_found;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
continue;
|
|
||||||
correct_atom_found:
|
|
||||||
std::wstring Subject_Abbrev;
|
std::wstring Subject_Abbrev;
|
||||||
std::wstring Room_Abbrev;
|
std::wstring Room_Abbrev;
|
||||||
std::wstring Teacher_Abbrev;
|
std::wstring Teacher_Abbrev;
|
||||||
@@ -297,17 +661,13 @@ void timetable_page() {
|
|||||||
__asm__("nop");
|
__asm__("nop");
|
||||||
}
|
}
|
||||||
|
|
||||||
wprint_in_middle(cells[i][j], cell_height / 2,
|
wprint_in_middle(cells[i][j], cell_height / 2, 0, cell_width,
|
||||||
cell_width / 2 - wcslen(Subject_Abbrev.c_str()) / 2,
|
Subject_Abbrev.c_str(), COLOR_PAIR(0));
|
||||||
wcslen(Subject_Abbrev.c_str()), Subject_Abbrev.c_str(),
|
mvwaddwstr(cells[i][j], cell_height - 2,
|
||||||
COLOR_PAIR(0));
|
cell_width - wcslen(Room_Abbrev.c_str()) - 1,
|
||||||
wprint_in_middle(cells[i][j], cell_height - 2,
|
Room_Abbrev.c_str());
|
||||||
cell_width - wcslen(Room_Abbrev.c_str()) - 1,
|
|
||||||
wcslen(Room_Abbrev.c_str()), Room_Abbrev.c_str(),
|
mvwaddwstr(cells[i][j], cell_height - 2, 1, Teacher_Abbrev.c_str());
|
||||||
COLOR_PAIR(0));
|
|
||||||
wprint_in_middle(cells[i][j], cell_height - 2, 1,
|
|
||||||
wcslen(Teacher_Abbrev.c_str()), Teacher_Abbrev.c_str(),
|
|
||||||
COLOR_PAIR(0));
|
|
||||||
wrefresh(cells[i][j]);
|
wrefresh(cells[i][j]);
|
||||||
} catch (const std::exception &e) {
|
} catch (const std::exception &e) {
|
||||||
std::cerr << RED "[ERROR]" << RESET " " << e.what() << "\n";
|
std::cerr << RED "[ERROR]" << RESET " " << e.what() << "\n";
|
||||||
@@ -316,104 +676,4 @@ void timetable_page() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
refresh();
|
|
||||||
|
|
||||||
SelectorType selected_cell(0, 0, 0, num_of_columns - 1, 0, num_of_days - 1);
|
|
||||||
std::array<WINDOW *, 4> selector_windows;
|
|
||||||
std::array<PANEL *, 4> selector_panels;
|
|
||||||
|
|
||||||
{
|
|
||||||
const chtype corners[] = {
|
|
||||||
ACS_ULCORNER, /* Upper left corner */
|
|
||||||
ACS_URCORNER, /* Upper right corner */
|
|
||||||
ACS_LLCORNER, /* Lower left corner */
|
|
||||||
ACS_LRCORNER /* Lower right corner */
|
|
||||||
};
|
|
||||||
|
|
||||||
unsigned short x_offset, y_offset;
|
|
||||||
for (uint8_t i = 0; i < selector_windows.size(); i++) {
|
|
||||||
|
|
||||||
if (!(i % 2 == 0)) {
|
|
||||||
x_offset = cell_width - 1;
|
|
||||||
} else {
|
|
||||||
x_offset = 0;
|
|
||||||
}
|
|
||||||
if (!(i < 2)) {
|
|
||||||
y_offset = cell_height - 1;
|
|
||||||
} else {
|
|
||||||
y_offset = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
selector_windows[i] =
|
|
||||||
newwin(1, 1, DEFAULT_OFFSET + y_offset, DEFAULT_OFFSET + x_offset);
|
|
||||||
selector_panels[i] = new_panel(selector_windows[i]);
|
|
||||||
wattron(selector_windows[i], COLOR_PAIR(COLOR_RED));
|
|
||||||
mvwaddch(selector_windows[i], 0, 0, corners[i]);
|
|
||||||
wattroff(selector_windows[i], COLOR_PAIR(COLOR_RED));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
update_panels();
|
|
||||||
doupdate();
|
|
||||||
|
|
||||||
int ch;
|
|
||||||
while ((ch = getch()) != KEY_F(1)) {
|
|
||||||
run_loop_again:
|
|
||||||
switch (ch) {
|
|
||||||
case KEY_UP:
|
|
||||||
case 'k':
|
|
||||||
selected_cell.y--;
|
|
||||||
break;
|
|
||||||
case KEY_DOWN:
|
|
||||||
case 'j':
|
|
||||||
selected_cell.y++;
|
|
||||||
break;
|
|
||||||
case KEY_LEFT:
|
|
||||||
case 'h':
|
|
||||||
selected_cell.x--;
|
|
||||||
break;
|
|
||||||
case KEY_RIGHT:
|
|
||||||
case 'l':
|
|
||||||
selected_cell.x++;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
{ // print selected indicator
|
|
||||||
chtype top_left_corner =
|
|
||||||
mvwinch(cells[selected_cell.y][selected_cell.x], 0, 0);
|
|
||||||
|
|
||||||
if (!((top_left_corner & A_CHARTEXT) == 32)) {
|
|
||||||
for (uint8_t i = 0; i < selector_panels.size(); i++) {
|
|
||||||
unsigned short x_offset, y_offset;
|
|
||||||
if (!(i % 2 == 0)) {
|
|
||||||
x_offset = cell_width - 1;
|
|
||||||
} else {
|
|
||||||
x_offset = 0;
|
|
||||||
}
|
|
||||||
if (!(i < 2)) {
|
|
||||||
y_offset = cell_height - 1;
|
|
||||||
} else {
|
|
||||||
y_offset = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
move_panel(selector_panels[i],
|
|
||||||
DEFAULT_OFFSET + y_offset + selected_cell.y * cell_height,
|
|
||||||
DEFAULT_OFFSET + x_offset + selected_cell.x * cell_width);
|
|
||||||
}
|
|
||||||
for (uint8_t i = 0; i < num_of_days; i++) {
|
|
||||||
for (uint8_t j = 0; j < num_of_columns; j++) {
|
|
||||||
wrefresh(cells[i][j]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
update_panels();
|
|
||||||
doupdate();
|
|
||||||
|
|
||||||
} else {
|
|
||||||
// skip if the cell is empty
|
|
||||||
goto run_loop_again;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
delete[] day_windows;
|
|
||||||
delete[] lesson_windows;
|
|
||||||
endwin();
|
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user