2018-08-08 23:54:58 +02:00
|
|
|
#include "modules/clock.hpp"
|
2020-05-22 18:52:26 +02:00
|
|
|
|
2022-01-08 07:25:15 +01:00
|
|
|
#include <fmt/chrono.h>
|
2022-07-13 07:20:49 +02:00
|
|
|
#include <spdlog/spdlog.h>
|
2020-05-22 18:52:26 +02:00
|
|
|
|
2022-01-08 07:25:15 +01:00
|
|
|
#include <ctime>
|
2022-07-13 07:20:49 +02:00
|
|
|
#include <iomanip>
|
2022-12-10 12:02:15 +01:00
|
|
|
#include <regex>
|
2022-12-10 14:48:22 +01:00
|
|
|
#include <sstream>
|
2020-02-05 20:02:42 +01:00
|
|
|
#include <type_traits>
|
2022-01-08 07:25:15 +01:00
|
|
|
|
2021-01-31 20:53:53 +01:00
|
|
|
#include "util/ustring_clen.hpp"
|
2022-01-08 03:09:44 +01:00
|
|
|
#include "util/waybar_time.hpp"
|
2020-02-04 01:58:18 +01:00
|
|
|
#ifdef HAVE_LANGINFO_1STDAY
|
|
|
|
#include <langinfo.h>
|
2020-02-05 20:02:42 +01:00
|
|
|
#include <locale.h>
|
2020-02-04 01:58:18 +01:00
|
|
|
#endif
|
2020-01-31 17:54:41 +01:00
|
|
|
|
2022-01-08 03:09:44 +01:00
|
|
|
using waybar::waybar_time;
|
2018-08-08 23:54:58 +02:00
|
|
|
|
2018-12-18 17:30:54 +01:00
|
|
|
waybar::modules::Clock::Clock(const std::string& id, const Json::Value& config)
|
2022-11-24 12:28:52 +01:00
|
|
|
: ALabel(config, "clock", id, "{:%H:%M}", 60, false, false, true),
|
2021-10-03 18:48:21 +02:00
|
|
|
current_time_zone_idx_(0),
|
2021-10-05 11:46:52 +02:00
|
|
|
is_calendar_in_tooltip_(false),
|
2022-04-06 08:37:19 +02:00
|
|
|
is_timezoned_list_in_tooltip_(false) {
|
2021-10-03 05:27:54 +02:00
|
|
|
if (config_["timezones"].isArray() && !config_["timezones"].empty()) {
|
2022-04-06 08:37:19 +02:00
|
|
|
for (const auto& zone_name : config_["timezones"]) {
|
2021-10-03 18:48:21 +02:00
|
|
|
if (!zone_name.isString() || zone_name.asString().empty()) {
|
|
|
|
time_zones_.push_back(nullptr);
|
|
|
|
continue;
|
|
|
|
}
|
2022-04-06 08:37:19 +02:00
|
|
|
time_zones_.push_back(date::locate_zone(zone_name.asString()));
|
2021-10-03 18:48:21 +02:00
|
|
|
}
|
2021-10-05 12:20:06 +02:00
|
|
|
} else if (config_["timezone"].isString() && !config_["timezone"].asString().empty()) {
|
2022-04-06 08:37:19 +02:00
|
|
|
time_zones_.push_back(date::locate_zone(config_["timezone"].asString()));
|
2021-10-03 05:27:54 +02:00
|
|
|
}
|
2021-10-03 18:48:21 +02:00
|
|
|
|
2022-04-06 08:37:19 +02:00
|
|
|
// If all timezones are parsed and no one is good, add nullptr to the timezones vector, to mark
|
|
|
|
// that local time should be shown.
|
2021-10-05 12:20:06 +02:00
|
|
|
if (!time_zones_.size()) {
|
|
|
|
time_zones_.push_back(nullptr);
|
|
|
|
}
|
|
|
|
|
2021-10-03 18:48:21 +02:00
|
|
|
if (!is_timezone_fixed()) {
|
2022-04-06 08:37:19 +02:00
|
|
|
spdlog::warn(
|
|
|
|
"As using a timezone, some format args may be missing as the date library haven't got a "
|
|
|
|
"release since 2018.");
|
2020-02-05 20:02:42 +01:00
|
|
|
}
|
2020-01-31 17:54:41 +01:00
|
|
|
|
2022-04-06 08:37:19 +02:00
|
|
|
// Check if a particular placeholder is present in the tooltip format, to know what to calculate
|
|
|
|
// on update.
|
2021-10-03 18:48:21 +02:00
|
|
|
if (config_["tooltip-format"].isString()) {
|
2021-10-05 11:55:30 +02:00
|
|
|
std::string trimmed_format = config_["tooltip-format"].asString();
|
2022-04-06 08:37:19 +02:00
|
|
|
trimmed_format.erase(std::remove_if(trimmed_format.begin(), trimmed_format.end(),
|
|
|
|
[](unsigned char x) { return std::isspace(x); }),
|
|
|
|
trimmed_format.end());
|
2021-10-05 11:55:30 +02:00
|
|
|
if (trimmed_format.find("{" + kCalendarPlaceholder + "}") != std::string::npos) {
|
2021-10-03 18:48:21 +02:00
|
|
|
is_calendar_in_tooltip_ = true;
|
|
|
|
}
|
2021-10-05 11:46:52 +02:00
|
|
|
if (trimmed_format.find("{" + KTimezonedTimeListPlaceholder + "}") != std::string::npos) {
|
|
|
|
is_timezoned_list_in_tooltip_ = true;
|
|
|
|
}
|
2021-10-03 18:48:21 +02:00
|
|
|
}
|
|
|
|
|
2022-08-06 12:55:20 +02:00
|
|
|
if (is_calendar_in_tooltip_) {
|
|
|
|
if (config_["on-scroll"][kCalendarPlaceholder].isInt()) {
|
|
|
|
calendar_shift_init_ =
|
|
|
|
date::months{config_["on-scroll"].get(kCalendarPlaceholder, 0).asInt()};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-13 21:58:00 +02:00
|
|
|
if (config_["locale"].isString()) {
|
|
|
|
locale_ = std::locale(config_["locale"].asString());
|
|
|
|
} else {
|
2022-05-13 21:58:38 +02:00
|
|
|
locale_ = std::locale("");
|
2020-02-05 20:02:42 +01:00
|
|
|
}
|
2020-01-31 17:54:41 +01:00
|
|
|
|
2022-12-10 12:02:15 +01:00
|
|
|
if (config_["format-calendar-weeks"].isString()) {
|
2022-12-10 14:48:22 +01:00
|
|
|
weeks_format_ =
|
|
|
|
std::regex_replace(config_["format-calendar-weeks"].asString(), std::regex("\\{\\}"),
|
|
|
|
(first_day_of_week() == date::Monday) ? "{:%V}" : "{:%U}");
|
|
|
|
weeks_format_left_gaps =
|
|
|
|
std::regex_replace(weeks_format_, std::regex(".*<b>|</b>.*|\\{.?+\\}"), "").length();
|
2022-12-10 12:02:15 +01:00
|
|
|
} else {
|
|
|
|
weeks_format_ = "";
|
|
|
|
}
|
|
|
|
|
2018-11-23 11:57:37 +01:00
|
|
|
thread_ = [this] {
|
2018-08-20 14:50:45 +02:00
|
|
|
dp.emit();
|
2022-03-31 17:14:29 +02:00
|
|
|
auto now = std::chrono::system_clock::now();
|
|
|
|
/* difference with projected wakeup time */
|
|
|
|
auto diff = now.time_since_epoch() % interval_;
|
|
|
|
/* sleep until the next projected time */
|
|
|
|
thread_.sleep_for(interval_ - diff);
|
2018-08-08 23:54:58 +02:00
|
|
|
};
|
2018-08-18 11:43:48 +02:00
|
|
|
}
|
2018-08-08 23:54:58 +02:00
|
|
|
|
2021-10-03 18:48:21 +02:00
|
|
|
const date::time_zone* waybar::modules::Clock::current_timezone() {
|
2022-04-06 08:37:19 +02:00
|
|
|
return time_zones_[current_time_zone_idx_] ? time_zones_[current_time_zone_idx_]
|
|
|
|
: date::current_zone();
|
2021-10-03 18:48:21 +02:00
|
|
|
}
|
2020-05-22 18:52:26 +02:00
|
|
|
|
2021-10-03 18:48:21 +02:00
|
|
|
bool waybar::modules::Clock::is_timezone_fixed() {
|
|
|
|
return time_zones_[current_time_zone_idx_] != nullptr;
|
|
|
|
}
|
2020-01-31 17:54:41 +01:00
|
|
|
|
2021-10-03 18:48:21 +02:00
|
|
|
auto waybar::modules::Clock::update() -> void {
|
|
|
|
auto time_zone = current_timezone();
|
|
|
|
auto now = std::chrono::system_clock::now();
|
2022-08-06 12:55:20 +02:00
|
|
|
waybar_time wtime = {locale_, date::make_zoned(time_zone, date::floor<std::chrono::seconds>(now) +
|
|
|
|
calendar_shift_)};
|
2021-10-03 18:48:21 +02:00
|
|
|
std::string text = "";
|
|
|
|
if (!is_timezone_fixed()) {
|
2020-05-22 18:52:26 +02:00
|
|
|
// As date dep is not fully compatible, prefer fmt
|
|
|
|
tzset();
|
|
|
|
auto localtime = fmt::localtime(std::chrono::system_clock::to_time_t(now));
|
2022-01-14 19:36:24 +01:00
|
|
|
text = fmt::format(locale_, format_, localtime);
|
2020-05-22 18:52:26 +02:00
|
|
|
} else {
|
2020-05-22 18:56:32 +02:00
|
|
|
text = fmt::format(format_, wtime);
|
2020-05-22 18:52:26 +02:00
|
|
|
}
|
2022-11-24 12:28:52 +01:00
|
|
|
label_.set_markup(text);
|
2019-04-18 17:52:00 +02:00
|
|
|
|
2019-02-24 09:25:34 +01:00
|
|
|
if (tooltipEnabled()) {
|
|
|
|
if (config_["tooltip-format"].isString()) {
|
2022-08-06 12:55:20 +02:00
|
|
|
std::string calendar_lines{""};
|
|
|
|
std::string timezoned_time_lines{""};
|
2022-12-06 01:47:28 +01:00
|
|
|
if (is_calendar_in_tooltip_) {
|
|
|
|
calendar_lines = calendar_text(wtime);
|
|
|
|
}
|
|
|
|
if (is_timezoned_list_in_tooltip_) {
|
|
|
|
timezoned_time_lines = timezones_text(&now);
|
|
|
|
}
|
2021-10-03 18:48:21 +02:00
|
|
|
auto tooltip_format = config_["tooltip-format"].asString();
|
2022-04-06 08:37:19 +02:00
|
|
|
text =
|
|
|
|
fmt::format(tooltip_format, wtime, fmt::arg(kCalendarPlaceholder.c_str(), calendar_lines),
|
|
|
|
fmt::arg(KTimezonedTimeListPlaceholder.c_str(), timezoned_time_lines));
|
2022-11-24 12:28:52 +01:00
|
|
|
label_.set_tooltip_markup(text);
|
2019-02-24 09:25:34 +01:00
|
|
|
}
|
|
|
|
}
|
2021-10-03 18:48:21 +02:00
|
|
|
|
2020-04-12 18:30:21 +02:00
|
|
|
// Call parent update
|
2022-11-24 12:28:52 +01:00
|
|
|
ALabel::update();
|
2018-08-09 12:05:48 +02:00
|
|
|
}
|
2020-02-02 23:44:26 +01:00
|
|
|
|
2022-04-06 08:37:19 +02:00
|
|
|
bool waybar::modules::Clock::handleScroll(GdkEventScroll* e) {
|
2020-08-13 04:46:51 +02:00
|
|
|
// defer to user commands if set
|
|
|
|
if (config_["on-scroll-up"].isString() || config_["on-scroll-down"].isString()) {
|
|
|
|
return AModule::handleScroll(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto dir = AModule::getScrollDir(e);
|
2021-10-03 18:48:21 +02:00
|
|
|
|
2022-08-06 12:55:20 +02:00
|
|
|
// Shift calendar date
|
2022-11-28 04:24:56 +01:00
|
|
|
if (calendar_shift_init_.count() != 0) {
|
2022-08-06 12:55:20 +02:00
|
|
|
if (dir == SCROLL_DIR::UP)
|
|
|
|
calendar_shift_ += calendar_shift_init_;
|
|
|
|
else
|
|
|
|
calendar_shift_ -= calendar_shift_init_;
|
2020-08-13 04:46:51 +02:00
|
|
|
} else {
|
2022-08-06 12:55:20 +02:00
|
|
|
// Change time zone
|
|
|
|
if (dir != SCROLL_DIR::UP && dir != SCROLL_DIR::DOWN) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (time_zones_.size() == 1) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto nr_zones = time_zones_.size();
|
|
|
|
if (dir == SCROLL_DIR::UP) {
|
|
|
|
size_t new_idx = current_time_zone_idx_ + 1;
|
|
|
|
current_time_zone_idx_ = new_idx == nr_zones ? 0 : new_idx;
|
|
|
|
} else {
|
|
|
|
current_time_zone_idx_ =
|
|
|
|
current_time_zone_idx_ == 0 ? nr_zones - 1 : current_time_zone_idx_ - 1;
|
|
|
|
}
|
2020-08-13 04:46:51 +02:00
|
|
|
}
|
2021-10-03 18:48:21 +02:00
|
|
|
|
2020-08-13 04:46:51 +02:00
|
|
|
update();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-02-05 20:02:42 +01:00
|
|
|
auto waybar::modules::Clock::calendar_text(const waybar_time& wtime) -> std::string {
|
2020-01-31 17:54:41 +01:00
|
|
|
const auto daypoint = date::floor<date::days>(wtime.ztime.get_local_time());
|
2022-08-06 12:55:20 +02:00
|
|
|
const auto ymd{date::year_month_day{daypoint}};
|
|
|
|
|
2022-12-06 01:47:28 +01:00
|
|
|
if (calendar_cached_ymd_ == ymd) {
|
|
|
|
return calendar_cached_text_;
|
|
|
|
}
|
2020-02-02 23:44:26 +01:00
|
|
|
|
2022-11-28 04:24:56 +01:00
|
|
|
const auto curr_day{(calendar_shift_init_.count() != 0 && calendar_shift_.count() != 0)
|
2022-08-06 12:55:20 +02:00
|
|
|
? date::day{0}
|
|
|
|
: ymd.day()};
|
|
|
|
const date::year_month ym{ymd.year(), ymd.month()};
|
2022-12-10 12:02:15 +01:00
|
|
|
const auto first_dow = first_day_of_week();
|
2020-01-31 17:54:41 +01:00
|
|
|
|
|
|
|
std::stringstream os;
|
2022-04-10 11:54:50 +02:00
|
|
|
|
2022-12-10 12:02:15 +01:00
|
|
|
enum class WeeksSide {
|
2022-12-06 01:47:28 +01:00
|
|
|
LEFT,
|
|
|
|
RIGHT,
|
|
|
|
HIDDEN,
|
|
|
|
};
|
2022-12-10 12:02:15 +01:00
|
|
|
WeeksSide weeks_pos = WeeksSide::HIDDEN;
|
2022-06-24 15:44:06 +02:00
|
|
|
|
2022-03-26 21:33:15 +01:00
|
|
|
if (config_["calendar-weeks-pos"].isString()) {
|
|
|
|
if (config_["calendar-weeks-pos"].asString() == "left") {
|
2022-12-10 12:02:15 +01:00
|
|
|
weeks_pos = WeeksSide::LEFT;
|
2022-03-24 13:41:50 +01:00
|
|
|
// Add paddings before the header
|
2022-12-10 12:02:15 +01:00
|
|
|
os << std::string(3 + weeks_format_left_gaps, ' ');
|
2022-03-26 21:33:15 +01:00
|
|
|
} else if (config_["calendar-weeks-pos"].asString() == "right") {
|
2022-12-10 12:02:15 +01:00
|
|
|
weeks_pos = WeeksSide::RIGHT;
|
2022-03-24 13:41:50 +01:00
|
|
|
}
|
|
|
|
}
|
2022-04-15 14:39:13 +02:00
|
|
|
|
2022-12-10 12:02:15 +01:00
|
|
|
weekdays_header(first_dow, os);
|
2022-06-24 15:44:06 +02:00
|
|
|
|
2022-12-10 12:02:15 +01:00
|
|
|
// First week day prefixed with spaces if needed.
|
2022-12-10 14:48:22 +01:00
|
|
|
date::sys_days print_wd{ym / 1};
|
2022-12-10 12:02:15 +01:00
|
|
|
auto wd{date::weekday{print_wd}};
|
|
|
|
auto empty_days = (wd - first_dow).count();
|
2022-06-24 15:44:06 +02:00
|
|
|
|
2022-03-25 14:12:11 +01:00
|
|
|
/* Print weeknumber on the left for the first row*/
|
2022-12-10 12:02:15 +01:00
|
|
|
if (weeks_pos == WeeksSide::LEFT) {
|
|
|
|
os << fmt::format(weeks_format_, print_wd) << ' ';
|
2022-03-25 14:12:11 +01:00
|
|
|
}
|
2022-06-24 15:44:06 +02:00
|
|
|
|
2020-01-31 17:54:41 +01:00
|
|
|
if (empty_days > 0) {
|
|
|
|
os << std::string(empty_days * 3 - 1, ' ');
|
|
|
|
}
|
2022-12-10 12:02:15 +01:00
|
|
|
|
2022-12-06 01:47:28 +01:00
|
|
|
const auto last_day = (ym / date::literals::last).day();
|
2022-12-10 12:02:15 +01:00
|
|
|
|
|
|
|
for (auto d{date::day{1}}; d <= last_day; ++d, ++wd) {
|
|
|
|
if (wd != first_dow) {
|
2020-02-02 23:55:37 +01:00
|
|
|
os << ' ';
|
2022-12-10 14:48:22 +01:00
|
|
|
} else if (unsigned(d) != 1) {
|
2022-12-10 12:02:15 +01:00
|
|
|
if (weeks_pos == WeeksSide::RIGHT) {
|
|
|
|
os << ' ' << fmt::format(weeks_format_, print_wd);
|
2022-03-24 13:41:50 +01:00
|
|
|
}
|
|
|
|
|
2022-12-10 12:02:15 +01:00
|
|
|
os << '\n';
|
|
|
|
|
2022-12-10 15:55:21 +01:00
|
|
|
print_wd = (ym / d);
|
2022-03-24 13:41:50 +01:00
|
|
|
|
2022-12-10 12:02:15 +01:00
|
|
|
if (weeks_pos == WeeksSide::LEFT) {
|
|
|
|
os << fmt::format(weeks_format_, print_wd) << ' ';
|
2022-03-24 13:41:50 +01:00
|
|
|
}
|
2020-02-02 23:55:37 +01:00
|
|
|
}
|
2022-12-10 12:02:15 +01:00
|
|
|
|
2020-01-31 17:54:41 +01:00
|
|
|
if (d == curr_day) {
|
2020-08-06 02:31:36 +02:00
|
|
|
if (config_["today-format"].isString()) {
|
|
|
|
auto today_format = config_["today-format"].asString();
|
|
|
|
os << fmt::format(today_format, date::format("%e", d));
|
|
|
|
} else {
|
|
|
|
os << "<b><u>" << date::format("%e", d) << "</u></b>";
|
|
|
|
}
|
2022-03-26 21:33:15 +01:00
|
|
|
} else if (config_["format-calendar"].isString()) {
|
|
|
|
os << fmt::format(config_["format-calendar"].asString(), date::format("%e", d));
|
2022-12-06 01:47:28 +01:00
|
|
|
} else {
|
2022-04-15 14:39:13 +02:00
|
|
|
os << date::format("%e", d);
|
2022-12-06 01:47:28 +01:00
|
|
|
}
|
2022-03-24 13:41:50 +01:00
|
|
|
/*Print weeks on the right when the endings with spaces*/
|
2022-12-10 12:02:15 +01:00
|
|
|
if (weeks_pos == WeeksSide::RIGHT && d == last_day) {
|
|
|
|
empty_days = 6 - (wd.c_encoding() - first_dow.c_encoding());
|
|
|
|
if (empty_days > 0) {
|
|
|
|
os << std::string(empty_days * 3, ' ');
|
|
|
|
}
|
|
|
|
|
|
|
|
os << ' ' << fmt::format(weeks_format_, print_wd);
|
2022-03-24 13:41:50 +01:00
|
|
|
}
|
2020-01-31 17:54:41 +01:00
|
|
|
}
|
|
|
|
|
2020-02-02 23:44:26 +01:00
|
|
|
auto result = os.str();
|
2022-08-06 12:55:20 +02:00
|
|
|
calendar_cached_ymd_ = ymd;
|
|
|
|
calendar_cached_text_ = result;
|
2020-02-02 23:44:26 +01:00
|
|
|
return result;
|
2020-01-31 17:54:41 +01:00
|
|
|
}
|
|
|
|
|
2022-12-06 01:47:28 +01:00
|
|
|
auto waybar::modules::Clock::weekdays_header(const date::weekday& first_week_day, std::ostream& os)
|
2020-05-22 18:52:26 +02:00
|
|
|
-> void {
|
2022-03-26 21:33:15 +01:00
|
|
|
std::stringstream res;
|
2022-12-06 01:47:28 +01:00
|
|
|
auto wd = first_week_day;
|
2020-02-05 20:02:42 +01:00
|
|
|
do {
|
2022-12-06 01:47:28 +01:00
|
|
|
if (wd != first_week_day) {
|
|
|
|
res << ' ';
|
|
|
|
}
|
2020-02-05 20:02:42 +01:00
|
|
|
Glib::ustring wd_ustring(date::format(locale_, "%a", wd));
|
2021-01-31 20:53:53 +01:00
|
|
|
auto clen = ustring_clen(wd_ustring);
|
|
|
|
auto wd_len = wd_ustring.length();
|
|
|
|
while (clen > 2) {
|
2022-04-06 08:37:19 +02:00
|
|
|
wd_ustring = wd_ustring.substr(0, wd_len - 1);
|
2021-01-31 20:53:53 +01:00
|
|
|
wd_len--;
|
|
|
|
clen = ustring_clen(wd_ustring);
|
2020-02-05 20:02:42 +01:00
|
|
|
}
|
2021-01-31 20:53:53 +01:00
|
|
|
const std::string pad(2 - clen, ' ');
|
2022-03-26 21:33:15 +01:00
|
|
|
res << pad << wd_ustring;
|
2022-12-06 01:47:28 +01:00
|
|
|
} while (++wd != first_week_day);
|
2022-12-10 12:02:15 +01:00
|
|
|
res << '\n';
|
2022-03-26 21:33:15 +01:00
|
|
|
|
|
|
|
if (config_["format-calendar-weekdays"].isString()) {
|
|
|
|
os << fmt::format(config_["format-calendar-weekdays"].asString(), res.str());
|
2022-04-15 14:39:13 +02:00
|
|
|
} else
|
|
|
|
os << res.str();
|
2020-01-31 17:54:41 +01:00
|
|
|
}
|
2018-08-08 23:54:58 +02:00
|
|
|
|
2022-04-06 08:37:19 +02:00
|
|
|
auto waybar::modules::Clock::timezones_text(std::chrono::system_clock::time_point* now)
|
|
|
|
-> std::string {
|
2021-10-05 11:46:52 +02:00
|
|
|
if (time_zones_.size() == 1) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
std::stringstream os;
|
|
|
|
waybar_time wtime;
|
|
|
|
for (size_t time_zone_idx = 0; time_zone_idx < time_zones_.size(); ++time_zone_idx) {
|
|
|
|
if (static_cast<int>(time_zone_idx) == current_time_zone_idx_) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const date::time_zone* timezone = time_zones_[time_zone_idx];
|
|
|
|
if (!timezone) {
|
|
|
|
timezone = date::current_zone();
|
|
|
|
}
|
|
|
|
wtime = {locale_, date::make_zoned(timezone, date::floor<std::chrono::seconds>(*now))};
|
2022-12-10 12:02:15 +01:00
|
|
|
os << fmt::format(format_, wtime) << '\n';
|
2021-10-05 11:46:52 +02:00
|
|
|
}
|
|
|
|
return os.str();
|
|
|
|
}
|
|
|
|
|
2020-02-05 20:02:42 +01:00
|
|
|
#ifdef HAVE_LANGINFO_1STDAY
|
|
|
|
template <auto fn>
|
|
|
|
using deleter_from_fn = std::integral_constant<decltype(fn), fn>;
|
2020-01-21 23:48:16 +01:00
|
|
|
|
2020-02-05 20:02:42 +01:00
|
|
|
template <typename T, auto fn>
|
|
|
|
using deleting_unique_ptr = std::unique_ptr<T, deleter_from_fn<fn>>;
|
|
|
|
#endif
|
2019-04-18 17:52:00 +02:00
|
|
|
|
2020-02-05 20:02:42 +01:00
|
|
|
// Computations done similarly to Linux cal utility.
|
|
|
|
auto waybar::modules::Clock::first_day_of_week() -> date::weekday {
|
|
|
|
#ifdef HAVE_LANGINFO_1STDAY
|
2020-05-22 18:52:26 +02:00
|
|
|
deleting_unique_ptr<std::remove_pointer<locale_t>::type, freelocale> posix_locale{
|
|
|
|
newlocale(LC_ALL, locale_.name().c_str(), nullptr)};
|
2020-02-05 20:02:42 +01:00
|
|
|
if (posix_locale) {
|
2020-05-22 18:52:26 +02:00
|
|
|
const int i = (std::intptr_t)nl_langinfo_l(_NL_TIME_WEEK_1STDAY, posix_locale.get());
|
2022-04-06 08:37:19 +02:00
|
|
|
auto ymd = date::year(i / 10000) / (i / 100 % 100) / (i % 100);
|
|
|
|
auto wd = date::weekday(ymd);
|
|
|
|
uint8_t j = *nl_langinfo_l(_NL_TIME_FIRST_WEEKDAY, posix_locale.get());
|
2020-02-05 20:02:42 +01:00
|
|
|
return wd + date::days(j - 1);
|
2019-02-24 09:25:34 +01:00
|
|
|
}
|
2020-02-05 20:02:42 +01:00
|
|
|
#endif
|
|
|
|
return date::Sunday;
|
2018-08-09 12:05:48 +02:00
|
|
|
}
|