diff options
Diffstat (limited to 'libs/ImGuiDatePicker/ImGuiDatePicker.cpp')
| -rw-r--r-- | libs/ImGuiDatePicker/ImGuiDatePicker.cpp | 393 |
1 files changed, 393 insertions, 0 deletions
diff --git a/libs/ImGuiDatePicker/ImGuiDatePicker.cpp b/libs/ImGuiDatePicker/ImGuiDatePicker.cpp new file mode 100644 index 0000000..8c81b2b --- /dev/null +++ b/libs/ImGuiDatePicker/ImGuiDatePicker.cpp @@ -0,0 +1,393 @@ +#include "ImGuiDatePicker.hpp" +#include <imgui_internal.h> +#include <cstdint> +#include <chrono> +#include <vector> +#include <unordered_map> + + +#define GET_DAY(timePoint) int(timePoint.tm_mday) +#define GET_MONTH_UNSCALED(timePoint) timePoint.tm_mon +#define GET_MONTH(timePoint) int(GET_MONTH_UNSCALED(timePoint) + 1) +#define GET_YEAR(timePoint) int(timePoint.tm_year + 1900) + +#define SET_DAY(timePoint, day) timePoint.tm_mday = day +#define SET_MONTH(timePoint, month) timePoint.tm_mon = month - 1 +#define SET_YEAR(timePoint, year) timePoint.tm_year = year - 1900 + +namespace ImGui +{ + static const std::vector<std::string> MONTHS = + { + "January", + "February", + "March", + "April", + "May", + "June", + "July", + "August", + "September", + "October", + "November", + "December" + }; + + static const std::vector<std::string> DAYS = + { + "Mo", + "Tu", + "We", + "Th", + "Fr", + "Sa", + "Su" + }; + + // Implements Zeller's Congruence to determine the day of week [1, 7](Mon-Sun) from the given parameters + inline static int DayOfWeek(int dayOfMonth, int month, int year) noexcept + { + if ((month == 1) || (month == 2)) + { + month += 12; + year -= 1; + } + + int h = (dayOfMonth + + static_cast<int>(std::floor((13 * (month + 1)) / 5.0)) + + year + + static_cast<int>(std::floor(year / 4.0)) + - static_cast<int>(std::floor(year / 100.0)) + + static_cast<int>(std::floor(year / 400.0))) % 7; + + return static_cast<int>(std::floor(((h + 5) % 7) + 1)); + } + + constexpr static bool IsLeapYear(int year) noexcept + { + if ((year % 400) == 0) + return true; + + if ((year % 4 == 0) && ((year % 100) != 0)) + return true; + + return false; + } + + inline static int NumDaysInMonth(int month, int year) + { + if (month == 2) + return IsLeapYear(year) ? 29 : 28; + + // Month index paired to the number of days in that month excluding February + static const std::unordered_map<int, int> monthDayMap = + { + { 1, 31 }, + { 3, 31 }, + { 4, 30 }, + { 5, 31 }, + { 6, 30 }, + { 7, 31 }, + { 8, 31 }, + { 9, 30 }, + { 10, 31 }, + { 11, 30 }, + { 12, 31 } + }; + + return monthDayMap.at(month); + } + + // Returns the number of calendar weeks spanned by month in the specified year + inline static int NumWeeksInMonth(int month, int year) + { + int days = NumDaysInMonth(month, year); + int firstDay = DayOfWeek(1, month, year); + + return static_cast<int>(std::ceil((days + firstDay - 1) / 7.0)); + } + + // Returns a vector containing dates as they would appear on the calendar for a given week. Populates 0 if there is no day. + inline static std::vector<int> CalendarWeek(int week, int startDay, int daysInMonth) + { + std::vector<int> res(7, 0); + int startOfWeek = 7 * (week - 1) + 2 - startDay; + + if (startOfWeek >= 1) + res[0] = startOfWeek; + + for (int i = 1; i < 7; ++i) + { + int day = startOfWeek + i; + if ((day >= 1) && (day <= daysInMonth)) + res[i] = day; + } + + return res; + } + + constexpr static tm EncodeTimePoint(int dayOfMonth, int month, int year) noexcept + { + tm res{ }; + res.tm_isdst = -1; + SET_DAY(res, dayOfMonth); + SET_MONTH(res, month); + SET_YEAR(res, year); + + return res; + } + + inline static std::string TimePointToLongString(const tm& timePoint) noexcept + { + std::string day = std::to_string(GET_DAY(timePoint)); + std::string month = MONTHS[GET_MONTH_UNSCALED(timePoint)]; + std::string year = std::to_string(GET_YEAR(timePoint)); + + return std::string(day + " " + month + " " + year); + } + + inline static tm Today() noexcept + { + std::chrono::system_clock::time_point now = std::chrono::system_clock::now(); + std::time_t currentTime = std::chrono::system_clock::to_time_t(now); + + tm res; + gmtime_s(&res, ¤tTime); + + return res; + } + + inline static tm PreviousMonth(const tm& timePoint) noexcept + { + int month = GET_MONTH(timePoint); + int year = GET_YEAR(timePoint); + + if (month == 1) + { + int newDay = std::min(GET_DAY(timePoint), NumDaysInMonth(12, --year)); + return EncodeTimePoint(newDay, 12, year); + } + + int newDay = std::min(GET_DAY(timePoint), NumDaysInMonth(--month, year)); + return EncodeTimePoint(newDay, month, year); + } + + inline static tm NextMonth(const tm& timePoint) noexcept + { + int month = GET_MONTH(timePoint); + int year = GET_YEAR(timePoint); + + if (month == 12) + { + int newDay = std::min(GET_DAY(timePoint), NumDaysInMonth(1, ++year)); + return EncodeTimePoint(newDay, 1, year); + } + + int newDay = std::min(GET_DAY(timePoint), NumDaysInMonth(++month, year)); + return EncodeTimePoint(newDay, month, year); + } + + constexpr static bool IsMinDate(const tm& timePoint) noexcept + { + return (GET_MONTH(timePoint) == 1) && (GET_YEAR(timePoint) == IMGUI_DATEPICKER_YEAR_MIN); + } + + constexpr static bool IsMaxDate(const tm& timePoint) noexcept + { + return (GET_MONTH(timePoint) == 12) && (GET_YEAR(timePoint) == IMGUI_DATEPICKER_YEAR_MAX); + } + + static bool ComboBox(const std::string& label, const std::vector<std::string>& items, int& v, ImFont* altFont) + { + bool res = false; + + ImGui::PushFont(altFont); + if (ImGui::BeginCombo(label.c_str(), items[v].c_str())) + { + for (int i = 0; i < items.size(); ++i) + { + bool selected = (items[v] == items[i]); + if (ImGui::Selectable(items[i].c_str(), &selected)) + { + v = i; + res = true; + } + + if (selected) + ImGui::SetItemDefaultFocus(); + } + + ImGui::EndCombo(); + } + + ImGui::PopFont(); + return res; + } + + bool DatePickerEx(const std::string& label, tm& v, ImFont* altFont, bool clampToBorder, float itemSpacing) + { + bool res = false; + + ImGuiWindow* window = GetCurrentWindow(); + if (window->SkipItems) + return false; + + bool hiddenLabel = label.substr(0, 2) == "##"; + std::string myLabel = (hiddenLabel) ? label.substr(2) : label; + + if (!hiddenLabel) + { + Text("%s", label.c_str()); + SameLine((itemSpacing == 0.0f) ? 0.0f : GetCursorPos().x + itemSpacing); + } + + if (clampToBorder) + SetNextItemWidth(GetContentRegionAvail().x); + + const ImVec2 windowSize = ImVec2(274.5f, 301.5f); + SetNextWindowSize(windowSize); + + if (BeginCombo(std::string("##" + myLabel).c_str(), TimePointToLongString(v).c_str())) + { + int monthIdx = GET_MONTH_UNSCALED(v); + int year = GET_YEAR(v); + + PushItemWidth((GetContentRegionAvail().x * 0.5f)); + + if (ComboBox("##CmbMonth_" + myLabel, MONTHS, monthIdx, altFont)) + { + SET_MONTH(v, monthIdx + 1); + res = true; + } + + PopItemWidth(); + SameLine(); + PushItemWidth(GetContentRegionAvail().x); + + if (InputInt(std::string("##IntYear_" + myLabel).c_str(), &year)) + { + SET_YEAR(v, std::min(std::max(IMGUI_DATEPICKER_YEAR_MIN, year), IMGUI_DATEPICKER_YEAR_MAX)); + res = true; + } + + PopItemWidth(); + + const float contentWidth = GetContentRegionAvail().x; + const float arrowSize = GetFrameHeight(); + const float arrowButtonWidth = arrowSize * 2.0f + GetStyle().ItemSpacing.x; + const float bulletSize = arrowSize - 5.0f; + const float bulletButtonWidth = bulletSize + GetStyle().ItemSpacing.x; + const float combinedWidth = arrowButtonWidth + bulletButtonWidth; + const float offset = (contentWidth - combinedWidth) * 0.5f; + + SetCursorPosX(GetCursorPosX() + offset); + PushStyleVar(ImGuiStyleVar_FrameRounding, 20.0f); + PushStyleColor(ImGuiCol_Button, ImVec4(0.0f, 0.0f, 0.0f, 0.0f)); + PushStyleColor(ImGuiCol_Border, ImVec4(0.0f, 0.0f, 0.0f, 0.0f)); + BeginDisabled(IsMinDate(v)); + + if (ArrowButtonEx(std::string("##ArrowLeft_" + myLabel).c_str(), ImGuiDir_Left, ImVec2(arrowSize, arrowSize))) + { + v = PreviousMonth(v); + res = true; + } + + EndDisabled(); + PopStyleColor(2); + SameLine(); + PushStyleColor(ImGuiCol_Button, GetStyleColorVec4(ImGuiCol_Text)); + SetCursorPosY(GetCursorPosY() + 2.0f); + + if (ButtonEx(std::string("##ArrowMid_" + myLabel).c_str(), ImVec2(bulletSize, bulletSize))) + { + v = Today(); + res = true; + CloseCurrentPopup(); + } + + PopStyleColor(); + SameLine(); + PushStyleColor(ImGuiCol_Button, ImVec4(0.0f, 0.0f, 0.0f, 0.0f)); + PushStyleColor(ImGuiCol_Border, ImVec4(0.0f, 0.0f, 0.0f, 0.0f)); + BeginDisabled(IsMaxDate(v)); + + if (ArrowButtonEx(std::string("##ArrowRight_" + myLabel).c_str(), ImGuiDir_Right, ImVec2(arrowSize, arrowSize))) + { + v = NextMonth(v); + res = true; + } + + EndDisabled(); + PopStyleColor(2); + PopStyleVar(); + + constexpr ImGuiTableFlags TABLE_FLAGS = ImGuiTableFlags_BordersOuter | ImGuiTableFlags_SizingFixedFit | + ImGuiTableFlags_NoHostExtendX | ImGuiTableFlags_NoHostExtendY; + + if (BeginTable(std::string("##Table_" + myLabel).c_str(), 7, TABLE_FLAGS, GetContentRegionAvail())) + { + for (const auto& day : DAYS) + TableSetupColumn(day.c_str(), ImGuiTableColumnFlags_WidthFixed | ImGuiTableColumnFlags_NoHeaderWidth, 30.0f); + + PushStyleColor(ImGuiCol_HeaderHovered, GetStyleColorVec4(ImGuiCol_TableHeaderBg)); + PushStyleColor(ImGuiCol_HeaderActive, GetStyleColorVec4(ImGuiCol_TableHeaderBg)); + PushFont(altFont); + TableHeadersRow(); + PopStyleColor(2); + PopFont(); + + TableNextRow(); + TableSetColumnIndex(0); + + int month = monthIdx + 1; + int firstDayOfMonth = DayOfWeek(1, month, year); + int numDaysInMonth = NumDaysInMonth(month, year); + int numWeeksInMonth = NumWeeksInMonth(month, year); + + for (int i = 1; i <= numWeeksInMonth; ++i) + { + for (const auto& day : CalendarWeek(i, firstDayOfMonth, numDaysInMonth)) + { + if (day != 0) + { + PushStyleVar(ImGuiStyleVar_FrameRounding, 20.0f); + + const bool selected = day == GET_DAY(v); + if (!selected) + { + PushStyleColor(ImGuiCol_Button, ImVec4(0.0f, 0.0f, 0.0f, 0.0f)); + PushStyleColor(ImGuiCol_Border, ImVec4(0.0f, 0.0f, 0.0f, 0.0f)); + } + + if (Button(std::to_string(day).c_str(), ImVec2(GetContentRegionAvail().x, GetTextLineHeightWithSpacing() + 5.0f))) + { + v = EncodeTimePoint(day, month, year); + res = true; + CloseCurrentPopup(); + } + + if (!selected) + PopStyleColor(2); + + PopStyleVar(); + } + + if (day != numDaysInMonth) + TableNextColumn(); + } + } + + EndTable(); + } + + EndCombo(); + } + + return res; + } + + bool DatePicker(const std::string& label, tm& v, bool clampToBorder, float itemSpacing) + { + return DatePickerEx(label, v, nullptr, clampToBorder, itemSpacing); + } +} |
