index.tsx
296 lines
| 1 | import {createContext, useRef, useState} from 'react'; |
| 2 | import {__} from '@wordpress/i18n'; |
| 3 | import {A11yDialog} from 'react-a11y-dialog'; |
| 4 | import A11yDialogInstance from 'a11y-dialog'; |
| 5 | import {GiveIcon} from '@givewp/components'; |
| 6 | import {ListTable} from '../ListTable'; |
| 7 | import Pagination from '../Pagination'; |
| 8 | import {Filter, getInitialFilterState} from '../Filters'; |
| 9 | import useDebounce from '../hooks/useDebounce'; |
| 10 | import {useResetPage} from '../hooks/useResetPage'; |
| 11 | import ListTableApi from '../api'; |
| 12 | import styles from './ListTablePage.module.scss'; |
| 13 | import cx from 'classnames'; |
| 14 | import {BulkActionSelect} from '@givewp/components/ListTable/BulkActions/BulkActionSelect'; |
| 15 | import ToggleSwitch from '@givewp/components/ListTable/ToggleSwitch'; |
| 16 | |
| 17 | export interface ListTablePageProps { |
| 18 | //required |
| 19 | title: string; |
| 20 | apiSettings: {apiRoot; apiNonce; table}; |
| 21 | |
| 22 | //optional |
| 23 | bulkActions?: Array<BulkActionsConfig> | null; |
| 24 | pluralName?: string; |
| 25 | singleName?: string; |
| 26 | children?: JSX.Element | JSX.Element[] | null; |
| 27 | rowActions?: JSX.Element | JSX.Element[] | Function | null; |
| 28 | filterSettings?; |
| 29 | align?: 'start' | 'center' | 'end'; |
| 30 | paymentMode?: boolean; |
| 31 | listTableBlankSlate: JSX.Element; |
| 32 | productRecommendation?: JSX.Element; |
| 33 | columnFilters?: Array<ColumnFilterConfig>; |
| 34 | banner?: () => JSX.Element; |
| 35 | } |
| 36 | |
| 37 | export interface FilterConfig { |
| 38 | // required |
| 39 | name: string; |
| 40 | type: 'select' | 'formselect' | 'search' | 'checkbox'; |
| 41 | |
| 42 | // optional |
| 43 | ariaLabel?: string; |
| 44 | inlineSize?: string; |
| 45 | text?: string; |
| 46 | options?: Array<{text: string; value: string}>; |
| 47 | } |
| 48 | |
| 49 | export interface ColumnFilterConfig { |
| 50 | column: string; |
| 51 | filter: Function |
| 52 | } |
| 53 | |
| 54 | export interface BulkActionsConfig { |
| 55 | //required |
| 56 | label: string; |
| 57 | value: string | number; |
| 58 | action: (selected: Array<string | number>) => Promise<{errors: string | number; successes: string | number}>; |
| 59 | confirm: (selected: Array<string | number>, names?: Array<string>) => JSX.Element | JSX.Element[] | string; |
| 60 | |
| 61 | //optional |
| 62 | isVisible?: (data, parameters) => Boolean; |
| 63 | type?: 'normal' | 'warning' | 'danger'; |
| 64 | } |
| 65 | |
| 66 | export const ShowConfirmModalContext = createContext((label, confirm, action, type = null) => {}); |
| 67 | export const CheckboxContext = createContext(null); |
| 68 | |
| 69 | export default function ListTablePage({ |
| 70 | title, |
| 71 | apiSettings, |
| 72 | bulkActions = null, |
| 73 | filterSettings = [], |
| 74 | singleName = __('item', 'give'), |
| 75 | pluralName = __('items', 'give'), |
| 76 | rowActions = null, |
| 77 | children = null, |
| 78 | align = 'start', |
| 79 | paymentMode, |
| 80 | listTableBlankSlate, |
| 81 | productRecommendation, |
| 82 | columnFilters = [], |
| 83 | banner |
| 84 | }: ListTablePageProps) { |
| 85 | const [page, setPage] = useState<number>(1); |
| 86 | const [perPage, setPerPage] = useState<number>(30); |
| 87 | const [filters, setFilters] = useState(getInitialFilterState(filterSettings)); |
| 88 | const [modalContent, setModalContent] = useState<{confirm; action; label; type?: 'normal' | 'warning' | 'danger'}>({ |
| 89 | confirm: (selected) => {}, |
| 90 | action: (selected) => {}, |
| 91 | label: '', |
| 92 | }); |
| 93 | const [selectedAction, setSelectedAction] = useState<string>(''); |
| 94 | const [selectedIds, setSelectedIds] = useState([]); |
| 95 | const [selectedNames, setSelectedNames] = useState([]); |
| 96 | const dialog = useRef() as {current: A11yDialogInstance}; |
| 97 | const checkboxRefs = useRef([]); |
| 98 | const [sortField, setSortField] = useState<{sortColumn: string; sortDirection: string}>({ |
| 99 | sortColumn: 'id', |
| 100 | sortDirection: 'desc', |
| 101 | }); |
| 102 | const [testMode, setTestMode] = useState(paymentMode); |
| 103 | |
| 104 | const {sortColumn, sortDirection} = sortField; |
| 105 | const locale = navigator.language || navigator.languages[0]; |
| 106 | const testModeFilter = filterSettings.find((filter) => filter.name === 'toggle'); |
| 107 | |
| 108 | const parameters = { |
| 109 | page, |
| 110 | perPage, |
| 111 | sortColumn, |
| 112 | sortDirection, |
| 113 | locale, |
| 114 | testMode, |
| 115 | ...filters, |
| 116 | }; |
| 117 | |
| 118 | const archiveApi = useRef(new ListTableApi(apiSettings)).current; |
| 119 | |
| 120 | const {data, error, isValidating, mutate} = archiveApi.useListTable(parameters); |
| 121 | |
| 122 | useResetPage(data, page, setPage, filters); |
| 123 | |
| 124 | const handleFilterChange = (name, value) => { |
| 125 | setFilters((prevState) => ({...prevState, [name]: value})); |
| 126 | }; |
| 127 | |
| 128 | const handleDebouncedFilterChange = useDebounce(handleFilterChange); |
| 129 | |
| 130 | const showConfirmActionModal = (label, confirm, action, type: 'normal' | 'warning' | 'danger' | null = null) => { |
| 131 | setModalContent({confirm, action, label, type}); |
| 132 | dialog.current.show(); |
| 133 | }; |
| 134 | |
| 135 | const openBulkActionModal = (event) => { |
| 136 | event.preventDefault(); |
| 137 | |
| 138 | if (window.GiveDonations && window.GiveDonations.addonsBulkActions) { |
| 139 | bulkActions = [...bulkActions, ...window.GiveDonations.addonsBulkActions]; |
| 140 | } |
| 141 | |
| 142 | const actionIndex = bulkActions.findIndex((config) => selectedAction === config.value); |
| 143 | |
| 144 | if (actionIndex < 0) return; |
| 145 | |
| 146 | const selected = []; |
| 147 | const names = []; |
| 148 | checkboxRefs.current.forEach((checkbox) => { |
| 149 | if (checkbox.checked) { |
| 150 | selected.push(checkbox.dataset.id); |
| 151 | names.push(checkbox.dataset.name); |
| 152 | } |
| 153 | }); |
| 154 | setSelectedIds(selected); |
| 155 | setSelectedNames(names); |
| 156 | if (selected.length) { |
| 157 | setModalContent({...bulkActions[actionIndex]}); |
| 158 | dialog.current.show(); |
| 159 | } |
| 160 | }; |
| 161 | |
| 162 | const setSortDirectionForColumn = (column, direction) => { |
| 163 | setSortField((previousState) => { |
| 164 | return { |
| 165 | ...previousState, |
| 166 | sortColumn: column, |
| 167 | sortDirection: direction, |
| 168 | }; |
| 169 | }); |
| 170 | }; |
| 171 | |
| 172 | const showPagination = () => ( |
| 173 | <Pagination |
| 174 | currentPage={page} |
| 175 | totalPages={data ? data.totalPages : 1} |
| 176 | disabled={!data} |
| 177 | totalItems={data ? parseInt(data.totalItems) : -1} |
| 178 | setPage={setPage} |
| 179 | singleName={singleName} |
| 180 | pluralName={pluralName} |
| 181 | /> |
| 182 | ); |
| 183 | |
| 184 | const PageActions = ({PageActionsTop}: {PageActionsTop?: boolean}) => { |
| 185 | return ( |
| 186 | <div className={cx(styles.pageActions, {[styles.alignEnd]: !bulkActions})}> |
| 187 | <BulkActionSelect |
| 188 | selectedState={[selectedAction, setSelectedAction]} |
| 189 | parameters={parameters} |
| 190 | data={data} |
| 191 | bulkActions={bulkActions} |
| 192 | showModal={openBulkActionModal} |
| 193 | /> |
| 194 | {PageActionsTop && testModeFilter && <TestModeFilter />} |
| 195 | {page && setPage && showPagination()} |
| 196 | </div> |
| 197 | ); |
| 198 | }; |
| 199 | |
| 200 | const TestModeFilter = () => ( |
| 201 | <ToggleSwitch ariaLabel={testModeFilter?.ariaLabel} onChange={setTestMode} checked={testMode} /> |
| 202 | ); |
| 203 | |
| 204 | const TestModeBadge = () => <span>{testModeFilter?.text}</span>; |
| 205 | |
| 206 | return ( |
| 207 | <> |
| 208 | <article className={styles.page}> |
| 209 | <header className={styles.pageHeader}> |
| 210 | <div className={styles.flexRow}> |
| 211 | <GiveIcon size={'1.875rem'} /> |
| 212 | <h1 className={styles.pageTitle}>{title}</h1> |
| 213 | {testModeFilter && testMode && <TestModeBadge />} |
| 214 | </div> |
| 215 | {children && <div className={styles.flexRow}>{children}</div>} |
| 216 | </header> |
| 217 | {banner && ( |
| 218 | <section role="banner"> |
| 219 | {banner()} |
| 220 | </section> |
| 221 | )} |
| 222 | <section role="search" id={styles.searchContainer}> |
| 223 | {filterSettings.map((filter) => ( |
| 224 | <Filter |
| 225 | key={filter.name} |
| 226 | value={filters[filter.name]} |
| 227 | filter={filter} |
| 228 | onChange={handleFilterChange} |
| 229 | debouncedOnChange={handleDebouncedFilterChange} |
| 230 | /> |
| 231 | ))} |
| 232 | </section> |
| 233 | <div className={cx('wp-header-end', 'hidden')} /> |
| 234 | <div className={styles.pageContent}> |
| 235 | <PageActions PageActionsTop /> |
| 236 | <CheckboxContext.Provider value={checkboxRefs}> |
| 237 | <ShowConfirmModalContext.Provider value={showConfirmActionModal}> |
| 238 | <ListTable |
| 239 | apiSettings={apiSettings} |
| 240 | sortField={sortField} |
| 241 | setSortDirectionForColumn={setSortDirectionForColumn} |
| 242 | singleName={singleName} |
| 243 | pluralName={pluralName} |
| 244 | title={title} |
| 245 | rowActions={rowActions} |
| 246 | parameters={parameters} |
| 247 | data={data} |
| 248 | error={error} |
| 249 | isLoading={isValidating} |
| 250 | align={align} |
| 251 | testMode={testMode} |
| 252 | listTableBlankSlate={listTableBlankSlate} |
| 253 | productRecommendation={productRecommendation} |
| 254 | columnFilters={columnFilters} |
| 255 | /> |
| 256 | </ShowConfirmModalContext.Provider> |
| 257 | </CheckboxContext.Provider> |
| 258 | <PageActions /> |
| 259 | </div> |
| 260 | </article> |
| 261 | <A11yDialog |
| 262 | id="giveListTableModal" |
| 263 | dialogRef={(instance) => (dialog.current = instance)} |
| 264 | title={modalContent.label} |
| 265 | titleId={styles.modalTitle} |
| 266 | classNames={{ |
| 267 | container: styles.container, |
| 268 | overlay: styles.overlay, |
| 269 | dialog: cx(styles.dialog, { |
| 270 | [styles.warning]: modalContent?.type === 'warning', |
| 271 | [styles.danger]: modalContent?.type === 'danger', |
| 272 | }), |
| 273 | closeButton: 'hidden', |
| 274 | }} |
| 275 | > |
| 276 | <div className={styles.modalContent}>{modalContent?.confirm(selectedIds, selectedNames) || null}</div> |
| 277 | <div className={styles.gutter}> |
| 278 | <button id={styles.cancel} onClick={(event) => dialog.current?.hide()}> |
| 279 | {__('Cancel', 'give')} |
| 280 | </button> |
| 281 | <button |
| 282 | id={styles.confirm} |
| 283 | onClick={async (event) => { |
| 284 | dialog.current?.hide(); |
| 285 | await modalContent.action(selectedIds); |
| 286 | await mutate(); |
| 287 | }} |
| 288 | > |
| 289 | {__('Confirm', 'give')} |
| 290 | </button> |
| 291 | </div> |
| 292 | </A11yDialog> |
| 293 | </> |
| 294 | ); |
| 295 | } |
| 296 |