Compare commits

...

16 Commits

Author SHA1 Message Date
Gutts-n
2bbf313489 Fixed error to remove anchor from document 2024-01-25 16:45:39 -03:00
Gutts-n
c26b76368d Fixed error to remove anchor from document 2024-01-25 16:43:47 -03:00
Gutts-n
058d23678a Added changeset to the PR 2024-01-25 16:16:49 -03:00
Gutts-n
540a08934c Changed behavior of the download data bucket viewer component 2024-01-25 16:10:22 -03:00
Leonardo Yuri Farias
7d010cfee4 Merge pull request #1078 from datopian/changeset-release/main
Version Packages
2024-01-24 17:23:14 -03:00
github-actions[bot]
dd79da1c6b Version Packages 2024-01-24 20:22:57 +00:00
Leonardo Yuri Farias
a58e2b81f7 Merge pull request #1077 from datopian/feature/download-loading-message
Created property to present a component while loading the download of the file and fixed download bug on pagination
2024-01-24 17:20:09 -03:00
Gutts-n
6d7acd27ed Created property to present a component while is loading the download of the file and fixed download bug on pagination 2024-01-24 17:15:14 -03:00
Leonardo Yuri Farias
7c30842c7d Merge pull request #1076 from datopian/changeset-release/main
Version Packages
2024-01-24 11:08:44 -03:00
github-actions[bot]
35ca1d6dfd Version Packages 2024-01-24 14:08:13 +00:00
Leonardo Yuri Farias
a7e90b64af Merge pull request #1075 from datopian/fix/download-button-presented-on-start-of-bucket-viewer
Fixed problem presenting the download component in the first load of …
2024-01-24 11:05:18 -03:00
Gutts-n
26dcffc279 Fixed problem presenting the download component in the first load of the bucket viewer 2024-01-24 11:03:08 -03:00
Leonardo Yuri Farias
d18e3dd486 Merge pull request #1074 from datopian/changeset-release/main
Version Packages
2024-01-23 16:54:19 -03:00
github-actions[bot]
8d7059acb4 Version Packages 2024-01-23 19:53:24 +00:00
Leonardo Yuri Farias
09d5324d4e Merge pull request #1073 from datopian/feature/search-and-pagination-for-bucket-viewer
Fixed bug on filter by startDate
2024-01-23 16:50:37 -03:00
Gutts-n
cf24042a91 Fixed bug on filter by startDate 2024-01-23 16:49:15 -03:00
6 changed files with 173 additions and 154 deletions

View File

@@ -0,0 +1,5 @@
---
'@portaljs/components': patch
---
Fixed error to remove anchor from document

View File

@@ -0,0 +1,5 @@
---
'@portaljs/components': patch
---
Changed the download behaviour of the bucket viewer component and removed loading component while downloading

View File

@@ -1,5 +1,23 @@
# @portaljs/components # @portaljs/components
## 0.5.7
### Patch Changes
- [#1077](https://github.com/datopian/portaljs/pull/1077) [`6d7acd27`](https://github.com/datopian/portaljs/commit/6d7acd27ed9299cbcc14eab906f2f0eb414656b8) Thanks [@Gutts-n](https://github.com/Gutts-n)! - Created property to present a component while is loading the download of the file and fixed download bug on pagination
## 0.5.6
### Patch Changes
- [#1075](https://github.com/datopian/portaljs/pull/1075) [`26dcffc2`](https://github.com/datopian/portaljs/commit/26dcffc279057f80a579134e862085ba042c06c3) Thanks [@Gutts-n](https://github.com/Gutts-n)! - Fixed problem presenting the download component in the first load of the bucket viewer
## 0.5.5
### Patch Changes
- [#1073](https://github.com/datopian/portaljs/pull/1073) [`cf24042a`](https://github.com/datopian/portaljs/commit/cf24042a910567e98eeb75ade42ce0149bdb62d1) Thanks [@Gutts-n](https://github.com/Gutts-n)! - Fixed filter by startDate error
## 0.5.4 ## 0.5.4
### Patch Changes ### Patch Changes

View File

@@ -1,6 +1,6 @@
{ {
"name": "@portaljs/components", "name": "@portaljs/components",
"version": "0.5.4", "version": "0.5.7",
"type": "module", "type": "module",
"description": "https://portaljs.org", "description": "https://portaljs.org",
"keywords": [ "keywords": [

View File

@@ -9,9 +9,11 @@ export interface BucketViewerFilterSearchedDataEvent {
export interface BucketViewerProps { export interface BucketViewerProps {
onLoadTotalNumberOfItems?: (total: number) => void; onLoadTotalNumberOfItems?: (total: number) => void;
domain: string; domain: string;
downloadConfig?: {
hoverOfTheFileComponent?: ReactNode;
};
suffix?: string; suffix?: string;
className?: string; className?: string;
downloadComponent?: ReactNode;
paginationConfig?: BucketViewerPaginationConfig; paginationConfig?: BucketViewerPaginationConfig;
filterState?: BucketViewerFilterSearchedDataEvent; filterState?: BucketViewerFilterSearchedDataEvent;
dataMapperFn: (rawData: Response) => Promise<BucketViewerData[]>; dataMapperFn: (rawData: Response) => Promise<BucketViewerData[]>;
@@ -34,20 +36,20 @@ export interface BucketViewerData {
export function BucketViewer({ export function BucketViewer({
domain, domain,
downloadComponent,
suffix, suffix,
dataMapperFn, dataMapperFn,
className, className,
filterState, filterState,
paginationConfig, paginationConfig,
onLoadTotalNumberOfItems downloadConfig,
onLoadTotalNumberOfItems,
}: BucketViewerProps) { }: BucketViewerProps) {
suffix = suffix ?? '/'; suffix = suffix ?? '/';
downloadComponent = downloadComponent ?? <></>;
const { hoverOfTheFileComponent } = downloadConfig ?? {};
const [isLoading, setIsLoading] = useState<boolean>(false); const [isLoading, setIsLoading] = useState<boolean>(false);
const [showDownloadComponentOnLine, setShowDownloadComponentOnLine] = useState(0); const [showDownloadComponentOnLine, setShowDownloadComponentOnLine] =
useState(-1);
const [currentPage, setCurrentPage] = useState<number>(0); const [currentPage, setCurrentPage] = useState<number>(0);
const [lastPage, setLastPage] = useState<number>(0); const [lastPage, setLastPage] = useState<number>(0);
const [bucketFiles, setBucketFiles] = useState<BucketViewerData[]>([]); const [bucketFiles, setBucketFiles] = useState<BucketViewerData[]>([]);
@@ -65,9 +67,8 @@ export function BucketViewer({
.finally(() => setIsLoading(false)); .finally(() => setIsLoading(false));
}, [domain, suffix]); }, [domain, suffix]);
useEffect( useEffect(() => {
() => { if (paginationConfig) {
if(paginationConfig) {
const startIndex = paginationConfig const startIndex = paginationConfig
? currentPage * paginationConfig.itemsPerPage ? currentPage * paginationConfig.itemsPerPage
: 0; : 0;
@@ -76,45 +77,49 @@ export function BucketViewer({
? startIndex + paginationConfig.itemsPerPage ? startIndex + paginationConfig.itemsPerPage
: 0; : 0;
setLastPage(Math.ceil(filteredData.length / paginationConfig.itemsPerPage) - 1); setLastPage(
Math.ceil(filteredData.length / paginationConfig.itemsPerPage) - 1
);
setPaginatedData(filteredData.slice(startIndex, endIndex)); setPaginatedData(filteredData.slice(startIndex, endIndex));
} }
}, }, [currentPage, filteredData]);
[currentPage, filteredData]
);
useEffect(
() => {
if(onLoadTotalNumberOfItems) onLoadTotalNumberOfItems(filteredData.length);
},
[filteredData]
)
useEffect(() => { useEffect(() => {
if(!filterState) return; if (onLoadTotalNumberOfItems) onLoadTotalNumberOfItems(filteredData.length);
}, [filteredData]);
if(filterState.startDate && filterState.startDate) { useEffect(() => {
setFilteredData(bucketFiles.filter(({ dateProps }) => if (!filterState) return;
if (filterState.startDate && filterState.endDate) {
setFilteredData(
bucketFiles.filter(({ dateProps }) =>
dateProps dateProps
? ? dateProps.date.getTime() >= filterState.startDate.getTime() &&
dateProps.date.getTime() >= filterState.startDate.getTime() dateProps.date.getTime() <= filterState.endDate.getTime()
&& dateProps.date.getTime() <= filterState.endDate.getTime()
: true : true
)); )
} else if(filterState.startDate) { );
setFilteredData(bucketFiles.filter(({ dateProps }) => } else if (filterState.startDate) {
dateProps ? dateProps.date.getTime() >= filterState.startDate.getTime() : true setFilteredData(
)); bucketFiles.filter(({ dateProps }) =>
} else if(filterState.endDate) { dateProps
setFilteredData(bucketFiles.filter(({ dateProps }) => ? dateProps.date.getTime() >= filterState.startDate.getTime()
dateProps ? dateProps.date.getTime() <= filterState.endDate.getTime() : true : true
)); )
);
} else if (filterState.endDate) {
setFilteredData(
bucketFiles.filter(({ dateProps }) =>
dateProps
? dateProps.date.getTime() <= filterState.endDate.getTime()
: true
)
);
} else { } else {
setFilteredData(bucketFiles); setFilteredData(bucketFiles);
} }
}, }, [filterState]);
[filterState]
)
return isLoading ? ( return isLoading ? (
<div className="w-full flex items-center justify-center h-[300px]"> <div className="w-full flex items-center justify-center h-[300px]">
@@ -122,30 +127,17 @@ export function BucketViewer({
</div> </div>
) : bucketFiles ? ( ) : bucketFiles ? (
<> <>
{...(paginationConfig && bucketFiles {...(paginationConfig && bucketFiles ? paginatedData : filteredData)?.map(
? paginatedData (data, i) => (
: filteredData
)?.map((data, i) => (
<ul <ul
onClick={() => { onClick={() => {
const anchorId = `download_anchor_${i}`; const a: HTMLAnchorElement = document.createElement('a');
const a: HTMLAnchorElement = a.href = data.downloadFileUri;
(document.getElementById(anchorId) as HTMLAnchorElement | null) ?? a.target = `_blank`;
document.createElement('a'); a.download = data.fileName;
a.id = anchorId;
if (a.download) a.click();
else {
setIsLoading(true);
fetch(data.downloadFileUri)
.then((res) => res.blob())
.then((res) => {
a.href = URL.createObjectURL(res);
a.download = res.name ?? data.fileName;
document.body.appendChild(a); document.body.appendChild(a);
a.click(); a.click();
}) document.body.removeChild(a);
.finally(() => setIsLoading(false));
}
}} }}
key={i} key={i}
onMouseEnter={() => setShowDownloadComponentOnLine(i)} onMouseEnter={() => setShowDownloadComponentOnLine(i)}
@@ -155,11 +147,12 @@ export function BucketViewer({
'mb-2 border-b-[2px] border-b-[red] hover:cursor-pointer' 'mb-2 border-b-[2px] border-b-[red] hover:cursor-pointer'
}`} }`}
> >
{ {hoverOfTheFileComponent && showDownloadComponentOnLine === i ? (
downloadComponent && showDownloadComponentOnLine === i hoverOfTheFileComponent
? downloadComponent ) : (
: <></> <></>
} )}
<div className="flex justify-between w-full items-center">
<div> <div>
<li>{data.fileName}</li> <li>{data.fileName}</li>
{data.dateProps && data.dateProps.dateFormatter ? ( {data.dateProps && data.dateProps.dateFormatter ? (
@@ -168,13 +161,16 @@ export function BucketViewer({
<></> <></>
)} )}
</div> </div>
</div>
</ul> </ul>
))} )
)}
{paginationConfig ? ( {paginationConfig ? (
<ul className={ <ul
className={
paginationConfig.containerClassName paginationConfig.containerClassName
? paginationConfig.containerClassName ? paginationConfig.containerClassName
: "flex justify-end gap-x-[0.5rem] w-full" : 'flex justify-end gap-x-[0.5rem] w-full'
} }
style={paginationConfig.containerStyles ?? {}} style={paginationConfig.containerStyles ?? {}}
> >
@@ -182,7 +178,10 @@ export function BucketViewer({
<button <button
className="hover:cursor-pointer hover:disabled:cursor-not-allowed" className="hover:cursor-pointer hover:disabled:cursor-not-allowed"
disabled={currentPage === 0} disabled={currentPage === 0}
onClick={() => setCurrentPage(0)}>First</button> onClick={() => setCurrentPage(0)}
>
First
</button>
</li> </li>
<li> <li>
<button <button
@@ -193,9 +192,7 @@ export function BucketViewer({
Previous Previous
</button> </button>
</li> </li>
<label> <label>{currentPage + 1}</label>
{currentPage + 1}
</label>
<li> <li>
<button <button

View File

@@ -1,6 +1,9 @@
import { type Meta, type StoryObj } from '@storybook/react'; import { type Meta, type StoryObj } from '@storybook/react';
import { BucketViewer, BucketViewerProps } from '../src/components/BucketViewer'; import {
BucketViewer,
BucketViewerProps,
} from '../src/components/BucketViewer';
import LoadingSpinner from '../src/components/LoadingSpinner'; import LoadingSpinner from '../src/components/LoadingSpinner';
// More on how to set up stories at: https://storybook.js.org/docs/react/writing-stories/introduction // More on how to set up stories at: https://storybook.js.org/docs/react/writing-stories/introduction
@@ -10,19 +13,16 @@ const meta: Meta = {
tags: ['autodocs'], tags: ['autodocs'],
argTypes: { argTypes: {
domain: { domain: {
description: description: 'Bucket domain URI',
'Bucket domain URI',
}, },
suffix: { suffix: {
description: description: 'Suffix of bucket domain',
'Suffix of bucket domain',
}, },
downloadComponent: { downloadConfig: {
description: description: `Bucket file download configuration`,
'Component to be displayed on hover of each bucket data',
}, },
filterState: { filterState: {
description: `State with values used to filter the bucket files` description: `State with values used to filter the bucket files`,
}, },
paginationConfig: { paginationConfig: {
description: `Configuration to show and stylise the pagination on the component`, description: `Configuration to show and stylise the pagination on the component`,
@@ -42,17 +42,15 @@ export const Normal: Story = {
suffix: '/', suffix: '/',
dataMapperFn: async (rawData: Response) => { dataMapperFn: async (rawData: Response) => {
const result = await rawData.json(); const result = await rawData.json();
return result.objects.map( return result.objects.map((e) => ({
e => ({
downloadFileUri: e.downloadLink, downloadFileUri: e.downloadLink,
fileName: e.key.replace(/^(\w+\/)/g, '') , fileName: e.key.replace(/^(\w+\/)/g, ''),
dateProps: { dateProps: {
date: new Date(e.uploaded), date: new Date(e.uploaded),
dateFormatter: (date) => date.toLocaleDateString() dateFormatter: (date) => date.toLocaleDateString(),
} },
}) }));
) },
}
}, },
}; };
@@ -62,21 +60,19 @@ export const WithPagination: Story = {
domain: 'https://ssen-smart-meter.datopian.workers.dev', domain: 'https://ssen-smart-meter.datopian.workers.dev',
suffix: '/', suffix: '/',
paginationConfig: { paginationConfig: {
itemsPerPage: 3 itemsPerPage: 3,
}, },
dataMapperFn: async (rawData: Response) => { dataMapperFn: async (rawData: Response) => {
const result = await rawData.json(); const result = await rawData.json();
return result.objects.map( return result.objects.map((e) => ({
e => ({
downloadFileUri: e.downloadLink, downloadFileUri: e.downloadLink,
fileName: e.key.replace(/^(\w+\/)/g, '') , fileName: e.key.replace(/^(\w+\/)/g, ''),
dateProps: { dateProps: {
date: new Date(e.uploaded), date: new Date(e.uploaded),
dateFormatter: (date) => date.toLocaleDateString() dateFormatter: (date) => date.toLocaleDateString(),
} },
}) }));
) },
}
}, },
}; };
@@ -85,19 +81,17 @@ export const WithComponentOnHoverOfEachBucketFile: Story = {
args: { args: {
domain: 'https://ssen-smart-meter.datopian.workers.dev', domain: 'https://ssen-smart-meter.datopian.workers.dev',
suffix: '/', suffix: '/',
downloadComponent: LoadingSpinner(), downloadConfig: { hoverOfTheFileComponent: `HOVER COMPONENT` },
dataMapperFn: async (rawData: Response) => { dataMapperFn: async (rawData: Response) => {
const result = await rawData.json(); const result = await rawData.json();
return result.objects.map( return result.objects.map((e) => ({
e => ({
downloadFileUri: e.downloadLink, downloadFileUri: e.downloadLink,
fileName: e.key.replace(/^(\w+\/)/g, '') , fileName: e.key.replace(/^(\w+\/)/g, ''),
dateProps: { dateProps: {
date: new Date(e.uploaded), date: new Date(e.uploaded),
dateFormatter: (date) => date.toLocaleDateString() dateFormatter: (date) => date.toLocaleDateString(),
} },
}) }));
) },
}
}, },
}; };