Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Simplified Collect][Taxes] Create WorkspaceTaxesPage #37870

Merged
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
049c487
Create WorkspaceTaxesPage
filip-solecki Mar 7, 2024
a491463
Remove unnecessary check
filip-solecki Mar 7, 2024
bc18001
Remove unnecessary import
filip-solecki Mar 7, 2024
0ff7067
Fix icon fill color
filip-solecki Mar 7, 2024
2d43808
CR fixes
filip-solecki Mar 7, 2024
b63bcbb
use ListItem type
filip-solecki Mar 7, 2024
a97c874
Fix tax icon
filip-solecki Mar 7, 2024
60aa7d3
Fix icon tax fill
filip-solecki Mar 7, 2024
e933a60
Header buttons
filip-solecki Mar 7, 2024
be9c328
Merge branch 'main' into wave8/WorkspaceTaxesPage
filip-solecki Mar 8, 2024
f06d07b
Merge branch 'main' into wave8/WorkspaceTaxesPage
filip-solecki Mar 11, 2024
c160d40
Add translations and adjust SelectionList
filip-solecki Mar 11, 2024
6838836
Merge branch 'main' into wave8/WorkspaceTaxesPage
filip-solecki Mar 11, 2024
940370c
Merge branch 'main' into wave8/WorkspaceTaxesPage
filip-solecki Mar 12, 2024
fd6716b
Fix failing TS
filip-solecki Mar 12, 2024
8bfd4ef
Fix Lint
filip-solecki Mar 12, 2024
e0ca61f
CR fixes
filip-solecki Mar 12, 2024
4ba47a8
Add possibility to remove foreign currency default tax
filip-solecki Mar 12, 2024
606c12d
Rename prop
filip-solecki Mar 12, 2024
2687bfb
handle isOffline mode and fetching taxes data
filip-solecki Mar 12, 2024
f3ab59c
Merge branch 'main' into wave8/WorkspaceTaxesPage
filip-solecki Mar 12, 2024
9fd30e5
Merge branch 'main' into wave8/WorkspaceTaxesPage
filip-solecki Mar 13, 2024
d772221
Fix type for WorkspaceTaxesPage
filip-solecki Mar 13, 2024
d696446
Refactor isDisabledCheckbox
filip-solecki Mar 13, 2024
14072ae
Merge branch 'main' into wave8/WorkspaceTaxesPage
kosmydel Mar 13, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
23 changes: 23 additions & 0 deletions assets/images/simple-illustrations/simple-illustration__coins.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
8 changes: 8 additions & 0 deletions assets/images/tax.svg
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
4 changes: 4 additions & 0 deletions src/ROUTES.ts
Original file line number Diff line number Diff line change
Expand Up @@ -554,6 +554,10 @@ const ROUTES = {
route: 'workspace/:policyID/tags',
getRoute: (policyID: string) => `workspace/${policyID}/tags` as const,
},
WORKSPACE_TAXES: {
route: 'workspace/:policyID/taxes',
getRoute: (policyID: string) => `workspace/${policyID}/taxes` as const,
},
// Referral program promotion
REFERRAL_DETAILS_MODAL: {
route: 'referral/:contentType',
Expand Down
1 change: 1 addition & 0 deletions src/SCREENS.ts
Original file line number Diff line number Diff line change
Expand Up @@ -215,6 +215,7 @@ const SCREENS = {
INVITE_MESSAGE: 'Workspace_Invite_Message',
CATEGORIES: 'Workspace_Categories',
TAGS: 'Workspace_Tags',
TAXES: 'Workspace_Taxes',
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
CURRENCY: 'Workspace_Profile_Currency',
WORKFLOWS: 'Workspace_Workflows',
WORKFLOWS_APPROVER: 'Workspace_Workflows_Approver',
Expand Down
2 changes: 2 additions & 0 deletions src/components/Icon/Expensicons.ts
Original file line number Diff line number Diff line change
Expand Up @@ -137,6 +137,7 @@ import Stopwatch from '@assets/images/stopwatch.svg';
import Sync from '@assets/images/sync.svg';
import Tag from '@assets/images/tag.svg';
import Task from '@assets/images/task.svg';
import Tax from '@assets/images/tax.svg';
import ThreeDots from '@assets/images/three-dots.svg';
import ThumbsUp from '@assets/images/thumbs-up.svg';
import Transfer from '@assets/images/transfer.svg';
Expand Down Expand Up @@ -224,6 +225,7 @@ export {
Fullscreen,
Folder,
Tag,
Tax,
Gallery,
Gear,
Globe,
Expand Down
2 changes: 2 additions & 0 deletions src/components/Icon/Illustrations.ts
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ import BigRocket from '@assets/images/simple-illustrations/simple-illustration__
import PinkBill from '@assets/images/simple-illustrations/simple-illustration__bill.svg';
import ChatBubbles from '@assets/images/simple-illustrations/simple-illustration__chatbubbles.svg';
import CoffeeMug from '@assets/images/simple-illustrations/simple-illustration__coffeemug.svg';
import Coins from '@assets/images/simple-illustrations/simple-illustration__coins.svg';
import CommentBubbles from '@assets/images/simple-illustrations/simple-illustration__commentbubbles.svg';
import ConciergeBubble from '@assets/images/simple-illustrations/simple-illustration__concierge-bubble.svg';
import ConciergeNew from '@assets/images/simple-illustrations/simple-illustration__concierge.svg';
Expand Down Expand Up @@ -148,4 +149,5 @@ export {
ThreeLeggedLaptopWoman,
House,
Tag,
Coins,
};
4 changes: 4 additions & 0 deletions src/languages/en.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1732,6 +1732,7 @@ export default {
reimburse: 'Reimbursements',
categories: 'Categories',
tags: 'Tags',
taxes: 'Taxes',
bills: 'Bills',
invoices: 'Invoices',
travel: 'Travel',
Expand Down Expand Up @@ -1784,6 +1785,9 @@ export default {
subtitle: 'Add a tag to track projects, locations, departments, and more.',
},
},
taxes: {
subtitle: 'Add tax names, rates, and set defaults.',
},
emptyWorkspace: {
title: 'Create a workspace',
subtitle: 'Workspaces are where you’ll chat with your team, reimburse expenses, issue cards, send invoices, pay bills, and more - all in one place.',
Expand Down
4 changes: 4 additions & 0 deletions src/languages/es.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1756,6 +1756,7 @@ export default {
reimburse: 'Reembolsos',
categories: 'Categorías',
tags: 'Etiquetas',
taxes: 'Impuestos',
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
bills: 'Pagar facturas',
invoices: 'Enviar facturas',
travel: 'Viajes',
Expand Down Expand Up @@ -1808,6 +1809,9 @@ export default {
subtitle: 'Añade una etiqueta para realizar el seguimiento de proyectos, ubicaciones, departamentos y otros.',
},
},
taxes: {
subtitle: 'Añada nombres de impuestos, tipos impositivos y establezca valores por defecto.',
},
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
emptyWorkspace: {
title: 'Crea un espacio de trabajo',
subtitle: 'En los espacios de trabajo podrás chatear con tu equipo, reembolsar gastos, emitir tarjetas, enviar y pagar facturas, y mucho más - todo en un mismo lugar.',
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ const workspaceSettingsScreens = {
[SCREENS.WORKSPACE.MEMBERS]: () => require('../../../../../pages/workspace/WorkspaceMembersPage').default as React.ComponentType,
[SCREENS.WORKSPACE.CATEGORIES]: () => require('../../../../../pages/workspace/categories/WorkspaceCategoriesPage').default as React.ComponentType,
[SCREENS.WORKSPACE.TAGS]: () => require('../../../../../pages/workspace/tags/WorkspaceTagsPage').default as React.ComponentType,
[SCREENS.WORKSPACE.TAXES]: () => require('../../../../../pages/workspace/taxes/WorkspaceTaxesPage').default as React.ComponentType,
} satisfies Screens;

function BaseCentralPaneNavigator() {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ const TAB_TO_CENTRAL_PANE_MAPPING: Record<BottomTabName, CentralPaneName[]> = {
SCREENS.WORKSPACE.TRAVEL,
SCREENS.WORKSPACE.MEMBERS,
SCREENS.WORKSPACE.CATEGORIES,
SCREENS.WORKSPACE.TAXES,
],
};

Expand Down
3 changes: 3 additions & 0 deletions src/libs/Navigation/linkingConfig/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,9 @@ const config: LinkingOptions<RootStackParamList>['config'] = {
[SCREENS.WORKSPACE.TAGS]: {
path: ROUTES.WORKSPACE_TAGS.route,
},
[SCREENS.WORKSPACE.TAXES]: {
path: ROUTES.WORKSPACE_TAXES.route,
},
},
},
[SCREENS.NOT_FOUND]: '*',
Expand Down
3 changes: 3 additions & 0 deletions src/libs/Navigation/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,9 @@ type CentralPaneNavigatorParamList = {
policyID: string;
categoryName: string;
};
[SCREENS.WORKSPACE.TAXES]: {
policyID: string;
};
};

type WorkspaceSwitcherNavigatorParamList = {
Expand Down
6 changes: 6 additions & 0 deletions src/pages/workspace/WorkspaceInitialPage.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -161,6 +161,12 @@ function WorkspaceInitialPage({policyDraft, policy: policyProp, policyMembers, r
action: singleExecution(waitForNavigate(() => Navigation.navigate(ROUTES.WORKSPACE_TAGS.getRoute(policyID)))),
routeName: SCREENS.WORKSPACE.TAGS,
},
{
translationKey: 'workspace.common.taxes',
icon: Expensicons.Tax,
action: singleExecution(waitForNavigate(() => Navigation.navigate(ROUTES.WORKSPACE_TAXES.getRoute(policyID)))),
routeName: SCREENS.WORKSPACE.TAXES,
},
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
];

const menuItems: WorkspaceMenuItem[] = [
Expand Down
125 changes: 125 additions & 0 deletions src/pages/workspace/taxes/WorkspaceTaxesPage.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,125 @@
import type {StackScreenProps} from '@react-navigation/stack';
import React, {useMemo, useState} from 'react';
import {View} from 'react-native';
import HeaderWithBackButton from '@components/HeaderWithBackButton';
import Icon from '@components/Icon';
import * as Expensicons from '@components/Icon/Expensicons';
import * as Illustrations from '@components/Icon/Illustrations';
import ScreenWrapper from '@components/ScreenWrapper';
import SelectionList from '@components/SelectionList';
import TableListItem from '@components/SelectionList/TableListItem';
import Text from '@components/Text';
import useLocalize from '@hooks/useLocalize';
import useTheme from '@hooks/useTheme';
import useThemeStyles from '@hooks/useThemeStyles';
import useWindowDimensions from '@hooks/useWindowDimensions';
import type {CentralPaneNavigatorParamList} from '@navigation/types';
import AdminPolicyAccessOrNotFoundWrapper from '@pages/workspace/AdminPolicyAccessOrNotFoundWrapper';
import PaidPolicyAccessOrNotFoundWrapper from '@pages/workspace/PaidPolicyAccessOrNotFoundWrapper';
import withPolicyAndFullscreenLoading from '@pages/workspace/withPolicyAndFullscreenLoading';
import type {WithPolicyAndFullscreenLoadingProps} from '@pages/workspace/withPolicyAndFullscreenLoading';
import type SCREENS from '@src/SCREENS';

type PolicyForList = {
value: string;
text: string;
keyForList: string;
isSelected: boolean;
rightElement: React.ReactNode;
};

type WorkspaceTaxesPageProps = WithPolicyAndFullscreenLoadingProps & StackScreenProps<CentralPaneNavigatorParamList, typeof SCREENS.WORKSPACE.TAXES>;

function WorkspaceTaxesPage({policy}: WorkspaceTaxesPageProps) {
const {isSmallScreenWidth} = useWindowDimensions();
const styles = useThemeStyles();
const theme = useTheme();
const {translate} = useLocalize();
const [selectedTaxes, setSelectedTaxes] = useState<string[]>([]);
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved

const taxesList = useMemo<PolicyForList[]>(
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
() =>
Object.entries(policy?.taxRates?.taxes ?? {}).map(([key, value]) => ({
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
// TODO: Clean up: check if all properties are needed
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
value: value.name,
text: value.name,
keyForList: key,
isSelected: !!selectedTaxes.includes(key),
rightElement: (
// TODO: Extract this into a separate component together with WorkspaceCategoriesPage
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
<View style={styles.flexRow}>
<Text style={[styles.disabledText, styles.alignSelfCenter]}>{value.isDisabled ? translate('workspace.common.disabled') : translate('workspace.common.enabled')}</Text>
<View style={[styles.p1, styles.pl2]}>
<Icon
src={Expensicons.ArrowRight}
fill={theme.icon}
/>
</View>
</View>
),
})),
[policy?.taxRates?.taxes, selectedTaxes, styles, theme.icon, translate],
);

const toggleTax = (tax: PolicyForList) => {
setSelectedTaxes((prev) => {
if (prev.includes(tax.keyForList)) {
return prev.filter((item) => item !== tax.keyForList);
}
return [...prev, tax.keyForList];
});
};

const toggleAllTaxes = () => {
const isAllSelected = selectedTaxes.length === taxesList.length;
if (isAllSelected) {
setSelectedTaxes([]);
} else {
setSelectedTaxes(taxesList.map((item) => item.keyForList));
}
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
};

const getCustomListHeader = () => (
<View style={[styles.flex1, styles.flexRow, styles.justifyContentBetween, styles.pl3, styles.pr9]}>
<Text style={styles.searchInputStyle}>{translate('common.name')}</Text>
<Text style={[styles.searchInputStyle, styles.textAlignCenter]}>{translate('statusPage.status')}</Text>
</View>
);

return (
<AdminPolicyAccessOrNotFoundWrapper policyID={policy?.id ?? ''}>
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
<PaidPolicyAccessOrNotFoundWrapper policyID={policy?.id ?? ''}>
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
<ScreenWrapper
includeSafeAreaPaddingBottom={false}
style={[styles.defaultModalContainer]}
testID={WorkspaceTaxesPage.displayName}
shouldShowOfflineIndicatorInWideScreen
>
<HeaderWithBackButton
icon={Illustrations.Coins}
title={translate('workspace.common.taxes')}
shouldShowBackButton={isSmallScreenWidth}
/>
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
<View style={[styles.ph5, styles.pb5]}>
<Text style={[styles.textNormal, styles.colorMuted]}>{translate('workspace.taxes.subtitle')}</Text>
</View>
<SelectionList
filip-solecki marked this conversation as resolved.
Show resolved Hide resolved
canSelectMultiple
sections={[{data: taxesList, indexOffset: 0, isDisabled: false}]}
onCheckboxPress={toggleTax}
onSelectRow={() => {}}
onSelectAll={toggleAllTaxes}
showScrollIndicator
ListItem={TableListItem}
customListHeader={getCustomListHeader()}
listHeaderWrapperStyle={[styles.ph9, styles.pv3, styles.pb5]}
/>
</ScreenWrapper>
</PaidPolicyAccessOrNotFoundWrapper>
</AdminPolicyAccessOrNotFoundWrapper>
);
}

WorkspaceTaxesPage.displayName = 'WorkspaceTaxesPage';

export default withPolicyAndFullscreenLoading(WorkspaceTaxesPage);
Loading