mirror of
https://github.com/emo2007/block-accounting.git
synced 2025-01-18 07:26:27 +00:00
org emp multi back intg
This commit is contained in:
parent
1082e2fa6c
commit
e6611d88fa
@ -32,6 +32,13 @@ export type PublicKey = {
|
||||
export type DeployResponse = {
|
||||
title: string;
|
||||
owners: PublicKey[];
|
||||
confirmations: number;
|
||||
confirmations: string;
|
||||
};
|
||||
export type MultiSigResponse = {};
|
||||
|
||||
export type InvitationResponse = {
|
||||
token: string;
|
||||
token_expired_at: number;
|
||||
refresh_token: string;
|
||||
refresh_token_expired_at: number;
|
||||
};
|
||||
|
@ -1,6 +1,7 @@
|
||||
import { globalService } from "../axios/global-api";
|
||||
import { AxiosResponse } from "axios";
|
||||
import Cookies from "js-cookie";
|
||||
|
||||
import {
|
||||
LoginResponse,
|
||||
OrganizationsResponse,
|
||||
@ -9,6 +10,7 @@ import {
|
||||
AddPartResponse,
|
||||
DeployResponse,
|
||||
MultiSigResponse,
|
||||
InvitationResponse,
|
||||
} from "./api-types";
|
||||
export class AccountingService {
|
||||
async login(seedKey: string): Promise<AxiosResponse<LoginResponse>> {
|
||||
@ -44,10 +46,16 @@ export class AccountingService {
|
||||
}
|
||||
|
||||
async getEmployees(
|
||||
organizationId: string
|
||||
organizationId: string,
|
||||
ids: string[]
|
||||
): Promise<AxiosResponse<ParticipantsResponse>> {
|
||||
return await globalService.get(
|
||||
`organizations/${organizationId}/participants`
|
||||
return await globalService.post(
|
||||
`organizations/${organizationId}/participants/fetch`,
|
||||
{
|
||||
data: {
|
||||
ids: [...ids, organizationId],
|
||||
},
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
@ -69,16 +77,19 @@ export class AccountingService {
|
||||
|
||||
// POST /organizations/{organization_id}/multisig
|
||||
async deployMultisig(
|
||||
organization_id: string,
|
||||
organizationId: string,
|
||||
title: string,
|
||||
owners: string[],
|
||||
confirmations: number
|
||||
): Promise<AxiosResponse<DeployResponse>> {
|
||||
return await globalService.post(`${organization_id}/multi-sig`, {
|
||||
title,
|
||||
owners,
|
||||
confirmations,
|
||||
});
|
||||
return await globalService.post(
|
||||
`organizations/${organizationId}/multisig`,
|
||||
{
|
||||
title,
|
||||
owners,
|
||||
confirmations,
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
async getAllMultisigsByOrganizationId(
|
||||
@ -86,6 +97,25 @@ export class AccountingService {
|
||||
): Promise<AxiosResponse<MultiSigResponse>> {
|
||||
return await globalService.get(`organizations/${organizationId}/multisig`);
|
||||
}
|
||||
|
||||
async sentInvitation(
|
||||
hash: string,
|
||||
name: string,
|
||||
credentals: {
|
||||
email: string;
|
||||
phone: string;
|
||||
telegram: string;
|
||||
},
|
||||
mnemonic: string
|
||||
): Promise<AxiosResponse<InvitationResponse>> {
|
||||
return await globalService.post(`/invite/${hash}/join`, {
|
||||
data: {
|
||||
name,
|
||||
credentals,
|
||||
mnemonic,
|
||||
},
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
export const apiService = new AccountingService();
|
||||
|
@ -14,17 +14,32 @@ export function LoginPage() {
|
||||
const { passwordVisible, setPasswordVisible } = useLoginHooks();
|
||||
const [inp, setInp] = useState("");
|
||||
//
|
||||
const [seed, setSeed] = useState<string[]>(["airport","donate","language","disagree","dumb","access","insect","tribe","ozone","humor","foot","jealous","much","digital",]);
|
||||
const [seed, setSeed] = useState<string[]>([
|
||||
"airport",
|
||||
"donate",
|
||||
"language",
|
||||
"disagree",
|
||||
"dumb",
|
||||
"access",
|
||||
"insect",
|
||||
"tribe",
|
||||
"ozone",
|
||||
"humor",
|
||||
"foot",
|
||||
"jealous",
|
||||
"much",
|
||||
"digital",
|
||||
]);
|
||||
|
||||
// const [seed, setSeed] = useState<string[]>(["melody","correct","brain","slide","flip","polar","asset","know","pencil","major","smile","vital","nominee","merge","addict"]);
|
||||
const [disabled, setDisabled] = useState(true);
|
||||
const router = useRouter();
|
||||
|
||||
const onLogin = async () => {
|
||||
const result = await apiService.login(seed.join(' '));
|
||||
const result = await apiService.login(seed.join(" "));
|
||||
// if (result) {
|
||||
Cookies.set("accessToken", result.data.token);
|
||||
router.push("/organization");
|
||||
Cookies.set("accessToken", result.data.token);
|
||||
router.push("/organization");
|
||||
// }
|
||||
};
|
||||
const onRegister = async () => {
|
||||
@ -33,7 +48,7 @@ export function LoginPage() {
|
||||
Cookies.set("accessToken", result.data.token);
|
||||
router.push("/organization");
|
||||
}
|
||||
}
|
||||
};
|
||||
// const getSeed = (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||
// setSeed(inp);
|
||||
// };
|
||||
@ -44,7 +59,7 @@ export function LoginPage() {
|
||||
}, [inp]);
|
||||
|
||||
const onSubmitHandler = async () => {
|
||||
setSeed((prevState) => prevState ? [...prevState, inp] : [inp]);
|
||||
setSeed((prevState) => (prevState ? [...prevState, inp] : [inp]));
|
||||
setInp("");
|
||||
};
|
||||
|
||||
@ -91,14 +106,17 @@ export function LoginPage() {
|
||||
</Button>
|
||||
</Space.Compact>
|
||||
</div>
|
||||
{ <div
|
||||
className="flex flex-row w-[700px] gap-3 content-box flex-wrap
|
||||
{
|
||||
<div
|
||||
className="flex flex-row w-[700px] gap-3 content-box flex-wrap
|
||||
"
|
||||
>
|
||||
{seed && seed.map((element: string, index: number) => (
|
||||
<SeedItem key={index} seed={element} />
|
||||
))}
|
||||
</div> }
|
||||
>
|
||||
{seed &&
|
||||
seed.map((element: string, index: number) => (
|
||||
<SeedItem key={index} seed={element} />
|
||||
))}
|
||||
</div>
|
||||
}
|
||||
<Button
|
||||
onClick={onLogin}
|
||||
style={{ width: "150px" }}
|
||||
@ -108,10 +126,10 @@ export function LoginPage() {
|
||||
Login
|
||||
</Button>
|
||||
<Button
|
||||
onClick={onRegister}
|
||||
style={{ width: "150px" }}
|
||||
type="primary"
|
||||
size="large"
|
||||
onClick={onRegister}
|
||||
style={{ width: "150px" }}
|
||||
type="primary"
|
||||
size="large"
|
||||
>
|
||||
Join
|
||||
</Button>
|
||||
|
@ -14,7 +14,6 @@ type OrgItemProps = {
|
||||
export const OrganizationCard: FC<OrgItemProps> = ({ element }) => {
|
||||
const router = useRouter();
|
||||
const id: any = element.id;
|
||||
|
||||
return (
|
||||
<>
|
||||
<Card
|
||||
@ -22,7 +21,10 @@ export const OrganizationCard: FC<OrgItemProps> = ({ element }) => {
|
||||
type="inner"
|
||||
extra={
|
||||
<Link
|
||||
href={{ pathname: "/organization/overview/dashboard/", query: id }}
|
||||
href={{
|
||||
pathname: "/organization/overview/dashboard/",
|
||||
query: { id: id },
|
||||
}}
|
||||
>
|
||||
Go To Overview
|
||||
</Link>
|
||||
|
@ -13,6 +13,7 @@ import { apiService } from "../axios/global.service";
|
||||
import { OrgForm } from "./OrgForm";
|
||||
import { OrganizationCard } from "./OrgCard";
|
||||
import { FolderOpenTwoTone } from "@ant-design/icons";
|
||||
import useOrganizationsHooks from "@/hooks/organizations";
|
||||
import {
|
||||
Organization,
|
||||
NewOrgResponse,
|
||||
@ -20,7 +21,8 @@ import {
|
||||
} from "../axios/api-types";
|
||||
import Cookies from "js-cookie";
|
||||
export function OrgCreatePage() {
|
||||
const [organizations, setOrganizations] = useState<Organization[]>([]);
|
||||
//const [organizations, setOrganizations] = useState<Organization[]>([]);
|
||||
const { organizations, setOrganizations } = useOrganizationsHooks();
|
||||
const [isModalOpen, setIsModalOpen] = useState(false);
|
||||
const [formData, setFormData] = useState({
|
||||
name: "",
|
||||
@ -47,16 +49,19 @@ export function OrgCreatePage() {
|
||||
setIsModalOpen(false);
|
||||
};
|
||||
|
||||
console.log(organizations);
|
||||
|
||||
const handleCancel = () => {
|
||||
setIsModalOpen(false);
|
||||
};
|
||||
useEffect(() => {
|
||||
loadOrganizations();
|
||||
}, []);
|
||||
|
||||
const loadOrganizations = async () => {
|
||||
const result = await apiService.getOrganizations();
|
||||
if (result) {
|
||||
setOrganizations(result.data.items);
|
||||
setOrganizations(result.data.items || []);
|
||||
}
|
||||
};
|
||||
return (
|
||||
@ -69,7 +74,7 @@ export function OrgCreatePage() {
|
||||
</div>
|
||||
<div></div>
|
||||
<div className="flex flex-col relative w-full h-3/4 items-center overflow-scroll gap-10 p-10 z-0">
|
||||
{organizations.length === 0 && (
|
||||
{organizations.length && (
|
||||
<FolderOpenTwoTone style={{ fontSize: "400%" }} />
|
||||
)}
|
||||
{organizations.length ? (
|
||||
|
@ -1,10 +1,17 @@
|
||||
"use client";
|
||||
import React, { useState, useEffect } from "react";
|
||||
import { useRouter, useSearchParams } from "next/navigation";
|
||||
import { Card } from "antd";
|
||||
import { useRouter } from "next/navigation";
|
||||
import { useSearchParams } from "next/navigation";
|
||||
import { UserOutlined } from "@ant-design/icons";
|
||||
import { Button, List, Divider, Typography, Avatar, Skeleton } from "antd";
|
||||
import { OrganizationCard } from "@/app/orgCreate/OrgCard";
|
||||
import { Organization } from "@/app/axios/api-types";
|
||||
import useOrganizationsHooks from "@/hooks/organizations";
|
||||
import { apiService } from "@/app/axios/global.service";
|
||||
|
||||
type OrgItemProps = {
|
||||
element: Organization;
|
||||
};
|
||||
interface DataType {
|
||||
gender?: string;
|
||||
name: {
|
||||
@ -35,53 +42,51 @@ const data = [
|
||||
const { Title } = Typography;
|
||||
|
||||
export function OrgProfile() {
|
||||
const { organizations, setOrganizations, loadOrganizations } =
|
||||
useOrganizationsHooks();
|
||||
const [organization, setOrganization] = useState<Organization>({
|
||||
id: "",
|
||||
name: "",
|
||||
address: "",
|
||||
});
|
||||
const [initLoading, setInitLoading] = useState(true);
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [dataEmployees, setData] = useState<DataType[]>([]);
|
||||
const [list, setList] = useState<DataType[]>([]);
|
||||
const [list, setList] = useState<any[]>([]);
|
||||
|
||||
const router = useRouter();
|
||||
const pathname = useSearchParams();
|
||||
console.log(pathname.getAll("query"));
|
||||
const searchParams = useSearchParams();
|
||||
useEffect(() => {
|
||||
const id = searchParams.get("id") || "";
|
||||
const filteredOrganization = organizations.find(
|
||||
(element) => element.id === id
|
||||
);
|
||||
|
||||
if (filteredOrganization) {
|
||||
loadEmployees(id);
|
||||
setOrganization(filteredOrganization);
|
||||
setInitLoading(false);
|
||||
}
|
||||
}, [organizations]);
|
||||
|
||||
useEffect(() => {
|
||||
loadOrganizations();
|
||||
}, []);
|
||||
|
||||
const loadEmployees = async (id: string) => {
|
||||
const data: any = await apiService.getEmployees(id, []);
|
||||
setList(data.data.participants);
|
||||
};
|
||||
|
||||
const onNextPageHandler = () => {
|
||||
router.push("/organization/overview/employees");
|
||||
};
|
||||
const onMultisigPageHandler = () => {
|
||||
router.push("/organization/overview/multiSig");
|
||||
router.push("/organization/overview/multiSig/?id=" + organization.id);
|
||||
};
|
||||
useEffect(() => {
|
||||
fetch(fakeDataUrl)
|
||||
.then((res) => res.json())
|
||||
.then((res) => {
|
||||
setInitLoading(false);
|
||||
setData(res.results);
|
||||
setList(res.results);
|
||||
});
|
||||
}, []);
|
||||
const onLoadMore = () => {
|
||||
setLoading(true);
|
||||
setList(
|
||||
dataEmployees.concat(
|
||||
[...new Array(count)].map(() => ({
|
||||
loading: true,
|
||||
name: {},
|
||||
picture: {},
|
||||
}))
|
||||
)
|
||||
);
|
||||
fetch(fakeDataUrl)
|
||||
.then((res) => res.json())
|
||||
.then((res) => {
|
||||
const newData = dataEmployees.concat(res.results);
|
||||
setData(newData);
|
||||
setList(newData);
|
||||
setLoading(false);
|
||||
// Resetting window's offsetTop so as to display react-virtualized demo underfloor.
|
||||
// In real scene, you can using public method of react-virtualized:
|
||||
// https://stackoverflow.com/questions/46700726/how-to-use-public-method-updateposition-of-react-virtualized
|
||||
window.dispatchEvent(new Event("resize"));
|
||||
});
|
||||
loadEmployees(organization.id);
|
||||
setInitLoading(false);
|
||||
};
|
||||
const loadMore =
|
||||
!initLoading && !loading ? (
|
||||
@ -101,7 +106,12 @@ export function OrgProfile() {
|
||||
<div className="flex flex-row w-full h-full bg-slate-50 p-8">
|
||||
<div className="flex flex-col w-11/12 ">
|
||||
<Title style={{ color: "#302d43", textIndent: 15 }}>Dashboard</Title>
|
||||
<Card
|
||||
|
||||
<div style={{ width: "60%" }} className="flex flex-col ">
|
||||
{organization && <OrganizationCard element={organization} />}
|
||||
</div>
|
||||
|
||||
{/* <Card
|
||||
title="Organization Name"
|
||||
bordered={false}
|
||||
style={{ width: "60%" }}
|
||||
@ -109,7 +119,7 @@ export function OrgProfile() {
|
||||
<p>Address</p>
|
||||
<p>Phone</p>
|
||||
<p>Description</p>
|
||||
</Card>
|
||||
</Card> */}
|
||||
<div className="flex w-full justify-end ">
|
||||
<Button
|
||||
type="primary"
|
||||
@ -161,23 +171,27 @@ export function OrgProfile() {
|
||||
itemLayout="horizontal"
|
||||
loadMore={loadMore}
|
||||
dataSource={list}
|
||||
renderItem={(item) => (
|
||||
<List.Item
|
||||
actions={[
|
||||
<a key="list-loadmore-edit">edit</a>,
|
||||
<a key="list-loadmore-more">more</a>,
|
||||
]}
|
||||
>
|
||||
<Skeleton avatar title={false} loading={item.loading} active>
|
||||
<List.Item.Meta
|
||||
avatar={<Avatar icon={<UserOutlined />} />}
|
||||
title={<a href="https://ant.design">{item.name?.last}</a>}
|
||||
description="1Lbcfr7sAHTD9CgdQo3HTMTkV8LK4ZnX71"
|
||||
/>
|
||||
<div>wallet address</div>
|
||||
</Skeleton>
|
||||
</List.Item>
|
||||
)}
|
||||
renderItem={(item) => {
|
||||
console.log(item);
|
||||
|
||||
return (
|
||||
<List.Item
|
||||
actions={[
|
||||
<a key="list-loadmore-edit">edit</a>,
|
||||
<a key="list-loadmore-more">more</a>,
|
||||
]}
|
||||
>
|
||||
<Skeleton avatar title={false} loading={item.loading} active>
|
||||
<List.Item.Meta
|
||||
avatar={<Avatar icon={<UserOutlined />} />}
|
||||
title={<a href="https://ant.design">{item.name}</a>}
|
||||
description="1Lbcfr7sAHTD9CgdQo3HTMTkV8LK4ZnX71"
|
||||
/>
|
||||
<div>wallet address</div>
|
||||
</Skeleton>
|
||||
</List.Item>
|
||||
);
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
@ -1,6 +1,7 @@
|
||||
"use client";
|
||||
import React from "react";
|
||||
import { OrgProfile } from "./id";
|
||||
import { OrgProfile } from "./[id]";
|
||||
import { useSearchParams } from "next/navigation";
|
||||
|
||||
export default function Home() {
|
||||
return (
|
||||
|
@ -6,8 +6,26 @@
|
||||
import React, { useState, useEffect } from "react";
|
||||
import { Input, Button, Typography, Space } from "antd";
|
||||
import { MailOutlined, CopyOutlined } from "@ant-design/icons";
|
||||
import { apiService } from "@/app/axios/global.service";
|
||||
|
||||
export function EmployeeInvitatonPage() {
|
||||
const { Text } = Typography;
|
||||
const [values, setValues] = useState({
|
||||
email: "ower@gmail.com",
|
||||
phone: "79999999999",
|
||||
telegram: "@ower",
|
||||
});
|
||||
|
||||
async function sendInvite() {
|
||||
const data = await apiService.sentInvitation(
|
||||
"",
|
||||
"Alex",
|
||||
values,
|
||||
"short orient camp maple lend pole balance token pledge fat analyst badge art happy sadsad"
|
||||
);
|
||||
console.log(data);
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="flex relative overflow-hidden flex-col w-2/3 h-3/4 items-center justify-center gap-10 bg-white border-solid border rounded-md border-neutral-300 p-10 text-neutral-500">
|
||||
<div className="w-full h-20 bg-[#1677FF] absolute top-0 flex items-center justify-center">
|
||||
@ -16,12 +34,21 @@ export function EmployeeInvitatonPage() {
|
||||
<div className="flex flex-col w-9/12 pb-36 gap-5 items-start ">
|
||||
<Text type="secondary">Invite new Employee</Text>
|
||||
<Space.Compact style={{ width: "100%" }}>
|
||||
<Input size="large" addonBefore="E-mail" />
|
||||
<Input
|
||||
size="large"
|
||||
name="email"
|
||||
addonBefore="E-mail"
|
||||
value={values.email}
|
||||
onChange={(e) =>
|
||||
setValues((prev) => ({ ...prev, email: e.target.value }))
|
||||
}
|
||||
/>
|
||||
<Button
|
||||
size="large"
|
||||
icon={<MailOutlined />}
|
||||
style={{ width: 210 }}
|
||||
type="primary"
|
||||
onClick={sendInvite}
|
||||
>
|
||||
Send Invite
|
||||
</Button>
|
||||
|
@ -4,6 +4,10 @@ import { Button, Menu, List, Typography, Avatar, Skeleton } from "antd";
|
||||
import { UserOutlined } from "@ant-design/icons";
|
||||
import type { MenuProps } from "antd";
|
||||
import { PayOutBtn } from "@/app/ui/PayOutBtn";
|
||||
import { apiService } from "@/app/axios/global.service";
|
||||
import useOrganizationsHooks from "@/hooks/organizations";
|
||||
import { useSearchParams } from "next/navigation";
|
||||
import { Organization } from "@/app/axios/api-types";
|
||||
const count = 8;
|
||||
const fakeDataUrl = `https://randomuser.me/api/?results=${count}&inc=name,gender,email,nat,picture&noinfo`;
|
||||
interface DataType {
|
||||
@ -25,11 +29,35 @@ interface DataType {
|
||||
const { Title } = Typography;
|
||||
type MenuItem = Required<MenuProps>["items"][number];
|
||||
export function EmployeeList() {
|
||||
const {
|
||||
organizations,
|
||||
filteredOrganization,
|
||||
setOrganizations,
|
||||
loadOrganizations,
|
||||
} = useOrganizationsHooks();
|
||||
const [collapsed, setCollapsed] = useState(true);
|
||||
const [initLoading, setInitLoading] = useState(true);
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [dataEmployees, setData] = useState<DataType[]>([]);
|
||||
const [list, setList] = useState<DataType[]>([]);
|
||||
const [organization, setOrganization] = useState<Organization>({
|
||||
id: "",
|
||||
name: "",
|
||||
address: "",
|
||||
});
|
||||
const loadEmployees = async (id: string) => {
|
||||
const data: any = await apiService.getEmployees(id, []);
|
||||
setList(data.data.participants);
|
||||
};
|
||||
const searchParams = useSearchParams();
|
||||
useEffect(() => {
|
||||
const id = searchParams.get("id") || "";
|
||||
if (filteredOrganization) {
|
||||
loadEmployees(id);
|
||||
setOrganization(filteredOrganization);
|
||||
setInitLoading(false);
|
||||
}
|
||||
}, [organizations]);
|
||||
useEffect(() => {
|
||||
fetch(fakeDataUrl)
|
||||
.then((res) => res.json())
|
||||
@ -41,27 +69,7 @@ export function EmployeeList() {
|
||||
}, []);
|
||||
const onLoadMore = () => {
|
||||
setLoading(true);
|
||||
setList(
|
||||
dataEmployees.concat(
|
||||
[...new Array(count)].map(() => ({
|
||||
loading: true,
|
||||
name: {},
|
||||
picture: {},
|
||||
}))
|
||||
)
|
||||
);
|
||||
fetch(fakeDataUrl)
|
||||
.then((res) => res.json())
|
||||
.then((res) => {
|
||||
const newData = dataEmployees.concat(res.results);
|
||||
setData(newData);
|
||||
setList(newData);
|
||||
setLoading(false);
|
||||
// Resetting window's offsetTop so as to display react-virtualized demo underfloor.
|
||||
// In real scene, you can using public method of react-virtualized:
|
||||
// https://stackoverflow.com/questions/46700726/how-to-use-public-method-updateposition-of-react-virtualized
|
||||
window.dispatchEvent(new Event("resize"));
|
||||
});
|
||||
loadEmployees(organization.id);
|
||||
};
|
||||
const loadMore =
|
||||
!initLoading && !loading ? (
|
||||
@ -77,7 +85,7 @@ export function EmployeeList() {
|
||||
</div>
|
||||
) : null;
|
||||
return (
|
||||
<div className="flex flex-col w-full h-full gap-5 pb-20 px-30 p-8">
|
||||
<div className="flex flex-col w-full h-full gap-5 pb-20 px-30 p-10">
|
||||
<Title style={{ color: "#302d43", textIndent: 15 }}>Employee List</Title>
|
||||
<List
|
||||
className="demo-loadmore-list"
|
||||
@ -85,18 +93,22 @@ export function EmployeeList() {
|
||||
itemLayout="horizontal"
|
||||
loadMore={loadMore}
|
||||
dataSource={list}
|
||||
renderItem={(item) => (
|
||||
<List.Item actions={[<PayOutBtn />]}>
|
||||
<Skeleton avatar title={false} loading={item.loading} active>
|
||||
<List.Item.Meta
|
||||
avatar={<Avatar icon={<UserOutlined />} />}
|
||||
title={<a href="https://ant.design">{item.name?.last}</a>}
|
||||
description="1Lbcfr7sAHTD9CgdQo3HTMTkV8LK4ZnX71"
|
||||
/>
|
||||
<div>wallet address</div>
|
||||
</Skeleton>
|
||||
</List.Item>
|
||||
)}
|
||||
renderItem={(item) => {
|
||||
console.log(item);
|
||||
|
||||
return (
|
||||
<List.Item actions={[<PayOutBtn />]}>
|
||||
<Skeleton avatar title={false} loading={item.loading} active>
|
||||
<List.Item.Meta
|
||||
avatar={<Avatar icon={<UserOutlined />} />}
|
||||
title={<a href="https://ant.design">{item.name}</a>}
|
||||
description="1Lbcfr7sAHTD9CgdQo3HTMTkV8LK4ZnX71"
|
||||
/>
|
||||
<div>wallet address</div>
|
||||
</Skeleton>
|
||||
</List.Item>
|
||||
);
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
|
@ -6,14 +6,37 @@ import { Card } from "antd";
|
||||
import { WalletOutlined } from "@ant-design/icons";
|
||||
import type { InputNumberProps } from "antd";
|
||||
import { Col, InputNumber, Row, Slider, Select, Button } from "antd";
|
||||
|
||||
import { apiService } from "@/app/axios/global.service";
|
||||
import Cookies from "js-cookie";
|
||||
import { useRouter, useSearchParams } from "next/navigation";
|
||||
const { Title } = Typography;
|
||||
|
||||
export function MultisigPage() {
|
||||
const searchParams = useSearchParams();
|
||||
const [multiSigData, setmultiSigData] = useState({
|
||||
title: "",
|
||||
owners: [],
|
||||
confirmations: 0,
|
||||
organizationId: searchParams.get("id") || "",
|
||||
});
|
||||
const router = useRouter();
|
||||
const [inputValue, setInputValue] = useState(1);
|
||||
const onChange: InputNumberProps["onChange"] = (newValue) => {
|
||||
setInputValue(newValue as number);
|
||||
};
|
||||
|
||||
const onDeployMultisig = async () => {
|
||||
const result = await apiService.deployMultisig(
|
||||
multiSigData.organizationId,
|
||||
multiSigData.title,
|
||||
multiSigData.owners,
|
||||
multiSigData.confirmations
|
||||
);
|
||||
if (result) {
|
||||
//Cookies.set("multisig", {result.data.confirmations, result.data.owners, result.data.organizationId, result.data.title});
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="flex flex-col w-full h-full p-8 gap-10 ">
|
||||
<div className="flex flex-col w-1/3">
|
||||
@ -93,7 +116,7 @@ export function MultisigPage() {
|
||||
</Card>
|
||||
</div>
|
||||
<div className="flex w-full justify-end pr-5">
|
||||
<Button size={"large"} type="primary">
|
||||
<Button onClick={onDeployMultisig} size={"large"} type="primary">
|
||||
Create Multisig
|
||||
</Button>
|
||||
</div>
|
||||
|
31
front/src/hooks/organizations.tsx
Normal file
31
front/src/hooks/organizations.tsx
Normal file
@ -0,0 +1,31 @@
|
||||
import { useState } from "react";
|
||||
import { Organization } from "@/app/axios/api-types";
|
||||
import { apiService } from "@/app/axios/global.service";
|
||||
import { useSearchParams } from "next/navigation";
|
||||
|
||||
export default function useOrganizationsHooks() {
|
||||
const [organizations, setOrganizations] = useState<Organization[]>([]);
|
||||
const [filteredOrganization, setFilteredOrganization] = useState<
|
||||
Organization[]
|
||||
>([]);
|
||||
const loadOrganizations = async () => {
|
||||
const result = await apiService.getOrganizations();
|
||||
if (result) {
|
||||
setOrganizations(result.data.items || []);
|
||||
}
|
||||
const searchParams = useSearchParams();
|
||||
const id = searchParams.get("id") || "";
|
||||
const filteredOrg = organizations.find((element) => element.id === id);
|
||||
|
||||
if (filteredOrg) {
|
||||
setFilteredOrganization(filteredOrg || []);
|
||||
}
|
||||
};
|
||||
|
||||
return {
|
||||
organizations,
|
||||
filteredOrganization,
|
||||
setOrganizations,
|
||||
loadOrganizations,
|
||||
};
|
||||
}
|
Loading…
Reference in New Issue
Block a user