mirror of
https://github.com/status-im/dappconnect-sdks.git
synced 2025-01-12 09:44:43 +00:00
[website][status-web] Calendar (#410)
* feat: adds calendar component to components package * feat: add datepicker component to use in insights * feat: clean calendar code and fix spacing
This commit is contained in:
parent
f54c22abf0
commit
1866ca8c42
45
apps/website/src/components/datepicker/datepicker.tsx
Normal file
45
apps/website/src/components/datepicker/datepicker.tsx
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
import { Calendar } from '@status-im/components/src/calendar/calendar'
|
||||||
|
import { Popover } from '@status-im/components/src/popover'
|
||||||
|
import { EditIcon } from '@status-im/icons'
|
||||||
|
|
||||||
|
import { formatDate } from '../chart/utils/format-time'
|
||||||
|
|
||||||
|
import type { DateRange } from '@status-im/components/src/calendar/calendar'
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
selected?: DateRange
|
||||||
|
onSelect: (selected?: DateRange) => void
|
||||||
|
}
|
||||||
|
|
||||||
|
const DatePicker = (props: Props) => {
|
||||||
|
const { selected, onSelect } = props
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="sticky bottom-5 flex justify-center">
|
||||||
|
<Popover alignOffset={8} align="center" sideOffset={8}>
|
||||||
|
<button className="border-neutral-80/5 bg-blur-neutral-5/70 active inline-flex min-h-[30px] cursor-pointer items-center justify-center gap-2 rounded-xl border-solid pl-3 pr-2 uppercase text-neutral-100 backdrop-blur-sm">
|
||||||
|
<span className="text-blur-neutral-80/80 text-[13px] font-medium">
|
||||||
|
Filter between
|
||||||
|
</span>
|
||||||
|
<span className="text-[13px] font-medium text-neutral-100">
|
||||||
|
{`${selected?.from ? formatDate(selected.from) : 'Start Date'} → ${
|
||||||
|
selected?.to ? formatDate(selected.to) : 'End Date'
|
||||||
|
}`}
|
||||||
|
</span>
|
||||||
|
<div className="bg-neutral-80/5 h-full w-[1px]" />
|
||||||
|
<EditIcon size={20} color="$neutral-80-opa-40" />
|
||||||
|
</button>
|
||||||
|
<Popover.Content>
|
||||||
|
<Calendar
|
||||||
|
mode="range"
|
||||||
|
selected={selected}
|
||||||
|
onSelect={onSelect}
|
||||||
|
fixedWeeks
|
||||||
|
/>
|
||||||
|
</Popover.Content>
|
||||||
|
</Popover>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export { DatePicker }
|
@ -1,3 +1,5 @@
|
|||||||
|
import { useState } from 'react'
|
||||||
|
|
||||||
import { IconButton, Shadow, Tag, Text } from '@status-im/components'
|
import { IconButton, Shadow, Tag, Text } from '@status-im/components'
|
||||||
import {
|
import {
|
||||||
DoneIcon,
|
DoneIcon,
|
||||||
@ -7,9 +9,11 @@ import {
|
|||||||
SortIcon,
|
SortIcon,
|
||||||
} from '@status-im/icons'
|
} from '@status-im/icons'
|
||||||
|
|
||||||
|
import { DatePicker } from '@/components/datepicker/datepicker'
|
||||||
import { EpicOverview } from '@/components/epic-overview'
|
import { EpicOverview } from '@/components/epic-overview'
|
||||||
import { InsightsLayout } from '@/layouts/insights-layout'
|
import { InsightsLayout } from '@/layouts/insights-layout'
|
||||||
|
|
||||||
|
import type { DateRange } from '@status-im/components/src/calendar/calendar'
|
||||||
import type { Page } from 'next'
|
import type { Page } from 'next'
|
||||||
|
|
||||||
const epics = [
|
const epics = [
|
||||||
@ -27,6 +31,8 @@ const epics = [
|
|||||||
]
|
]
|
||||||
|
|
||||||
const EpicsPage: Page = () => {
|
const EpicsPage: Page = () => {
|
||||||
|
const [selectedDates, setSelectedDates] = useState<DateRange>()
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="space-y-4 p-10">
|
<div className="space-y-4 p-10">
|
||||||
<Text size={27} weight="semibold">
|
<Text size={27} weight="semibold">
|
||||||
@ -53,6 +59,8 @@ const EpicsPage: Page = () => {
|
|||||||
</Shadow>
|
</Shadow>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<DatePicker selected={selectedDates} onSelect={setSelectedDates} />
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -46,8 +46,10 @@
|
|||||||
"@tamagui/react-native-media-driver": "1.11.1",
|
"@tamagui/react-native-media-driver": "1.11.1",
|
||||||
"@tamagui/shorthands": "1.11.1",
|
"@tamagui/shorthands": "1.11.1",
|
||||||
"@tamagui/theme-base": "1.11.1",
|
"@tamagui/theme-base": "1.11.1",
|
||||||
|
"date-fns": "^2.30.0",
|
||||||
"expo-blur": "^12.2.2",
|
"expo-blur": "^12.2.2",
|
||||||
"expo-linear-gradient": "^12.1.2",
|
"expo-linear-gradient": "^12.1.2",
|
||||||
|
"react-day-picker": "^8.7.1",
|
||||||
"tamagui": "1.11.1",
|
"tamagui": "1.11.1",
|
||||||
"zustand": "^4.3.7"
|
"zustand": "^4.3.7"
|
||||||
},
|
},
|
||||||
|
354
packages/components/src/calendar/calendar.css
Normal file
354
packages/components/src/calendar/calendar.css
Normal file
@ -0,0 +1,354 @@
|
|||||||
|
.rdp {
|
||||||
|
--rdp-cell-size: 32px;
|
||||||
|
--rdp-caption-font-size: 15px;
|
||||||
|
--rdp-accent-color: #2a4af5;
|
||||||
|
--rdp-background-color: #e7edff;
|
||||||
|
--rdp-background-color-selected-secondary: #f5f6f8;
|
||||||
|
--rdp-hover-color: #f5f6f8;
|
||||||
|
--rdp-hover-color-darker: #f0f2f5;
|
||||||
|
--rdp-accent-color-dark: #223bc4;
|
||||||
|
--rdp-outline: 2px solid var(--rdp-accent-color); /* Outline border for focused elements */
|
||||||
|
--rdp-outline-selected: 3px solid var(--rdp-accent-color); /* Outline border for focused _and_ selected elements */
|
||||||
|
--rdp-text-color: #09101c;
|
||||||
|
color: var(--rdp-text-color);
|
||||||
|
font-family: Inter, sans-serif;
|
||||||
|
font-weight: 500;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Hide elements for devices that are not screen readers */
|
||||||
|
.rdp-vhidden {
|
||||||
|
box-sizing: border-box;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
background: transparent;
|
||||||
|
border: 0;
|
||||||
|
-moz-appearance: none;
|
||||||
|
-webkit-appearance: none;
|
||||||
|
appearance: none;
|
||||||
|
position: absolute !important;
|
||||||
|
top: 0;
|
||||||
|
width: 1px !important;
|
||||||
|
height: 1px !important;
|
||||||
|
padding: 0 !important;
|
||||||
|
overflow: hidden !important;
|
||||||
|
clip: rect(1px, 1px, 1px, 1px) !important;
|
||||||
|
border: 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Buttons */
|
||||||
|
.rdp-button_reset {
|
||||||
|
appearance: none;
|
||||||
|
position: relative;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
cursor: default;
|
||||||
|
color: inherit;
|
||||||
|
background: none;
|
||||||
|
font: inherit;
|
||||||
|
|
||||||
|
-moz-appearance: none;
|
||||||
|
-webkit-appearance: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-button_reset:focus-visible {
|
||||||
|
/* Make sure to reset outline only when :focus-visible is supported */
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-button {
|
||||||
|
border: 2px solid transparent;
|
||||||
|
font-size: 0.8125rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-button[disabled]:not(.rdp-day_selected) {
|
||||||
|
opacity: 0.25;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-button:not([disabled]) {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-button:focus-visible:not([disabled]) {
|
||||||
|
color: inherit;
|
||||||
|
background-color: var(--rdp-background-color);
|
||||||
|
border: var(--rdp-outline);
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-button:hover:not([disabled]):not(.rdp-day_selected) {
|
||||||
|
background-color: var(--rdp-hover-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-months {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-month {
|
||||||
|
margin: 0 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-month:first-child {
|
||||||
|
margin-left: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-month:last-child {
|
||||||
|
margin-right: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-table {
|
||||||
|
margin: 0;
|
||||||
|
max-width: calc(var(--rdp-cell-size) * 7);
|
||||||
|
border-collapse: separate;
|
||||||
|
border-spacing: 0 2px;
|
||||||
|
padding: 0 0.75rem 0.625rem 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-with_weeknumber .rdp-table {
|
||||||
|
max-width: calc(var(--rdp-cell-size) * 8);
|
||||||
|
border-collapse: collapse;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-caption {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 0;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-multiple_months .rdp-caption {
|
||||||
|
position: relative;
|
||||||
|
display: block;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-caption_dropdowns {
|
||||||
|
position: relative;
|
||||||
|
display: inline-flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-caption_label {
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
margin: 0;
|
||||||
|
white-space: nowrap;
|
||||||
|
color: currentColor;
|
||||||
|
font-family: inherit;
|
||||||
|
font-size: var(--rdp-caption-font-size);
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-nav {
|
||||||
|
white-space: nowrap;
|
||||||
|
padding: 0.375rem 0.375rem 0 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-multiple_months .rdp-caption_start .rdp-nav {
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
left: 0;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-multiple_months .rdp-caption_end .rdp-nav {
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
right: 0;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-nav_button {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
padding: 0.25em;
|
||||||
|
border-radius: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ---------- */
|
||||||
|
/* Dropdowns */
|
||||||
|
/* ---------- */
|
||||||
|
|
||||||
|
.rdp-dropdown_year,
|
||||||
|
.rdp-dropdown_month {
|
||||||
|
position: relative;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-dropdown {
|
||||||
|
appearance: none;
|
||||||
|
position: absolute;
|
||||||
|
z-index: 2;
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
cursor: inherit;
|
||||||
|
opacity: 0;
|
||||||
|
border: none;
|
||||||
|
background-color: transparent;
|
||||||
|
font-family: inherit;
|
||||||
|
font-size: inherit;
|
||||||
|
line-height: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-dropdown[disabled] {
|
||||||
|
opacity: unset;
|
||||||
|
color: unset;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-dropdown:focus-visible:not([disabled]) + .rdp-caption_label {
|
||||||
|
background-color: var(--rdp-background-color);
|
||||||
|
border: var(--rdp-outline);
|
||||||
|
border-radius: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-dropdown_icon {
|
||||||
|
margin: 0 0 0 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-head {
|
||||||
|
border: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-head_row,
|
||||||
|
.rdp-row {
|
||||||
|
height: 100%;
|
||||||
|
padding-bottom: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-head_cell {
|
||||||
|
vertical-align: middle;
|
||||||
|
font-size: 0.8125em;
|
||||||
|
font-weight: 500;
|
||||||
|
|
||||||
|
text-align: center;
|
||||||
|
height: var(--rdp-cell-size);
|
||||||
|
padding: 0;
|
||||||
|
|
||||||
|
line-height: 140%;
|
||||||
|
color: #647084;
|
||||||
|
letter-spacing: -0.003em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-tbody {
|
||||||
|
border: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-tfoot {
|
||||||
|
margin: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-cell {
|
||||||
|
width: var(--rdp-cell-size);
|
||||||
|
height: var(--rdp-cell-size);
|
||||||
|
text-align: center;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-cell_selected_start {
|
||||||
|
border-top-left-radius: 10px;
|
||||||
|
border-bottom-left-radius: 10px;
|
||||||
|
background-color: var(--rdp-background-color-selected-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-cell_selected_end {
|
||||||
|
border-top-right-radius: 10px;
|
||||||
|
border-bottom-right-radius: 10px;
|
||||||
|
background-color: var(--rdp-background-color-selected-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-cell_selected_range {
|
||||||
|
background-color: var(--rdp-background-color-selected-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-weeknumber {
|
||||||
|
font-size: 0.75em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-weeknumber,
|
||||||
|
.rdp-day {
|
||||||
|
display: flex;
|
||||||
|
overflow: hidden;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
box-sizing: border-box;
|
||||||
|
width: var(--rdp-cell-size);
|
||||||
|
max-width: var(--rdp-cell-size);
|
||||||
|
height: var(--rdp-cell-size);
|
||||||
|
margin: 0;
|
||||||
|
border: 2px solid transparent;
|
||||||
|
border-radius: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-day_today:not(.rdp-day_outside) {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
.rdp-day_today:not(.rdp-day_outside)::after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
bottom: 3px;
|
||||||
|
left: 50%;
|
||||||
|
width: 4px;
|
||||||
|
height: 2px;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
|
||||||
|
border-radius: 10px;
|
||||||
|
background-color: var(--rdp-accent-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-day_selected {
|
||||||
|
opacity: 1;
|
||||||
|
background-color: var(--rdp-accent-color);
|
||||||
|
color: var(--rdp-background-color);
|
||||||
|
transition: all 150ms ease-in-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-day_selected:focus-visible,
|
||||||
|
.rdp-day_selected:hover {
|
||||||
|
opacity: 1;
|
||||||
|
background-color: var(--rdp-accent-color-dark);
|
||||||
|
border-radius: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-day_outside {
|
||||||
|
opacity: 0.3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-day_selected:focus-visible {
|
||||||
|
/* Since the background is the same use again the outline */
|
||||||
|
outline: var(--rdp-outline);
|
||||||
|
outline-offset: 2px;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp[dir='rtl'] .rdp-day_range_start:not(.rdp-day_range_end) {
|
||||||
|
border-top-left-radius: 0;
|
||||||
|
border-bottom-left-radius: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp[dir='rtl'] .rdp-day_range_end:not(.rdp-day_range_start) {
|
||||||
|
border-top-right-radius: 0;
|
||||||
|
border-bottom-right-radius: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-day_range_end.rdp-day_range_start {
|
||||||
|
border-radius: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-day_range_middle {
|
||||||
|
color: var(--rdp-text-color);
|
||||||
|
background-color: var(--rdp-background-color-selected-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.rdp-day_range_middle:hover {
|
||||||
|
color: var(--rdp-text-color);
|
||||||
|
background-color: var(--rdp-hover-color-darker);
|
||||||
|
}
|
86
packages/components/src/calendar/calendar.stories.tsx
Normal file
86
packages/components/src/calendar/calendar.stories.tsx
Normal file
@ -0,0 +1,86 @@
|
|||||||
|
import { useState } from 'react'
|
||||||
|
|
||||||
|
import { Stack } from '@tamagui/core'
|
||||||
|
import { format } from 'date-fns'
|
||||||
|
|
||||||
|
import { Text } from '../text'
|
||||||
|
import { Calendar } from './calendar'
|
||||||
|
|
||||||
|
import type { Meta, StoryObj } from '@storybook/react'
|
||||||
|
import type { DateRange } from 'react-day-picker'
|
||||||
|
|
||||||
|
// More on how to set up stories at: https://storybook.js.org/docs/7.0/react/writing-stories/introduction
|
||||||
|
const meta: Meta<typeof Calendar> = {
|
||||||
|
component: Calendar,
|
||||||
|
args: {},
|
||||||
|
argTypes: {
|
||||||
|
disabled: {
|
||||||
|
defaultValue: false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// More on writing stories with args: https://storybook.js.org/docs/7.0/react/writing-stories/args
|
||||||
|
type Story = StoryObj<typeof Calendar>
|
||||||
|
|
||||||
|
const SingleCalendarWithHooks = () => {
|
||||||
|
const [selected, setSelected] = useState<Date>()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Calendar mode="single" selected={selected} onSelect={setSelected} />
|
||||||
|
<Stack pt="$2">
|
||||||
|
<Text size={15} color="$neutral-50">
|
||||||
|
Selected date: {selected && format(selected, 'do MMM yyyy')}
|
||||||
|
</Text>
|
||||||
|
</Stack>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const MultipleCalendarWithHooks = () => {
|
||||||
|
const [selected, setSelected] = useState<Date[]>()
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Calendar mode="multiple" selected={selected} onSelect={setSelected} />
|
||||||
|
<Stack pt="$2">
|
||||||
|
<Text size={15} color="$neutral-50">
|
||||||
|
Selected dates:{' '}
|
||||||
|
{selected?.map(date => format(date, 'do MMM yyyy')).join(', ')}
|
||||||
|
</Text>
|
||||||
|
</Stack>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const RangeCalendarWithHooks = () => {
|
||||||
|
const [selected, setSelected] = useState<DateRange>()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Calendar mode="range" selected={selected} onSelect={setSelected} />
|
||||||
|
<Stack pt="$2">
|
||||||
|
<Text size={15} color="$neutral-50">
|
||||||
|
Start date: {selected?.from && format(selected.from, 'do MMM yyyy')}
|
||||||
|
</Text>
|
||||||
|
<Text size={15} color="$neutral-50">
|
||||||
|
End Date: {selected?.to && format(selected.to, 'do MMM yyyy')}
|
||||||
|
</Text>
|
||||||
|
</Stack>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SinglePicker: Story = {
|
||||||
|
render: () => <SingleCalendarWithHooks />,
|
||||||
|
}
|
||||||
|
|
||||||
|
export const MultiplePicker: Story = {
|
||||||
|
render: () => <MultipleCalendarWithHooks />,
|
||||||
|
}
|
||||||
|
|
||||||
|
export const RangePicker: Story = {
|
||||||
|
render: () => <RangeCalendarWithHooks />,
|
||||||
|
}
|
||||||
|
|
||||||
|
export default meta
|
34
packages/components/src/calendar/calendar.tsx
Normal file
34
packages/components/src/calendar/calendar.tsx
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
import './calendar.css'
|
||||||
|
|
||||||
|
import { DayPicker } from 'react-day-picker'
|
||||||
|
|
||||||
|
import { Shadow } from '../shadow'
|
||||||
|
import { CustomCaption } from './components/caption'
|
||||||
|
import { CustomRow } from './components/row'
|
||||||
|
|
||||||
|
import type { DateRange, DayPickerProps } from 'react-day-picker'
|
||||||
|
|
||||||
|
type Props = DayPickerProps
|
||||||
|
|
||||||
|
const Calendar = (props: Props): JSX.Element => {
|
||||||
|
return (
|
||||||
|
<Shadow
|
||||||
|
borderWidth={1}
|
||||||
|
borderColor="$neutral-10"
|
||||||
|
borderRadius="$12"
|
||||||
|
display="inline-flex"
|
||||||
|
>
|
||||||
|
<DayPicker
|
||||||
|
{...props}
|
||||||
|
showOutsideDays
|
||||||
|
components={{
|
||||||
|
Row: CustomRow,
|
||||||
|
Caption: CustomCaption,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Shadow>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export { Calendar }
|
||||||
|
export type { Props as CalendarProps, DateRange }
|
44
packages/components/src/calendar/components/caption.tsx
Normal file
44
packages/components/src/calendar/components/caption.tsx
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
import { ChevronLeftIcon, ChevronRightIcon } from '@status-im/icons'
|
||||||
|
import { Stack } from '@tamagui/core'
|
||||||
|
import { format } from 'date-fns'
|
||||||
|
import { useNavigation } from 'react-day-picker'
|
||||||
|
|
||||||
|
import { IconButton } from '../../icon-button'
|
||||||
|
|
||||||
|
import type { CaptionProps } from 'react-day-picker'
|
||||||
|
|
||||||
|
const CustomCaption = (props: CaptionProps): JSX.Element => {
|
||||||
|
const { goToMonth, nextMonth, previousMonth } = useNavigation()
|
||||||
|
return (
|
||||||
|
<div className="rdp-nav">
|
||||||
|
<div className="rdp-caption">
|
||||||
|
<div
|
||||||
|
className="rdp-caption_label"
|
||||||
|
aria-live="polite"
|
||||||
|
role="presentation"
|
||||||
|
>
|
||||||
|
{format(props.displayMonth, 'MMM yyy')}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Stack flexDirection="row">
|
||||||
|
<IconButton
|
||||||
|
variant="ghost"
|
||||||
|
aria-label="Go to previous month"
|
||||||
|
icon={<ChevronLeftIcon size={20} />}
|
||||||
|
disabled={!previousMonth}
|
||||||
|
onPress={() => previousMonth && goToMonth(previousMonth)}
|
||||||
|
/>
|
||||||
|
<IconButton
|
||||||
|
variant="ghost"
|
||||||
|
aria-label="Go to next month"
|
||||||
|
icon={<ChevronRightIcon size={20} />}
|
||||||
|
disabled={!nextMonth}
|
||||||
|
onPress={() => nextMonth && goToMonth(nextMonth)}
|
||||||
|
/>
|
||||||
|
</Stack>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export { CustomCaption }
|
54
packages/components/src/calendar/components/row.tsx
Normal file
54
packages/components/src/calendar/components/row.tsx
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
import { getUnixTime, isEqual } from 'date-fns'
|
||||||
|
import { Day, useDayPicker } from 'react-day-picker'
|
||||||
|
|
||||||
|
import type { DateRange, RowProps } from 'react-day-picker'
|
||||||
|
|
||||||
|
const CustomRow = (props: RowProps): JSX.Element => {
|
||||||
|
const { styles, classNames, selected } = useDayPicker()
|
||||||
|
|
||||||
|
const castSelected = selected as DateRange
|
||||||
|
|
||||||
|
return (
|
||||||
|
<tr className={classNames.row} style={styles.row}>
|
||||||
|
{props.dates.map(date => {
|
||||||
|
const isSelectedStartDate =
|
||||||
|
castSelected?.from && isEqual(date, castSelected.from)
|
||||||
|
const isSelectedEndDate =
|
||||||
|
castSelected?.to && isEqual(date, castSelected.to)
|
||||||
|
|
||||||
|
const cellClassNames = () => {
|
||||||
|
if (isSelectedStartDate) {
|
||||||
|
return classNames.cell + ' ' + 'rdp-cell_selected_start'
|
||||||
|
}
|
||||||
|
if (isSelectedEndDate) {
|
||||||
|
return classNames.cell + ' ' + 'rdp-cell_selected_end'
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
castSelected?.from &&
|
||||||
|
castSelected?.to &&
|
||||||
|
date > castSelected.from &&
|
||||||
|
date < castSelected.to
|
||||||
|
) {
|
||||||
|
return classNames.cell + ' ' + 'rdp-cell_selected_range'
|
||||||
|
}
|
||||||
|
|
||||||
|
return classNames.cell
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<td
|
||||||
|
className={cellClassNames()}
|
||||||
|
style={styles.cell}
|
||||||
|
key={getUnixTime(date)}
|
||||||
|
role="presentation"
|
||||||
|
>
|
||||||
|
<Day displayMonth={props.displayMonth} date={date} />
|
||||||
|
</td>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</tr>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export { CustomRow }
|
1
packages/components/src/calendar/index.tsx
Normal file
1
packages/components/src/calendar/index.tsx
Normal file
@ -0,0 +1 @@
|
|||||||
|
export { Calendar } from './calendar'
|
@ -1,6 +1,7 @@
|
|||||||
export * from './anchor-actions'
|
export * from './anchor-actions'
|
||||||
export * from './avatar'
|
export * from './avatar'
|
||||||
export * from './button'
|
export * from './button'
|
||||||
|
export * from './calendar'
|
||||||
export * from './community'
|
export * from './community'
|
||||||
export * from './composer'
|
export * from './composer'
|
||||||
export * from './context-tag'
|
export * from './context-tag'
|
||||||
@ -21,6 +22,5 @@ export * from './tag'
|
|||||||
export * from './text'
|
export * from './text'
|
||||||
export * from './toast'
|
export * from './toast'
|
||||||
export * from './user-list'
|
export * from './user-list'
|
||||||
|
|
||||||
// eslint-disable-next-line simple-import-sort/exports
|
// eslint-disable-next-line simple-import-sort/exports
|
||||||
export { config } from './tamagui.config'
|
export { config } from './tamagui.config'
|
||||||
|
19
yarn.lock
19
yarn.lock
@ -1818,6 +1818,13 @@
|
|||||||
dependencies:
|
dependencies:
|
||||||
regenerator-runtime "^0.13.11"
|
regenerator-runtime "^0.13.11"
|
||||||
|
|
||||||
|
"@babel/runtime@^7.21.0":
|
||||||
|
version "7.21.5"
|
||||||
|
resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.21.5.tgz#8492dddda9644ae3bda3b45eabe87382caee7200"
|
||||||
|
integrity sha512-8jI69toZqqcsnqGGqwGS4Qb1VwLOEp4hz+CXPywcvjs60u3B4Pom/U/7rm4W8tMOYEB+E9wgD0mW1l3r8qlI9Q==
|
||||||
|
dependencies:
|
||||||
|
regenerator-runtime "^0.13.11"
|
||||||
|
|
||||||
"@babel/template@7.18.10", "@babel/template@^7.18.10":
|
"@babel/template@7.18.10", "@babel/template@^7.18.10":
|
||||||
version "7.18.10"
|
version "7.18.10"
|
||||||
resolved "https://registry.yarnpkg.com/@babel/template/-/template-7.18.10.tgz#6f9134835970d1dbf0835c0d100c9f38de0c5e71"
|
resolved "https://registry.yarnpkg.com/@babel/template/-/template-7.18.10.tgz#6f9134835970d1dbf0835c0d100c9f38de0c5e71"
|
||||||
@ -10246,6 +10253,13 @@ datastore-core@^8.0.1:
|
|||||||
it-take "^1.0.1"
|
it-take "^1.0.1"
|
||||||
uint8arrays "^3.0.0"
|
uint8arrays "^3.0.0"
|
||||||
|
|
||||||
|
date-fns@^2.30.0:
|
||||||
|
version "2.30.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/date-fns/-/date-fns-2.30.0.tgz#f367e644839ff57894ec6ac480de40cae4b0f4d0"
|
||||||
|
integrity sha512-fnULvOpxnC5/Vg3NCiWelDsLiUc9bRwAPs/+LfTLNvetFCtCTN+yQz15C/fs4AwX1R9K5GLtLfn8QW+dWisaAw==
|
||||||
|
dependencies:
|
||||||
|
"@babel/runtime" "^7.21.0"
|
||||||
|
|
||||||
dayjs@^1.8.15:
|
dayjs@^1.8.15:
|
||||||
version "1.11.7"
|
version "1.11.7"
|
||||||
resolved "https://registry.yarnpkg.com/dayjs/-/dayjs-1.11.7.tgz#4b296922642f70999544d1144a2c25730fce63e2"
|
resolved "https://registry.yarnpkg.com/dayjs/-/dayjs-1.11.7.tgz#4b296922642f70999544d1144a2c25730fce63e2"
|
||||||
@ -16867,6 +16881,11 @@ react-colorful@^5.1.2:
|
|||||||
resolved "https://registry.yarnpkg.com/react-colorful/-/react-colorful-5.6.1.tgz#7dc2aed2d7c72fac89694e834d179e32f3da563b"
|
resolved "https://registry.yarnpkg.com/react-colorful/-/react-colorful-5.6.1.tgz#7dc2aed2d7c72fac89694e834d179e32f3da563b"
|
||||||
integrity sha512-1exovf0uGTGyq5mXQT0zgQ80uvj2PCwvF8zY1RN9/vbJVSjSo3fsB/4L3ObbF7u70NduSiK4xu4Y6q1MHoUGEw==
|
integrity sha512-1exovf0uGTGyq5mXQT0zgQ80uvj2PCwvF8zY1RN9/vbJVSjSo3fsB/4L3ObbF7u70NduSiK4xu4Y6q1MHoUGEw==
|
||||||
|
|
||||||
|
react-day-picker@^8.7.1:
|
||||||
|
version "8.7.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/react-day-picker/-/react-day-picker-8.7.1.tgz#b8388c2afa69d4a6da4c5fde5323fae884acfe2f"
|
||||||
|
integrity sha512-Gv426AW8b151CZfh3aP5RUGztLwHB/EyJgWZ5iMgtzbFBkjHfG6Y66CIQFMWGLnYjsQ9DYSJRmJ5S0Pg5HWKjA==
|
||||||
|
|
||||||
react-devtools-core@4.24.0:
|
react-devtools-core@4.24.0:
|
||||||
version "4.24.0"
|
version "4.24.0"
|
||||||
resolved "https://registry.yarnpkg.com/react-devtools-core/-/react-devtools-core-4.24.0.tgz#7daa196bdc64f3626b3f54f2ff2b96f7c4fdf017"
|
resolved "https://registry.yarnpkg.com/react-devtools-core/-/react-devtools-core-4.24.0.tgz#7daa196bdc64f3626b3f54f2ff2b96f7c4fdf017"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user