forked from github/plane
chore: date and time standardization all across the platform. (#3283)
* chore: date and time standardization all across the platform. * chore: update `renderFormattedTime` function. * remove unwanted code. * fix: build errors * chore: update `renderFormattedTime` function params.
This commit is contained in:
parent
d9ee692ce9
commit
1539340113
@ -1,11 +1,10 @@
|
|||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
|
|
||||||
// mobx store
|
// mobx store
|
||||||
import { useMobxStore } from "lib/mobx/store-provider";
|
import { useMobxStore } from "lib/mobx/store-provider";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderEmoji } from "helpers/emoji.helper";
|
import { renderEmoji } from "helpers/emoji.helper";
|
||||||
import { renderShortDate } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// constants
|
// constants
|
||||||
import { NETWORK_CHOICES } from "constants/project";
|
import { NETWORK_CHOICES } from "constants/project";
|
||||||
|
|
||||||
@ -37,7 +36,7 @@ export const CustomAnalyticsSidebarHeader = observer(() => {
|
|||||||
<h6 className="text-custom-text-200">Start Date</h6>
|
<h6 className="text-custom-text-200">Start Date</h6>
|
||||||
<span>
|
<span>
|
||||||
{cycleDetails.start_date && cycleDetails.start_date !== ""
|
{cycleDetails.start_date && cycleDetails.start_date !== ""
|
||||||
? renderShortDate(cycleDetails.start_date)
|
? renderFormattedDate(cycleDetails.start_date)
|
||||||
: "No start date"}
|
: "No start date"}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
@ -45,7 +44,7 @@ export const CustomAnalyticsSidebarHeader = observer(() => {
|
|||||||
<h6 className="text-custom-text-200">Target Date</h6>
|
<h6 className="text-custom-text-200">Target Date</h6>
|
||||||
<span>
|
<span>
|
||||||
{cycleDetails.end_date && cycleDetails.end_date !== ""
|
{cycleDetails.end_date && cycleDetails.end_date !== ""
|
||||||
? renderShortDate(cycleDetails.end_date)
|
? renderFormattedDate(cycleDetails.end_date)
|
||||||
: "No end date"}
|
: "No end date"}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
@ -63,7 +62,7 @@ export const CustomAnalyticsSidebarHeader = observer(() => {
|
|||||||
<h6 className="text-custom-text-200">Start Date</h6>
|
<h6 className="text-custom-text-200">Start Date</h6>
|
||||||
<span>
|
<span>
|
||||||
{moduleDetails.start_date && moduleDetails.start_date !== ""
|
{moduleDetails.start_date && moduleDetails.start_date !== ""
|
||||||
? renderShortDate(moduleDetails.start_date)
|
? renderFormattedDate(moduleDetails.start_date)
|
||||||
: "No start date"}
|
: "No start date"}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
@ -71,7 +70,7 @@ export const CustomAnalyticsSidebarHeader = observer(() => {
|
|||||||
<h6 className="text-custom-text-200">Target Date</h6>
|
<h6 className="text-custom-text-200">Target Date</h6>
|
||||||
<span>
|
<span>
|
||||||
{moduleDetails.target_date && moduleDetails.target_date !== ""
|
{moduleDetails.target_date && moduleDetails.target_date !== ""
|
||||||
? renderShortDate(moduleDetails.target_date)
|
? renderFormattedDate(moduleDetails.target_date)
|
||||||
: "No end date"}
|
: "No end date"}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
@ -14,7 +14,7 @@ import { Button, LayersIcon } from "@plane/ui";
|
|||||||
// icons
|
// icons
|
||||||
import { CalendarDays, Download, RefreshCw } from "lucide-react";
|
import { CalendarDays, Download, RefreshCw } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDate } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IAnalyticsParams, IAnalyticsResponse, IExportAnalyticsFormData, IWorkspace } from "types";
|
import { IAnalyticsParams, IAnalyticsResponse, IExportAnalyticsFormData, IWorkspace } from "types";
|
||||||
// fetch-keys
|
// fetch-keys
|
||||||
@ -156,7 +156,7 @@ export const CustomAnalyticsSidebar: React.FC<Props> = observer(
|
|||||||
{isProjectLevel && (
|
{isProjectLevel && (
|
||||||
<div className="flex items-center gap-1 rounded-md bg-custom-background-80 px-3 py-1 text-xs text-custom-text-200">
|
<div className="flex items-center gap-1 rounded-md bg-custom-background-80 px-3 py-1 text-xs text-custom-text-200">
|
||||||
<CalendarDays className="h-3.5 w-3.5" />
|
<CalendarDays className="h-3.5 w-3.5" />
|
||||||
{renderShortDate(
|
{renderFormattedDate(
|
||||||
(cycleId
|
(cycleId
|
||||||
? cycleDetails?.created_at
|
? cycleDetails?.created_at
|
||||||
: moduleId
|
: moduleId
|
||||||
|
@ -48,7 +48,7 @@ export const CreateApiTokenModal: React.FC<Props> = (props) => {
|
|||||||
const csvData = {
|
const csvData = {
|
||||||
Title: data.label,
|
Title: data.label,
|
||||||
Description: data.description,
|
Description: data.description,
|
||||||
Expiry: data.expired_at ? renderFormattedDate(data.expired_at) : "Never expires",
|
Expiry: data.expired_at ? renderFormattedDate(data.expired_at)?.replace(",", " ") ?? "" : "Never expires",
|
||||||
"Secret key": data.token ?? "",
|
"Secret key": data.token ?? "",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -5,7 +5,7 @@ import { DeleteApiTokenModal } from "components/api-token";
|
|||||||
// ui
|
// ui
|
||||||
import { Tooltip } from "@plane/ui";
|
import { Tooltip } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderFormattedDate, timeAgo } from "helpers/date-time.helper";
|
import { renderFormattedDate, calculateTimeAgo } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IApiToken } from "types/api_token";
|
import { IApiToken } from "types/api_token";
|
||||||
|
|
||||||
@ -49,7 +49,7 @@ export const ApiTokenListItem: React.FC<Props> = (props) => {
|
|||||||
? token.expired_at
|
? token.expired_at
|
||||||
? `Expires ${renderFormattedDate(token.expired_at!)}`
|
? `Expires ${renderFormattedDate(token.expired_at!)}`
|
||||||
: "Never expires"
|
: "Never expires"
|
||||||
: `Expired ${timeAgo(token.expired_at)}`}
|
: `Expired ${calculateTimeAgo(token.expired_at)}`}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
|
|
||||||
import useSWR from "swr";
|
import useSWR from "swr";
|
||||||
|
|
||||||
// headless ui
|
// headless ui
|
||||||
import { Dialog, Transition } from "@headlessui/react";
|
import { Dialog, Transition } from "@headlessui/react";
|
||||||
// services
|
// services
|
||||||
@ -12,7 +10,7 @@ import { Loader } from "@plane/ui";
|
|||||||
// icons
|
// icons
|
||||||
import { X } from "lucide-react";
|
import { X } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderLongDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
@ -69,7 +67,7 @@ export const ProductUpdatesModal: React.FC<Props> = ({ isOpen, setIsOpen }) => {
|
|||||||
<span className="flex items-center rounded-full border border-custom-border-200 bg-custom-background-90 px-3 py-1.5 text-xs">
|
<span className="flex items-center rounded-full border border-custom-border-200 bg-custom-background-90 px-3 py-1.5 text-xs">
|
||||||
{item.tag_name}
|
{item.tag_name}
|
||||||
</span>
|
</span>
|
||||||
<span>{renderLongDateFormat(item.published_at)}</span>
|
<span>{renderFormattedDate(item.published_at)}</span>
|
||||||
{index === 0 && (
|
{index === 0 && (
|
||||||
<span className="flex items-center rounded-full border border-custom-border-200 bg-custom-primary px-3 py-1.5 text-xs text-white">
|
<span className="flex items-center rounded-full border border-custom-border-200 bg-custom-primary px-3 py-1.5 text-xs text-white">
|
||||||
New
|
New
|
||||||
|
@ -22,7 +22,7 @@ import {
|
|||||||
UsersIcon,
|
UsersIcon,
|
||||||
} from "lucide-react";
|
} from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
import { capitalizeFirstLetter } from "helpers/string.helper";
|
import { capitalizeFirstLetter } from "helpers/string.helper";
|
||||||
// types
|
// types
|
||||||
import { IIssueActivity } from "types";
|
import { IIssueActivity } from "types";
|
||||||
@ -597,7 +597,7 @@ const activityDetails: {
|
|||||||
<>
|
<>
|
||||||
set the start date to{" "}
|
set the start date to{" "}
|
||||||
<span className="font-medium text-custom-text-100">
|
<span className="font-medium text-custom-text-100">
|
||||||
{renderShortDateWithYearFormat(activity.new_value)}
|
{renderFormattedDate(activity.new_value)}
|
||||||
</span>
|
</span>
|
||||||
{showIssue && (
|
{showIssue && (
|
||||||
<>
|
<>
|
||||||
@ -646,7 +646,7 @@ const activityDetails: {
|
|||||||
<>
|
<>
|
||||||
set the due date to{" "}
|
set the due date to{" "}
|
||||||
<span className="font-medium text-custom-text-100">
|
<span className="font-medium text-custom-text-100">
|
||||||
{renderShortDateWithYearFormat(activity.new_value)}
|
{renderFormattedDate(activity.new_value)}
|
||||||
</span>
|
</span>
|
||||||
{showIssue && (
|
{showIssue && (
|
||||||
<>
|
<>
|
||||||
|
@ -2,7 +2,6 @@ import { Fragment } from "react";
|
|||||||
import { Controller, useForm } from "react-hook-form";
|
import { Controller, useForm } from "react-hook-form";
|
||||||
import DatePicker from "react-datepicker";
|
import DatePicker from "react-datepicker";
|
||||||
import { Dialog, Transition } from "@headlessui/react";
|
import { Dialog, Transition } from "@headlessui/react";
|
||||||
|
|
||||||
// components
|
// components
|
||||||
import { DateFilterSelect } from "./date-filter-select";
|
import { DateFilterSelect } from "./date-filter-select";
|
||||||
// ui
|
// ui
|
||||||
@ -10,7 +9,7 @@ import { Button } from "@plane/ui";
|
|||||||
// icons
|
// icons
|
||||||
import { X } from "lucide-react";
|
import { X } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderDateFormat, renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate, renderFormattedDate } from "helpers/date-time.helper";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
title: string;
|
title: string;
|
||||||
@ -39,8 +38,8 @@ export const DateFilterModal: React.FC<Props> = ({ title, handleClose, isOpen, o
|
|||||||
const handleFormSubmit = (formData: TFormValues) => {
|
const handleFormSubmit = (formData: TFormValues) => {
|
||||||
const { filterType, date1, date2 } = formData;
|
const { filterType, date1, date2 } = formData;
|
||||||
|
|
||||||
if (filterType === "range") onSelect([`${renderDateFormat(date1)};after`, `${renderDateFormat(date2)};before`]);
|
if (filterType === "range") onSelect([`${renderFormattedPayloadDate(date1)};after`, `${renderFormattedPayloadDate(date2)};before`]);
|
||||||
else onSelect([`${renderDateFormat(date1)};${filterType}`]);
|
else onSelect([`${renderFormattedPayloadDate(date1)};${filterType}`]);
|
||||||
|
|
||||||
handleClose();
|
handleClose();
|
||||||
};
|
};
|
||||||
@ -121,9 +120,9 @@ export const DateFilterModal: React.FC<Props> = ({ title, handleClose, isOpen, o
|
|||||||
{watch("filterType") === "range" && (
|
{watch("filterType") === "range" && (
|
||||||
<h6 className="flex items-center gap-1 text-xs">
|
<h6 className="flex items-center gap-1 text-xs">
|
||||||
<span className="text-custom-text-200">After:</span>
|
<span className="text-custom-text-200">After:</span>
|
||||||
<span>{renderShortDateWithYearFormat(watch("date1"))}</span>
|
<span>{renderFormattedDate(watch("date1"))}</span>
|
||||||
<span className="ml-1 text-custom-text-200">Before:</span>
|
<span className="ml-1 text-custom-text-200">Before:</span>
|
||||||
{!isInvalid && <span>{renderShortDateWithYearFormat(watch("date2"))}</span>}
|
{!isInvalid && <span>{renderFormattedDate(watch("date2"))}</span>}
|
||||||
</h6>
|
</h6>
|
||||||
)}
|
)}
|
||||||
<div className="flex justify-end gap-4">
|
<div className="flex justify-end gap-4">
|
||||||
|
@ -3,7 +3,7 @@ import { ExternalLinkIcon, Tooltip } from "@plane/ui";
|
|||||||
// icons
|
// icons
|
||||||
import { Pencil, Trash2, LinkIcon } from "lucide-react";
|
import { Pencil, Trash2, LinkIcon } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { timeAgo } from "helpers/date-time.helper";
|
import { calculateTimeAgo } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { ILinkDetails, UserAuth } from "types";
|
import { ILinkDetails, UserAuth } from "types";
|
||||||
// hooks
|
// hooks
|
||||||
@ -89,7 +89,7 @@ export const LinksList: React.FC<Props> = ({ links, handleDeleteLink, handleEdit
|
|||||||
</div>
|
</div>
|
||||||
<div className="px-5">
|
<div className="px-5">
|
||||||
<p className="mt-0.5 stroke-[1.5] text-xs text-custom-text-300">
|
<p className="mt-0.5 stroke-[1.5] text-xs text-custom-text-300">
|
||||||
Added {timeAgo(link.created_at)}
|
Added {calculateTimeAgo(link.created_at)}
|
||||||
<br />
|
<br />
|
||||||
by{" "}
|
by{" "}
|
||||||
{link.created_by_detail.is_bot
|
{link.created_by_detail.is_bot
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
|
import { eachDayOfInterval } from "date-fns";
|
||||||
// ui
|
// ui
|
||||||
import { LineGraph } from "components/ui";
|
import { LineGraph } from "components/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { getDatesInRange, renderShortNumericDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedDateWithoutYear } from "helpers/date-time.helper";
|
||||||
//types
|
//types
|
||||||
import { TCompletionChartDistribution } from "types";
|
import { TCompletionChartDistribution } from "types";
|
||||||
|
|
||||||
@ -42,25 +42,25 @@ const DashedLine = ({ series, lineGenerator, xScale, yScale }: any) =>
|
|||||||
|
|
||||||
const ProgressChart: React.FC<Props> = ({ distribution, startDate, endDate, totalIssues }) => {
|
const ProgressChart: React.FC<Props> = ({ distribution, startDate, endDate, totalIssues }) => {
|
||||||
const chartData = Object.keys(distribution).map((key) => ({
|
const chartData = Object.keys(distribution).map((key) => ({
|
||||||
currentDate: renderShortNumericDateFormat(key),
|
currentDate: renderFormattedDateWithoutYear(key),
|
||||||
pending: distribution[key],
|
pending: distribution[key],
|
||||||
}));
|
}));
|
||||||
|
|
||||||
const generateXAxisTickValues = () => {
|
const generateXAxisTickValues = () => {
|
||||||
const dates = getDatesInRange(startDate, endDate);
|
const dates = eachDayOfInterval({ start: new Date(startDate), end: new Date(endDate) });
|
||||||
|
|
||||||
const maxDates = 4;
|
const maxDates = 4;
|
||||||
const totalDates = dates.length;
|
const totalDates = dates.length;
|
||||||
|
|
||||||
if (totalDates <= maxDates) return dates.map((d) => renderShortNumericDateFormat(d));
|
if (totalDates <= maxDates) return dates.map((d) => renderFormattedDateWithoutYear(d));
|
||||||
else {
|
else {
|
||||||
const interval = Math.ceil(totalDates / maxDates);
|
const interval = Math.ceil(totalDates / maxDates);
|
||||||
const limitedDates = [];
|
const limitedDates = [];
|
||||||
|
|
||||||
for (let i = 0; i < totalDates; i += interval) limitedDates.push(renderShortNumericDateFormat(dates[i]));
|
for (let i = 0; i < totalDates; i += interval) limitedDates.push(renderFormattedDateWithoutYear(dates[i]));
|
||||||
|
|
||||||
if (!limitedDates.includes(renderShortNumericDateFormat(dates[totalDates - 1])))
|
if (!limitedDates.includes(renderFormattedDateWithoutYear(dates[totalDates - 1])))
|
||||||
limitedDates.push(renderShortNumericDateFormat(dates[totalDates - 1]));
|
limitedDates.push(renderFormattedDateWithoutYear(dates[totalDates - 1]));
|
||||||
|
|
||||||
return limitedDates;
|
return limitedDates;
|
||||||
}
|
}
|
||||||
|
@ -28,7 +28,7 @@ import { ViewIssueLabel } from "components/issues";
|
|||||||
// icons
|
// icons
|
||||||
import { AlarmClock, AlertTriangle, ArrowRight, CalendarDays, Star, Target } from "lucide-react";
|
import { AlarmClock, AlertTriangle, ArrowRight, CalendarDays, Star, Target } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDateWithYearFormat, findHowManyDaysLeft } from "helpers/date-time.helper";
|
import { renderFormattedDate, findHowManyDaysLeft } from "helpers/date-time.helper";
|
||||||
import { truncateText } from "helpers/string.helper";
|
import { truncateText } from "helpers/string.helper";
|
||||||
// types
|
// types
|
||||||
import { ICycle } from "types";
|
import { ICycle } from "types";
|
||||||
@ -267,12 +267,12 @@ export const ActiveCycleDetails: React.FC<IActiveCycleDetails> = observer((props
|
|||||||
<div className="flex items-center justify-start gap-5 text-custom-text-200">
|
<div className="flex items-center justify-start gap-5 text-custom-text-200">
|
||||||
<div className="flex items-start gap-1">
|
<div className="flex items-start gap-1">
|
||||||
<CalendarDays className="h-4 w-4" />
|
<CalendarDays className="h-4 w-4" />
|
||||||
<span>{renderShortDateWithYearFormat(startDate)}</span>
|
<span>{renderFormattedDate(startDate)}</span>
|
||||||
</div>
|
</div>
|
||||||
<ArrowRight className="h-4 w-4 text-custom-text-200" />
|
<ArrowRight className="h-4 w-4 text-custom-text-200" />
|
||||||
<div className="flex items-start gap-1">
|
<div className="flex items-start gap-1">
|
||||||
<Target className="h-4 w-4" />
|
<Target className="h-4 w-4" />
|
||||||
<span>{renderShortDateWithYearFormat(endDate)}</span>
|
<span>{renderFormattedDate(endDate)}</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -10,7 +10,7 @@ import { Avatar, AvatarGroup, CustomMenu, Tooltip, LayersIcon, CycleGroupIcon }
|
|||||||
// icons
|
// icons
|
||||||
import { Info, LinkIcon, Pencil, Star, Trash2 } from "lucide-react";
|
import { Info, LinkIcon, Pencil, Star, Trash2 } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { findHowManyDaysLeft, renderShortDate, renderShortMonthDate } from "helpers/date-time.helper";
|
import { findHowManyDaysLeft, renderFormattedDate } from "helpers/date-time.helper";
|
||||||
import { copyTextToClipboard } from "helpers/string.helper";
|
import { copyTextToClipboard } from "helpers/string.helper";
|
||||||
// types
|
// types
|
||||||
import { ICycle, TCycleGroups } from "types";
|
import { ICycle, TCycleGroups } from "types";
|
||||||
@ -53,8 +53,6 @@ export const CyclesBoardCard: FC<ICyclesBoardCard> = (props) => {
|
|||||||
|
|
||||||
const currentCycle = CYCLE_STATUS.find((status) => status.value === cycleStatus);
|
const currentCycle = CYCLE_STATUS.find((status) => status.value === cycleStatus);
|
||||||
|
|
||||||
const areYearsEqual = startDate.getFullYear() === endDate.getFullYear();
|
|
||||||
|
|
||||||
const cycleTotalIssues =
|
const cycleTotalIssues =
|
||||||
cycle.backlog_issues +
|
cycle.backlog_issues +
|
||||||
cycle.unstarted_issues +
|
cycle.unstarted_issues +
|
||||||
@ -228,8 +226,7 @@ export const CyclesBoardCard: FC<ICyclesBoardCard> = (props) => {
|
|||||||
<div className="flex items-center justify-between">
|
<div className="flex items-center justify-between">
|
||||||
{isDateValid ? (
|
{isDateValid ? (
|
||||||
<span className="text-xs text-custom-text-300">
|
<span className="text-xs text-custom-text-300">
|
||||||
{areYearsEqual ? renderShortDate(startDate, "_ _") : renderShortMonthDate(startDate, "_ _")} -{" "}
|
{renderFormattedDate(startDate) ?? "_ _"} - {renderFormattedDate(endDate) ?? "_ _"}
|
||||||
{areYearsEqual ? renderShortDate(endDate, "_ _") : renderShortMonthDate(endDate, "_ _")}
|
|
||||||
</span>
|
</span>
|
||||||
) : (
|
) : (
|
||||||
<span className="text-xs text-custom-text-400">No due date</span>
|
<span className="text-xs text-custom-text-400">No due date</span>
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
import { FC, MouseEvent, useState } from "react";
|
import { FC, MouseEvent, useState } from "react";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
|
|
||||||
// stores
|
// stores
|
||||||
import { useMobxStore } from "lib/mobx/store-provider";
|
import { useMobxStore } from "lib/mobx/store-provider";
|
||||||
// hooks
|
// hooks
|
||||||
@ -13,7 +12,7 @@ import { CustomMenu, Tooltip, CircularProgressIndicator, CycleGroupIcon, AvatarG
|
|||||||
// icons
|
// icons
|
||||||
import { Check, Info, LinkIcon, Pencil, Star, Trash2, User2 } from "lucide-react";
|
import { Check, Info, LinkIcon, Pencil, Star, Trash2, User2 } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { findHowManyDaysLeft, renderShortDate, renderShortMonthDate } from "helpers/date-time.helper";
|
import { findHowManyDaysLeft, renderFormattedDate } from "helpers/date-time.helper";
|
||||||
import { copyTextToClipboard } from "helpers/string.helper";
|
import { copyTextToClipboard } from "helpers/string.helper";
|
||||||
// types
|
// types
|
||||||
import { ICycle, TCycleGroups } from "types";
|
import { ICycle, TCycleGroups } from "types";
|
||||||
@ -64,8 +63,6 @@ export const CyclesListItem: FC<TCyclesListItem> = (props) => {
|
|||||||
|
|
||||||
const renderDate = cycle.start_date || cycle.end_date;
|
const renderDate = cycle.start_date || cycle.end_date;
|
||||||
|
|
||||||
const areYearsEqual = startDate.getFullYear() === endDate.getFullYear();
|
|
||||||
|
|
||||||
const completionPercentage = (cycle.completed_issues / cycleTotalIssues) * 100;
|
const completionPercentage = (cycle.completed_issues / cycleTotalIssues) * 100;
|
||||||
|
|
||||||
const progress = isNaN(completionPercentage) ? 0 : Math.floor(completionPercentage);
|
const progress = isNaN(completionPercentage) ? 0 : Math.floor(completionPercentage);
|
||||||
@ -204,10 +201,8 @@ export const CyclesListItem: FC<TCyclesListItem> = (props) => {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
{renderDate && (
|
{renderDate && (
|
||||||
<span className="flex w-28 items-center justify-center gap-2 text-xs text-custom-text-300">
|
<span className="flex w-40 items-center justify-center gap-2 text-xs text-custom-text-300">
|
||||||
{areYearsEqual ? renderShortDate(startDate, "_ _") : renderShortMonthDate(startDate, "_ _")}
|
{renderFormattedDate(startDate) ?? "_ _"} - {renderFormattedDate(endDate) ?? "_ _"}
|
||||||
{" - "}
|
|
||||||
{areYearsEqual ? renderShortDate(endDate, "_ _") : renderShortMonthDate(endDate, "_ _")}
|
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
|
|
||||||
// ui
|
// ui
|
||||||
import { Tooltip, ContrastIcon } from "@plane/ui";
|
import { Tooltip, ContrastIcon } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDate } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { ICycle } from "types";
|
import { ICycle } from "types";
|
||||||
|
|
||||||
@ -35,7 +34,7 @@ export const CycleGanttBlock = ({ data }: { data: ICycle }) => {
|
|||||||
<div className="space-y-1">
|
<div className="space-y-1">
|
||||||
<h5>{data?.name}</h5>
|
<h5>{data?.name}</h5>
|
||||||
<div>
|
<div>
|
||||||
{renderShortDate(data?.start_date ?? "")} to {renderShortDate(data?.end_date ?? "")}
|
{renderFormattedDate(data?.start_date ?? "")} to {renderFormattedDate(data?.end_date ?? "")}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
|
@ -32,9 +32,8 @@ import { copyUrlToClipboard } from "helpers/string.helper";
|
|||||||
import {
|
import {
|
||||||
findHowManyDaysLeft,
|
findHowManyDaysLeft,
|
||||||
isDateGreaterThanToday,
|
isDateGreaterThanToday,
|
||||||
renderDateFormat,
|
renderFormattedPayloadDate,
|
||||||
renderShortDate,
|
renderFormattedDate,
|
||||||
renderShortMonthDate,
|
|
||||||
} from "helpers/date-time.helper";
|
} from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { ICycle, IIssueFilterOptions } from "types";
|
import { ICycle, IIssueFilterOptions } from "types";
|
||||||
@ -141,8 +140,8 @@ export const CycleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
|
|
||||||
if (isDateValidForExistingCycle) {
|
if (isDateValidForExistingCycle) {
|
||||||
submitChanges({
|
submitChanges({
|
||||||
start_date: renderDateFormat(`${watch("start_date")}`),
|
start_date: renderFormattedPayloadDate(`${watch("start_date")}`),
|
||||||
end_date: renderDateFormat(`${watch("end_date")}`),
|
end_date: renderFormattedPayloadDate(`${watch("end_date")}`),
|
||||||
});
|
});
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
type: "success",
|
type: "success",
|
||||||
@ -168,8 +167,8 @@ export const CycleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
|
|
||||||
if (isDateValid) {
|
if (isDateValid) {
|
||||||
submitChanges({
|
submitChanges({
|
||||||
start_date: renderDateFormat(`${watch("start_date")}`),
|
start_date: renderFormattedPayloadDate(`${watch("start_date")}`),
|
||||||
end_date: renderDateFormat(`${watch("end_date")}`),
|
end_date: renderFormattedPayloadDate(`${watch("end_date")}`),
|
||||||
});
|
});
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
type: "success",
|
type: "success",
|
||||||
@ -209,8 +208,8 @@ export const CycleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
|
|
||||||
if (isDateValidForExistingCycle) {
|
if (isDateValidForExistingCycle) {
|
||||||
submitChanges({
|
submitChanges({
|
||||||
start_date: renderDateFormat(`${watch("start_date")}`),
|
start_date: renderFormattedPayloadDate(`${watch("start_date")}`),
|
||||||
end_date: renderDateFormat(`${watch("end_date")}`),
|
end_date: renderFormattedPayloadDate(`${watch("end_date")}`),
|
||||||
});
|
});
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
type: "success",
|
type: "success",
|
||||||
@ -236,8 +235,8 @@ export const CycleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
|
|
||||||
if (isDateValid) {
|
if (isDateValid) {
|
||||||
submitChanges({
|
submitChanges({
|
||||||
start_date: renderDateFormat(`${watch("start_date")}`),
|
start_date: renderFormattedPayloadDate(`${watch("start_date")}`),
|
||||||
end_date: renderDateFormat(`${watch("end_date")}`),
|
end_date: renderFormattedPayloadDate(`${watch("end_date")}`),
|
||||||
});
|
});
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
type: "success",
|
type: "success",
|
||||||
@ -302,9 +301,6 @@ export const CycleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
const endDate = new Date(watch("end_date") ?? cycleDetails.end_date ?? "");
|
const endDate = new Date(watch("end_date") ?? cycleDetails.end_date ?? "");
|
||||||
const startDate = new Date(watch("start_date") ?? cycleDetails.start_date ?? "");
|
const startDate = new Date(watch("start_date") ?? cycleDetails.start_date ?? "");
|
||||||
|
|
||||||
const areYearsEqual =
|
|
||||||
startDate.getFullYear() === endDate.getFullYear() || isNaN(startDate.getFullYear()) || isNaN(endDate.getFullYear());
|
|
||||||
|
|
||||||
const currentCycle = CYCLE_STATUS.find((status) => status.value === cycleStatus);
|
const currentCycle = CYCLE_STATUS.find((status) => status.value === cycleStatus);
|
||||||
|
|
||||||
const issueCount =
|
const issueCount =
|
||||||
@ -396,19 +392,17 @@ export const CycleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
<div className="relative flex w-1/2 items-center rounded-sm">
|
<div className="relative flex w-1/2 items-center rounded-sm">
|
||||||
<Popover className="flex h-full w-full items-center justify-center rounded-lg">
|
<Popover className="flex h-full w-full items-center justify-center rounded-lg">
|
||||||
<Popover.Button
|
<Popover.Button
|
||||||
className={`text-sm font-medium text-custom-text-300 w-full rounded-sm cursor-pointer hover:bg-custom-background-80 ${
|
className={`w-full cursor-pointer rounded-sm text-sm font-medium text-custom-text-300 hover:bg-custom-background-80 ${
|
||||||
isEditingAllowed ? "cursor-pointer" : "cursor-not-allowed"
|
isEditingAllowed ? "cursor-pointer" : "cursor-not-allowed"
|
||||||
}`}
|
}`}
|
||||||
disabled={isCompleted || !isEditingAllowed}
|
disabled={isCompleted || !isEditingAllowed}
|
||||||
>
|
>
|
||||||
<span
|
<span
|
||||||
className={`group flex w-full items-center justify-between gap-2 py-1 px-1.5 text-sm ${
|
className={`group flex w-full items-center justify-between gap-2 px-1.5 py-1 text-sm ${
|
||||||
watch("start_date") ? "" : "text-custom-text-400"
|
watch("start_date") ? "" : "text-custom-text-400"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
{areYearsEqual
|
{renderFormattedDate(startDate) ?? "No date selected"}
|
||||||
? renderShortDate(startDate, "No date selected")
|
|
||||||
: renderShortMonthDate(startDate, "No date selected")}
|
|
||||||
</span>
|
</span>
|
||||||
</Popover.Button>
|
</Popover.Button>
|
||||||
|
|
||||||
@ -450,19 +444,17 @@ export const CycleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
<Popover className="flex h-full w-full items-center justify-center rounded-lg">
|
<Popover className="flex h-full w-full items-center justify-center rounded-lg">
|
||||||
<>
|
<>
|
||||||
<Popover.Button
|
<Popover.Button
|
||||||
className={`text-sm font-medium text-custom-text-300 w-full rounded-sm cursor-pointer hover:bg-custom-background-80 ${
|
className={`w-full cursor-pointer rounded-sm text-sm font-medium text-custom-text-300 hover:bg-custom-background-80 ${
|
||||||
isEditingAllowed ? "cursor-pointer" : "cursor-not-allowed"
|
isEditingAllowed ? "cursor-pointer" : "cursor-not-allowed"
|
||||||
}`}
|
}`}
|
||||||
disabled={isCompleted || !isEditingAllowed}
|
disabled={isCompleted || !isEditingAllowed}
|
||||||
>
|
>
|
||||||
<span
|
<span
|
||||||
className={`group flex w-full items-center justify-between gap-2 py-1 px-1.5 text-sm ${
|
className={`group flex w-full items-center justify-between gap-2 px-1.5 py-1 text-sm ${
|
||||||
watch("end_date") ? "" : "text-custom-text-400"
|
watch("end_date") ? "" : "text-custom-text-400"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
{areYearsEqual
|
{renderFormattedDate(endDate) ?? "No date selected"}
|
||||||
? renderShortDate(endDate, "No date selected")
|
|
||||||
: renderShortMonthDate(endDate, "No date selected")}
|
|
||||||
</span>
|
</span>
|
||||||
</Popover.Button>
|
</Popover.Button>
|
||||||
|
|
||||||
|
@ -136,7 +136,7 @@ export const TransferIssuesModal: React.FC<Props> = observer(({ isOpen, handleCl
|
|||||||
<div className="flex w-full justify-between">
|
<div className="flex w-full justify-between">
|
||||||
<span>{option?.name}</span>
|
<span>{option?.name}</span>
|
||||||
<span className=" flex items-center rounded-full bg-custom-background-80 px-2 capitalize">
|
<span className=" flex items-center rounded-full bg-custom-background-80 px-2 capitalize">
|
||||||
{option.status}
|
{option.status.toLocaleLowerCase()}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</button>
|
</button>
|
||||||
|
@ -2,7 +2,7 @@ import { useState, FC } from "react";
|
|||||||
// ui
|
// ui
|
||||||
import { Button } from "@plane/ui";
|
import { Button } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IExportData } from "types";
|
import { IExportData } from "types";
|
||||||
|
|
||||||
@ -50,7 +50,7 @@ export const SingleExport: FC<Props> = ({ service, refreshing }) => {
|
|||||||
</span>
|
</span>
|
||||||
</h4>
|
</h4>
|
||||||
<div className="mt-2 flex items-center gap-2 text-xs text-custom-text-200">
|
<div className="mt-2 flex items-center gap-2 text-xs text-custom-text-200">
|
||||||
<span>{renderShortDateWithYearFormat(service.created_at)}</span>|
|
<span>{renderFormattedDate(service.created_at)}</span>|
|
||||||
<span>Exported by {service?.initiated_by_detail?.display_name}</span>
|
<span>Exported by {service?.initiated_by_detail?.display_name}</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,10 +1,9 @@
|
|||||||
import { FC } from "react";
|
import { FC } from "react";
|
||||||
|
|
||||||
// hooks
|
// hooks
|
||||||
import { useChart } from "../hooks";
|
import { useChart } from "../hooks";
|
||||||
// helpers
|
// helpers
|
||||||
import { ChartDraggable } from "../helpers/draggable";
|
import { ChartDraggable } from "../helpers/draggable";
|
||||||
import { renderDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IBlockUpdateData, IGanttBlock } from "../types";
|
import { IBlockUpdateData, IGanttBlock } from "../types";
|
||||||
|
|
||||||
@ -64,8 +63,8 @@ export const GanttChartBlocks: FC<GanttChartBlocksProps> = (props) => {
|
|||||||
|
|
||||||
// call the block update handler with the updated dates
|
// call the block update handler with the updated dates
|
||||||
blockUpdateHandler(block.data, {
|
blockUpdateHandler(block.data, {
|
||||||
start_date: renderDateFormat(updatedStartDate),
|
start_date: renderFormattedPayloadDate(updatedStartDate) ?? undefined,
|
||||||
target_date: renderDateFormat(updatedTargetDate),
|
target_date: renderFormattedPayloadDate(updatedTargetDate) ?? undefined,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -94,7 +94,7 @@ export const CycleGanttSidebar: React.FC<Props> = (props) => {
|
|||||||
<>
|
<>
|
||||||
{blocks ? (
|
{blocks ? (
|
||||||
blocks.map((block, index) => {
|
blocks.map((block, index) => {
|
||||||
const duration = findTotalDaysInRange(block.start_date ?? "", block.target_date ?? "", true);
|
const duration = findTotalDaysInRange(block.start_date ?? "", block.target_date ?? "");
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Draggable
|
<Draggable
|
||||||
|
@ -94,7 +94,7 @@ export const ModuleGanttSidebar: React.FC<Props> = (props) => {
|
|||||||
<>
|
<>
|
||||||
{blocks ? (
|
{blocks ? (
|
||||||
blocks.map((block, index) => {
|
blocks.map((block, index) => {
|
||||||
const duration = findTotalDaysInRange(block.start_date ?? "", block.target_date ?? "", true);
|
const duration = findTotalDaysInRange(block.start_date ?? "", block.target_date ?? "");
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Draggable
|
<Draggable
|
||||||
|
@ -95,7 +95,7 @@ export const ProjectViewGanttSidebar: React.FC<Props> = (props) => {
|
|||||||
<>
|
<>
|
||||||
{blocks ? (
|
{blocks ? (
|
||||||
blocks.map((block, index) => {
|
blocks.map((block, index) => {
|
||||||
const duration = findTotalDaysInRange(block.start_date ?? "", block.target_date ?? "", true);
|
const duration = findTotalDaysInRange(block.start_date ?? "", block.target_date ?? "");
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Draggable
|
<Draggable
|
||||||
|
@ -111,7 +111,7 @@ export const IssueGanttSidebar: React.FC<Props> = (props) => {
|
|||||||
<>
|
<>
|
||||||
{blocks ? (
|
{blocks ? (
|
||||||
blocks.map((block, index) => {
|
blocks.map((block, index) => {
|
||||||
const duration = findTotalDaysInRange(block.start_date ?? "", block.target_date ?? "", true);
|
const duration = findTotalDaysInRange(block.start_date ?? "", block.target_date ?? "");
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Draggable
|
<Draggable
|
||||||
|
@ -1,12 +1,11 @@
|
|||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
|
|
||||||
// ui
|
// ui
|
||||||
import { Tooltip, PriorityIcon } from "@plane/ui";
|
import { Tooltip, PriorityIcon } from "@plane/ui";
|
||||||
// icons
|
// icons
|
||||||
import { AlertTriangle, CalendarDays, CheckCircle2, Clock, Copy, XCircle } from "lucide-react";
|
import { AlertTriangle, CalendarDays, CheckCircle2, Clock, Copy, XCircle } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IInboxIssue } from "types";
|
import { IInboxIssue } from "types";
|
||||||
// constants
|
// constants
|
||||||
@ -45,11 +44,11 @@ export const InboxIssueCard: React.FC<Props> = (props) => {
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
<Tooltip
|
<Tooltip
|
||||||
tooltipHeading="Created on"
|
tooltipHeading="Created on"
|
||||||
tooltipContent={`${renderShortDateWithYearFormat(issue.created_at ?? "")}`}
|
tooltipContent={`${renderFormattedDate(issue.created_at ?? "")}`}
|
||||||
>
|
>
|
||||||
<div className="flex items-center gap-1 rounded border border-custom-border-200 px-2 py-[0.19rem] text-xs text-custom-text-200 shadow-sm">
|
<div className="flex items-center gap-1 rounded border border-custom-border-200 px-2 py-[0.19rem] text-xs text-custom-text-200 shadow-sm">
|
||||||
<CalendarDays size={12} strokeWidth={1.5} />
|
<CalendarDays size={12} strokeWidth={1.5} />
|
||||||
<span>{renderShortDateWithYearFormat(issue.created_at ?? "")}</span>
|
<span>{renderFormattedDate(issue.created_at ?? "")}</span>
|
||||||
</div>
|
</div>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</div>
|
</div>
|
||||||
|
@ -4,7 +4,6 @@ import { observer } from "mobx-react-lite";
|
|||||||
import useSWR from "swr";
|
import useSWR from "swr";
|
||||||
import { useForm } from "react-hook-form";
|
import { useForm } from "react-hook-form";
|
||||||
import { AlertTriangle, CheckCircle2, Clock, Copy, ExternalLink, Inbox, XCircle } from "lucide-react";
|
import { AlertTriangle, CheckCircle2, Clock, Copy, ExternalLink, Inbox, XCircle } from "lucide-react";
|
||||||
|
|
||||||
// mobx store
|
// mobx store
|
||||||
import { useMobxStore } from "lib/mobx/store-provider";
|
import { useMobxStore } from "lib/mobx/store-provider";
|
||||||
// components
|
// components
|
||||||
@ -13,7 +12,7 @@ import { InboxIssueActivity } from "components/inbox";
|
|||||||
// ui
|
// ui
|
||||||
import { Loader, StateGroupIcon } from "@plane/ui";
|
import { Loader, StateGroupIcon } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IInboxIssue, IIssue } from "types";
|
import { IInboxIssue, IIssue } from "types";
|
||||||
import { EUserWorkspaceRoles } from "constants/workspace";
|
import { EUserWorkspaceRoles } from "constants/workspace";
|
||||||
@ -193,12 +192,12 @@ export const InboxMainContent: React.FC = observer(() => {
|
|||||||
{new Date(issueDetails.issue_inbox[0].snoozed_till ?? "") < new Date() ? (
|
{new Date(issueDetails.issue_inbox[0].snoozed_till ?? "") < new Date() ? (
|
||||||
<p>
|
<p>
|
||||||
This issue was snoozed till{" "}
|
This issue was snoozed till{" "}
|
||||||
{renderShortDateWithYearFormat(issueDetails.issue_inbox[0].snoozed_till ?? "")}.
|
{renderFormattedDate(issueDetails.issue_inbox[0].snoozed_till ?? "")}.
|
||||||
</p>
|
</p>
|
||||||
) : (
|
) : (
|
||||||
<p>
|
<p>
|
||||||
This issue has been snoozed till{" "}
|
This issue has been snoozed till{" "}
|
||||||
{renderShortDateWithYearFormat(issueDetails.issue_inbox[0].snoozed_till ?? "")}.
|
{renderFormattedDate(issueDetails.issue_inbox[0].snoozed_till ?? "")}.
|
||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
|
@ -3,7 +3,7 @@ import { CustomMenu } from "@plane/ui";
|
|||||||
// icons
|
// icons
|
||||||
import { Trash2 } from "lucide-react";
|
import { Trash2 } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IImporterService } from "types";
|
import { IImporterService } from "types";
|
||||||
// constants
|
// constants
|
||||||
@ -39,7 +39,7 @@ export const SingleImport: React.FC<Props> = ({ service, refreshing, handleDelet
|
|||||||
</span>
|
</span>
|
||||||
</h4>
|
</h4>
|
||||||
<div className="mt-2 flex items-center gap-2 text-xs text-custom-text-200">
|
<div className="mt-2 flex items-center gap-2 text-xs text-custom-text-200">
|
||||||
<span>{renderShortDateWithYearFormat(service.created_at)}</span>|
|
<span>{renderFormattedDate(service.created_at)}</span>|
|
||||||
<span>Imported by {service.initiated_by_detail.display_name}</span>
|
<span>Imported by {service.initiated_by_detail.display_name}</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -9,7 +9,7 @@ import { CommentCard } from "components/issues/comment";
|
|||||||
// ui
|
// ui
|
||||||
import { Loader, Tooltip } from "@plane/ui";
|
import { Loader, Tooltip } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { render24HourFormatTime, renderLongDateFormat, timeAgo } from "helpers/date-time.helper";
|
import { renderFormattedTime, renderFormattedDate, calculateTimeAgo } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IIssueActivity } from "types";
|
import { IIssueActivity } from "types";
|
||||||
import { History } from "lucide-react";
|
import { History } from "lucide-react";
|
||||||
@ -114,11 +114,11 @@ export const IssueActivitySection: React.FC<Props> = ({
|
|||||||
)}{" "}
|
)}{" "}
|
||||||
{message}{" "}
|
{message}{" "}
|
||||||
<Tooltip
|
<Tooltip
|
||||||
tooltipContent={`${renderLongDateFormat(activityItem.created_at)}, ${render24HourFormatTime(
|
tooltipContent={`${renderFormattedDate(activityItem.created_at)}, ${renderFormattedTime(
|
||||||
activityItem.created_at
|
activityItem.created_at
|
||||||
)}`}
|
)}`}
|
||||||
>
|
>
|
||||||
<span className="whitespace-nowrap">{timeAgo(activityItem.created_at)}</span>
|
<span className="whitespace-nowrap">{calculateTimeAgo(activityItem.created_at)}</span>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -15,7 +15,7 @@ import { ProjectMemberService } from "services/project";
|
|||||||
import { ISSUE_ATTACHMENTS, PROJECT_MEMBERS } from "constants/fetch-keys";
|
import { ISSUE_ATTACHMENTS, PROJECT_MEMBERS } from "constants/fetch-keys";
|
||||||
// helper
|
// helper
|
||||||
import { truncateText } from "helpers/string.helper";
|
import { truncateText } from "helpers/string.helper";
|
||||||
import { renderLongDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
import { convertBytesToSize, getFileExtension, getFileName } from "helpers/attachment.helper";
|
import { convertBytesToSize, getFileExtension, getFileName } from "helpers/attachment.helper";
|
||||||
// type
|
// type
|
||||||
import { IIssueAttachment } from "types";
|
import { IIssueAttachment } from "types";
|
||||||
@ -77,7 +77,7 @@ export const IssueAttachments: React.FC<Props> = (props) => {
|
|||||||
<Tooltip
|
<Tooltip
|
||||||
tooltipContent={`${
|
tooltipContent={`${
|
||||||
people?.find((person) => person.member.id === file.updated_by)?.member.display_name ?? ""
|
people?.find((person) => person.member.id === file.updated_by)?.member.display_name ?? ""
|
||||||
} uploaded on ${renderLongDateFormat(file.updated_at)}`}
|
} uploaded on ${renderFormattedDate(file.updated_at)}`}
|
||||||
>
|
>
|
||||||
<span>
|
<span>
|
||||||
<AlertCircle className="h-3 w-3" />
|
<AlertCircle className="h-3 w-3" />
|
||||||
|
@ -1,21 +1,20 @@
|
|||||||
import React, { useEffect, useState } from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { useForm } from "react-hook-form";
|
import { useForm } from "react-hook-form";
|
||||||
|
|
||||||
// services
|
// services
|
||||||
import { FileService } from "services/file.service";
|
import { FileService } from "services/file.service";
|
||||||
// icons
|
// icons
|
||||||
import { Check, Globe2, Lock, MessageSquare, Pencil, Trash2, X } from "lucide-react";
|
import { Check, Globe2, Lock, MessageSquare, Pencil, Trash2, X } from "lucide-react";
|
||||||
// hooks
|
// hooks
|
||||||
import useUser from "hooks/use-user";
|
import useUser from "hooks/use-user";
|
||||||
|
import useEditorSuggestions from "hooks/use-editor-suggestions";
|
||||||
// ui
|
// ui
|
||||||
import { CustomMenu } from "@plane/ui";
|
import { CustomMenu } from "@plane/ui";
|
||||||
import { CommentReaction } from "components/issues";
|
import { CommentReaction } from "components/issues";
|
||||||
import { LiteTextEditorWithRef, LiteReadOnlyEditorWithRef } from "@plane/lite-text-editor";
|
import { LiteTextEditorWithRef, LiteReadOnlyEditorWithRef } from "@plane/lite-text-editor";
|
||||||
// helpers
|
// helpers
|
||||||
import { timeAgo } from "helpers/date-time.helper";
|
import { calculateTimeAgo } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import type { IIssueActivity } from "types";
|
import type { IIssueActivity } from "types";
|
||||||
import useEditorSuggestions from "hooks/use-editor-suggestions";
|
|
||||||
|
|
||||||
// services
|
// services
|
||||||
const fileService = new FileService();
|
const fileService = new FileService();
|
||||||
@ -98,7 +97,7 @@ export const CommentCard: React.FC<Props> = ({
|
|||||||
<div className="text-xs">
|
<div className="text-xs">
|
||||||
{comment.actor_detail.is_bot ? comment.actor_detail.first_name + " Bot" : comment.actor_detail.display_name}
|
{comment.actor_detail.is_bot ? comment.actor_detail.first_name + " Bot" : comment.actor_detail.display_name}
|
||||||
</div>
|
</div>
|
||||||
<p className="mt-0.5 text-xs text-custom-text-200">commented {timeAgo(comment.created_at)}</p>
|
<p className="mt-0.5 text-xs text-custom-text-200">commented {calculateTimeAgo(comment.created_at)}</p>
|
||||||
</div>
|
</div>
|
||||||
<div className="issue-comments-section p-0">
|
<div className="issue-comments-section p-0">
|
||||||
<form className={`flex-col gap-2 ${isEditing ? "flex" : "hidden"}`}>
|
<form className={`flex-col gap-2 ${isEditing ? "flex" : "hidden"}`}>
|
||||||
|
@ -4,7 +4,7 @@ import { Droppable } from "@hello-pangea/dnd";
|
|||||||
// components
|
// components
|
||||||
import { CalendarIssueBlocks, ICalendarDate, CalendarQuickAddIssueForm } from "components/issues";
|
import { CalendarIssueBlocks, ICalendarDate, CalendarQuickAddIssueForm } from "components/issues";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
// constants
|
// constants
|
||||||
import { MONTHS_LIST } from "constants/calendar";
|
import { MONTHS_LIST } from "constants/calendar";
|
||||||
import { IIssue } from "types";
|
import { IIssue } from "types";
|
||||||
@ -52,7 +52,9 @@ export const CalendarDayTile: React.FC<Props> = observer((props) => {
|
|||||||
const [showAllIssues, setShowAllIssues] = useState(false);
|
const [showAllIssues, setShowAllIssues] = useState(false);
|
||||||
const calendarLayout = issuesFilterStore?.issueFilters?.displayFilters?.calendar?.layout ?? "month";
|
const calendarLayout = issuesFilterStore?.issueFilters?.displayFilters?.calendar?.layout ?? "month";
|
||||||
|
|
||||||
const issueIdList = groupedIssueIds ? groupedIssueIds[renderDateFormat(date.date)] : null;
|
const formattedDatePayload = renderFormattedPayloadDate(date.date);
|
||||||
|
if (!formattedDatePayload) return null;
|
||||||
|
const issueIdList = groupedIssueIds ? groupedIssueIds[formattedDatePayload] : null;
|
||||||
|
|
||||||
const totalIssues = issueIdList?.length ?? 0;
|
const totalIssues = issueIdList?.length ?? 0;
|
||||||
return (
|
return (
|
||||||
@ -78,7 +80,7 @@ export const CalendarDayTile: React.FC<Props> = observer((props) => {
|
|||||||
|
|
||||||
{/* content */}
|
{/* content */}
|
||||||
<div className="h-full w-full">
|
<div className="h-full w-full">
|
||||||
<Droppable droppableId={renderDateFormat(date.date)} isDropDisabled={false}>
|
<Droppable droppableId={formattedDatePayload} isDropDisabled={false}>
|
||||||
{(provided, snapshot) => (
|
{(provided, snapshot) => (
|
||||||
<div
|
<div
|
||||||
className={`h-full w-full select-none overflow-y-auto ${
|
className={`h-full w-full select-none overflow-y-auto ${
|
||||||
@ -100,9 +102,9 @@ export const CalendarDayTile: React.FC<Props> = observer((props) => {
|
|||||||
<div className="px-2 py-1">
|
<div className="px-2 py-1">
|
||||||
<CalendarQuickAddIssueForm
|
<CalendarQuickAddIssueForm
|
||||||
formKey="target_date"
|
formKey="target_date"
|
||||||
groupId={renderDateFormat(date.date)}
|
groupId={formattedDatePayload}
|
||||||
prePopulatedData={{
|
prePopulatedData={{
|
||||||
target_date: renderDateFormat(date.date),
|
target_date: renderFormattedPayloadDate(date.date),
|
||||||
}}
|
}}
|
||||||
quickAddCallback={quickAddCallback}
|
quickAddCallback={quickAddCallback}
|
||||||
viewId={viewId}
|
viewId={viewId}
|
||||||
|
@ -2,7 +2,7 @@ import { observer } from "mobx-react-lite";
|
|||||||
// components
|
// components
|
||||||
import { CalendarDayTile } from "components/issues";
|
import { CalendarDayTile } from "components/issues";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { ICalendarDate, ICalendarWeek } from "./types";
|
import { ICalendarDate, ICalendarWeek } from "./types";
|
||||||
import { IIssue } from "types";
|
import { IIssue } from "types";
|
||||||
@ -65,7 +65,7 @@ export const CalendarWeekDays: React.FC<Props> = observer((props) => {
|
|||||||
return (
|
return (
|
||||||
<CalendarDayTile
|
<CalendarDayTile
|
||||||
issuesFilterStore={issuesFilterStore}
|
issuesFilterStore={issuesFilterStore}
|
||||||
key={renderDateFormat(date.date)}
|
key={renderFormattedPayloadDate(date.date)}
|
||||||
date={date}
|
date={date}
|
||||||
issues={issues}
|
issues={issues}
|
||||||
groupedIssueIds={groupedIssueIds}
|
groupedIssueIds={groupedIssueIds}
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
|
|
||||||
// icons
|
// icons
|
||||||
import { X } from "lucide-react";
|
import { X } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderLongDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
import { capitalizeFirstLetter } from "helpers/string.helper";
|
import { capitalizeFirstLetter } from "helpers/string.helper";
|
||||||
// constants
|
// constants
|
||||||
import { DATE_FILTER_OPTIONS } from "constants/filters";
|
import { DATE_FILTER_OPTIONS } from "constants/filters";
|
||||||
@ -28,7 +27,7 @@ export const AppliedDateFilters: React.FC<Props> = observer((props) => {
|
|||||||
if (dateParts.length === 2) {
|
if (dateParts.length === 2) {
|
||||||
const [date, time] = dateParts;
|
const [date, time] = dateParts;
|
||||||
|
|
||||||
dateLabel = `${capitalizeFirstLetter(time)} ${renderLongDateFormat(date)}`;
|
dateLabel = `${capitalizeFirstLetter(time)} ${renderFormattedDate(date)}`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2,7 +2,7 @@ import { useRouter } from "next/router";
|
|||||||
// ui
|
// ui
|
||||||
import { Tooltip, StateGroupIcon } from "@plane/ui";
|
import { Tooltip, StateGroupIcon } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDate } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IIssue } from "types";
|
import { IIssue } from "types";
|
||||||
|
|
||||||
@ -34,15 +34,13 @@ export const IssueGanttBlock = ({ data }: { data: IIssue }) => {
|
|||||||
<div className="space-y-1">
|
<div className="space-y-1">
|
||||||
<h5>{data?.name}</h5>
|
<h5>{data?.name}</h5>
|
||||||
<div>
|
<div>
|
||||||
{renderShortDate(data?.start_date ?? "")} to {renderShortDate(data?.target_date ?? "")}
|
{renderFormattedDate(data?.start_date ?? "")} to {renderFormattedDate(data?.target_date ?? "")}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
position="top-left"
|
position="top-left"
|
||||||
>
|
>
|
||||||
<Tooltip tooltipHeading="Title" tooltipContent={data.name}>
|
<div className="relative w-full truncate px-2.5 py-1 text-sm text-custom-text-100">{data?.name}</div>
|
||||||
<div className="relative w-full truncate px-2.5 py-1 text-sm text-custom-text-100">{data?.name}</div>
|
|
||||||
</Tooltip>
|
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -11,11 +11,10 @@ import useKeypress from "hooks/use-keypress";
|
|||||||
import useProjectDetails from "hooks/use-project-details";
|
import useProjectDetails from "hooks/use-project-details";
|
||||||
import useOutsideClickDetector from "hooks/use-outside-click-detector";
|
import useOutsideClickDetector from "hooks/use-outside-click-detector";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
|
import { createIssuePayload } from "helpers/issue.helper";
|
||||||
// types
|
// types
|
||||||
import { IIssue } from "types";
|
import { IIssue } from "types";
|
||||||
// helpers
|
|
||||||
import { createIssuePayload } from "helpers/issue.helper";
|
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
prePopulatedData?: Partial<IIssue>;
|
prePopulatedData?: Partial<IIssue>;
|
||||||
@ -116,8 +115,8 @@ export const GanttInlineCreateIssueForm: React.FC<Props> = observer((props) => {
|
|||||||
const payload = createIssuePayload(workspaceDetail!, projectDetails!, {
|
const payload = createIssuePayload(workspaceDetail!, projectDetails!, {
|
||||||
...(prePopulatedData ?? {}),
|
...(prePopulatedData ?? {}),
|
||||||
...formData,
|
...formData,
|
||||||
start_date: renderDateFormat(new Date()),
|
start_date: renderFormattedPayloadDate(new Date()),
|
||||||
target_date: renderDateFormat(new Date(new Date().getTime() + 24 * 60 * 60 * 1000)),
|
target_date: renderFormattedPayloadDate(new Date(new Date().getTime() + 24 * 60 * 60 * 1000)),
|
||||||
});
|
});
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
@ -12,7 +12,7 @@ import { Tooltip } from "@plane/ui";
|
|||||||
// hooks
|
// hooks
|
||||||
import useDynamicDropdownPosition from "hooks/use-dynamic-dropdown";
|
import useDynamicDropdownPosition from "hooks/use-dynamic-dropdown";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderDateFormat, renderFormattedDate } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate, renderFormattedDate } from "helpers/date-time.helper";
|
||||||
|
|
||||||
export interface IIssuePropertyDate {
|
export interface IIssuePropertyDate {
|
||||||
value: string | null;
|
value: string | null;
|
||||||
@ -105,7 +105,7 @@ export const IssuePropertyDate: React.FC<IIssuePropertyDate> = observer((props)
|
|||||||
onChange={(val, e) => {
|
onChange={(val, e) => {
|
||||||
e?.stopPropagation();
|
e?.stopPropagation();
|
||||||
if (onChange && val) {
|
if (onChange && val) {
|
||||||
onChange(renderDateFormat(val));
|
onChange(renderFormattedPayloadDate(val));
|
||||||
close();
|
close();
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
|
|
||||||
// hooks
|
// hooks
|
||||||
import useSubIssue from "hooks/use-sub-issue";
|
import useSubIssue from "hooks/use-sub-issue";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderLongDetailDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IIssue } from "types";
|
import { IIssue } from "types";
|
||||||
|
|
||||||
@ -20,7 +19,7 @@ export const SpreadsheetCreatedOnColumn: React.FC<Props> = ({ issue, expandedIss
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="flex h-11 w-full items-center justify-center text-xs border-b-[0.5px] border-custom-border-200 hover:bg-custom-background-80">
|
<div className="flex h-11 w-full items-center justify-center text-xs border-b-[0.5px] border-custom-border-200 hover:bg-custom-background-80">
|
||||||
{renderLongDetailDateFormat(issue.created_at)}
|
{renderFormattedDate(issue.created_at)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{isExpanded &&
|
{isExpanded &&
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
|
|
||||||
// hooks
|
// hooks
|
||||||
import useSubIssue from "hooks/use-sub-issue";
|
import useSubIssue from "hooks/use-sub-issue";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderLongDetailDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IIssue } from "types";
|
import { IIssue } from "types";
|
||||||
|
|
||||||
@ -22,7 +21,7 @@ export const SpreadsheetUpdatedOnColumn: React.FC<Props> = (props) => {
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="flex h-11 w-full items-center justify-center text-xs border-b-[0.5px] border-custom-border-200 hover:bg-custom-background-80">
|
<div className="flex h-11 w-full items-center justify-center text-xs border-b-[0.5px] border-custom-border-200 hover:bg-custom-background-80">
|
||||||
{renderLongDetailDateFormat(issue.updated_at)}
|
{renderFormattedDate(issue.updated_at)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{isExpanded &&
|
{isExpanded &&
|
||||||
|
@ -7,7 +7,7 @@ import { Loader, Tooltip } from "@plane/ui";
|
|||||||
import { ActivityIcon, ActivityMessage } from "components/core";
|
import { ActivityIcon, ActivityMessage } from "components/core";
|
||||||
import { IssueCommentCard } from "./comment-card";
|
import { IssueCommentCard } from "./comment-card";
|
||||||
// helpers
|
// helpers
|
||||||
import { render24HourFormatTime, renderLongDateFormat, timeAgo } from "helpers/date-time.helper";
|
import { renderFormattedTime, renderFormattedDate, calculateTimeAgo } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IIssueActivity, IUser } from "types";
|
import { IIssueActivity, IUser } from "types";
|
||||||
|
|
||||||
@ -104,11 +104,11 @@ export const IssueActivityCard: FC<IIssueActivityCard> = (props) => {
|
|||||||
)}
|
)}
|
||||||
{message}
|
{message}
|
||||||
<Tooltip
|
<Tooltip
|
||||||
tooltipContent={`${renderLongDateFormat(activityItem.created_at)}, ${render24HourFormatTime(
|
tooltipContent={`${renderFormattedDate(activityItem.created_at)}, ${renderFormattedTime(
|
||||||
activityItem.created_at
|
activityItem.created_at
|
||||||
)}`}
|
)}`}
|
||||||
>
|
>
|
||||||
<span className="whitespace-nowrap">{timeAgo(activityItem.created_at)}</span>
|
<span className="whitespace-nowrap">{calculateTimeAgo(activityItem.created_at)}</span>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -10,7 +10,7 @@ import { LiteTextEditorWithRef, LiteReadOnlyEditorWithRef } from "@plane/lite-te
|
|||||||
// components
|
// components
|
||||||
import { IssueCommentReaction } from "./comment-reaction";
|
import { IssueCommentReaction } from "./comment-reaction";
|
||||||
// helpers
|
// helpers
|
||||||
import { timeAgo } from "helpers/date-time.helper";
|
import { calculateTimeAgo } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import type { IIssueActivity, IUser } from "types";
|
import type { IIssueActivity, IUser } from "types";
|
||||||
|
|
||||||
@ -106,7 +106,7 @@ export const IssueCommentCard: React.FC<IIssueCommentCard> = (props) => {
|
|||||||
<div className="text-xs">
|
<div className="text-xs">
|
||||||
{comment.actor_detail.is_bot ? comment.actor_detail.first_name + " Bot" : comment.actor_detail.display_name}
|
{comment.actor_detail.is_bot ? comment.actor_detail.first_name + " Bot" : comment.actor_detail.display_name}
|
||||||
</div>
|
</div>
|
||||||
<p className="mt-0.5 text-xs text-custom-text-200">commented {timeAgo(comment.created_at)}</p>
|
<p className="mt-0.5 text-xs text-custom-text-200">commented {calculateTimeAgo(comment.created_at)}</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="issue-comments-section p-0">
|
<div className="issue-comments-section p-0">
|
||||||
|
@ -4,8 +4,8 @@ import { Popover, Transition } from "@headlessui/react";
|
|||||||
import { CalendarDays, X } from "lucide-react";
|
import { CalendarDays, X } from "lucide-react";
|
||||||
// react-datepicker
|
// react-datepicker
|
||||||
import DatePicker from "react-datepicker";
|
import DatePicker from "react-datepicker";
|
||||||
// import "react-datepicker/dist/react-datepicker.css";
|
// helpers
|
||||||
import { renderDateFormat, renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate, renderFormattedDate } from "helpers/date-time.helper";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
label: string;
|
label: string;
|
||||||
@ -35,7 +35,7 @@ export const IssueDateSelect: FC<Props> = ({ label, maxDate, minDate, onChange,
|
|||||||
{value ? (
|
{value ? (
|
||||||
<>
|
<>
|
||||||
<CalendarDays className="h-3 w-3 flex-shrink-0" />
|
<CalendarDays className="h-3 w-3 flex-shrink-0" />
|
||||||
<span>{renderShortDateWithYearFormat(value)}</span>
|
<span>{renderFormattedDate(value)}</span>
|
||||||
<button onClick={() => onChange(null)}>
|
<button onClick={() => onChange(null)}>
|
||||||
<X className="h-3 w-3 flex-shrink-0" />
|
<X className="h-3 w-3 flex-shrink-0" />
|
||||||
</button>
|
</button>
|
||||||
@ -69,7 +69,7 @@ export const IssueDateSelect: FC<Props> = ({ label, maxDate, minDate, onChange,
|
|||||||
selected={value ? new Date(value) : null}
|
selected={value ? new Date(value) : null}
|
||||||
onChange={(val) => {
|
onChange={(val) => {
|
||||||
if (!val) onChange("");
|
if (!val) onChange("");
|
||||||
else onChange(renderDateFormat(val));
|
else onChange(renderFormattedPayloadDate(val));
|
||||||
|
|
||||||
close();
|
close();
|
||||||
}}
|
}}
|
||||||
|
@ -3,12 +3,7 @@ import { CustomDatePicker } from "components/ui";
|
|||||||
import { Tooltip } from "@plane/ui";
|
import { Tooltip } from "@plane/ui";
|
||||||
import { CalendarCheck } from "lucide-react";
|
import { CalendarCheck } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import {
|
import { findHowManyDaysLeft, renderFormattedDate } from "helpers/date-time.helper";
|
||||||
findHowManyDaysLeft,
|
|
||||||
renderShortDate,
|
|
||||||
renderShortDateWithYearFormat,
|
|
||||||
renderShortMonthDate,
|
|
||||||
} from "helpers/date-time.helper";
|
|
||||||
// types
|
// types
|
||||||
import { IIssue } from "types";
|
import { IIssue } from "types";
|
||||||
|
|
||||||
@ -36,14 +31,10 @@ export const ViewDueDateSelect: React.FC<Props> = ({
|
|||||||
const minDate = issue.start_date ? new Date(issue.start_date) : null;
|
const minDate = issue.start_date ? new Date(issue.start_date) : null;
|
||||||
minDate?.setDate(minDate.getDate());
|
minDate?.setDate(minDate.getDate());
|
||||||
|
|
||||||
const today = new Date();
|
|
||||||
const endDate = new Date(issue.target_date ?? "");
|
|
||||||
const areYearsEqual = endDate.getFullYear() === today.getFullYear();
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
tooltipHeading="Due date"
|
tooltipHeading="Due date"
|
||||||
tooltipContent={issue.target_date ? renderShortDateWithYearFormat(issue.target_date) ?? "N/A" : "N/A"}
|
tooltipContent={issue.target_date ? renderFormattedDate(issue.target_date) ?? "N/A" : "N/A"}
|
||||||
position={tooltipPosition}
|
position={tooltipPosition}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
@ -68,11 +59,7 @@ export const ViewDueDateSelect: React.FC<Props> = ({
|
|||||||
{issue.target_date ? (
|
{issue.target_date ? (
|
||||||
<>
|
<>
|
||||||
<CalendarCheck className="h-3.5 w-3.5 flex-shrink-0" />
|
<CalendarCheck className="h-3.5 w-3.5 flex-shrink-0" />
|
||||||
<span>
|
<span>{renderFormattedDate(issue.target_date) ?? "_ _"}</span>
|
||||||
{areYearsEqual
|
|
||||||
? renderShortDate(issue.target_date ?? "", "_ _")
|
|
||||||
: renderShortMonthDate(issue.target_date ?? "", "_ _")}
|
|
||||||
</span>
|
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
|
@ -3,7 +3,7 @@ import { CustomDatePicker } from "components/ui";
|
|||||||
import { Tooltip } from "@plane/ui";
|
import { Tooltip } from "@plane/ui";
|
||||||
import { CalendarClock } from "lucide-react";
|
import { CalendarClock } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDate, renderShortDateWithYearFormat, renderShortMonthDate } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IIssue } from "types";
|
import { IIssue } from "types";
|
||||||
|
|
||||||
@ -30,14 +30,11 @@ export const ViewStartDateSelect: React.FC<Props> = ({
|
|||||||
}) => {
|
}) => {
|
||||||
const maxDate = issue.target_date ? new Date(issue.target_date) : null;
|
const maxDate = issue.target_date ? new Date(issue.target_date) : null;
|
||||||
maxDate?.setDate(maxDate.getDate());
|
maxDate?.setDate(maxDate.getDate());
|
||||||
const today = new Date();
|
|
||||||
const startDate = new Date(issue.start_date ?? "");
|
|
||||||
const areYearsEqual = startDate.getFullYear() === today.getFullYear();
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
tooltipHeading="Start date"
|
tooltipHeading="Start date"
|
||||||
tooltipContent={issue.start_date ? renderShortDateWithYearFormat(issue.start_date) ?? "N/A" : "N/A"}
|
tooltipContent={issue.start_date ? renderFormattedDate(issue.start_date) ?? "N/A" : "N/A"}
|
||||||
position={tooltipPosition}
|
position={tooltipPosition}
|
||||||
>
|
>
|
||||||
<div className={`group max-w-[6.5rem] flex-shrink-0 ${className}`}>
|
<div className={`group max-w-[6.5rem] flex-shrink-0 ${className}`}>
|
||||||
@ -56,11 +53,7 @@ export const ViewStartDateSelect: React.FC<Props> = ({
|
|||||||
{issue?.start_date ? (
|
{issue?.start_date ? (
|
||||||
<>
|
<>
|
||||||
<CalendarClock className="h-3.5 w-3.5 flex-shrink-0" />
|
<CalendarClock className="h-3.5 w-3.5 flex-shrink-0" />
|
||||||
<span>
|
<span>{renderFormattedDate(issue?.start_date ?? "_ _")}</span>
|
||||||
{areYearsEqual
|
|
||||||
? renderShortDate(issue?.start_date, "_ _")
|
|
||||||
: renderShortMonthDate(issue?.start_date, "_ _")}
|
|
||||||
</span>
|
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
|
|
||||||
// ui
|
// ui
|
||||||
import { Tooltip, ModuleStatusIcon } from "@plane/ui";
|
import { Tooltip, ModuleStatusIcon } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDate } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IModule } from "types";
|
import { IModule } from "types";
|
||||||
// constants
|
// constants
|
||||||
@ -25,7 +24,7 @@ export const ModuleGanttBlock = ({ data }: { data: IModule }) => {
|
|||||||
<div className="space-y-1">
|
<div className="space-y-1">
|
||||||
<h5>{data?.name}</h5>
|
<h5>{data?.name}</h5>
|
||||||
<div>
|
<div>
|
||||||
{renderShortDate(data?.start_date ?? "")} to {renderShortDate(data?.target_date ?? "")}
|
{renderFormattedDate(data?.start_date ?? "")} to {renderFormattedDate(data?.target_date ?? "")}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,7 @@ import { Avatar, AvatarGroup, CustomMenu, LayersIcon, Tooltip } from "@plane/ui"
|
|||||||
import { Info, LinkIcon, Pencil, Star, Trash2 } from "lucide-react";
|
import { Info, LinkIcon, Pencil, Star, Trash2 } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { copyUrlToClipboard } from "helpers/string.helper";
|
import { copyUrlToClipboard } from "helpers/string.helper";
|
||||||
import { renderShortDate, renderShortMonthDate } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IModule } from "types";
|
import { IModule } from "types";
|
||||||
// constants
|
// constants
|
||||||
@ -56,8 +56,6 @@ export const ModuleCardItem: React.FC<Props> = observer((props) => {
|
|||||||
|
|
||||||
const isDateValid = module.target_date || module.start_date;
|
const isDateValid = module.target_date || module.start_date;
|
||||||
|
|
||||||
const areYearsEqual = startDate.getFullYear() === endDate.getFullYear();
|
|
||||||
|
|
||||||
const moduleStatus = MODULE_STATUS.find((status) => status.value === module.status);
|
const moduleStatus = MODULE_STATUS.find((status) => status.value === module.status);
|
||||||
|
|
||||||
const issueCount = module
|
const issueCount = module
|
||||||
@ -213,8 +211,7 @@ export const ModuleCardItem: React.FC<Props> = observer((props) => {
|
|||||||
{isDateValid ? (
|
{isDateValid ? (
|
||||||
<>
|
<>
|
||||||
<span className="text-xs text-custom-text-300">
|
<span className="text-xs text-custom-text-300">
|
||||||
{areYearsEqual ? renderShortDate(startDate, "_ _") : renderShortMonthDate(startDate, "_ _")} -{" "}
|
{renderFormattedDate(startDate) ?? "_ _"} - {renderFormattedDate(endDate) ?? "_ _"}
|
||||||
{areYearsEqual ? renderShortDate(endDate, "_ _") : renderShortMonthDate(endDate, "_ _")}
|
|
||||||
</span>
|
</span>
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
|
@ -14,7 +14,7 @@ import { Avatar, AvatarGroup, CircularProgressIndicator, CustomMenu, Tooltip } f
|
|||||||
import { Check, Info, LinkIcon, Pencil, Star, Trash2, User2 } from "lucide-react";
|
import { Check, Info, LinkIcon, Pencil, Star, Trash2, User2 } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { copyUrlToClipboard } from "helpers/string.helper";
|
import { copyUrlToClipboard } from "helpers/string.helper";
|
||||||
import { renderShortDate, renderShortMonthDate } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IModule } from "types";
|
import { IModule } from "types";
|
||||||
// constants
|
// constants
|
||||||
@ -49,8 +49,6 @@ export const ModuleListItem: React.FC<Props> = observer((props) => {
|
|||||||
|
|
||||||
const renderDate = module.start_date || module.target_date;
|
const renderDate = module.start_date || module.target_date;
|
||||||
|
|
||||||
const areYearsEqual = startDate.getFullYear() === endDate.getFullYear();
|
|
||||||
|
|
||||||
const moduleStatus = MODULE_STATUS.find((status) => status.value === module.status);
|
const moduleStatus = MODULE_STATUS.find((status) => status.value === module.status);
|
||||||
|
|
||||||
const progress = isNaN(completionPercentage) ? 0 : Math.floor(completionPercentage);
|
const progress = isNaN(completionPercentage) ? 0 : Math.floor(completionPercentage);
|
||||||
@ -176,10 +174,8 @@ export const ModuleListItem: React.FC<Props> = observer((props) => {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
{renderDate && (
|
{renderDate && (
|
||||||
<span className="flex w-28 items-center justify-center gap-2 text-xs text-custom-text-300">
|
<span className="flex w-40 items-center justify-center gap-2 text-xs text-custom-text-300">
|
||||||
{areYearsEqual ? renderShortDate(startDate, "_ _") : renderShortMonthDate(startDate, "_ _")}
|
{renderFormattedDate(startDate) ?? "_ _"} - {renderFormattedDate(endDate) ?? "_ _"}
|
||||||
{" - "}
|
|
||||||
{areYearsEqual ? renderShortDate(endDate, "_ _") : renderShortMonthDate(endDate, "_ _")}
|
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
@ -27,12 +27,7 @@ import {
|
|||||||
Trash2,
|
Trash2,
|
||||||
} from "lucide-react";
|
} from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import {
|
import { isDateGreaterThanToday, renderFormattedPayloadDate, renderFormattedDate } from "helpers/date-time.helper";
|
||||||
isDateGreaterThanToday,
|
|
||||||
renderDateFormat,
|
|
||||||
renderShortDate,
|
|
||||||
renderShortMonthDate,
|
|
||||||
} from "helpers/date-time.helper";
|
|
||||||
import { copyUrlToClipboard } from "helpers/string.helper";
|
import { copyUrlToClipboard } from "helpers/string.helper";
|
||||||
// types
|
// types
|
||||||
import { IIssueFilterOptions, ILinkDetails, IModule, ModuleLink } from "types";
|
import { IIssueFilterOptions, ILinkDetails, IModule, ModuleLink } from "types";
|
||||||
@ -187,8 +182,8 @@ export const ModuleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
submitChanges({
|
submitChanges({
|
||||||
start_date: renderDateFormat(`${watch("start_date")}`),
|
start_date: renderFormattedPayloadDate(`${watch("start_date")}`),
|
||||||
target_date: renderDateFormat(`${watch("target_date")}`),
|
target_date: renderFormattedPayloadDate(`${watch("target_date")}`),
|
||||||
});
|
});
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
type: "success",
|
type: "success",
|
||||||
@ -212,8 +207,8 @@ export const ModuleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
submitChanges({
|
submitChanges({
|
||||||
start_date: renderDateFormat(`${watch("start_date")}`),
|
start_date: renderFormattedPayloadDate(`${watch("start_date")}`),
|
||||||
target_date: renderDateFormat(`${watch("target_date")}`),
|
target_date: renderFormattedPayloadDate(`${watch("target_date")}`),
|
||||||
});
|
});
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
type: "success",
|
type: "success",
|
||||||
@ -285,9 +280,6 @@ export const ModuleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
const startDate = new Date(watch("start_date") ?? moduleDetails.start_date ?? "");
|
const startDate = new Date(watch("start_date") ?? moduleDetails.start_date ?? "");
|
||||||
const endDate = new Date(watch("target_date") ?? moduleDetails.target_date ?? "");
|
const endDate = new Date(watch("target_date") ?? moduleDetails.target_date ?? "");
|
||||||
|
|
||||||
const areYearsEqual =
|
|
||||||
startDate.getFullYear() === endDate.getFullYear() || isNaN(startDate.getFullYear()) || isNaN(endDate.getFullYear());
|
|
||||||
|
|
||||||
const moduleStatus = MODULE_STATUS.find((status) => status.value === moduleDetails.status);
|
const moduleStatus = MODULE_STATUS.find((status) => status.value === moduleDetails.status);
|
||||||
|
|
||||||
const issueCount =
|
const issueCount =
|
||||||
@ -400,19 +392,17 @@ export const ModuleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
<div className="relative flex w-1/2 items-center rounded-sm">
|
<div className="relative flex w-1/2 items-center rounded-sm">
|
||||||
<Popover className="flex h-full w-full items-center justify-center rounded-lg">
|
<Popover className="flex h-full w-full items-center justify-center rounded-lg">
|
||||||
<Popover.Button
|
<Popover.Button
|
||||||
className={`text-sm font-medium text-custom-text-300 w-full rounded-sm cursor-pointer hover:bg-custom-background-80 ${
|
className={`w-full cursor-pointer rounded-sm text-sm font-medium text-custom-text-300 hover:bg-custom-background-80 ${
|
||||||
isEditingAllowed ? "cursor-pointer" : "cursor-not-allowed"
|
isEditingAllowed ? "cursor-pointer" : "cursor-not-allowed"
|
||||||
}`}
|
}`}
|
||||||
disabled={!isEditingAllowed}
|
disabled={!isEditingAllowed}
|
||||||
>
|
>
|
||||||
<span
|
<span
|
||||||
className={`group flex w-full items-center justify-between gap-2 py-1 px-1.5 text-sm ${
|
className={`group flex w-full items-center justify-between gap-2 px-1.5 py-1 text-sm ${
|
||||||
watch("start_date") ? "" : "text-custom-text-400"
|
watch("start_date") ? "" : "text-custom-text-400"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
{areYearsEqual
|
{renderFormattedDate(startDate) ?? "No date selected"}
|
||||||
? renderShortDate(startDate, "No date selected")
|
|
||||||
: renderShortMonthDate(startDate, "No date selected")}
|
|
||||||
</span>
|
</span>
|
||||||
</Popover.Button>
|
</Popover.Button>
|
||||||
|
|
||||||
@ -453,19 +443,17 @@ export const ModuleDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
<Popover className="flex h-full w-full items-center justify-center rounded-lg">
|
<Popover className="flex h-full w-full items-center justify-center rounded-lg">
|
||||||
<>
|
<>
|
||||||
<Popover.Button
|
<Popover.Button
|
||||||
className={`text-sm font-medium text-custom-text-300 w-full rounded-sm cursor-pointer hover:bg-custom-background-80 ${
|
className={`w-full cursor-pointer rounded-sm text-sm font-medium text-custom-text-300 hover:bg-custom-background-80 ${
|
||||||
isEditingAllowed ? "cursor-pointer" : "cursor-not-allowed"
|
isEditingAllowed ? "cursor-pointer" : "cursor-not-allowed"
|
||||||
}`}
|
}`}
|
||||||
disabled={!isEditingAllowed}
|
disabled={!isEditingAllowed}
|
||||||
>
|
>
|
||||||
<span
|
<span
|
||||||
className={`group flex w-full items-center justify-between gap-2 py-1 px-1.5 text-sm ${
|
className={`group flex w-full items-center justify-between gap-2 px-1.5 py-1 text-sm ${
|
||||||
watch("target_date") ? "" : "text-custom-text-400"
|
watch("target_date") ? "" : "text-custom-text-400"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
{areYearsEqual
|
{renderFormattedDate(endDate) ?? "No date selected"}
|
||||||
? renderShortDate(endDate, "No date selected")
|
|
||||||
: renderShortMonthDate(endDate, "No date selected")}
|
|
||||||
</span>
|
</span>
|
||||||
</Popover.Button>
|
</Popover.Button>
|
||||||
|
|
||||||
|
@ -11,11 +11,9 @@ import { snoozeOptions } from "constants/notification";
|
|||||||
// helper
|
// helper
|
||||||
import { replaceUnderscoreIfSnakeCase, truncateText, stripAndTruncateHTML } from "helpers/string.helper";
|
import { replaceUnderscoreIfSnakeCase, truncateText, stripAndTruncateHTML } from "helpers/string.helper";
|
||||||
import {
|
import {
|
||||||
formatDateDistance,
|
calculateTimeAgo,
|
||||||
render12HourFormatTime,
|
renderFormattedTime,
|
||||||
renderLongDateFormat,
|
renderFormattedDate,
|
||||||
renderShortDate,
|
|
||||||
renderShortDateWithYearFormat,
|
|
||||||
} from "helpers/date-time.helper";
|
} from "helpers/date-time.helper";
|
||||||
// type
|
// type
|
||||||
import type { IUserNotification } from "types";
|
import type { IUserNotification } from "types";
|
||||||
@ -112,7 +110,7 @@ export const NotificationCard: React.FC<NotificationCardProps> = (props) => {
|
|||||||
{notification.data.issue_activity.field !== "None" ? (
|
{notification.data.issue_activity.field !== "None" ? (
|
||||||
notification.data.issue_activity.field !== "comment" ? (
|
notification.data.issue_activity.field !== "comment" ? (
|
||||||
notification.data.issue_activity.field === "target_date" ? (
|
notification.data.issue_activity.field === "target_date" ? (
|
||||||
renderShortDateWithYearFormat(notification.data.issue_activity.new_value)
|
renderFormattedDate(notification.data.issue_activity.new_value)
|
||||||
) : notification.data.issue_activity.field === "attachment" ? (
|
) : notification.data.issue_activity.field === "attachment" ? (
|
||||||
"the issue"
|
"the issue"
|
||||||
) : notification.data.issue_activity.field === "description" ? (
|
) : notification.data.issue_activity.field === "description" ? (
|
||||||
@ -151,11 +149,11 @@ export const NotificationCard: React.FC<NotificationCardProps> = (props) => {
|
|||||||
<p className="flex flex-shrink-0 items-center justify-end gap-x-1 text-custom-text-300">
|
<p className="flex flex-shrink-0 items-center justify-end gap-x-1 text-custom-text-300">
|
||||||
<Clock className="h-4 w-4" />
|
<Clock className="h-4 w-4" />
|
||||||
<span>
|
<span>
|
||||||
Till {renderShortDate(notification.snoozed_till)}, {render12HourFormatTime(notification.snoozed_till)}
|
Till {renderFormattedDate(notification.snoozed_till)}, {renderFormattedTime(notification.snoozed_till, '12-hour')}
|
||||||
</span>
|
</span>
|
||||||
</p>
|
</p>
|
||||||
) : (
|
) : (
|
||||||
<p className="flex-shrink-0 text-custom-text-300">{formatDateDistance(notification.created_at)}</p>
|
<p className="flex-shrink-0 text-custom-text-300">{calculateTimeAgo(notification.created_at)}</p>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -233,7 +231,7 @@ export const NotificationCard: React.FC<NotificationCardProps> = (props) => {
|
|||||||
|
|
||||||
markSnoozeNotification(notification.id, item.value).then(() => {
|
markSnoozeNotification(notification.id, item.value).then(() => {
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
title: `Notification snoozed till ${renderLongDateFormat(item.value)}`,
|
title: `Notification snoozed till ${renderFormattedDate(item.value)}`,
|
||||||
type: "success",
|
type: "success",
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -3,8 +3,8 @@ import { useRouter } from "next/router";
|
|||||||
import { useForm, Controller } from "react-hook-form";
|
import { useForm, Controller } from "react-hook-form";
|
||||||
import { Transition, Dialog } from "@headlessui/react";
|
import { Transition, Dialog } from "@headlessui/react";
|
||||||
import { X } from "lucide-react";
|
import { X } from "lucide-react";
|
||||||
// date helper
|
// constants
|
||||||
import { getAllTimeIn30MinutesInterval } from "helpers/date-time.helper";
|
import { allTimeIn30MinutesInterval12HoursFormat } from "constants/notification";
|
||||||
// hooks
|
// hooks
|
||||||
import useToast from "hooks/use-toast";
|
import useToast from "hooks/use-toast";
|
||||||
// ui
|
// ui
|
||||||
@ -33,7 +33,7 @@ const defaultValues: FormValues = {
|
|||||||
period: "AM",
|
period: "AM",
|
||||||
};
|
};
|
||||||
|
|
||||||
const timeStamps = getAllTimeIn30MinutesInterval();
|
const timeStamps = allTimeIn30MinutesInterval12HoursFormat;
|
||||||
|
|
||||||
export const SnoozeNotificationModal: FC<SnoozeModalProps> = (props) => {
|
export const SnoozeNotificationModal: FC<SnoozeModalProps> = (props) => {
|
||||||
const { isOpen, onClose, notification, onSuccess, onSubmit: handleSubmitSnooze } = props;
|
const { isOpen, onClose, notification, onSuccess, onSubmit: handleSubmitSnooze } = props;
|
||||||
|
@ -19,7 +19,7 @@ import { useMobxStore } from "lib/mobx/store-provider";
|
|||||||
import useToast from "hooks/use-toast";
|
import useToast from "hooks/use-toast";
|
||||||
// helpers
|
// helpers
|
||||||
import { copyUrlToClipboard } from "helpers/string.helper";
|
import { copyUrlToClipboard } from "helpers/string.helper";
|
||||||
import { render24HourFormatTime, renderFormattedDate } from "helpers/date-time.helper";
|
import { renderFormattedTime, renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// ui
|
// ui
|
||||||
import { CustomMenu, Tooltip } from "@plane/ui";
|
import { CustomMenu, Tooltip } from "@plane/ui";
|
||||||
// components
|
// components
|
||||||
@ -194,19 +194,19 @@ export const PagesListItem: FC<IPagesListItem> = observer((props) => {
|
|||||||
<div className="flex items-center gap-2.5">
|
<div className="flex items-center gap-2.5">
|
||||||
{page.archived_at ? (
|
{page.archived_at ? (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
tooltipContent={`Archived at ${render24HourFormatTime(page.archived_at)} on ${renderFormattedDate(
|
tooltipContent={`Archived at ${renderFormattedTime(page.archived_at)} on ${renderFormattedDate(
|
||||||
page.archived_at
|
page.archived_at
|
||||||
)}`}
|
)}`}
|
||||||
>
|
>
|
||||||
<p className="text-sm text-custom-text-200">{render24HourFormatTime(page.archived_at)}</p>
|
<p className="text-sm text-custom-text-200">{renderFormattedTime(page.archived_at)}</p>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
) : (
|
) : (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
tooltipContent={`Last updated at ${render24HourFormatTime(
|
tooltipContent={`Last updated at ${renderFormattedTime(
|
||||||
page.updated_at
|
page.updated_at
|
||||||
)} on ${renderFormattedDate(page.updated_at)}`}
|
)} on ${renderFormattedDate(page.updated_at)}`}
|
||||||
>
|
>
|
||||||
<p className="text-sm text-custom-text-200">{render24HourFormatTime(page.updated_at)}</p>
|
<p className="text-sm text-custom-text-200">{renderFormattedTime(page.updated_at)}</p>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
)}
|
)}
|
||||||
{isEditingAllowed && (
|
{isEditingAllowed && (
|
||||||
|
@ -10,7 +10,7 @@ import { Loader } from "@plane/ui";
|
|||||||
// image
|
// image
|
||||||
import recentActivityEmptyState from "public/empty-state/recent_activity.svg";
|
import recentActivityEmptyState from "public/empty-state/recent_activity.svg";
|
||||||
// helpers
|
// helpers
|
||||||
import { timeAgo } from "helpers/date-time.helper";
|
import { calculateTimeAgo } from "helpers/date-time.helper";
|
||||||
// fetch-keys
|
// fetch-keys
|
||||||
import { USER_PROFILE_ACTIVITY } from "constants/fetch-keys";
|
import { USER_PROFILE_ACTIVITY } from "constants/fetch-keys";
|
||||||
|
|
||||||
@ -71,7 +71,7 @@ export const ProfileActivity = () => {
|
|||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
</p>
|
</p>
|
||||||
<p className="text-xs text-custom-text-200">{timeAgo(activity.created_at)}</p>
|
<p className="text-xs text-custom-text-200">{calculateTimeAgo(activity.created_at)}</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
))}
|
))}
|
||||||
|
@ -14,7 +14,7 @@ import { Loader, Tooltip } from "@plane/ui";
|
|||||||
// icons
|
// icons
|
||||||
import { ChevronDown, Pencil } from "lucide-react";
|
import { ChevronDown, Pencil } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderLongDetailDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
import { renderEmoji } from "helpers/emoji.helper";
|
import { renderEmoji } from "helpers/emoji.helper";
|
||||||
// fetch-keys
|
// fetch-keys
|
||||||
import { USER_PROFILE_PROJECT_SEGREGATION } from "constants/fetch-keys";
|
import { USER_PROFILE_PROJECT_SEGREGATION } from "constants/fetch-keys";
|
||||||
@ -48,7 +48,7 @@ export const ProfileSidebar = () => {
|
|||||||
const userDetails = [
|
const userDetails = [
|
||||||
{
|
{
|
||||||
label: "Joined on",
|
label: "Joined on",
|
||||||
value: renderLongDetailDateFormat(userProjectsData?.user_data.date_joined ?? ""),
|
value: renderFormattedDate(userProjectsData?.user_data.date_joined ?? ""),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Timezone",
|
label: "Timezone",
|
||||||
|
@ -8,7 +8,7 @@ import { Button, CustomSelect, Input, TextArea } from "@plane/ui";
|
|||||||
import { IProject, IWorkspace } from "types";
|
import { IProject, IWorkspace } from "types";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderEmoji } from "helpers/emoji.helper";
|
import { renderEmoji } from "helpers/emoji.helper";
|
||||||
import { renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// constants
|
// constants
|
||||||
import { NETWORK_CHOICES } from "constants/project";
|
import { NETWORK_CHOICES } from "constants/project";
|
||||||
// services
|
// services
|
||||||
@ -310,7 +310,7 @@ export const ProjectDetailsForm: FC<IProjectDetailsForm> = (props) => {
|
|||||||
{isSubmitting ? "Updating Project..." : "Update Project"}
|
{isSubmitting ? "Updating Project..." : "Update Project"}
|
||||||
</Button>
|
</Button>
|
||||||
<span className="text-sm italic text-custom-sidebar-text-400">
|
<span className="text-sm italic text-custom-sidebar-text-400">
|
||||||
Created on {renderShortDateWithYearFormat(project?.created_at)}
|
Created on {renderFormattedDate(project?.created_at)}
|
||||||
</span>
|
</span>
|
||||||
</>
|
</>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,12 +1,11 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
|
|
||||||
import { Popover, Transition } from "@headlessui/react";
|
import { Popover, Transition } from "@headlessui/react";
|
||||||
// react-datepicker
|
// react-datepicker
|
||||||
import DatePicker from "react-datepicker";
|
import DatePicker from "react-datepicker";
|
||||||
// icons
|
// icons
|
||||||
import { CalendarDays, X } from "lucide-react";
|
import { CalendarDays, X } from "lucide-react";
|
||||||
// import "react-datepicker/dist/react-datepicker.css";
|
// helpers
|
||||||
import { renderDateFormat, renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate, renderFormattedDate } from "helpers/date-time.helper";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
value: string | null;
|
value: string | null;
|
||||||
@ -25,7 +24,7 @@ export const DateSelect: React.FC<Props> = ({ value, onChange, label, minDate, m
|
|||||||
{value ? (
|
{value ? (
|
||||||
<>
|
<>
|
||||||
<CalendarDays className="h-3 w-3 flex-shrink-0" />
|
<CalendarDays className="h-3 w-3 flex-shrink-0" />
|
||||||
<span>{renderShortDateWithYearFormat(value)}</span>
|
<span>{renderFormattedDate(value)}</span>
|
||||||
<button onClick={() => onChange(null)}>
|
<button onClick={() => onChange(null)}>
|
||||||
<X className="h-3 w-3" />
|
<X className="h-3 w-3" />
|
||||||
</button>
|
</button>
|
||||||
@ -52,7 +51,7 @@ export const DateSelect: React.FC<Props> = ({ value, onChange, label, minDate, m
|
|||||||
selected={value ? new Date(value) : null}
|
selected={value ? new Date(value) : null}
|
||||||
onChange={(val) => {
|
onChange={(val) => {
|
||||||
if (!val) onChange("");
|
if (!val) onChange("");
|
||||||
else onChange(renderDateFormat(val));
|
else onChange(renderFormattedPayloadDate(val));
|
||||||
|
|
||||||
if (closeOnSelect) close();
|
if (closeOnSelect) close();
|
||||||
}}
|
}}
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
import DatePicker from "react-datepicker";
|
import DatePicker from "react-datepicker";
|
||||||
import "react-datepicker/dist/react-datepicker.css";
|
import "react-datepicker/dist/react-datepicker.css";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
renderAs?: "input" | "button";
|
renderAs?: "input" | "button";
|
||||||
@ -45,7 +45,7 @@ export const CustomDatePicker: React.FC<Props> = ({
|
|||||||
selected={value ? new Date(value) : null}
|
selected={value ? new Date(value) : null}
|
||||||
onChange={(val) => {
|
onChange={(val) => {
|
||||||
if (!val) onChange(null);
|
if (!val) onChange(null);
|
||||||
else onChange(renderDateFormat(val));
|
else onChange(renderFormattedPayloadDate(val));
|
||||||
}}
|
}}
|
||||||
onCalendarOpen={handleOnOpen}
|
onCalendarOpen={handleOnOpen}
|
||||||
onCalendarClose={handleOnClose}
|
onCalendarClose={handleOnClose}
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
|
|
||||||
import useSWR from "swr";
|
import useSWR from "swr";
|
||||||
|
|
||||||
// headless ui
|
// headless ui
|
||||||
import { Dialog, Transition } from "@headlessui/react";
|
import { Dialog, Transition } from "@headlessui/react";
|
||||||
// services
|
// services
|
||||||
@ -12,7 +10,7 @@ import { Loader } from "@plane/ui";
|
|||||||
// icons
|
// icons
|
||||||
import { X } from "lucide-react";
|
import { X } from "lucide-react";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderLongDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
@ -68,7 +66,7 @@ export const ProductUpdatesModal: React.FC<Props> = ({ isOpen, setIsOpen }) => {
|
|||||||
<span className="flex items-center rounded-full border border-custom-border-200 bg-custom-background-90 px-3 py-1.5 text-xs">
|
<span className="flex items-center rounded-full border border-custom-border-200 bg-custom-background-90 px-3 py-1.5 text-xs">
|
||||||
{item.tag_name}
|
{item.tag_name}
|
||||||
</span>
|
</span>
|
||||||
<span>{renderLongDateFormat(item.published_at)}</span>
|
<span>{renderFormattedDate(item.published_at)}</span>
|
||||||
{index === 0 && (
|
{index === 0 && (
|
||||||
<span className="flex items-center rounded-full border border-custom-border-200 bg-custom-primary px-3 py-1.5 text-xs text-white">
|
<span className="flex items-center rounded-full border border-custom-border-200 bg-custom-primary px-3 py-1.5 text-xs text-white">
|
||||||
New
|
New
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
import DatePicker from "react-datepicker";
|
import DatePicker from "react-datepicker";
|
||||||
import "react-datepicker/dist/react-datepicker.css";
|
import "react-datepicker/dist/react-datepicker.css";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
renderAs?: "input" | "button";
|
renderAs?: "input" | "button";
|
||||||
@ -38,7 +38,7 @@ export const CustomRangeDatePicker: React.FC<Props> = ({
|
|||||||
selected={value ? new Date(value) : null}
|
selected={value ? new Date(value) : null}
|
||||||
onChange={(val) => {
|
onChange={(val) => {
|
||||||
if (!val) onChange(null);
|
if (!val) onChange(null);
|
||||||
else onChange(renderDateFormat(val));
|
else onChange(renderFormattedPayloadDate(val));
|
||||||
}}
|
}}
|
||||||
className={`${
|
className={`${
|
||||||
renderAs === "input"
|
renderAs === "input"
|
||||||
|
@ -1,4 +1,6 @@
|
|||||||
import { renderDateFormat } from "helpers/date-time.helper";
|
// helpers
|
||||||
|
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
|
// types
|
||||||
import { IWebhook, IWorkspace } from "types";
|
import { IWebhook, IWorkspace } from "types";
|
||||||
|
|
||||||
export const getCurrentHookAsCSV = (
|
export const getCurrentHookAsCSV = (
|
||||||
@ -8,8 +10,8 @@ export const getCurrentHookAsCSV = (
|
|||||||
) => ({
|
) => ({
|
||||||
id: webhook?.id || "",
|
id: webhook?.id || "",
|
||||||
url: webhook?.url || "",
|
url: webhook?.url || "",
|
||||||
created_at: renderDateFormat(webhook?.created_at),
|
created_at: renderFormattedPayloadDate(webhook?.created_at || "") ?? "",
|
||||||
updated_at: renderDateFormat(webhook?.updated_at),
|
updated_at: renderFormattedPayloadDate(webhook?.updated_at || "") ?? "",
|
||||||
is_active: webhook?.is_active?.toString() || "",
|
is_active: webhook?.is_active?.toString() || "",
|
||||||
secret_key: secretKey || "",
|
secret_key: secretKey || "",
|
||||||
project: webhook?.project?.toString() || "",
|
project: webhook?.project?.toString() || "",
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
import { useEffect, useRef, useState } from "react";
|
import { useEffect, useRef, useState } from "react";
|
||||||
|
|
||||||
// ui
|
// ui
|
||||||
import { Tooltip } from "@plane/ui";
|
import { Tooltip } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderDateFormat, renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate, renderFormattedDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { IUserActivity } from "types";
|
import { IUserActivity } from "types";
|
||||||
// constants
|
// constants
|
||||||
@ -35,7 +34,7 @@ export const ActivityGraph: React.FC<Props> = ({ activities }) => {
|
|||||||
const date = new Date(year, month, 1);
|
const date = new Date(year, month, 1);
|
||||||
|
|
||||||
while (date.getMonth() === month && date < new Date()) {
|
while (date.getMonth() === month && date < new Date()) {
|
||||||
dates.push(renderDateFormat(new Date(date)));
|
dates.push(renderFormattedPayloadDate(new Date(date)) ?? "");
|
||||||
date.setDate(date.getDate() + 1);
|
date.setDate(date.getDate() + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -102,7 +101,7 @@ export const ActivityGraph: React.FC<Props> = ({ activities }) => {
|
|||||||
key={`${date}-${index}`}
|
key={`${date}-${index}`}
|
||||||
tooltipContent={`${
|
tooltipContent={`${
|
||||||
isActive ? isActive.activity_count : 0
|
isActive ? isActive.activity_count : 0
|
||||||
} activities on ${renderShortDateWithYearFormat(date)}`}
|
} activities on ${renderFormattedDate(date)}`}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className={`${date === "" ? "pointer-events-none opacity-0" : ""} h-4 w-4 rounded ${
|
className={`${date === "" ? "pointer-events-none opacity-0" : ""} h-4 w-4 rounded ${
|
||||||
|
@ -1,11 +1,10 @@
|
|||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
|
|
||||||
// icons
|
// icons
|
||||||
import { AlertTriangle } from "lucide-react";
|
import { AlertTriangle } from "lucide-react";
|
||||||
import { LayersIcon, Loader } from "@plane/ui";
|
import { LayersIcon, Loader } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
import { renderFormattedDate } from "helpers/date-time.helper";
|
||||||
import { truncateText } from "helpers/string.helper";
|
import { truncateText } from "helpers/string.helper";
|
||||||
// types
|
// types
|
||||||
import { IIssueLite } from "types";
|
import { IIssueLite } from "types";
|
||||||
@ -67,7 +66,7 @@ export const IssuesList: React.FC<Props> = ({ issues, type }) => {
|
|||||||
</h5>
|
</h5>
|
||||||
<h5 className="col-span-2">{truncateText(issue.name, 30)}</h5>
|
<h5 className="col-span-2">{truncateText(issue.name, 30)}</h5>
|
||||||
<h5 className="cursor-default">
|
<h5 className="cursor-default">
|
||||||
{renderShortDateWithYearFormat(new Date(date?.toString() ?? ""))}
|
{renderFormattedDate(new Date(date?.toString() ?? ""))}
|
||||||
</h5>
|
</h5>
|
||||||
</div>
|
</div>
|
||||||
</span>
|
</span>
|
||||||
|
@ -24,3 +24,34 @@ export const snoozeOptions = [
|
|||||||
value: null,
|
value: null,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
// Constant for all time values in 30 minutes interval in 12 hours format
|
||||||
|
export const allTimeIn30MinutesInterval12HoursFormat: Array<{
|
||||||
|
label: string;
|
||||||
|
value: string;
|
||||||
|
}> = [
|
||||||
|
{ label: "12:00", value: "12:00" },
|
||||||
|
{ label: "12:30", value: "12:30" },
|
||||||
|
{ label: "01:00", value: "01:00" },
|
||||||
|
{ label: "01:30", value: "01:30" },
|
||||||
|
{ label: "02:00", value: "02:00" },
|
||||||
|
{ label: "02:30", value: "02:30" },
|
||||||
|
{ label: "03:00", value: "03:00" },
|
||||||
|
{ label: "03:30", value: "03:30" },
|
||||||
|
{ label: "04:00", value: "04:00" },
|
||||||
|
{ label: "04:30", value: "04:30" },
|
||||||
|
{ label: "05:00", value: "05:00" },
|
||||||
|
{ label: "05:30", value: "05:30" },
|
||||||
|
{ label: "06:00", value: "06:00" },
|
||||||
|
{ label: "06:30", value: "06:30" },
|
||||||
|
{ label: "07:00", value: "07:00" },
|
||||||
|
{ label: "07:30", value: "07:30" },
|
||||||
|
{ label: "08:00", value: "08:00" },
|
||||||
|
{ label: "08:30", value: "08:30" },
|
||||||
|
{ label: "09:00", value: "09:00" },
|
||||||
|
{ label: "09:30", value: "09:30" },
|
||||||
|
{ label: "10:00", value: "10:00" },
|
||||||
|
{ label: "10:30", value: "10:30" },
|
||||||
|
{ label: "11:00", value: "11:00" },
|
||||||
|
{ label: "11:30", value: "11:30" },
|
||||||
|
];
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// helpers
|
// helpers
|
||||||
import { getWeekNumberOfDate, renderDateFormat } from "helpers/date-time.helper";
|
import { getWeekNumberOfDate, renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { ICalendarDate, ICalendarPayload } from "components/issues";
|
import { ICalendarDate, ICalendarPayload } from "components/issues";
|
||||||
|
|
||||||
@ -73,16 +73,18 @@ export const generateCalendarData = (currentStructure: ICalendarPayload | null,
|
|||||||
|
|
||||||
const date = new Date(year, month, dayNumber + 1);
|
const date = new Date(year, month, dayNumber + 1);
|
||||||
|
|
||||||
currentWeekObject[renderDateFormat(date)] = {
|
const formattedDatePayload = renderFormattedPayloadDate(date);
|
||||||
date,
|
if (formattedDatePayload)
|
||||||
year,
|
currentWeekObject[formattedDatePayload] = {
|
||||||
month,
|
date,
|
||||||
day: dayNumber + 1,
|
year,
|
||||||
week: weekNumber,
|
month,
|
||||||
is_current_month: date.getMonth() === month,
|
day: dayNumber + 1,
|
||||||
is_current_week: getWeekNumberOfDate(date) === getWeekNumberOfDate(new Date()),
|
week: weekNumber,
|
||||||
is_today: date.toDateString() === new Date().toDateString(),
|
is_current_month: date.getMonth() === month,
|
||||||
};
|
is_current_week: getWeekNumberOfDate(date) === getWeekNumberOfDate(new Date()),
|
||||||
|
is_today: date.toDateString() === new Date().toDateString(),
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
calendarData[`y-${year}`][`m-${month}`][`w-${weekNumber}`] = currentWeekObject;
|
calendarData[`y-${year}`][`m-${month}`][`w-${weekNumber}`] = currentWeekObject;
|
||||||
|
@ -1,384 +1,83 @@
|
|||||||
import { format } from "date-fns";
|
import { differenceInDays, format, formatDistanceToNow, isAfter, isValid, parseISO } from "date-fns";
|
||||||
|
|
||||||
export const addDays = ({ date, days }: { date: Date; days: number }): Date => {
|
// Format Date Helpers
|
||||||
date.setDate(date.getDate() + days);
|
/**
|
||||||
return date;
|
* @returns {string | null} formatted date in the format of MMM dd, yyyy
|
||||||
|
* @description Returns date in the formatted format
|
||||||
|
* @param {Date | string} date
|
||||||
|
* @example renderFormattedDate("2024-01-01") // Jan 01, 2024
|
||||||
|
*/
|
||||||
|
export const renderFormattedDate = (date: string | Date): string | null => {
|
||||||
|
if (!date) return null;
|
||||||
|
// Parse the date to check if it is valid
|
||||||
|
const parsedDate = new Date(date);
|
||||||
|
// Check if the parsed date is valid before formatting
|
||||||
|
if (!isValid(parsedDate)) return null; // Return null for invalid dates
|
||||||
|
// Format the date in format (MMM dd, yyyy)
|
||||||
|
const formattedDate = format(parsedDate, "MMM dd, yyyy");
|
||||||
|
return formattedDate;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const renderDateFormat = (date: string | Date | null | undefined, dayFirst: boolean = false) => {
|
/**
|
||||||
if (!date) return "N/A";
|
* @returns {string} formatted date in the format of MMM dd
|
||||||
|
* @description Returns date in the formatted format
|
||||||
var d = new Date(date),
|
* @param {string | Date} date
|
||||||
month = "" + (d.getMonth() + 1),
|
* @example renderShortDateFormat("2024-01-01") // Jan 01
|
||||||
day = "" + d.getDate(),
|
*/
|
||||||
year = d.getFullYear();
|
export const renderFormattedDateWithoutYear = (date: string | Date): string => {
|
||||||
|
if (!date) return "";
|
||||||
if (month.length < 2) month = "0" + month;
|
// Parse the date to check if it is valid
|
||||||
if (day.length < 2) day = "0" + day;
|
const parsedDate = new Date(date);
|
||||||
|
// Check if the parsed date is valid before formatting
|
||||||
return dayFirst ? [day, month, year].join("-") : [year, month, day].join("-");
|
if (!isValid(parsedDate)) return ""; // Return empty string for invalid dates
|
||||||
|
// Format the date in short format (MMM dd)
|
||||||
|
const formattedDate = format(parsedDate, "MMM dd");
|
||||||
|
return formattedDate;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const renderShortNumericDateFormat = (date: string | Date) =>
|
/**
|
||||||
new Date(date).toLocaleDateString("en-UK", {
|
* @returns {string | null} formatted date in the format of yyyy-mm-dd to be used in payload
|
||||||
day: "numeric",
|
* @description Returns date in the formatted format to be used in payload
|
||||||
month: "short",
|
* @param {Date | string} date
|
||||||
});
|
* @example renderFormattedPayloadDate("Jan 01, 20224") // "2024-01-01"
|
||||||
|
*/
|
||||||
export const renderLongDetailDateFormat = (date: string | Date) =>
|
export const renderFormattedPayloadDate = (date: Date | string): string | null => {
|
||||||
new Date(date).toLocaleDateString("en-UK", {
|
if (!date) return null;
|
||||||
day: "numeric",
|
// Parse the date to check if it is valid
|
||||||
month: "long",
|
const parsedDate = new Date(date);
|
||||||
year: "numeric",
|
// Check if the parsed date is valid before formatting
|
||||||
});
|
if (!isValid(parsedDate)) return null; // Return null for invalid dates
|
||||||
|
// Format the date in payload format (yyyy-mm-dd)
|
||||||
export const findHowManyDaysLeft = (date: string | Date) => {
|
const formattedDate = format(parsedDate, "yyyy-MM-dd");
|
||||||
const today = new Date();
|
return formattedDate;
|
||||||
const eventDate = new Date(date);
|
|
||||||
const timeDiff = Math.abs(eventDate.getTime() - today.getTime());
|
|
||||||
return Math.ceil(timeDiff / (1000 * 3600 * 24));
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getDatesInRange = (startDate: string | Date, endDate: string | Date) => {
|
// Format Time Helpers
|
||||||
startDate = new Date(startDate);
|
/**
|
||||||
endDate = new Date(endDate);
|
* @returns {string} formatted date in the format of hh:mm a or HH:mm
|
||||||
|
* @description Returns date in 12 hour format if in12HourFormat is true else 24 hour format
|
||||||
const date = new Date(startDate.getTime());
|
* @param {string | Date} date
|
||||||
const dates = [];
|
* @param {boolean} timeFormat (optional) // default 24 hour
|
||||||
|
* @example renderFormattedTime("2024-01-01 13:00:00") // 13:00
|
||||||
while (date <= endDate) {
|
* @example renderFormattedTime("2024-01-01 13:00:00", "12-hour") // 01:00 PM
|
||||||
dates.push(new Date(date));
|
*/
|
||||||
date.setDate(date.getDate() + 1);
|
export const renderFormattedTime = (date: string | Date, timeFormat: "12-hour" | "24-hour" = "24-hour"): string => {
|
||||||
}
|
|
||||||
|
|
||||||
return dates;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const timeAgo = (time: any) => {
|
|
||||||
switch (typeof time) {
|
|
||||||
case "number":
|
|
||||||
break;
|
|
||||||
case "string":
|
|
||||||
time = +new Date(time);
|
|
||||||
break;
|
|
||||||
case "object":
|
|
||||||
if (time.constructor === Date) time = time.getTime();
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
time = +new Date();
|
|
||||||
}
|
|
||||||
|
|
||||||
var time_formats = [
|
|
||||||
[60, "seconds", 1], // 60
|
|
||||||
[120, "1 minute ago", "1 minute from now"], // 60*2
|
|
||||||
[3600, "minutes", 60], // 60*60, 60
|
|
||||||
[7200, "1 hour ago", "1 hour from now"], // 60*60*2
|
|
||||||
[86400, "hours", 3600], // 60*60*24, 60*60
|
|
||||||
[172800, "Yesterday", "Tomorrow"], // 60*60*24*2
|
|
||||||
[604800, "days", 86400], // 60*60*24*7, 60*60*24
|
|
||||||
[1209600, "Last week", "Next week"], // 60*60*24*7*4*2
|
|
||||||
[2419200, "weeks", 604800], // 60*60*24*7*4, 60*60*24*7
|
|
||||||
[4838400, "Last month", "Next month"], // 60*60*24*7*4*2
|
|
||||||
[29030400, "months", 2419200], // 60*60*24*7*4*12, 60*60*24*7*4
|
|
||||||
[58060800, "Last year", "Next year"], // 60*60*24*7*4*12*2
|
|
||||||
[2903040000, "years", 29030400], // 60*60*24*7*4*12*100, 60*60*24*7*4*12
|
|
||||||
[5806080000, "Last century", "Next century"], // 60*60*24*7*4*12*100*2
|
|
||||||
[58060800000, "centuries", 2903040000], // 60*60*24*7*4*12*100*20, 60*60*24*7*4*12*100
|
|
||||||
];
|
|
||||||
|
|
||||||
var seconds = (+new Date() - time) / 1000,
|
|
||||||
token = "ago",
|
|
||||||
list_choice = 1;
|
|
||||||
|
|
||||||
if (seconds == 0) {
|
|
||||||
return "Just now";
|
|
||||||
}
|
|
||||||
if (seconds < 0) {
|
|
||||||
seconds = Math.abs(seconds);
|
|
||||||
token = "from now";
|
|
||||||
list_choice = 2;
|
|
||||||
}
|
|
||||||
var i = 0,
|
|
||||||
format: any[];
|
|
||||||
while ((format = time_formats[i++]))
|
|
||||||
if (seconds < format[0]) {
|
|
||||||
if (typeof format[2] == "string") return format[list_choice];
|
|
||||||
else return Math.floor(seconds / format[2]) + " " + format[1] + " " + token;
|
|
||||||
}
|
|
||||||
return time;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const formatDateDistance = (date: string | Date) => {
|
|
||||||
const today = new Date();
|
|
||||||
const eventDate = new Date(date);
|
|
||||||
const timeDiff = Math.abs(eventDate.getTime() - today.getTime());
|
|
||||||
const days = Math.ceil(timeDiff / (1000 * 3600 * 24));
|
|
||||||
|
|
||||||
if (days < 1) {
|
|
||||||
const hours = Math.ceil(timeDiff / (1000 * 3600));
|
|
||||||
if (hours < 1) {
|
|
||||||
const minutes = Math.ceil(timeDiff / (1000 * 60));
|
|
||||||
if (minutes < 1) {
|
|
||||||
return "Just now";
|
|
||||||
} else {
|
|
||||||
return `${minutes}m`;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return `${hours}h`;
|
|
||||||
}
|
|
||||||
} else if (days < 7) {
|
|
||||||
return `${days}d`;
|
|
||||||
} else if (days < 30) {
|
|
||||||
return `${Math.floor(days / 7)}w`;
|
|
||||||
} else if (days < 365) {
|
|
||||||
return `${Math.floor(days / 30)}m`;
|
|
||||||
} else {
|
|
||||||
return `${Math.floor(days / 365)}y`;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const formatLongDateDistance = (date: string | Date) => {
|
|
||||||
const today = new Date();
|
|
||||||
const eventDate = new Date(date);
|
|
||||||
const timeDiff = Math.abs(eventDate.getTime() - today.getTime());
|
|
||||||
const days = Math.ceil(timeDiff / (1000 * 3600 * 24));
|
|
||||||
|
|
||||||
if (days < 1) {
|
|
||||||
const hours = Math.ceil(timeDiff / (1000 * 3600));
|
|
||||||
if (hours < 1) {
|
|
||||||
const minutes = Math.ceil(timeDiff / (1000 * 60));
|
|
||||||
if (minutes < 1) {
|
|
||||||
return "Just now";
|
|
||||||
} else {
|
|
||||||
return `${minutes} minutes`;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return `${hours} hours`;
|
|
||||||
}
|
|
||||||
} else if (days < 7) {
|
|
||||||
if (days === 1) return `${days} day`;
|
|
||||||
return `${days} days`;
|
|
||||||
} else if (days < 30) {
|
|
||||||
if (Math.floor(days / 7) === 1) return `${Math.floor(days / 7)} week`;
|
|
||||||
return `${Math.floor(days / 7)} weeks`;
|
|
||||||
} else if (days < 365) {
|
|
||||||
if (Math.floor(days / 30) === 1) return `${Math.floor(days / 30)} month`;
|
|
||||||
return `${Math.floor(days / 30)} months`;
|
|
||||||
} else {
|
|
||||||
if (Math.floor(days / 365) === 1) return `${Math.floor(days / 365)} year`;
|
|
||||||
return `${Math.floor(days / 365)} years`;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const renderShortDateWithYearFormat = (date: string | Date, placeholder?: string) => {
|
|
||||||
if (!date || date === "") return null;
|
|
||||||
|
|
||||||
date = new Date(date);
|
|
||||||
|
|
||||||
const months = ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"];
|
|
||||||
const day = date.getDate();
|
|
||||||
const month = months[date.getMonth()];
|
|
||||||
const year = date.getFullYear();
|
|
||||||
|
|
||||||
return isNaN(date.getTime()) ? placeholder ?? "N/A" : ` ${month} ${day}, ${year}`;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const renderShortDate = (date: string | Date, placeholder?: string) => {
|
|
||||||
if (!date || date === "") return null;
|
|
||||||
|
|
||||||
date = new Date(date);
|
|
||||||
|
|
||||||
const months = ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"];
|
|
||||||
const day = date.getDate();
|
|
||||||
const month = months[date.getMonth()];
|
|
||||||
|
|
||||||
return isNaN(date.getTime()) ? placeholder ?? "N/A" : `${day} ${month}`;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const renderShortMonthDate = (date: string | Date, placeholder?: string) => {
|
|
||||||
if (!date || date === "") return null;
|
|
||||||
|
|
||||||
date = new Date(date);
|
|
||||||
|
|
||||||
const months = ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"];
|
|
||||||
const month = months[date.getMonth()];
|
|
||||||
const year = date.getFullYear();
|
|
||||||
|
|
||||||
return isNaN(date.getTime()) ? placeholder ?? "N/A" : `${month} ${year}`;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const render12HourFormatTime = (date: string | Date): string => {
|
|
||||||
if (!date || date === "") return "";
|
if (!date || date === "") return "";
|
||||||
|
// Parse the date to check if it is valid
|
||||||
date = new Date(date);
|
const parsedDate = new Date(date);
|
||||||
|
// Check if the parsed date is valid
|
||||||
let hours = date.getHours();
|
if (!isValid(parsedDate)) return ""; // Return empty string for invalid dates
|
||||||
const minutes = date.getMinutes();
|
// Format the date in 12 hour format if in12HourFormat is true
|
||||||
|
if (timeFormat === "12-hour") {
|
||||||
let period = "AM";
|
const formattedTime = format(parsedDate, "hh:mm a");
|
||||||
|
return formattedTime;
|
||||||
if (hours >= 12) {
|
|
||||||
period = "PM";
|
|
||||||
|
|
||||||
if (hours > 12) hours -= 12;
|
|
||||||
}
|
}
|
||||||
|
// Format the date in 24 hour format
|
||||||
return hours + ":" + (minutes < 10 ? `0${minutes}` : minutes) + " " + period;
|
const formattedTime = format(parsedDate, "HH:mm");
|
||||||
|
return formattedTime;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const render24HourFormatTime = (date: string | Date): string => {
|
// Date Difference Helpers
|
||||||
if (!date || date === "") return "";
|
|
||||||
|
|
||||||
date = new Date(date);
|
|
||||||
|
|
||||||
const hours = date.getHours();
|
|
||||||
let minutes: any = date.getMinutes();
|
|
||||||
|
|
||||||
// add leading zero to single digit minutes
|
|
||||||
if (minutes < 10) minutes = "0" + minutes;
|
|
||||||
|
|
||||||
return hours + ":" + minutes;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const isDateRangeValid = (startDate: string, endDate: string) => new Date(startDate) < new Date(endDate);
|
|
||||||
|
|
||||||
export const isDateGreaterThanToday = (dateStr: string) => {
|
|
||||||
const date = new Date(dateStr);
|
|
||||||
const today = new Date();
|
|
||||||
return date > today;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const renderLongDateFormat = (dateString: string | Date) => {
|
|
||||||
const date = new Date(dateString);
|
|
||||||
const day = date.getDate();
|
|
||||||
const year = date.getFullYear();
|
|
||||||
const monthNames = [
|
|
||||||
"January",
|
|
||||||
"February",
|
|
||||||
"March",
|
|
||||||
"April",
|
|
||||||
"May",
|
|
||||||
"June",
|
|
||||||
"July",
|
|
||||||
"August",
|
|
||||||
"September",
|
|
||||||
"October",
|
|
||||||
"November",
|
|
||||||
"December",
|
|
||||||
];
|
|
||||||
const monthIndex = date.getMonth();
|
|
||||||
const monthName = monthNames[monthIndex];
|
|
||||||
const suffixes = ["st", "nd", "rd", "th"];
|
|
||||||
let suffix = "";
|
|
||||||
if (day === 1 || day === 21 || day === 31) {
|
|
||||||
suffix = suffixes[0];
|
|
||||||
} else if (day === 2 || day === 22) {
|
|
||||||
suffix = suffixes[1];
|
|
||||||
} else if (day === 3 || day === 23) {
|
|
||||||
suffix = suffixes[2];
|
|
||||||
} else {
|
|
||||||
suffix = suffixes[3];
|
|
||||||
}
|
|
||||||
return `${day}${suffix} ${monthName} ${year}`;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @returns {Array} Array of time objects with label and value as keys
|
|
||||||
*/
|
|
||||||
|
|
||||||
export const getTimestampAfterCurrentTime = (): Array<{
|
|
||||||
label: string;
|
|
||||||
value: Date;
|
|
||||||
}> => {
|
|
||||||
const current = new Date();
|
|
||||||
const time = [];
|
|
||||||
for (let i = 0; i < 24; i++) {
|
|
||||||
const newTime = new Date(current.getTime() + i * 60 * 60 * 1000);
|
|
||||||
time.push({
|
|
||||||
label: newTime.toLocaleTimeString([], { hour: "2-digit", minute: "2-digit" }),
|
|
||||||
value: newTime,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
return time;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @returns {Array} Array of date objects with label and value as keys
|
|
||||||
* @description Returns an array of date objects starting from current date to 7 days after
|
|
||||||
*/
|
|
||||||
|
|
||||||
export const getDatesAfterCurrentDate = (): Array<{
|
|
||||||
label: string;
|
|
||||||
value: Date;
|
|
||||||
}> => {
|
|
||||||
const current = new Date();
|
|
||||||
const date = [];
|
|
||||||
for (let i = 0; i < 7; i++) {
|
|
||||||
const newDate = new Date(Math.round(current.getTime() / (30 * 60 * 1000)) * 30 * 60 * 1000);
|
|
||||||
date.push({
|
|
||||||
label: newDate.toLocaleDateString([], {
|
|
||||||
day: "numeric",
|
|
||||||
month: "long",
|
|
||||||
year: "numeric",
|
|
||||||
}),
|
|
||||||
value: newDate,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
return date;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @returns {boolean} true if date is valid
|
|
||||||
* @description Returns true if date is valid
|
|
||||||
* @param {string} date
|
|
||||||
* @example checkIfStringIsDate("2021-01-01") // true
|
|
||||||
* @example checkIfStringIsDate("2021-01-32") // false
|
|
||||||
*/
|
|
||||||
|
|
||||||
export const checkIfStringIsDate = (date: string): boolean => new Date(date).toString() !== "Invalid Date";
|
|
||||||
|
|
||||||
// return an array of dates starting from 12:00 to 23:30 with 30 minutes interval as dates
|
|
||||||
export const getDatesWith30MinutesInterval = (): Array<Date> => {
|
|
||||||
const dates = [];
|
|
||||||
const current = new Date();
|
|
||||||
for (let i = 0; i < 24; i++) {
|
|
||||||
const newDate = new Date(current.getTime() + i * 60 * 60 * 1000);
|
|
||||||
dates.push(newDate);
|
|
||||||
}
|
|
||||||
return dates;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getAllTimeIn30MinutesInterval = (): Array<{
|
|
||||||
label: string;
|
|
||||||
value: string;
|
|
||||||
}> => [
|
|
||||||
{ label: "12:00", value: "12:00" },
|
|
||||||
{ label: "12:30", value: "12:30" },
|
|
||||||
{ label: "01:00", value: "01:00" },
|
|
||||||
{ label: "01:30", value: "01:30" },
|
|
||||||
{ label: "02:00", value: "02:00" },
|
|
||||||
{ label: "02:30", value: "02:30" },
|
|
||||||
{ label: "03:00", value: "03:00" },
|
|
||||||
{ label: "03:30", value: "03:30" },
|
|
||||||
{ label: "04:00", value: "04:00" },
|
|
||||||
{ label: "04:30", value: "04:30" },
|
|
||||||
{ label: "05:00", value: "05:00" },
|
|
||||||
{ label: "05:30", value: "05:30" },
|
|
||||||
{ label: "06:00", value: "06:00" },
|
|
||||||
{ label: "06:30", value: "06:30" },
|
|
||||||
{ label: "07:00", value: "07:00" },
|
|
||||||
{ label: "07:30", value: "07:30" },
|
|
||||||
{ label: "08:00", value: "08:00" },
|
|
||||||
{ label: "08:30", value: "08:30" },
|
|
||||||
{ label: "09:00", value: "09:00" },
|
|
||||||
{ label: "09:30", value: "09:30" },
|
|
||||||
{ label: "10:00", value: "10:00" },
|
|
||||||
{ label: "10:30", value: "10:30" },
|
|
||||||
{ label: "11:00", value: "11:00" },
|
|
||||||
{ label: "11:30", value: "11:30" },
|
|
||||||
];
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @returns {number} total number of days in range
|
* @returns {number} total number of days in range
|
||||||
* @description Returns total number of days in range
|
* @description Returns total number of days in range
|
||||||
@ -387,25 +86,73 @@ export const getAllTimeIn30MinutesInterval = (): Array<{
|
|||||||
* @param {boolean} inclusive
|
* @param {boolean} inclusive
|
||||||
* @example checkIfStringIsDate("2021-01-01", "2021-01-08") // 8
|
* @example checkIfStringIsDate("2021-01-01", "2021-01-08") // 8
|
||||||
*/
|
*/
|
||||||
|
export const findTotalDaysInRange = (
|
||||||
export const findTotalDaysInRange = (startDate: Date | string, endDate: Date | string, inclusive: boolean): number => {
|
startDate: Date | string,
|
||||||
|
endDate: Date | string,
|
||||||
|
inclusive: boolean = true
|
||||||
|
): number => {
|
||||||
if (!startDate || !endDate) return 0;
|
if (!startDate || !endDate) return 0;
|
||||||
|
// Parse the dates to check if they are valid
|
||||||
startDate = new Date(startDate);
|
const parsedStartDate = new Date(startDate);
|
||||||
endDate = new Date(endDate);
|
const parsedEndDate = new Date(endDate);
|
||||||
|
// Check if the parsed dates are valid before calculating the difference
|
||||||
// find number of days between startDate and endDate
|
if (!isValid(parsedStartDate) || !isValid(parsedEndDate)) return 0; // Return 0 for invalid dates
|
||||||
const diffInTime = endDate.getTime() - startDate.getTime();
|
// Calculate the difference in days
|
||||||
const diffInDays = Math.floor(diffInTime / (1000 * 3600 * 24));
|
const diffInDays = differenceInDays(parsedEndDate, parsedStartDate);
|
||||||
|
// Return the difference in days based on inclusive flag
|
||||||
// if inclusive is true, add 1 to diffInDays
|
return inclusive ? diffInDays + 1 : diffInDays;
|
||||||
if (inclusive) return diffInDays + 1;
|
|
||||||
|
|
||||||
return diffInDays;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getUserTimeZoneFromWindow = () => Intl.DateTimeFormat().resolvedOptions().timeZone;
|
/**
|
||||||
|
* @returns {number} number of days left from today
|
||||||
|
* @description Returns number of days left from today
|
||||||
|
* @param {string | Date} date
|
||||||
|
* @param {boolean} inclusive (optional) // default true
|
||||||
|
* @example findHowManyDaysLeft("2024-01-01") // 3
|
||||||
|
*/
|
||||||
|
export const findHowManyDaysLeft = (date: string | Date, inclusive: boolean = true): number => {
|
||||||
|
if (!date) return 0;
|
||||||
|
// Pass the date to findTotalDaysInRange function to find the total number of days in range from today
|
||||||
|
return findTotalDaysInRange(new Date(), date, inclusive);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Time Difference Helpers
|
||||||
|
/**
|
||||||
|
* @returns {string} formatted date in the form of amount of time passed since the event happened
|
||||||
|
* @description Returns time passed since the event happened
|
||||||
|
* @param {string | Date} time
|
||||||
|
* @example calculateTimeAgo("2023-01-01") // 1 year ago
|
||||||
|
*/
|
||||||
|
export const calculateTimeAgo = (time: string | number | Date | null): string => {
|
||||||
|
if (!time) return "";
|
||||||
|
// Parse the time to check if it is valid
|
||||||
|
const parsedTime = typeof time === "string" || typeof time === "number" ? parseISO(String(time)) : time;
|
||||||
|
if (!parsedTime) return ""; // Return empty string for invalid dates
|
||||||
|
// Format the time in the form of amount of time passed since the event happened
|
||||||
|
const distance = formatDistanceToNow(parsedTime, { addSuffix: true });
|
||||||
|
return distance;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Date Validation Helpers
|
||||||
|
/**
|
||||||
|
* @returns {string} boolean value depending on whether the date is greater than today
|
||||||
|
* @description Returns boolean value depending on whether the date is greater than today
|
||||||
|
* @param {string} dateStr
|
||||||
|
* @example isDateGreaterThanToday("2024-01-01") // true
|
||||||
|
*/
|
||||||
|
export const isDateGreaterThanToday = (dateStr: string): boolean => {
|
||||||
|
// Return false if dateStr is not present
|
||||||
|
if (!dateStr) return false;
|
||||||
|
// Parse the date to check if it is valid
|
||||||
|
const date = parseISO(dateStr);
|
||||||
|
const today = new Date();
|
||||||
|
// Check if the parsed date is valid
|
||||||
|
if (!isValid(date)) return false; // Return false for invalid dates
|
||||||
|
// Return true if the date is greater than today
|
||||||
|
return isAfter(date, today);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Week Related Helpers
|
||||||
/**
|
/**
|
||||||
* @returns {number} week number of date
|
* @returns {number} week number of date
|
||||||
* @description Returns week number of date
|
* @description Returns week number of date
|
||||||
@ -414,73 +161,11 @@ export const getUserTimeZoneFromWindow = () => Intl.DateTimeFormat().resolvedOpt
|
|||||||
*/
|
*/
|
||||||
export const getWeekNumberOfDate = (date: Date): number => {
|
export const getWeekNumberOfDate = (date: Date): number => {
|
||||||
const currentDate = new Date(date);
|
const currentDate = new Date(date);
|
||||||
|
|
||||||
// Adjust the starting day to Sunday (0) instead of Monday (1)
|
// Adjust the starting day to Sunday (0) instead of Monday (1)
|
||||||
const startDate = new Date(currentDate.getFullYear(), 0, 1);
|
const startDate = new Date(currentDate.getFullYear(), 0, 1);
|
||||||
|
|
||||||
// Calculate the number of days between currentDate and startDate
|
// Calculate the number of days between currentDate and startDate
|
||||||
const days = Math.floor((currentDate.getTime() - startDate.getTime()) / (24 * 60 * 60 * 1000));
|
const days = Math.floor((currentDate.getTime() - startDate.getTime()) / (24 * 60 * 60 * 1000));
|
||||||
|
|
||||||
// Adjust the calculation for weekNumber
|
// Adjust the calculation for weekNumber
|
||||||
const weekNumber = Math.ceil((days + 1) / 7);
|
const weekNumber = Math.ceil((days + 1) / 7);
|
||||||
|
|
||||||
return weekNumber;
|
return weekNumber;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* @returns {Date} first date of week
|
|
||||||
* @description Returns week number of date
|
|
||||||
* @param {Date} date
|
|
||||||
* @example getFirstDateOfWeek(35, 2023) // 2023-08-27T00:00:00.000Z
|
|
||||||
*/
|
|
||||||
export const getFirstDateOfWeek = (date: Date): Date => {
|
|
||||||
const year = date.getFullYear();
|
|
||||||
const weekNumber = getWeekNumberOfDate(date);
|
|
||||||
|
|
||||||
const januaryFirst: Date = new Date(year, 0, 1); // January is month 0
|
|
||||||
const daysToAdd: number = (weekNumber - 1) * 7; // Subtract 1 from the week number since weeks are 0-indexed
|
|
||||||
const firstDateOfWeek: Date = new Date(januaryFirst);
|
|
||||||
firstDateOfWeek.setDate(januaryFirst.getDate() + daysToAdd);
|
|
||||||
|
|
||||||
// Adjust the date to Sunday (week start)
|
|
||||||
const dayOfWeek: number = firstDateOfWeek.getDay();
|
|
||||||
firstDateOfWeek.setDate(firstDateOfWeek.getDate() - dayOfWeek); // Move back to Sunday
|
|
||||||
|
|
||||||
return firstDateOfWeek;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @returns {string} formatted date in the format of MMM dd, yyyy
|
|
||||||
* @description Returns date in the formatted format
|
|
||||||
* @param {Date | string} date
|
|
||||||
* @example renderFormattedDate("2023-01-01") // Jan 01, 2023
|
|
||||||
*/
|
|
||||||
export const renderFormattedDate = (date: string | Date): string => {
|
|
||||||
if (!date) return "";
|
|
||||||
|
|
||||||
date = new Date(date);
|
|
||||||
|
|
||||||
const formattedDate = format(date, "MMM dd, yyyy");
|
|
||||||
|
|
||||||
return formattedDate;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @returns {string | null} formatted date in the format of yyyy-mm-dd to be used in payload
|
|
||||||
* @description Returns date in the formatted format to be used in payload
|
|
||||||
* @param {Date | string} date
|
|
||||||
* @example renderFormattedPayloadDate("2023-01-01") // "2023-01-01"
|
|
||||||
*/
|
|
||||||
export const renderFormattedPayloadDate = (date: Date | string): string | null => {
|
|
||||||
if (!date) return null;
|
|
||||||
|
|
||||||
var d = new Date(date),
|
|
||||||
month = "" + (d.getMonth() + 1),
|
|
||||||
day = "" + d.getDate(),
|
|
||||||
year = d.getFullYear();
|
|
||||||
|
|
||||||
if (month.length < 2) month = "0" + month;
|
|
||||||
if (day.length < 2) day = "0" + day;
|
|
||||||
|
|
||||||
return [year, month, day].join("-");
|
|
||||||
};
|
|
||||||
|
@ -20,7 +20,7 @@ import { Spinner } from "@plane/ui";
|
|||||||
// assets
|
// assets
|
||||||
import emptyPage from "public/empty-state/page.svg";
|
import emptyPage from "public/empty-state/page.svg";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { NextPageWithLayout } from "types/app";
|
import { NextPageWithLayout } from "types/app";
|
||||||
import { IPage, IIssue } from "types";
|
import { IPage, IIssue } from "types";
|
||||||
@ -279,7 +279,7 @@ const PageDetailsPage: NextPageWithLayout = observer(() => {
|
|||||||
mutatePageDetailsHelper(
|
mutatePageDetailsHelper(
|
||||||
pageService.archivePage(workspaceSlug.toString(), projectId.toString(), pageId.toString()),
|
pageService.archivePage(workspaceSlug.toString(), projectId.toString(), pageId.toString()),
|
||||||
{
|
{
|
||||||
archived_at: renderDateFormat(new Date()),
|
archived_at: renderFormattedPayloadDate(new Date()),
|
||||||
},
|
},
|
||||||
["description_html"],
|
["description_html"],
|
||||||
() =>
|
() =>
|
||||||
|
@ -15,7 +15,7 @@ import { ExternalLinkIcon, Loader } from "@plane/ui";
|
|||||||
// fetch-keys
|
// fetch-keys
|
||||||
import { USER_ACTIVITY } from "constants/fetch-keys";
|
import { USER_ACTIVITY } from "constants/fetch-keys";
|
||||||
// helper
|
// helper
|
||||||
import { timeAgo } from "helpers/date-time.helper";
|
import { calculateTimeAgo } from "helpers/date-time.helper";
|
||||||
// type
|
// type
|
||||||
import { NextPageWithLayout } from "types/app";
|
import { NextPageWithLayout } from "types/app";
|
||||||
|
|
||||||
@ -70,7 +70,7 @@ const ProfileActivityPage: NextPageWithLayout = () => {
|
|||||||
: activityItem.actor_detail.display_name}
|
: activityItem.actor_detail.display_name}
|
||||||
</div>
|
</div>
|
||||||
<p className="mt-0.5 text-xs text-custom-text-200">
|
<p className="mt-0.5 text-xs text-custom-text-200">
|
||||||
Commented {timeAgo(activityItem.created_at)}
|
Commented {calculateTimeAgo(activityItem.created_at)}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<div className="issue-comments-section p-0">
|
<div className="issue-comments-section p-0">
|
||||||
@ -165,7 +165,7 @@ const ProfileActivityPage: NextPageWithLayout = () => {
|
|||||||
<div className="flex gap-1 truncate">
|
<div className="flex gap-1 truncate">
|
||||||
{message}{" "}
|
{message}{" "}
|
||||||
<span className="flex-shrink-0 whitespace-nowrap">
|
<span className="flex-shrink-0 whitespace-nowrap">
|
||||||
{timeAgo(activityItem.created_at)}
|
{calculateTimeAgo(activityItem.created_at)}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,11 +1,10 @@
|
|||||||
import { observable, action, makeObservable, runInAction, computed } from "mobx";
|
import { observable, action, makeObservable, runInAction, computed } from "mobx";
|
||||||
|
|
||||||
// helpers
|
// helpers
|
||||||
import { generateCalendarData } from "helpers/calendar.helper";
|
import { generateCalendarData } from "helpers/calendar.helper";
|
||||||
|
import { getWeekNumberOfDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { RootStore } from "./root";
|
import { RootStore } from "./root";
|
||||||
import { ICalendarPayload, ICalendarWeek } from "components/issues";
|
import { ICalendarPayload, ICalendarWeek } from "components/issues";
|
||||||
import { getWeekNumberOfDate } from "helpers/date-time.helper";
|
|
||||||
|
|
||||||
export interface ICalendarStore {
|
export interface ICalendarStore {
|
||||||
calendarFilters: {
|
calendarFilters: {
|
||||||
|
@ -10,7 +10,7 @@ import { IIssueResponse } from "../types";
|
|||||||
// constants
|
// constants
|
||||||
import { ISSUE_PRIORITIES, ISSUE_STATE_GROUPS } from "constants/issue";
|
import { ISSUE_PRIORITIES, ISSUE_STATE_GROUPS } from "constants/issue";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
|
|
||||||
export interface IIssueBaseStore {
|
export interface IIssueBaseStore {
|
||||||
groupedIssues(
|
groupedIssues(
|
||||||
@ -201,7 +201,7 @@ export class IssueBaseStore implements IIssueBaseStore {
|
|||||||
if (Array.isArray(value)) {
|
if (Array.isArray(value)) {
|
||||||
if (value.length) return value;
|
if (value.length) return value;
|
||||||
else return ["None"];
|
else return ["None"];
|
||||||
} else if (isDate) return [renderDateFormat(value) || "None"];
|
} else if (isDate) return [renderFormattedPayloadDate(value ?? "") || "None"];
|
||||||
else return [value || "None"];
|
else return [value || "None"];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,7 +6,7 @@ import isThisWeek from "date-fns/isThisWeek";
|
|||||||
import { ProjectService } from "services/project";
|
import { ProjectService } from "services/project";
|
||||||
import { PageService } from "services/page.service";
|
import { PageService } from "services/page.service";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderDateFormat } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
// types
|
// types
|
||||||
import { RootStore } from "./root";
|
import { RootStore } from "./root";
|
||||||
import { IPage, IRecentPages } from "types";
|
import { IPage, IRecentPages } from "types";
|
||||||
@ -329,7 +329,7 @@ export class PageStore implements IPageStore {
|
|||||||
...this.archivedPages,
|
...this.archivedPages,
|
||||||
[projectId]: [
|
[projectId]: [
|
||||||
...this.archivedPages[projectId],
|
...this.archivedPages[projectId],
|
||||||
{ ...archivedPage, archived_at: renderDateFormat(new Date()) },
|
{ ...archivedPage, archived_at: renderFormattedPayloadDate(new Date()) },
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
Loading…
Reference in New Issue
Block a user