postiz/apps/frontend/src/components/launches/calendar.tsx

824 lines
27 KiB
TypeScript

'use client';
import React, {
FC,
Fragment,
memo,
useCallback,
useEffect,
useMemo,
useState,
} from 'react';
import {
CalendarContext,
Integrations,
useCalendar,
} from '@gitroom/frontend/components/launches/calendar.context';
import dayjs from 'dayjs';
import { useModals } from '@mantine/modals';
import { AddEditModal } from '@gitroom/frontend/components/launches/add.edit.model';
import clsx from 'clsx';
import { useFetch } from '@gitroom/helpers/utils/custom.fetch';
import { ExistingDataContextProvider } from '@gitroom/frontend/components/launches/helpers/use.existing.data';
import { useDrag, useDrop } from 'react-dnd';
import { Integration, Post, State } from '@prisma/client';
import { useAddProvider } from '@gitroom/frontend/components/launches/add.provider.component';
import { useToaster } from '@gitroom/react/toaster/toaster';
import { useUser } from '@gitroom/frontend/components/layout/user.context';
import { IntegrationContext } from '@gitroom/frontend/components/launches/helpers/use.integration';
import { PreviewPopup } from '@gitroom/frontend/components/marketplace/special.message';
import isSameOrAfter from 'dayjs/plugin/isSameOrAfter';
import isSameOrBefore from 'dayjs/plugin/isSameOrBefore';
import { groupBy, random, sortBy } from 'lodash';
import Image from 'next/image';
import { extend } from 'dayjs';
import { isUSCitizen } from './helpers/isuscitizen.utils';
import removeMd from 'remove-markdown';
import { useInterval } from '@mantine/hooks';
import { StatisticsModal } from '@gitroom/frontend/components/launches/statistics';
extend(isSameOrAfter);
extend(isSameOrBefore);
const convertTimeFormatBasedOnLocality = (time: number) => {
if (isUSCitizen()) {
return `${time === 12 ? 12 : time % 12}:00 ${time >= 12 ? 'PM' : 'AM'}`;
} else {
return `${time}:00`;
}
};
export const days = [
'Monday',
'Tuesday',
'Wednesday',
'Thursday',
'Friday',
'Saturday',
'Sunday',
];
export const hours = Array.from({ length: 24 }, (_, i) => i);
export const DayView = () => {
const calendar = useCalendar();
const { integrations, posts, currentYear, currentDay, currentWeek } =
calendar;
const options = useMemo(() => {
const createdPosts = posts.map((post) => ({
integration: [integrations.find((i) => i.id === post.integration.id)!],
image: post.integration.picture,
identifier: post.integration.providerIdentifier,
id: post.integration.id,
name: post.integration.name,
time: dayjs
.utc(post.publishDate)
.diff(dayjs.utc(post.publishDate).startOf('day'), 'minute'),
}));
return sortBy(
Object.values(
groupBy(
[
...createdPosts,
...integrations.flatMap((p) =>
p.time.flatMap((t) => ({
integration: p,
identifier: p.identifier,
name: p.name,
id: p.id,
image: p.picture,
time: t.time,
}))
),
],
(p: any) => p.time
)
),
(p) => p[0].time
);
}, [integrations, posts]);
return (
<div className="flex flex-col gap-[10px]">
{options.map((option) => (
<Fragment key={option[0].time}>
<div className="text-center text-[20px]">
{dayjs()
.utc()
.startOf('day')
.add(option[0].time, 'minute')
.local()
.format(isUSCitizen() ? 'hh:mm A' : 'HH:mm')}
</div>
<div
key={option[0].time}
className="bg-secondary min-h-[60px] border-[2px] border-secondary rounded-[10px] flex justify-center items-center gap-[10px] mb-[20px]"
>
<CalendarContext.Provider
value={{
...calendar,
integrations: option.flatMap((p) => p.integration),
}}
>
<CalendarColumn
getDate={dayjs()
.utc()
.year(currentYear)
.week(currentWeek)
.day(currentDay)
.startOf('day')
.add(option[0].time, 'minute')
.local()}
/>
</CalendarContext.Provider>
</div>
</Fragment>
))}
</div>
);
};
export const WeekView = () => {
const { currentYear, currentWeek } = useCalendar();
return (
<div className="flex flex-col h-screen overflow-hidden text-textColor flex-1">
<div className="flex-1">
<div className="grid grid-cols-8 bg-customColor31 gap-[1px] border-customColor31 border rounded-[10px]">
<div className="bg-customColor20 sticky top-0 z-10 bg-gray-900"></div>
{days.map((day, index) => (
<div
key={day}
className="sticky top-0 z-10 bg-customColor20 p-2 text-center"
>
<div>{day}</div>
</div>
))}
{hours.map((hour) => (
<Fragment key={hour}>
<div className="p-2 pr-4 bg-secondary text-center items-center justify-center flex">
{/* {hour.toString().padStart(2, '0')}:00 */}
{convertTimeFormatBasedOnLocality(hour)}
</div>
{days.map((day, indexDay) => (
<Fragment key={`${currentYear}-${currentWeek}-${day}-${hour}`}>
<div className="relative bg-secondary">
<CalendarColumn
getDate={dayjs()
.year(currentYear)
.week(currentWeek)
.day(indexDay + 1)
.hour(hour)
.startOf('hour')}
/>
</div>
</Fragment>
))}
</Fragment>
))}
</div>
</div>
</div>
);
};
export const MonthView = () => {
const { currentYear, currentMonth } = useCalendar();
const calendarDays = useMemo(() => {
const startOfMonth = dayjs(new Date(currentYear, currentMonth, 1));
// Calculate the day offset for Monday (isoWeekday() returns 1 for Monday)
const startDayOfWeek = startOfMonth.isoWeekday(); // 1 for Monday, 7 for Sunday
const daysBeforeMonth = startDayOfWeek - 1; // Days to show from the previous month
// Get the start date (Monday of the first week that includes this month)
const startDate = startOfMonth.subtract(daysBeforeMonth, 'day');
// Create an array to hold the calendar days (6 weeks * 7 days = 42 days max)
const calendarDays = [];
let currentDay = startDate;
for (let i = 0; i < 42; i++) {
let label = 'current-month';
if (currentDay.month() < currentMonth) label = 'previous-month';
if (currentDay.month() > currentMonth) label = 'next-month';
calendarDays.push({
day: currentDay,
label,
});
// Move to the next day
currentDay = currentDay.add(1, 'day');
}
return calendarDays;
}, [currentYear, currentMonth]);
return (
<div className="flex flex-col h-screen overflow-hidden text-textColor flex-1">
<div className="flex-1 flex">
<div className="grid grid-cols-7 grid-rows-[40px_auto] bg-customColor31 gap-[1px] border-customColor31 border rounded-[10px] flex-1">
{days.map((day) => (
<div
key={day}
className="sticky top-0 z-10 bg-customColor20 p-2 text-center"
>
<div>{day}</div>
</div>
))}
{calendarDays.map((date, index) => (
<div
key={index}
className="bg-secondary text-center items-center justify-center flex min-h-[100px]"
>
<CalendarColumn
getDate={dayjs(date.day).endOf('day')}
randomHour={true}
/>
</div>
))}
</div>
</div>
</div>
);
};
export const Calendar = () => {
const { display } = useCalendar();
return (
<>
{display === 'day' ? (
<DayView />
) : display === 'week' ? (
<WeekView />
) : (
<MonthView />
)}
</>
);
};
export const CalendarColumn: FC<{
getDate: dayjs.Dayjs;
randomHour?: boolean;
}> = memo((props) => {
const { getDate, randomHour } = props;
const [num, setNum] = useState(0);
const user = useUser();
const {
integrations,
posts,
trendings,
changeDate,
display,
reloadCalendarView,
} = useCalendar();
const toaster = useToaster();
const modal = useModals();
const fetch = useFetch();
const postList = useMemo(() => {
return posts.filter((post) => {
const pList = dayjs.utc(post.publishDate).local();
const check =
display === 'day'
? pList.format('YYYY-MM-DD HH:mm') ===
getDate.format('YYYY-MM-DD HH:mm')
: display === 'week'
? pList.isSameOrAfter(getDate.startOf('hour')) &&
pList.isBefore(getDate.endOf('hour'))
: pList.format('DD/MM/YYYY') === getDate.format('DD/MM/YYYY');
return check;
});
}, [posts, display, getDate]);
const [showAll, setShowAll] = useState(false);
const showAllFunc = useCallback(() => {
setShowAll(true);
}, []);
const showLessFunc = useCallback(() => {
setShowAll(false);
}, []);
const list = useMemo(() => {
if (showAll) {
return postList;
}
return postList.slice(0, 3);
}, [postList, showAll]);
const canBeTrending = useMemo(() => {
return !!trendings.find((trend) => {
return dayjs
.utc(trend)
.local()
.isBetween(getDate, getDate.add(10, 'minute'), 'minute', '[)');
});
}, [trendings]);
const isBeforeNow = useMemo(() => {
return getDate.startOf('hour').isBefore(dayjs().startOf('hour'));
}, [getDate, num]);
const { start, stop } = useInterval(
useCallback(() => {
if (isBeforeNow) {
return;
}
setNum(num + 1);
}, [isBeforeNow]),
random(120000, 150000)
);
useEffect(() => {
start();
return () => {
stop();
};
}, []);
const [{ canDrop }, drop] = useDrop(() => ({
accept: 'post',
drop: async (item: any) => {
if (isBeforeNow) return;
const { status } = await fetch(`/posts/${item.id}/date`, {
method: 'PUT',
body: JSON.stringify({
date: getDate.utc().format('YYYY-MM-DDTHH:mm:ss'),
}),
});
if (status !== 500) {
changeDate(item.id, getDate);
return;
}
toaster.show(
"Can't change date, remove post from publication",
'warning'
);
},
collect: (monitor) => ({
canDrop: isBeforeNow ? false : !!monitor.canDrop() && !!monitor.isOver(),
}),
}));
const getIntegration = useCallback(
async (post: Post & { integration: Integration }) => {
return (
await fetch(
`/integrations/${post.integration.id}?order=${post.submittedForOrderId}`,
{
method: 'GET',
}
)
).json();
},
[]
);
const previewPublication = useCallback(
async (postInfo: Post & { integration: Integration }) => {
const post = await (
await fetch(`/marketplace/posts/${postInfo.id}`)
).json();
const integration = await getIntegration(postInfo);
modal.openModal({
classNames: {
modal: 'bg-transparent text-textColor',
},
size: 'auto',
withCloseButton: false,
children: (
<IntegrationContext.Provider
value={{
allIntegrations: [],
date: dayjs(),
integration,
value: [],
}}
>
<PreviewPopup
providerId={post?.providerId!}
post={post}
postId={post.id}
/>
</IntegrationContext.Provider>
),
});
},
[]
);
const editPost = useCallback(
(post: Post & { integration: Integration }, isDuplicate?: boolean) =>
async () => {
if (user?.orgId === post.submittedForOrganizationId) {
return previewPublication(post);
}
const data = await (await fetch(`/posts/${post.id}`)).json();
const date = !isDuplicate
? null
: (await (await fetch('/posts/find-slot')).json()).date;
const publishDate = dayjs
.utc(date || data.posts[0].publishDate)
.local();
const ExistingData = !isDuplicate
? ExistingDataContextProvider
: Fragment;
modal.openModal({
closeOnClickOutside: false,
closeOnEscape: false,
withCloseButton: false,
classNames: {
modal: 'w-[100%] max-w-[1400px] bg-transparent text-textColor',
},
children: (
<ExistingData value={data}>
<AddEditModal
{...(isDuplicate
? {
onlyValues: data.posts.map(
({ image, settings, content }: any) => {
return {
image,
settings,
content,
};
}
),
}
: {})}
allIntegrations={integrations.map((p) => ({ ...p }))}
reopenModal={editPost(post)}
mutate={reloadCalendarView}
integrations={
isDuplicate
? integrations
: integrations
.slice(0)
.filter((f) => f.id === data.integration)
.map((p) => ({
...p,
picture: data.integrationPicture,
}))
}
date={publishDate}
/>
</ExistingData>
),
size: '80%',
title: ``,
});
},
[integrations]
);
const addModal = useCallback(() => {
modal.openModal({
closeOnClickOutside: false,
closeOnEscape: false,
withCloseButton: false,
classNames: {
modal: 'w-[100%] max-w-[1400px] bg-transparent text-textColor',
},
children: (
<AddEditModal
allIntegrations={integrations.map((p) => ({ ...p }))}
integrations={integrations.slice(0).map((p) => ({ ...p }))}
mutate={reloadCalendarView}
date={
randomHour ? getDate.hour(Math.floor(Math.random() * 24)) : getDate
}
reopenModal={() => ({})}
/>
),
size: '80%',
// title: `Adding posts for ${getDate.format('DD/MM/YYYY HH:mm')}`,
});
}, [integrations, getDate]);
const openStatistics = useCallback(
(id: string) => () => {
modal.openModal({
closeOnClickOutside: true,
closeOnEscape: true,
withCloseButton: false,
classNames: {
modal: 'w-[100%] max-w-[1400px] bg-transparent text-textColor',
},
children: <StatisticsModal postId={id} />,
size: '80%',
// title: `Adding posts for ${getDate.format('DD/MM/YYYY HH:mm')}`,
});
},
[]
);
const addProvider = useAddProvider();
return (
<div className="flex flex-col w-full min-h-full" ref={drop as any}>
{display === 'month' && (
<div className={clsx('pt-[5px]', isBeforeNow && 'bg-customColor23')}>
{getDate.date()}
</div>
)}
<div
className={clsx(
'relative flex flex-col flex-1 text-white',
canDrop && 'bg-white/80',
isBeforeNow && postList.length === 0 && 'cursor-not-allowed'
)}
>
<div
{...(canBeTrending
? {
'data-tooltip-id': 'tooltip',
'data-tooltip-content': 'Predicted GitHub Trending Change',
}
: {})}
className={clsx(
'flex-col text-[12px] pointer w-full flex scrollbar scrollbar-thumb-tableBorder scrollbar-track-secondary',
isBeforeNow ? 'bg-customColor23 flex-1' : 'cursor-pointer',
isBeforeNow && postList.length === 0 && 'col-calendar',
canBeTrending && 'bg-customColor24'
)}
>
{list.map((post) => (
<div
key={post.id}
className={clsx(
'text-textColor p-[2.5px] relative flex flex-col justify-center items-center'
)}
>
<div className="relative w-full flex flex-col items-center p-[2.5px] h-[66px]">
<CalendarItem
display={display as 'day' | 'week' | 'month'}
isBeforeNow={isBeforeNow}
date={getDate}
state={post.state}
statistics={openStatistics(post.id)}
editPost={editPost(post, false)}
duplicatePost={editPost(post, true)}
post={post}
integrations={integrations}
/>
</div>
</div>
))}
{!showAll && postList.length > 3 && (
<div
className="text-center hover:underline py-[5px] text-textColor"
onClick={showAllFunc}
>
+ Show more ({postList.length - 3})
</div>
)}
{showAll && postList.length > 3 && (
<div
className="text-center hover:underline py-[5px]"
onClick={showLessFunc}
>
- Show less
</div>
)}
</div>
{(display === 'day'
? !isBeforeNow && postList.length === 0
: !isBeforeNow) && (
<div
className="pb-[2.5px] px-[5px] flex-1 flex"
onClick={integrations.length ? addModal : addProvider}
>
<div
className={clsx(
display === ('month' as any)
? 'flex-1 min-h-[40px] w-full'
: !postList.length
? 'h-full w-full absolute left-0 top-0 p-[5px]'
: 'min-h-[40px] w-full',
'flex items-center justify-center cursor-pointer pb-[2.5px]'
)}
>
{display !== 'day' && (
<div
className={clsx(
'hover:before:content-["+"] w-full h-full text-seventh rounded-[10px] hover:border hover:border-seventh flex justify-center items-center'
)}
/>
)}
{display === 'day' && (
<div
className={`w-full h-full rounded-[10px] hover:border hover:border-seventh flex justify-center items-center gap-[20px] opacity-30 grayscale hover:grayscale-0 hover:opacity-100`}
>
{integrations.map((selectedIntegrations) => (
<div
className="relative"
key={selectedIntegrations.identifier}
>
<div
className={clsx(
'relative w-[34px] h-[34px] rounded-full flex justify-center items-center bg-fifth filter transition-all duration-500'
)}
>
<Image
src={selectedIntegrations.picture}
className="rounded-full"
alt={selectedIntegrations.identifier}
width={32}
height={32}
/>
{selectedIntegrations.identifier === 'youtube' ? (
<img
src="/icons/platforms/youtube.svg"
className="absolute z-10 -bottom-[5px] -right-[5px]"
width={20}
/>
) : (
<Image
src={`/icons/platforms/${selectedIntegrations.identifier}.png`}
className="rounded-full absolute z-10 -bottom-[5px] -right-[5px] border border-fifth"
alt={selectedIntegrations.identifier}
width={20}
height={20}
/>
)}
</div>
</div>
))}
</div>
)}
</div>
</div>
)}
</div>
</div>
);
});
const CalendarItem: FC<{
date: dayjs.Dayjs;
isBeforeNow: boolean;
editPost: () => void;
duplicatePost: () => void;
statistics: () => void;
integrations: Integrations[];
state: State;
display: 'day' | 'week' | 'month';
post: Post & { integration: Integration };
}> = memo((props) => {
const {
editPost,
statistics,
duplicatePost,
post,
date,
isBeforeNow,
state,
display,
} = props;
const preview = useCallback(() => {
window.open(`/p/` + post.id + '?share=true', '_blank');
}, [post]);
const [{ opacity }, dragRef] = useDrag(
() => ({
type: 'post',
item: { id: post.id, date },
collect: (monitor) => ({
opacity: monitor.isDragging() ? 0 : 1,
}),
}),
[]
);
return (
<div
// @ts-ignore
ref={dragRef}
className={clsx('w-full flex h-full flex-1 flex-col group', 'relative')}
style={{ opacity }}
>
<div className="text-white bg-forth text-[11px] h-[15px] w-full rounded-tr-[10px] rounded-tl-[10px] flex justify-center gap-[10px] px-[5px]">
<div
className="hidden group-hover:block hover:underline cursor-pointer"
onClick={duplicatePost}
>
<Duplicate />
</div>
<div
className="hidden group-hover:block hover:underline cursor-pointer"
onClick={preview}
>
<Preview />
</div>{' '}
<div
className="hidden group-hover:block hover:underline cursor-pointer"
onClick={statistics}
>
<Statistics />
</div>
</div>
<div
onClick={editPost}
className={clsx(
'gap-[5px] w-full flex h-full flex-1 rounded-br-[10px] rounded-bl-[10px] border border-seventh px-[5px] p-[2.5px]',
'relative',
isBeforeNow && '!grayscale'
)}
>
<div
className={clsx(
'relative min-w-[20px] h-[20px]',
display === 'day' ? 'h-[40px]' : 'h-[20px]'
)}
>
<img
className="w-[20px] h-[20px] rounded-full"
src={post.integration.picture!}
/>
<img
className="w-[12px] h-[12px] rounded-full absolute z-10 top-[10px] right-0 border border-fifth"
src={`/icons/platforms/${post.integration?.providerIdentifier}.png`}
/>
</div>
<div className="whitespace-nowrap line-clamp-2">
<div className="text-left">{state === 'DRAFT' ? 'Draft: ' : ''}</div>
<div className="w-full overflow-hidden overflow-ellipsis text-left">
{removeMd(post.content).replace(/\n/g, ' ')}
</div>
</div>
</div>
</div>
);
});
const Duplicate = () => {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
width="15"
height="15"
viewBox="0 0 32 32"
fill="none"
data-tooltip-id="tooltip"
data-tooltip-content="Duplicate Post"
>
<path
d="M27 5H9C8.46957 5 7.96086 5.21071 7.58579 5.58579C7.21071 5.96086 7 6.46957 7 7V9H5C4.46957 9 3.96086 9.21071 3.58579 9.58579C3.21071 9.96086 3 10.4696 3 11V25C3 25.5304 3.21071 26.0391 3.58579 26.4142C3.96086 26.7893 4.46957 27 5 27H23C23.5304 27 24.0391 26.7893 24.4142 26.4142C24.7893 26.0391 25 25.5304 25 25V23H27C27.5304 23 28.0391 22.7893 28.4142 22.4142C28.7893 22.0391 29 21.5304 29 21V7C29 6.46957 28.7893 5.96086 28.4142 5.58579C28.0391 5.21071 27.5304 5 27 5ZM23 11V13H5V11H23ZM23 25H5V15H23V25ZM27 21H25V11C25 10.4696 24.7893 9.96086 24.4142 9.58579C24.0391 9.21071 23.5304 9 23 9H9V7H27V21Z"
fill="white"
/>
</svg>
);
};
const Preview = () => {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
width="15"
height="15"
viewBox="0 0 32 32"
fill="none"
data-tooltip-id="tooltip"
data-tooltip-content="Preview Post"
>
<path
d="M30.9137 15.595C30.87 15.4963 29.8112 13.1475 27.4575 10.7937C24.3212 7.6575 20.36 6 16 6C11.64 6 7.67874 7.6575 4.54249 10.7937C2.18874 13.1475 1.12499 15.5 1.08624 15.595C1.02938 15.7229 1 15.8613 1 16.0012C1 16.1412 1.02938 16.2796 1.08624 16.4075C1.12999 16.5062 2.18874 18.8538 4.54249 21.2075C7.67874 24.3425 11.64 26 16 26C20.36 26 24.3212 24.3425 27.4575 21.2075C29.8112 18.8538 30.87 16.5062 30.9137 16.4075C30.9706 16.2796 31 16.1412 31 16.0012C31 15.8613 30.9706 15.7229 30.9137 15.595ZM16 24C12.1525 24 8.79124 22.6012 6.00874 19.8438C4.86704 18.7084 3.89572 17.4137 3.12499 16C3.89551 14.5862 4.86686 13.2915 6.00874 12.1562C8.79124 9.39875 12.1525 8 16 8C19.8475 8 23.2087 9.39875 25.9912 12.1562C27.1352 13.2912 28.1086 14.5859 28.8812 16C27.98 17.6825 24.0537 24 16 24ZM16 10C14.8133 10 13.6533 10.3519 12.6666 11.0112C11.6799 11.6705 10.9108 12.6075 10.4567 13.7039C10.0026 14.8003 9.88377 16.0067 10.1153 17.1705C10.3468 18.3344 10.9182 19.4035 11.7573 20.2426C12.5965 21.0818 13.6656 21.6532 14.8294 21.8847C15.9933 22.1162 17.1997 21.9974 18.2961 21.5433C19.3924 21.0892 20.3295 20.3201 20.9888 19.3334C21.6481 18.3467 22 17.1867 22 16C21.9983 14.4092 21.3657 12.884 20.2408 11.7592C19.1159 10.6343 17.5908 10.0017 16 10ZM16 20C15.2089 20 14.4355 19.7654 13.7777 19.3259C13.1199 18.8864 12.6072 18.2616 12.3045 17.5307C12.0017 16.7998 11.9225 15.9956 12.0768 15.2196C12.2312 14.4437 12.6122 13.731 13.1716 13.1716C13.731 12.6122 14.4437 12.2312 15.2196 12.0769C15.9956 11.9225 16.7998 12.0017 17.5307 12.3045C18.2616 12.6072 18.8863 13.1199 19.3259 13.7777C19.7654 14.4355 20 15.2089 20 16C20 17.0609 19.5786 18.0783 18.8284 18.8284C18.0783 19.5786 17.0609 20 16 20Z"
fill="white"
/>
</svg>
);
};
export const Statistics = () => {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
width="15"
height="15"
viewBox="0 0 32 32"
fill="none"
data-tooltip-id="tooltip"
data-tooltip-content="Post Statistics"
>
<path
d="M28 25H27V5C27 4.73478 26.8946 4.48043 26.7071 4.29289C26.5196 4.10536 26.2652 4 26 4H19C18.7348 4 18.4804 4.10536 18.2929 4.29289C18.1054 4.48043 18 4.73478 18 5V10H12C11.7348 10 11.4804 10.1054 11.2929 10.2929C11.1054 10.4804 11 10.7348 11 11V16H6C5.73478 16 5.48043 16.1054 5.29289 16.2929C5.10536 16.4804 5 16.7348 5 17V25H4C3.73478 25 3.48043 25.1054 3.29289 25.2929C3.10536 25.4804 3 25.7348 3 26C3 26.2652 3.10536 26.5196 3.29289 26.7071C3.48043 26.8946 3.73478 27 4 27H28C28.2652 27 28.5196 26.8946 28.7071 26.7071C28.8946 26.5196 29 26.2652 29 26C29 25.7348 28.8946 25.4804 28.7071 25.2929C28.5196 25.1054 28.2652 25 28 25ZM20 6H25V25H20V6ZM13 12H18V25H13V12ZM7 18H11V25H7V18Z"
fill="white"
/>
</svg>
);
};