Skip to content

Commit 6f49b99

Browse files
authored
Merge pull request #683 from codeforpdx/issue-682/create-documents-route
[Enhancement] - Issue-682/create documents route
2 parents 2b8c5da + 27e8cbc commit 6f49b99

13 files changed

+307
-122
lines changed

package-lock.json

+2-2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
{
22
"name": "pass",
33
"homepage": ".",
4-
"version": "1.0.1-alpha",
4+
"version": "1.1.0-alpha",
55
"description": "",
66
"scripts": {
77
"start": "concurrently --kill-others \"npm run podserver\" \"npm run dev\"",

src/AppRoutes.jsx

+2-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ import { useSession } from '@hooks';
77
import { CIVIC_FORM_LIST, FormLayout } from '@components/CivicProfileForms';
88
import { MESSAGE_PAGES_LIST, MessagesLayout } from '@components/Messages';
99
// Page Imports
10-
import { CivicProfile, Home, Contacts, Profile, Signup } from './pages';
10+
import { CivicProfile, Documents, Home, Contacts, Profile, Signup } from './pages';
1111

1212
const ProtectedRoute = ({ isLoggedIn, children }) =>
1313
isLoggedIn ? children ?? <Outlet /> : <Navigate to="/" replace />;
@@ -58,6 +58,7 @@ const AppRoutes = () => {
5858
))}
5959
</Route>
6060
<Route path="/profile" element={<Profile />} />
61+
<Route path="/documents" element={<Documents />} />
6162
{/* TODO: Remove blank Civic Profile page, ensure it directs Basic Information instead */}
6263
<Route path="/civic-profile" element={<CivicProfile />}>
6364
{CIVIC_FORM_LIST.map((formProps) => (

src/components/Documents/DocumentCard.jsx

+24-16
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
// React Imports
22
import React, { useState } from 'react';
3+
import { useLocation } from 'react-router-dom';
34
// Material UI Imports
45
import Box from '@mui/material/Box';
56
import Button from '@mui/material/Button';
@@ -41,6 +42,9 @@ import { truncateText, getTypeText } from '@utils';
4142
* @returns {React.JSX.Element} The DocumentCard component
4243
*/
4344
const DocumentCard = ({ document, onShare, onDelete, onPreview }) => {
45+
const location = useLocation();
46+
const profileWebId = decodeURIComponent(location.pathname.split('/')[2]);
47+
4448
const [anchorEl, setAnchorEl] = useState(null);
4549
const [openMenu, setOpenMenu] = useState(null);
4650

@@ -149,22 +153,26 @@ const DocumentCard = ({ document, onShare, onDelete, onPreview }) => {
149153
>
150154
Preview
151155
</MenuItem>
152-
<MenuItem
153-
component={Button}
154-
onClick={handleMenuItemClick(onShare, document)}
155-
startIcon={<ShareIcon sx={iconSize} />}
156-
sx={iconStyling}
157-
>
158-
Share
159-
</MenuItem>
160-
<MenuItem
161-
component={Button}
162-
onClick={handleMenuItemClick(onDelete, document)}
163-
startIcon={<DeleteOutlineOutlinedIcon sx={iconSize} />}
164-
sx={iconStyling}
165-
>
166-
Delete
167-
</MenuItem>
156+
{`${profileWebId}` === 'undefined' && (
157+
<MenuItem
158+
component={Button}
159+
onClick={handleMenuItemClick(onShare, document)}
160+
startIcon={<ShareIcon sx={iconSize} />}
161+
sx={iconStyling}
162+
>
163+
Share
164+
</MenuItem>
165+
)}
166+
{`${profileWebId}` === 'undefined' && (
167+
<MenuItem
168+
component={Button}
169+
onClick={handleMenuItemClick(onDelete, document)}
170+
startIcon={<DeleteOutlineOutlinedIcon sx={iconSize} />}
171+
sx={iconStyling}
172+
>
173+
Delete
174+
</MenuItem>
175+
)}
168176
</Menu>
169177
</Card>
170178
</Box>

src/components/Documents/DocumentsDesktop.jsx

+9
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
// React Import
22
import React from 'react';
3+
import { useLocation } from 'react-router-dom';
34
// Material UI Imports
45
import DeleteOutlineOutlinedIcon from '@mui/icons-material/DeleteOutlineOutlined';
56
import FileOpenIcon from '@mui/icons-material/FileOpen';
@@ -13,6 +14,8 @@ import {
1314
} from '@mui/x-data-grid';
1415
// Util Imports
1516
import { getTypeText } from '@utils';
17+
// Hooks Imports
18+
import { useSession } from '@hooks';
1619
// Theme Imports
1720
import theme from '../../theme';
1821

@@ -65,6 +68,10 @@ const CustomToolbar = () => (
6568
* @returns {React.JSX.Element} The DocumentsDesktop component
6669
*/
6770
const DocumentsDesktop = ({ documents, handlers }) => {
71+
const { session } = useSession();
72+
const location = useLocation();
73+
const profileWebId = decodeURIComponent(location.pathname.split('/')[2]);
74+
6875
const columnTitlesArray = [
6976
{ field: 'Name', minWidth: 120, flex: 1, headerAlign: 'center', align: 'center' },
7077
{ field: 'Type', minWidth: 120, flex: 1, headerAlign: 'center', align: 'center' },
@@ -109,6 +116,7 @@ const DocumentsDesktop = ({ documents, handlers }) => {
109116
onClick={() => handlers.onShare('document', name, type)}
110117
label="Share"
111118
data-testid={`share-button-${id}`}
119+
disabled={session.info.webId !== profileWebId}
112120
/>
113121
);
114122
}
@@ -130,6 +138,7 @@ const DocumentsDesktop = ({ documents, handlers }) => {
130138
onClick={() => handlers.onDelete(document)}
131139
label="Delete"
132140
data-testid={`delete-button-${document.id}`}
141+
disabled={session.info.webId !== profileWebId}
133142
/>
134143
);
135144
}

src/components/NavBar/NavMenu.jsx

+13
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import ContactsIcon from '@mui/icons-material/Contacts';
1010
import Divider from '@mui/material/Divider';
1111
import Drawer from '@mui/material/Drawer';
1212
import EmailIcon from '@mui/icons-material/Email';
13+
import InventoryIcon from '@mui/icons-material/Inventory';
1314
import LogoutIcon from '@mui/icons-material/Logout';
1415
import Menu from '@mui/material/Menu';
1516
import MenuItem from '@mui/material/MenuItem';
@@ -139,6 +140,18 @@ const NavMenu = ({
139140
Civic Profile
140141
</MenuItem>
141142
</Link>
143+
<Link
144+
to="/documents"
145+
style={{ textDecoration: 'none', color: theme.palette.primary.main }}
146+
>
147+
<MenuItem
148+
component={Button}
149+
startIcon={<InventoryIcon sx={iconSize} />}
150+
sx={iconStyling}
151+
>
152+
Documents
153+
</MenuItem>
154+
</Link>
142155
<Divider sx={{ my: '5px' }} />
143156
</div>
144157
)}

src/components/NavBar/NavbarLinks.jsx

+2-1
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,8 @@ const NavbarLinks = () => {
2323
// Array of current nav links for menus
2424
const routesArray = [
2525
{ label: 'Contacts', path: '/contacts' },
26-
{ label: 'Civic Profile', path: '/civic-profile/basic-info' }
26+
{ label: 'Civic Profile', path: '/civic-profile/basic-info' },
27+
{ label: 'Documents', path: '/documents' }
2728
];
2829

2930
return (

src/layouts/Layout.jsx

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
// React Imports
22
import React from 'react';
3+
import { useLocation } from 'react-router-dom';
34
// Inrupt Library Imports
45
import { useSession, useNotification } from '@hooks';
56
// Material UI Imports
@@ -14,6 +15,7 @@ import Main from './Main';
1415
const Layout = ({ ariaLabel, children }) => {
1516
const { session } = useSession();
1617
const { state } = useNotification();
18+
const location = useLocation();
1719

1820
return (
1921
<Box
@@ -24,7 +26,7 @@ const Layout = ({ ariaLabel, children }) => {
2426
>
2527
<NavBarSkipLink />
2628
<NavBar />
27-
{session.info.isLoggedIn && <Breadcrumbs />}
29+
{session.info.isLoggedIn && location.pathname !== '/profile' && <Breadcrumbs />}
2830
<Main>{children}</Main>
2931
{session.info.isLoggedIn && <InactivityMessage />}
3032
<Footer />

src/pages/Documents.jsx

+150
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,150 @@
1+
import React, { useContext, useEffect, useState } from 'react';
2+
import Box from '@mui/material/Box';
3+
import Button from '@mui/material/Button';
4+
import { useLocation } from 'react-router-dom';
5+
import { useTheme } from '@emotion/react';
6+
import { Container, useMediaQuery } from '@mui/material';
7+
import { ConfirmationModal, SetAclPermissionsModal, UploadDocumentModal } from '@components/Modals';
8+
import { useNotification, useSession } from '@hooks';
9+
import { DocumentListContext } from '@contexts';
10+
import { truncateText } from '@utils';
11+
import { DocumentTable } from '@components/Documents';
12+
13+
/**
14+
* Documents - Component that generates Documents Page for PASS
15+
*
16+
* @memberof Pages
17+
* @name Documents
18+
* @returns {React.JSX.Component} The Documents Page
19+
*/
20+
const Documents = () => {
21+
// Route related states
22+
const location = useLocation();
23+
if (location.pathname.split('/')[1] === 'contacts') {
24+
localStorage.setItem('restorePath', '/contacts');
25+
} else {
26+
localStorage.setItem('restorePath', '/profile');
27+
}
28+
const { setContact } = useContext(DocumentListContext);
29+
const theme = useTheme();
30+
const isSmallScreen = useMediaQuery(theme.breakpoints.down('sm'));
31+
32+
// Documents related states
33+
const { session } = useSession();
34+
const [showConfirmationModal, setShowConfirmationModal] = useState(false);
35+
const [processing, setProcessing] = useState(false);
36+
const { addNotification } = useNotification();
37+
const { removeDocument } = useContext(DocumentListContext);
38+
const [selectedDocToDelete, setSelectedDocToDelete] = useState(null);
39+
const [showAddDocModal, setShowAddDocModal] = useState(false);
40+
const [showAclPermissionModal, setShowAclPermissionModal] = useState(false);
41+
const [dataset, setDataset] = useState({
42+
modalType: '',
43+
docName: '',
44+
docType: ''
45+
});
46+
47+
useEffect(() => {
48+
setContact({
49+
familyName: '',
50+
givenName: '',
51+
podUrl: session.info.webId?.split('profile')[0],
52+
thingId: session.info.webId,
53+
webId: session.info.webId
54+
});
55+
}, [session]);
56+
57+
const handleSelectDeleteDoc = (document) => {
58+
setSelectedDocToDelete(document);
59+
setShowConfirmationModal(true);
60+
};
61+
62+
// Function for deleting documents
63+
const handleDeleteDoc = async () => {
64+
setProcessing(true);
65+
try {
66+
await removeDocument(selectedDocToDelete.name);
67+
addNotification('success', `${selectedDocToDelete?.name} deleted from the pod.`);
68+
} catch (e) {
69+
addNotification('error', `Document deletion failed. Reason: ${e.message}`);
70+
} finally {
71+
setShowConfirmationModal(false);
72+
setProcessing(false);
73+
}
74+
};
75+
76+
const handleAclPermissionsModal = (modalType, docName = '', docType = '') => {
77+
setDataset({
78+
modalType,
79+
docName,
80+
docType
81+
});
82+
setShowAclPermissionModal(true);
83+
};
84+
85+
const truncatedText = selectedDocToDelete?.name ? truncateText(selectedDocToDelete.name) : '';
86+
87+
return (
88+
<Container
89+
sx={{
90+
display: 'flex',
91+
flexDirection: 'column',
92+
alignItems: 'center',
93+
width: '100%'
94+
}}
95+
>
96+
<Box
97+
sx={{
98+
display: 'flex',
99+
gap: 2,
100+
flexDirection: 'row',
101+
paddingLeft: '0px'
102+
}}
103+
>
104+
<Button
105+
variant="outlined"
106+
color="primary"
107+
size="small"
108+
onClick={() => handleAclPermissionsModal('container')}
109+
sx={{
110+
width: isSmallScreen ? '165px' : '200px',
111+
borderColor: 'primary.main',
112+
padding: '6px 12px'
113+
}}
114+
>
115+
Share Documents
116+
</Button>
117+
<Button
118+
variant="contained"
119+
color="primary"
120+
size="small"
121+
onClick={() => setShowAddDocModal(true)}
122+
sx={{ width: isSmallScreen ? '140px' : '180px', padding: '6px 12px' }}
123+
>
124+
Add Document
125+
</Button>
126+
<UploadDocumentModal showModal={showAddDocModal} setShowModal={setShowAddDocModal} />
127+
<SetAclPermissionsModal
128+
showModal={showAclPermissionModal}
129+
setShowModal={setShowAclPermissionModal}
130+
dataset={dataset}
131+
/>
132+
<ConfirmationModal
133+
showModal={showConfirmationModal}
134+
setShowModal={setShowConfirmationModal}
135+
title="Delete Document"
136+
text={`You're about to delete "${truncatedText}" from the pod. Do you wish to continue?`}
137+
onConfirm={handleDeleteDoc}
138+
confirmButtonText="Delete"
139+
processing={processing}
140+
/>
141+
</Box>
142+
<DocumentTable
143+
handleAclPermissionsModal={handleAclPermissionsModal}
144+
handleSelectDeleteDoc={(document) => handleSelectDeleteDoc(document)}
145+
/>
146+
</Container>
147+
);
148+
};
149+
150+
export default Documents;

0 commit comments

Comments
 (0)