Skip to content

Commit

Permalink
add a new page component for redirecting disputes => details: (#7310)
Browse files Browse the repository at this point in the history
Co-authored-by: Rua Haszard <[email protected]>
Co-authored-by: Eric Jinks <[email protected]>
  • Loading branch information
3 people authored Oct 3, 2023
1 parent a10f4c0 commit 1e696f8
Show file tree
Hide file tree
Showing 13 changed files with 149 additions and 12 deletions.
4 changes: 4 additions & 0 deletions changelog/fix-6891-redirect-dispute-detail-to-transaction
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
Significance: patch
Type: dev

This work is part of a UI improvements to increase disputes response that is behind a feature flag. A changelog entry will be added to represent the work as a whole.
1 change: 1 addition & 0 deletions client/data/disputes/action-types.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

export default {
SET_DISPUTE: 'SET_DISPUTE',
SET_ERROR_FOR_DISPUTE: 'SET_ERROR_FOR_DISPUTE',
SET_DISPUTES: 'SET_DISPUTES',
SET_DISPUTES_SUMMARY: 'SET_DISPUTES_SUMMARY',
};
9 changes: 9 additions & 0 deletions client/data/disputes/actions.js
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,15 @@ export function updateDispute( data ) {
};
}

export function updateErrorForDispute( id, data, error ) {
return {
type: TYPES.SET_ERROR_FOR_DISPUTE,
id,
data,
error,
};
}

export function updateDisputes( query, data ) {
return {
type: TYPES.SET_DISPUTES,
Expand Down
15 changes: 10 additions & 5 deletions client/data/disputes/hooks.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ import type {
CachedDisputes,
DisputesSummary,
} from 'wcpay/types/disputes';
import type { ApiError } from 'wcpay/types/errors';
import { STORE_NAME } from '../constants';
import { disputeAwaitingResponseStatuses } from 'wcpay/disputes/filters/config';

Expand All @@ -25,16 +26,20 @@ import { disputeAwaitingResponseStatuses } from 'wcpay/disputes/filters/config';
export const useDispute = (
id: string
): {
dispute: Dispute;
dispute?: Dispute;
error?: ApiError;
isLoading: boolean;
doAccept: () => void;
} => {
const { dispute, isLoading } = useSelect(
const { dispute, error, isLoading } = useSelect(
( select ) => {
const { getDispute, isResolving } = select( STORE_NAME );
const { getDispute, getDisputeError, isResolving } = select(
STORE_NAME
);

return {
dispute: <Dispute>getDispute( id ),
dispute: <Dispute | undefined>getDispute( id ),
error: <ApiError | undefined>getDisputeError( id ),
isLoading: <boolean>isResolving( 'getDispute', [ id ] ),
};
},
Expand All @@ -44,7 +49,7 @@ export const useDispute = (
const { acceptDispute } = useDispatch( STORE_NAME );
const doAccept = () => acceptDispute( id );

return { dispute, isLoading, doAccept };
return { dispute, isLoading, error, doAccept };
};

/**
Expand Down
8 changes: 7 additions & 1 deletion client/data/disputes/reducer.js
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ const defaultState = { byId: {}, queries: {}, summary: {}, cached: {} };

const receiveDisputes = (
state = defaultState,
{ type, query = {}, data = [] }
{ type, query = {}, data = [], id, error }
) => {
const index = getResourceId( query );

Expand All @@ -25,6 +25,12 @@ const receiveDisputes = (
...state,
byId: { ...state.byId, [ data.id ]: data },
};
case TYPES.SET_ERROR_FOR_DISPUTE:
state = {
...state,
byId: { ...state.byId, [ id ]: { error } },
};
break;
case TYPES.SET_DISPUTES:
return {
...state,
Expand Down
2 changes: 2 additions & 0 deletions client/data/disputes/resolvers.js
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ import {
updateDispute,
updateDisputes,
updateDisputesSummary,
updateErrorForDispute,
} from './actions';

const formatQueryFilters = ( query ) => ( {
Expand Down Expand Up @@ -61,6 +62,7 @@ export function* getDispute( id ) {
'createErrorNotice',
__( 'Error retrieving dispute.', 'woocommerce-payments' )
);
yield updateErrorForDispute( id, undefined, e );
}
}

Expand Down
5 changes: 5 additions & 0 deletions client/data/disputes/selectors.js
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,11 @@ export const getDispute = ( state, id ) => {
return disputeById[ id ];
};

export const getDisputeError = ( state, id ) => {
const disputeById = getDisputesState( state ).byId || {};
return disputeById[ id ]?.error;
};

export const getCachedDispute = ( state, id ) => {
const disputeById = getDisputesState( state ).cached || {};
return disputeById[ id ];
Expand Down
4 changes: 4 additions & 0 deletions client/data/disputes/test/resolvers.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ import {
updateDispute,
updateDisputes,
updateDisputesSummary,
updateErrorForDispute,
} from '../actions';
import { getDispute, getDisputes, getDisputesSummary } from '../resolvers';

Expand Down Expand Up @@ -68,6 +69,9 @@ describe( 'getDispute resolver', () => {
expect.any( String )
)
);
expect( generator.next().value ).toEqual(
updateErrorForDispute( 'dp_mock1', undefined, errorResponse )
);
} );
} );
} );
Expand Down
4 changes: 2 additions & 2 deletions client/disputes/details/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ import { TestModeNotice, topics } from 'components/test-mode-notice';
import '../style.scss';
import { Dispute } from 'wcpay/types/disputes';

const DisputeDetails = ( {
const LegacyDisputeDetails = ( {
query: { id: disputeId },
}: {
query: { id: string };
Expand Down Expand Up @@ -156,4 +156,4 @@ const DisputeDetails = ( {
);
};

export default DisputeDetails;
export default LegacyDisputeDetails;
76 changes: 76 additions & 0 deletions client/disputes/redirect-to-transaction-details/index.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
/**
* External dependencies
*/
import React, { useEffect } from 'react';
import { getHistory } from '@woocommerce/navigation';
import { Spinner, Icon, Flex, FlexItem } from '@wordpress/components';

/**
* Internal dependencies.
*/
import Page from 'components/page';
import { useDispute } from 'data/index';
import { Charge } from 'wcpay/types/charges';
import { getAdminUrl } from 'wcpay/utils';

import './style.scss';
import warning from 'wcpay/components/icons/warning';

const RedirectToTransactionDetails: React.FC< { query: { id: string } } > = ( {
query: { id: disputeId },
} ) => {
const { dispute, error, isLoading } = useDispute( disputeId );

useEffect( () => {
if ( ! isLoading && dispute?.charge ) {
// Dispute type allows charge as nested object or string ID,
// so we have to hint we expect a Charge object here.
const chargeObject = dispute.charge as Charge;
const transactionDetailsUrl = getAdminUrl( {
page: 'wc-admin',
path: '/payments/transactions/details',
id: chargeObject.payment_intent,
transaction_id: chargeObject.balance_transaction,
type: 'dispute',
} );
getHistory().replace( transactionDetailsUrl );
}
}, [ dispute, isLoading ] );

return (
<Page>
<Flex
direction="column"
className="wcpay-dispute-detail-legacy-redirect"
>
{ error ? (
<>
<FlexItem>
<Icon icon={ warning } type="warning" size={ 32 } />
</FlexItem>
<FlexItem>
<div>
<b>Error retrieving dispute</b>
</div>
<div>Please check your network and try again.</div>
</FlexItem>
</>
) : (
<>
<FlexItem>
<Spinner />
</FlexItem>
<FlexItem>
<div>
<b>One moment please</b>
</div>
<div>Redirecting to payment details…</div>
</FlexItem>
</>
) }
</Flex>
</Page>
);
};

export default RedirectToTransactionDetails;
9 changes: 9 additions & 0 deletions client/disputes/redirect-to-transaction-details/style.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
.wcpay-dispute-detail-legacy-redirect {
// Shift the redirect spinner to approx center of viewport.
top: 10vh;
position: relative;

.components-flex-item {
text-align: center;
}
}
17 changes: 14 additions & 3 deletions client/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,8 @@ import DepositDetailsPage from 'deposits/details';
import TransactionsPage from 'transactions';
import PaymentDetailsPage from 'payment-details';
import DisputesPage from 'disputes';
import DisputeDetailsPage from 'disputes/details';
import LegacyDisputeDetailsPage from 'disputes/details';
import RedirectToTransactionDetails from 'disputes/redirect-to-transaction-details';
import DisputeEvidencePage from 'disputes/evidence';
import AdditionalMethodsPage from 'wcpay/additional-methods-setup';
import MultiCurrencySetupPage from 'wcpay/multi-currency-setup';
Expand Down Expand Up @@ -150,8 +151,15 @@ addFilter(
},
capability: 'manage_woocommerce',
} );

// If disputes on transaction page feature is enabled, set up a soft
// redirect component; otherwise register the (legacy) dispute details page.
const isDisputeOnTransactionPageEnabled =
window.wcpaySettings.featureFlags.isDisputeOnTransactionPageEnabled;
pages.push( {
container: DisputeDetailsPage,
container: isDisputeOnTransactionPageEnabled
? RedirectToTransactionDetails
: LegacyDisputeDetailsPage,
path: '/payments/disputes/details',
wpOpenMenu: menuID,
breadcrumbs: [
Expand All @@ -163,11 +171,14 @@ addFilter(
__( 'Dispute details', 'woocommerce-payments' ),
],
navArgs: {
id: 'wc-payments-disputes-details',
id: isDisputeOnTransactionPageEnabled
? 'wc-payments-disputes-details-legacy-redirect'
: 'wc-payments-disputes-details',
parentPath: '/payments/disputes',
},
capability: 'manage_woocommerce',
} );

pages.push( {
container: DisputeEvidencePage,
path: '/payments/disputes/challenge',
Expand Down
7 changes: 6 additions & 1 deletion includes/admin/class-wc-payments-admin.php
Original file line number Diff line number Diff line change
Expand Up @@ -472,14 +472,19 @@ public function add_payments_menu() {
'path' => '/payments/transactions/details',
]
);

$is_dispute_on_transaction_page_enabled = WC_Payments_Features::is_dispute_on_transaction_page_enabled();
wc_admin_register_page(
[
'id' => 'wc-payments-disputes-details',
'id' => $is_dispute_on_transaction_page_enabled
? 'wc-payments-disputes-details-legacy-redirect'
: 'wc-payments-disputes-details',
'title' => __( 'Dispute details', 'woocommerce-payments' ),
'parent' => 'wc-payments-disputes',
'path' => '/payments/disputes/details',
]
);

wc_admin_register_page(
[
'id' => 'wc-payments-disputes-challenge',
Expand Down

0 comments on commit 1e696f8

Please sign in to comment.