SubscriptionsListTable.tsx
154 lines
| 1 | import {__} from '@wordpress/i18n'; |
| 2 | import {ListTablePage} from '@givewp/components'; |
| 3 | import ListTableApi from '@givewp/components/ListTable/api'; |
| 4 | import tableStyles from '@givewp/components/ListTable/ListTablePage/ListTablePage.module.scss'; |
| 5 | import {BulkActionsConfig, FilterConfig} from '@givewp/components/ListTable/ListTablePage'; |
| 6 | import {SubscriptionsRowActions} from './SubscriptionsRowActions'; |
| 7 | import {IdBadge} from '@givewp/components/ListTable/TableCell'; |
| 8 | import {Interweave} from 'interweave'; |
| 9 | import BlankSlate from '@givewp/components/ListTable/BlankSlate'; |
| 10 | |
| 11 | declare global { |
| 12 | interface Window { |
| 13 | GiveSubscriptions: { |
| 14 | apiNonce: string; |
| 15 | apiRoot: string; |
| 16 | table: {columns: Array<object>}; |
| 17 | forms: Array<{value: string; text: string}>; |
| 18 | paymentMode: boolean; |
| 19 | pluginUrl: string; |
| 20 | }; |
| 21 | } |
| 22 | } |
| 23 | |
| 24 | const API = new ListTableApi(window.GiveSubscriptions); |
| 25 | |
| 26 | const filters: Array<FilterConfig> = [ |
| 27 | { |
| 28 | name: 'search', |
| 29 | type: 'search', |
| 30 | inlineSize: '14rem', |
| 31 | text: __('Name, Email, or ID', 'give'), |
| 32 | ariaLabel: __('search donations', 'give'), |
| 33 | }, |
| 34 | { |
| 35 | name: 'form', |
| 36 | type: 'formselect', |
| 37 | text: __('Select Form', 'give'), |
| 38 | ariaLabel: __('filter donation forms by status', 'give'), |
| 39 | options: window.GiveSubscriptions.forms, |
| 40 | }, |
| 41 | { |
| 42 | name: 'toggle', |
| 43 | type: 'checkbox', |
| 44 | text: __('Test', 'give'), |
| 45 | ariaLabel: __('View Test Subscriptions', 'give'), |
| 46 | }, |
| 47 | ]; |
| 48 | |
| 49 | const bulkActions: Array<BulkActionsConfig> = [ |
| 50 | { |
| 51 | label: __('Delete', 'give'), |
| 52 | value: 'delete', |
| 53 | type: 'danger', |
| 54 | action: async (selected) => { |
| 55 | const response = await API.fetchWithArgs('/delete', {ids: selected.join(',')}, 'DELETE'); |
| 56 | return response; |
| 57 | }, |
| 58 | confirm: (selected, names) => ( |
| 59 | <> |
| 60 | <p>{__('Really delete the following subscriptions?', 'give')}</p> |
| 61 | <ul role="document" tabIndex={0}> |
| 62 | {selected.map((donationId, index) => ( |
| 63 | <li key={donationId}> |
| 64 | <IdBadge id={donationId} />{' '} |
| 65 | <span> |
| 66 | {__('from ', 'give')} <Interweave content={names[index]} /> |
| 67 | </span> |
| 68 | </li> |
| 69 | ))} |
| 70 | </ul> |
| 71 | </> |
| 72 | ), |
| 73 | }, |
| 74 | ...(() => { |
| 75 | const subscriptionStatuses = { |
| 76 | active: __('Set To Active', 'give'), |
| 77 | expired: __('Set To Expired', 'give'), |
| 78 | completed: __('Set To Completed', 'give'), |
| 79 | cancelled: __('Set To Cancelled', 'give'), |
| 80 | pending: __('Set To Pending', 'give'), |
| 81 | failing: __('Set To Failing', 'give'), |
| 82 | suspended: __('Set To Suspended', 'give'), |
| 83 | abandoned: __('Set To Abandoned', 'give'), |
| 84 | }; |
| 85 | |
| 86 | return Object.entries(subscriptionStatuses).map(([value, label]) => { |
| 87 | return { |
| 88 | label, |
| 89 | value, |
| 90 | action: async (selected) => |
| 91 | await API.fetchWithArgs( |
| 92 | '/setStatus', |
| 93 | { |
| 94 | ids: selected.join(','), |
| 95 | status: value, |
| 96 | }, |
| 97 | 'POST' |
| 98 | ), |
| 99 | confirm: (selected, names) => ( |
| 100 | <> |
| 101 | <p>{__('Set status for the following donations?', 'give')}</p> |
| 102 | <ul role="document" tabIndex={0}> |
| 103 | {selected.map((donationId, index) => ( |
| 104 | <li key={donationId}> |
| 105 | <IdBadge id={donationId} /> <span>{__('from', 'give')}</span> |
| 106 | <Interweave content={names[index]} /> |
| 107 | </li> |
| 108 | ))} |
| 109 | </ul> |
| 110 | </> |
| 111 | ), |
| 112 | }; |
| 113 | }); |
| 114 | })(), |
| 115 | ]; |
| 116 | |
| 117 | /** |
| 118 | * Displays a blank slate for the Subscriptions table. |
| 119 | * @since 2.27.0 |
| 120 | */ |
| 121 | const ListTableBlankSlate = ( |
| 122 | <BlankSlate |
| 123 | imagePath={`${window.GiveSubscriptions.pluginUrl}/assets/dist/images/list-table/blank-slate-recurring-icon.svg`} |
| 124 | description={__('No subscriptions found', 'give')} |
| 125 | href={'https://docs.givewp.com/subscriptions'} |
| 126 | linkText={__('Recurring Donations.', 'give')} |
| 127 | /> |
| 128 | ); |
| 129 | |
| 130 | export default function SubscriptionsListTable() { |
| 131 | return ( |
| 132 | <ListTablePage |
| 133 | title={__('Subscriptions', 'give')} |
| 134 | singleName={__('subscription', 'give')} |
| 135 | pluralName={__('subscriptions', 'give')} |
| 136 | rowActions={SubscriptionsRowActions} |
| 137 | bulkActions={bulkActions} |
| 138 | apiSettings={window.GiveSubscriptions} |
| 139 | filterSettings={filters} |
| 140 | paymentMode={!!window.GiveSubscriptions.paymentMode} |
| 141 | listTableBlankSlate={ListTableBlankSlate} |
| 142 | > |
| 143 | <button className={tableStyles.addFormButton} onClick={showLegacyDonations}> |
| 144 | {__('Switch to Legacy View')} |
| 145 | </button> |
| 146 | </ListTablePage> |
| 147 | ); |
| 148 | } |
| 149 | |
| 150 | const showLegacyDonations = async (event) => { |
| 151 | await API.fetchWithArgs('/view', {isLegacy: 1}); |
| 152 | window.location.reload(); |
| 153 | }; |
| 154 |