From 212ce6bc7bbee4ad821a92b7d432452f8bd53b82 Mon Sep 17 00:00:00 2001 From: Tristan Cole Date: Thu, 27 May 2021 03:32:36 +0800 Subject: [PATCH] feat: improve master wallet management (#705) * master wallet management * bump * fix format money Co-authored-by: Michiel --- .../components/dashboard/MasterWalletCard.jsx | 12 +++ .../dashboard/MasterWalletManagementModal.jsx | 18 +++-- app/client/components/navBar/OrgSwitcher.tsx | 19 ++++- .../components/navBar/WalletListModal.tsx | 80 +++++++++++++++++++ app/client/nav.jsx | 11 +++ app/client/reducers/auth/types.ts | 1 + app/client/reducers/organisation/types.ts | 1 + app/client/sagas/authSagas.ts | 9 ++- app/server/schemas.py | 2 + config.py | 2 +- 10 files changed, 144 insertions(+), 11 deletions(-) create mode 100644 app/client/components/navBar/WalletListModal.tsx diff --git a/app/client/components/dashboard/MasterWalletCard.jsx b/app/client/components/dashboard/MasterWalletCard.jsx index 95d8337f1..a6216398e 100644 --- a/app/client/components/dashboard/MasterWalletCard.jsx +++ b/app/client/components/dashboard/MasterWalletCard.jsx @@ -11,6 +11,7 @@ import { HorizontalBar } from "react-chartjs-2"; import { formatMoney, getActiveToken } from "../../utils"; import MasterWalletManagementModal from "./MasterWalletManagementModal"; +import { browserHistory } from "../../createStore"; const { Text } = Typography; @@ -29,7 +30,18 @@ class MasterWalletCard extends React.Component { }; } + componentDidMount() { + if (window.location.pathname === "/manage") { + this.setState({ modalVisible: !this.state.modalVisible }); + } + } + toggleModal() { + if (window.location.pathname === "/manage") { + browserHistory.push("/"); + } else { + browserHistory.push("/manage"); + } this.setState({ modalVisible: !this.state.modalVisible }); } diff --git a/app/client/components/dashboard/MasterWalletManagementModal.jsx b/app/client/components/dashboard/MasterWalletManagementModal.jsx index ee59de490..8834201b5 100644 --- a/app/client/components/dashboard/MasterWalletManagementModal.jsx +++ b/app/client/components/dashboard/MasterWalletManagementModal.jsx @@ -1,5 +1,5 @@ import React, { Suspense, lazy } from "react"; -import { connect } from "react-redux"; +import { connect, useSelector } from "react-redux"; import { sempoObjects } from "../../reducers/rootReducer"; import { Modal, @@ -37,7 +37,11 @@ const MasterWalletManagementModal = props => { const [current, setCurrent] = React.useState(0); const [form] = Form.useForm(); - const orgName = "Reserve"; + const activeOrganisation = useSelector( + state => state.organisations.byId[Number(state.login.organisationId)] + ); + + const orgName = activeOrganisation && activeOrganisation.name; const onFinish = values => { const { recipient_blockchain_address, transfer_amount } = values; @@ -151,12 +155,12 @@ const MasterWalletManagementModal = props => { onCancel={props.handleCancel} footer={[ current > 0 && ( - ), current === 0 && ( - ), @@ -165,6 +169,7 @@ const MasterWalletManagementModal = props => { type="primary" onClick={() => form.submit()} loading={props.masterWallet.createStatus.isRequesting} + key={3} > Withdraw @@ -214,9 +219,9 @@ const MasterWalletManagementModal = props => { {current === 2 ? depositQr : null} {current === 0 || current === 2 ? depositAddress : null} {current === 0 - ? otherDetails.map(item => { + ? otherDetails.map((item, i) => { return ( - + {item.value} ); @@ -225,7 +230,6 @@ const MasterWalletManagementModal = props => { {current === 1 ? ( {withdrawal} ) : null} - {current === 0} diff --git a/app/client/components/navBar/OrgSwitcher.tsx b/app/client/components/navBar/OrgSwitcher.tsx index 6f22cc7a2..c96d10b5b 100644 --- a/app/client/components/navBar/OrgSwitcher.tsx +++ b/app/client/components/navBar/OrgSwitcher.tsx @@ -24,6 +24,7 @@ import { Organisation } from "../../reducers/organisation/types"; import { IntercomChat } from "../intercom/IntercomChat"; import { getActiveToken } from "../../utils"; +import { WalletListModal } from "./WalletListModal"; interface Props { icon: string; @@ -37,6 +38,7 @@ declare global { } const OrgSwitcher: React.FunctionComponent = props => { + const [modalVisible, setModalVisible] = React.useState(false); const [switcherActive, setSwitcherActive] = React.useState(false); const activeToken = useSelector((state: ReduxState) => getActiveToken(state)); @@ -81,6 +83,10 @@ const OrgSwitcher: React.FunctionComponent = props => { dispatch(LoginAction.updateActiveOrgRequest({ organisationIds })); }; + const toggleModal = () => { + setModalVisible(!modalVisible); + }; + let menu = ( = props => { : "" ]} > - + + Your Projects + + + } + > {Object.keys(tokenMap).map((id: string) => { const orgsForToken = tokenMap[id]; diff --git a/app/client/components/navBar/WalletListModal.tsx b/app/client/components/navBar/WalletListModal.tsx new file mode 100644 index 000000000..0b9b443ba --- /dev/null +++ b/app/client/components/navBar/WalletListModal.tsx @@ -0,0 +1,80 @@ +import React from "react"; +import { useSelector, useDispatch } from "react-redux"; +import { Modal, Tooltip, List } from "antd"; + +import { DollarCircleOutlined } from "@ant-design/icons"; +import { grey } from "@ant-design/colors"; +import { Organisation } from "../../reducers/organisation/types"; +import { ReduxState } from "../../reducers/rootReducer"; +import { LoginAction } from "../../reducers/auth/actions"; +import { formatMoney, getActiveToken } from "../../utils"; + +interface OuterProps { + isModalVisible: boolean; + handleOk: any; + handleCancel: any; +} + +export const WalletListModal = (props: OuterProps) => { + const dispatch: any = useDispatch(); + const organisations: Organisation[] = useSelector((state: ReduxState) => + Object.keys(state.organisations.byId).map( + id => state.organisations.byId[Number(id)] + ) + ); + const activeToken = useSelector((state: ReduxState) => getActiveToken(state)); + const symbol = activeToken && activeToken.symbol; + + let selectOrg = (organisationIds: number[]) => { + dispatch( + LoginAction.updateActiveOrgRequest({ + organisationIds, + isManageWallet: true + }) + ); + }; + + return ( + + + + + + + + ( + selectOrg([org.id])}> + Manage + + ]} + > + + + )} + /> + + + ); +}; diff --git a/app/client/nav.jsx b/app/client/nav.jsx index c1ae67f26..2193ab840 100644 --- a/app/client/nav.jsx +++ b/app/client/nav.jsx @@ -92,6 +92,17 @@ class Nav extends React.Component { title={"Dashboard"} isMultiOrg={true} /> + ) { try { + const isManageWallet = action.payload.isManageWallet; yield call(storeOrgIds, action.payload.organisationIds); // window.location.search = "?org=2" or "?query_organisations=1,2" @@ -166,9 +167,13 @@ function* saveOrgId( action.payload.organisationIds.toString() === query_params["query_organisations"] ) { - window.location.reload(); + isManageWallet + ? window.location.assign("/manage") + : window.location.reload(); } else { - window.location.assign("/"); + isManageWallet + ? window.location.assign("/manage") + : window.location.assign("/"); } } catch (e) { removeOrgIds(); diff --git a/app/server/schemas.py b/app/server/schemas.py index 44bf63d87..8608714e4 100644 --- a/app/server/schemas.py +++ b/app/server/schemas.py @@ -425,6 +425,8 @@ class OrganisationSchema(SchemaBase): valid_roles = fields.Raw() + master_wallet_balance = fields.Function(lambda obj: obj.queried_org_level_transfer_account.balance) + require_transfer_card = fields.Boolean(default=False) default_disbursement = fields.Function(lambda obj: int(obj.default_disbursement)) minimum_vendor_payout_withdrawal = fields.Function(lambda obj: int(obj.minimum_vendor_payout_withdrawal)) diff --git a/config.py b/config.py index b6034161e..98d37f73f 100755 --- a/config.py +++ b/config.py @@ -5,7 +5,7 @@ logging.basicConfig(level=env_loglevel) logg = logging.getLogger(__name__) -VERSION = '1.10.3' # Remember to bump this in every PR +VERSION = '1.10.4' # Remember to bump this in every PR logg.info('Loading configs at UTC {}'.format(datetime.datetime.utcnow()))