정산내역 상세 변경

This commit is contained in:
focp212@naver.com
2025-11-04 09:28:42 +09:00
parent c99b635950
commit cab65b19bc
10 changed files with 246 additions and 22 deletions

View File

@@ -186,9 +186,17 @@ export interface ListDateGroupProps {
date?: string; date?: string;
periodType: SettlementPeriodType; periodType: SettlementPeriodType;
items?: Array<SettlementsHistoryContent & SettlementsTransactionListContent> items?: Array<SettlementsHistoryContent & SettlementsTransactionListContent>
setDetailData?: (detailData: DetailData) => void;
};
export interface DetailData {
settlementId?: string;
tid?: string;
detailOn: boolean;
}; };
export interface ListItemProps extends SettlementsHistoryContent, SettlementsTransactionListContent { export interface ListItemProps extends SettlementsHistoryContent, SettlementsTransactionListContent {
periodType: SettlementPeriodType; periodType: SettlementPeriodType;
setDetailData?: (detailData: DetailData) => void;
}; };
export interface AmountInfoWrapProps { export interface AmountInfoWrapProps {

View File

@@ -0,0 +1,192 @@
import { useEffect, useState } from 'react';
import { motion } from 'framer-motion';
import { AmountInfoWrap } from '@/entities/settlement/ui/info-wrap/amount-info-wrap';
import { SettlementInfoWrap } from '@/entities/settlement/ui/info-wrap/settlement-info-wrap';
import { TransactionInfoWrap } from '@/entities/settlement/ui/info-wrap/transaction-info-wrap'
import {
AmountInfo,
InfoWrapKeys,
SettlementInfo,
SettlementPeriodType,
SettlementsHistoryDetailParams,
SettlementsHistoryDetailResponse,
SettlementsTransactionDetailParams,
SettlementsTransactionDetailResponse,
TransactionInfo
} from '@/entities/settlement/model/types';
import { useSettlementsHistoryDetailMutation } from '@/entities/settlement/api/use-settlements-history-detail-mutation';
import { useSettlementsTransactionDetailMutation } from '@/entities/settlement/api/use-settlements-transaction-detail-mutation';
import { NumericFormat } from 'react-number-format';
import moment from 'moment';
import { useTranslation } from 'react-i18next';
import { FullMenuClose } from '@/entities/common/ui/full-menu-close';
import { DetailMotionDuration, DetailMotionStyle, DetailMotionVariants } from '@/entities/common/model/constant';
export interface SettlementDetailProps {
detailOn: boolean;
setDetailOn: (detailOn: boolean) => void;
periodType: SettlementPeriodType;
settlementId?: string;
tid?: string;
};
export const SettlementDetail = ({
detailOn,
setDetailOn,
periodType,
settlementId,
tid
}: SettlementDetailProps) => {
const { t, i18n } = useTranslation();
const [amountInfo, setAmountInfo] = useState<AmountInfo>();
const [settlementInfo, setSettlementInfo] = useState<SettlementInfo>();
const [settlementAmount, setSettlementAmount] = useState<number>();
const [settlementDate, setSettlementDate] = useState<string>();
const [transactionInfo, setTransactionInfo] = useState<TransactionInfo>();
const [transactionAmount, setTransactionAmount] = useState<number>();
const [merchantName, setMerchantName] = useState<string>();
const [showSettlement, setShowSettlement] = useState<boolean>(false);
const [showTransaction, setShowTransaction] = useState<boolean>(false);
const { mutateAsync: settlementsHistoryDetail } = useSettlementsHistoryDetailMutation();
const { mutateAsync: settlementsTransactionDetail } = useSettlementsTransactionDetailMutation();
const callSettlementDetail = () => {
if(settlementId){
let params: SettlementsHistoryDetailParams = {
settlementId: settlementId
};
settlementsHistoryDetail(params).then((rs: SettlementsHistoryDetailResponse) => {
setAmountInfo(rs.amountInfo);
setSettlementInfo(rs.settlementInfo);
setSettlementAmount(rs.settlementAmount);
setSettlementDate(rs.settlementDate);
});
}
};
const callTransactionDetail = () => {
if(tid){
let params: SettlementsTransactionDetailParams = {
tid: tid
};
settlementsTransactionDetail(params).then((rs: SettlementsTransactionDetailResponse) => {
setAmountInfo(rs.amountInfo);
setTransactionInfo(rs.transactionInfo);
setTransactionAmount(rs.transactionAmount);
setMerchantName(rs.merchantName);
});
}
};
const onClickToShowInfo = (infoWrapKey: InfoWrapKeys) => {
if(infoWrapKey === InfoWrapKeys.Settlement){
setShowSettlement(!showSettlement);
}
else if(infoWrapKey === InfoWrapKeys.Transaction){
setShowTransaction(!showTransaction);
}
};
const onClickToClose = () => {
setDetailOn(false);
};
useEffect(() => {
if(periodType === SettlementPeriodType.SETTLEMENT_DATE && settlementId){
callSettlementDetail();
}
else if(periodType === SettlementPeriodType.TRANSACTION_DATE && tid){
callTransactionDetail();
}
}, [periodType, settlementId, tid]);
return (
<>
<motion.div
className="full-menu-modal"
initial="hidden"
animate={ (detailOn)? 'visible': 'hidden' }
variants={ DetailMotionVariants }
transition={ DetailMotionDuration }
style={ DetailMotionStyle }
>
<div className="full-menu-container pdw-16">
<div className="full-menu-header">
<div className="full-menu-title center">{ t('settlement.detailTitle') }</div>
<div className="full-menu-actions">
<FullMenuClose
addClass="full-menu-close"
onClickToCallback={ onClickToClose }
></FullMenuClose>
</div>
</div>
<div className="tab-pane sub active">
<div className="option-list">
<div className="txn-detail">
{ (periodType === SettlementPeriodType.SETTLEMENT_DATE) &&
<div className="txn-num-group">
<div className="txn-amount">
<div className="value">
{i18n.language === 'en' && <span className="unit">{t('home.currencySymbol')}</span>}
<NumericFormat
value={ settlementAmount }
thousandSeparator
displayType="text"
></NumericFormat>
{i18n.language !== 'en' && <span className="unit">{t('home.currencyWon')}</span>}
</div>
</div>
<div className="txn-date">{ moment(settlementDate).format('YYYY.MM.DD') }</div>
</div>
}
{ (periodType === SettlementPeriodType.TRANSACTION_DATE) &&
<div className="txn-num-group">
<div className="txn-amount">
<div className="value">
{i18n.language === 'en' && <span className="unit">{t('home.currencySymbol')}</span>}
<NumericFormat
value={ transactionAmount }
thousandSeparator
displayType="text"
></NumericFormat>
{i18n.language !== 'en' && <span className="unit">{t('home.currencyWon')}</span>}
</div>
</div>
<div className="txn-date">{ merchantName }</div>
</div>
}
<div className="txn-divider minus"></div>
<AmountInfoWrap
periodType={ periodType }
amountInfo={ amountInfo }
settlementAmount={ settlementAmount }
></AmountInfoWrap>
<div className="txn-divider"></div>
{ (periodType === SettlementPeriodType.SETTLEMENT_DATE) &&
<SettlementInfoWrap
settlementInfo={ settlementInfo }
isOpen={ showSettlement }
onClickToShowInfo={ (infoWrapKey) => onClickToShowInfo(infoWrapKey) }
></SettlementInfoWrap>
}
{ (periodType === SettlementPeriodType.TRANSACTION_DATE) &&
<TransactionInfoWrap
transactionInfo={ transactionInfo }
isOpen={ showTransaction }
onClickToShowInfo={ (infoWrapKey) => onClickToShowInfo(infoWrapKey) }
></TransactionInfoWrap>
}
</div>
</div>
</div>
</div>
</motion.div>
</>
);
};

View File

@@ -8,7 +8,8 @@ import { useTranslation } from 'react-i18next';
export const ListDateGroup = ({ export const ListDateGroup = ({
date, date,
periodType, periodType,
items items,
setDetailData
}: ListDateGroupProps) => { }: ListDateGroupProps) => {
const { i18n } = useTranslation(); const { i18n } = useTranslation();
@@ -38,6 +39,7 @@ export const ListDateGroup = ({
approvalNumber={ items[i]?.approvalNumber } approvalNumber={ items[i]?.approvalNumber }
approvalDate={ items[i]?.approvalDate } approvalDate={ items[i]?.approvalDate }
transactionAmount={ items[i]?.transactionAmount } transactionAmount={ items[i]?.transactionAmount }
setDetailData={ setDetailData }
></ListItem> ></ListItem>
) )
} }

View File

@@ -15,27 +15,28 @@ export const ListItem = ({
settlementAmount, settlementAmount,
approvalNumber, approvalNumber,
approvalDate, approvalDate,
transactionAmount transactionAmount,
setDetailData
}: ListItemProps) => { }: ListItemProps) => {
const { navigate } = useNavigate(); const { t } = useTranslation();
const { t, i18n } = useTranslation();
const onClickToNavigate = () => { const onClickToNavigate = () => {
let detailId; let detailId;
if(periodType === SettlementPeriodType.SETTLEMENT_DATE){ if(periodType === SettlementPeriodType.SETTLEMENT_DATE){
detailId = settlementId; if(setDetailData && settlementId){
setDetailData({
settlementId: settlementId,
detailOn: true
});
}
} }
else if(periodType === SettlementPeriodType.TRANSACTION_DATE){ else if(periodType === SettlementPeriodType.TRANSACTION_DATE){
detailId = approvalNumber; if(setDetailData && tid){
} setDetailData({
tid: tid,
navigate(PATHS.settlement.detail + detailId, { detailOn: true
state: { });
periodType: periodType,
settlementId: settlementId,
approvalNumber: approvalNumber,
tid: tid,
} }
}); }
}; };
const getDotClass = () => { const getDotClass = () => {
@@ -46,7 +47,6 @@ export const ListItem = ({
else if(periodType === SettlementPeriodType.TRANSACTION_DATE && transactionAmount){ else if(periodType === SettlementPeriodType.TRANSACTION_DATE && transactionAmount){
if(transactionAmount > 0) rs = 'blue'; if(transactionAmount > 0) rs = 'blue';
} }
return rs; return rs;
}; };

View File

@@ -27,13 +27,15 @@ import {
SettlementsHistoryResponse, SettlementsHistoryResponse,
SettlementsHistorySummaryParams, SettlementsHistorySummaryParams,
SettlementsHistorySummaryResponse, SettlementsHistorySummaryResponse,
SettlementsTransactionSummaryParams SettlementsTransactionSummaryParams,
DetailData
} from '../model/types'; } from '../model/types';
import { DefaultRequestPagination, SortTypeKeys } from '@/entities/common/model/types'; import { DefaultRequestPagination, SortTypeKeys } from '@/entities/common/model/types';
import { useStore } from '@/shared/model/store'; import { useStore } from '@/shared/model/store';
import { EmailBottomSheet } from '@/entities/common/ui/email-bottom-sheet'; import { EmailBottomSheet } from '@/entities/common/ui/email-bottom-sheet';
import useIntersectionObserver from '@/widgets/intersection-observer'; import useIntersectionObserver from '@/widgets/intersection-observer';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { SettlementDetail } from './detail/settlement-detail';
export interface ListWrapProps { export interface ListWrapProps {
startDateFromCalendar?: string; startDateFromCalendar?: string;
@@ -74,6 +76,10 @@ export const ListWrap = ({
const [preSettlementCancelOffset, setPreSettlementCancelOffset] = useState<number>(); const [preSettlementCancelOffset, setPreSettlementCancelOffset] = useState<number>();
const [isOpenSummary, setIsOpenSummary] = useState<boolean>(false); const [isOpenSummary, setIsOpenSummary] = useState<boolean>(false);
const [detailOn, setDetailOn] = useState<boolean>(false);
const [detailSettlementId, setDetailSettlementId] = useState<string>('');
const [detailTid, setDetailTid] = useState<string>('');
const [emailBottomSheetOn, setEmailBottomSheetOn] = useState<boolean>(false); const [emailBottomSheetOn, setEmailBottomSheetOn] = useState<boolean>(false);
const { mutateAsync: settlementsHistory } = useSettlementsHistoryMutation(); const { mutateAsync: settlementsHistory } = useSettlementsHistoryMutation();
@@ -241,6 +247,16 @@ export const ListWrap = ({
setPeriodType(val); setPeriodType(val);
} }
}; };
const setDetailData = (detailData: DetailData) => {
setDetailOn(detailData.detailOn);
if(detailData.settlementId){
setDetailSettlementId(detailData.settlementId);
}
if(detailData.tid){
setDetailTid(detailData.tid);
}
};
const getSettlementDateListDateGroup = () => { const getSettlementDateListDateGroup = () => {
let rs = []; let rs = [];
@@ -265,6 +281,7 @@ export const ListWrap = ({
periodType={ periodType } periodType={ periodType }
date={ date } date={ date }
items={ list } items={ list }
setDetailData={ setDetailData }
></ListDateGroup> ></ListDateGroup>
); );
} }
@@ -282,6 +299,7 @@ export const ListWrap = ({
periodType={ periodType } periodType={ periodType }
date={ date } date={ date }
items={ list } items={ list }
setDetailData={ setDetailData }
></ListDateGroup> ></ListDateGroup>
); );
} }
@@ -311,6 +329,7 @@ export const ListWrap = ({
periodType={ periodType } periodType={ periodType }
date={ date } date={ date }
items={ list } items={ list }
setDetailData={ setDetailData }
></ListDateGroup> ></ListDateGroup>
); );
} }
@@ -329,6 +348,7 @@ export const ListWrap = ({
periodType={ periodType } periodType={ periodType }
date={ date } date={ date }
items={ list } items={ list }
setDetailData={ setDetailData }
></ListDateGroup> ></ListDateGroup>
); );
} }
@@ -477,6 +497,13 @@ export const ListWrap = ({
setEndDate={ setEndDate } setEndDate={ setEndDate }
setPaymentMethod={ setPaymentMethod } setPaymentMethod={ setPaymentMethod }
></ListFilter> ></ListFilter>
<SettlementDetail
detailOn={ detailOn }
setDetailOn={ setDetailOn }
periodType={ periodType }
settlementId={ detailSettlementId }
tid={ detailTid }
></SettlementDetail>
{ !!emailBottomSheetOn && { !!emailBottomSheetOn &&
<EmailBottomSheet <EmailBottomSheet
bottomSheetOn={ emailBottomSheetOn } bottomSheetOn={ emailBottomSheetOn }

View File

@@ -5,7 +5,6 @@ import { useNavigate } from '@/shared/lib/hooks/use-navigate';
import { SettlementTab } from '@/entities/settlement/ui/settlement-tab'; import { SettlementTab } from '@/entities/settlement/ui/settlement-tab';
import { ListWrap } from '@/entities/settlement/ui/list-wrap'; import { ListWrap } from '@/entities/settlement/ui/list-wrap';
import { SettlementTabKeys } from '@/entities/settlement/model/types'; import { SettlementTabKeys } from '@/entities/settlement/model/types';
import { FooterItemActiveKey } from '@/entities/common/model/types';
import { HeaderType } from '@/entities/common/model/types'; import { HeaderType } from '@/entities/common/model/types';
import { import {
useSetHeaderTitle, useSetHeaderTitle,

View File

@@ -3,7 +3,6 @@ import { SentryRoutes } from '@/shared/configs/sentry';
import { ROUTE_NAMES } from '@/shared/constants/route-names'; import { ROUTE_NAMES } from '@/shared/constants/route-names';
import { CalendarPage } from './calendar/calendar-page'; import { CalendarPage } from './calendar/calendar-page';
import { ListPage } from './list/list-page'; import { ListPage } from './list/list-page';
import { DetailPage } from './list/detail-page';
export const SettlementPages = () => { export const SettlementPages = () => {
return ( return (
@@ -11,7 +10,6 @@ export const SettlementPages = () => {
<SentryRoutes> <SentryRoutes>
<Route path={ROUTE_NAMES.settlement.calendar} element={<CalendarPage />} /> <Route path={ROUTE_NAMES.settlement.calendar} element={<CalendarPage />} />
<Route path={ROUTE_NAMES.settlement.list} element={<ListPage />} /> <Route path={ROUTE_NAMES.settlement.list} element={<ListPage />} />
<Route path={ROUTE_NAMES.settlement.detail} element={<DetailPage />} />
</SentryRoutes> </SentryRoutes>
</> </>
); );

View File

@@ -59,7 +59,6 @@ export const PATHS: RouteNamesType = {
base: generatePath(ROUTE_NAMES.settlement.base), base: generatePath(ROUTE_NAMES.settlement.base),
calendar: generatePath(ROUTE_NAMES.settlement.base, ROUTE_NAMES.settlement.calendar), calendar: generatePath(ROUTE_NAMES.settlement.base, ROUTE_NAMES.settlement.calendar),
list: generatePath(ROUTE_NAMES.settlement.base, ROUTE_NAMES.settlement.list), list: generatePath(ROUTE_NAMES.settlement.base, ROUTE_NAMES.settlement.list),
detail: generatePath(ROUTE_NAMES.settlement.base, ROUTE_NAMES.settlement.detail),
}, },
merchant: { merchant: {
base: generatePath(ROUTE_NAMES.merchant.base), base: generatePath(ROUTE_NAMES.merchant.base),

View File

@@ -26,7 +26,6 @@ export const ROUTE_NAMES = {
base: '/settlement/*', base: '/settlement/*',
calendar: 'calendar', calendar: 'calendar',
list: 'list', list: 'list',
detail: 'detail/:detailId',
}, },
merchant: { merchant: {
base: '/merchant/*', base: '/merchant/*',