From 67abf8b4130f3a8eca78e8ab9de2d4214692b700 Mon Sep 17 00:00:00 2001 From: Monalisha Mishra <42746736+mishramonalisha76@users.noreply.github.com> Date: Tue, 10 Oct 2023 20:54:14 +0530 Subject: [PATCH] Create group chat comp (#766) * Chat profile refractor (#727) * fix: fixed minor bugs and made add wallets reusable * fix: refactored code * fix: fixed ens domains not working while adding more wallets to a group * fix: fixed back btn in add wallet and cofirm btn ui * fix: fixed theme in chat profile * fix: refractor of addWallet * fix: ens issues * fix: adde changes * fix: fixed lint issues * fix: fixed review comments * fix: fixed minor review comments --------- Co-authored-by: KlausMikhaelson * Refactoring of messageInput, chatBubble, connectButton,chatViewList, chatViewComponent (#718) * fix: added accoutn hook * fix: refactoring code of messageInput chatlist chatbubble * fix: fixed review comment * fix: fixed * fix: added reusables * fix: fixed toast * feat: create group modal * fix: fixed condition in messageInput * fix: fixed * fix: fixed code * feat: create group info validation done (#738) * fix: fixed autoconnect issues * Group criteria added (#764) * feat: created define criteria and add criteria * feat: added some more dropdowns in add criteria * fix: added critera options * feat: worked on criterias * Arbitrum changes (#735) * fix: inital implementation for arbitrum changes * fix: more changes * fix: added final changes * fix: fixed few ui issues * fix: fixed minor issue * fix: added operstor * fix: added dummy 2d data * fix: fixed test * feat: condition type added * feat: adder done * feat: all/any added on condition box * fix: minor fixes * fix: fixed typo * fix: fixed upload group image ui * Multiple group criteria (#747) * fix: fixed multiple criteria Ui * fix: fixed conditions code --------- Co-authored-by: KlausMikhaelson * docs: added Class Examples (#739) * docs: added documentation * docs: fixed chat class examples * docs: fix notif class * docs: fixed stream examples * docs: fix nft grp update * docs: fixed logs * Message Type Implementations (#730) * fix: added video & audio messages * fix: fixed meta, reaction & added intent & readReceipt * fix: added reply * fix: added composite, fixed receipt * fix: local tests (#744) * fix(env and index): fixes typo RECIPEINT to RECIPIENT (#743) fix #733 * Update README.md (#742) Changed the discord link * feat: add criteria done * fix: fixed themes * feat: rule deletion done * feat: condition update done * feat: add criteria update lablel done * feat: edit rule auto fill done * feat: pr changes done * feat: refactor type and hooks * fix: fixed mobilde view * fix: fixed scroll * feat: added icon in groupinfo to show if it is token gated or not (#748) * feat: added icon in groupinfo to show if it is token gated or not * fix: fixed minor text * feat: created group criteria info modal and fixed dropdown * fix: made changes as per the review * fix: removed more options from conditions in group info * fix: fixed minor issues --------- Co-authored-by: Monalisha Mishra * feat: drop down labels * fix: fixed * fix: fixed dropdown * Gp states fix (#763) * feat: conditions to chat added * feat: generation of payload done * refactor: rule generation done * feat: operator undefined fix * fix: fixed condition ui * feat: create group rest api call done * fix: fixed dropdown * feat: group detail info fix done * fix: added validation for guild --------- Co-authored-by: Abishek Bashyal --------- Co-authored-by: KlausMikhaelson Co-authored-by: Ashis Kumar Pradhan <38760485+akp111@users.noreply.github.com> Co-authored-by: Abishek Bashyal Co-authored-by: akp111 Co-authored-by: Aman Gupta Co-authored-by: strykerin Co-authored-by: Rahul Pandey Co-authored-by: dinesh <67892133+dinesh11515@users.noreply.github.com> Co-authored-by: Satyam <100528412+KlausMikhaelson@users.noreply.github.com> * feat: group msg fix (#765) * feat: group msg fix * fix: minor changes --------- Co-authored-by: Monalisha Mishra --------- Co-authored-by: KlausMikhaelson Co-authored-by: Abishek Bashyal Co-authored-by: Ashis Kumar Pradhan <38760485+akp111@users.noreply.github.com> Co-authored-by: akp111 Co-authored-by: Aman Gupta Co-authored-by: strykerin Co-authored-by: Rahul Pandey Co-authored-by: dinesh <67892133+dinesh11515@users.noreply.github.com> Co-authored-by: Satyam <100528412+KlausMikhaelson@users.noreply.github.com> --- .../sdk-backend-node/pushAPI/channel.ts | 192 +++ .../examples/sdk-backend-node/pushAPI/chat.ts | 227 +++ .../sdk-backend-node/pushAPI/encryption.ts | 38 + .../sdk-backend-node/pushAPI/notification.ts | 58 + .../sdk-backend-node/pushAPI/profile.ts | 38 + .../sdk-backend-node/pushAPI/stream.ts | 202 +++ .../src/app/ChatUITest/ChatProfile.tsx | 2 +- .../src/app/ChatUITest/ChatViewComponent.tsx | 9 +- .../sdk-frontend-react/src/app/app.tsx | 2 +- packages/restapi/README.md | 19 +- .../chat/ChatProfile/AddWalletContent.tsx | 525 +++--- .../chat/ChatProfile/ChatProfile.tsx | 410 ++--- .../chat/ChatProfile/GroupInfoModal.tsx | 1506 ++++++++++------- .../chat/ChatProfile/MemberListContainer.tsx | 294 ++-- .../chat/ChatProfile/MemberProfileCard.tsx | 154 ++ .../chat/ChatProfile/ProfileCard.tsx | 158 -- .../chat/ChatViewBubble/ChatViewBubble.tsx | 367 ++-- .../chat/ChatViewList/ChatViewList.tsx | 18 +- .../chat/ConnectButton/ConnectButton.tsx | 55 +- .../components/chat/ConnectButton/index.tsx | 33 +- .../chat/CreateGroup/AddButtons.tsx | 27 + .../chat/CreateGroup/AddCriteria.tsx | 557 ++++++ .../chat/CreateGroup/ConditionsComponent.tsx | 322 ++++ .../chat/CreateGroup/CreateGroupModal.tsx | 341 ++++ .../chat/CreateGroup/CreateGroupType.tsx | 275 +++ .../chat/CreateGroup/DefineCondition.tsx | 159 ++ .../chat/CreateGroup/OperatorContainer.tsx | 46 + .../lib/components/chat/CreateGroup/index.ts | 1 + .../chat/MessageInput/MessageInput.tsx | 634 ++++--- .../chat/MessageInput/VerificationFailed.tsx | 107 -- .../chat/ProfileHeader/AddWalletContent.tsx | 354 ---- .../lib/components/chat/constants/index.ts | 56 + .../src/lib/components/chat/exportedTypes.ts | 45 +- .../src/lib/components/chat/helpers/Modal.tsx | 101 -- .../src/lib/components/chat/helpers/Toast.tsx | 190 --- .../src/lib/components/chat/helpers/group.ts | 56 + .../src/lib/components/chat/helpers/helper.ts | 59 +- .../src/lib/components/chat/helpers/index.ts | 8 +- .../chat/helpers/tokenGatedGroup.ts | 67 + .../uiweb/src/lib/components/chat/index.ts | 1 + .../lib/components/chat/reusables/Button.tsx | 106 ++ .../chat/reusables/ChatSearchInput.tsx | 64 +- .../components/chat/reusables/Checkbox.tsx | 48 + .../{ChatProfile => reusables}/DropDown.tsx | 117 +- .../chat/reusables/DropDownInput.tsx | 116 ++ .../lib/components/chat/reusables/Modal.tsx | 138 ++ .../chat/{helpers => reusables}/NewToast.tsx | 48 +- .../chat/reusables/OptionButtons.tsx | 125 ++ .../chat/reusables/ProfileContainer.tsx | 47 + .../chat/reusables/QuantityInput.tsx | 117 ++ .../components/chat/reusables/TextArea.tsx | 82 + .../components/chat/reusables/TextInput.tsx | 107 ++ .../components/chat/reusables/ToggleInput.tsx | 114 ++ .../lib/components/chat/reusables/index.ts | 15 +- .../src/lib/components/chat/theme/index.ts | 118 +- .../src/lib/components/chat/types/index.ts | 76 + .../chat/types/tokenGatedGroupCreationType.ts | 75 + packages/uiweb/src/lib/config/constants.ts | 73 +- packages/uiweb/src/lib/helpers/chat/search.ts | 54 +- packages/uiweb/src/lib/hooks/chat/index.ts | 2 + .../uiweb/src/lib/hooks/chat/useAccount.ts | 35 + .../src/lib/hooks/chat/useCreateGatedGroup.ts | 43 + .../src/lib/hooks/chat/useCriteriaState.ts | 228 +++ .../src/lib/hooks/chat/usePushSendMessage.ts | 2 +- .../src/lib/hooks/chat/useUpdateGroup.ts | 52 + .../lib/hooks/chat/useVerifyAccessControl.ts | 2 +- packages/uiweb/src/lib/hooks/index.ts | 4 +- .../src/lib/hooks/useCreateChatProfile.ts | 33 + .../uiweb/src/lib/hooks/useDecryptPGPKey.ts | 39 + packages/uiweb/src/lib/icons/Arbitrum.tsx | 20 + packages/uiweb/src/lib/icons/Back.tsx | 6 +- packages/uiweb/src/lib/icons/Bsc.tsx | 17 + packages/uiweb/src/lib/icons/CaretUp.svg | 4 +- packages/uiweb/src/lib/icons/EditSvg.svg | 4 + packages/uiweb/src/lib/icons/Ethereum.tsx | 21 + packages/uiweb/src/lib/icons/MoreDark.tsx | 8 +- packages/uiweb/src/lib/icons/PolygonIcon.tsx | 16 + packages/uiweb/src/lib/icons/RemoveSvg.svg | 4 + packages/uiweb/src/lib/icons/SpamIcon.tsx | 8 + .../uiweb/src/lib/icons/TokenGatedIcon.svg | 3 + packages/uiweb/src/lib/icons/bsc.svg | 4 + packages/uiweb/src/lib/icons/ethereum.svg | 9 + packages/uiweb/src/lib/icons/optimisim.svg | 12 + packages/uiweb/src/lib/icons/polygon.svg | 13 + 84 files changed, 6884 insertions(+), 3058 deletions(-) create mode 100644 packages/examples/sdk-backend-node/pushAPI/channel.ts create mode 100644 packages/examples/sdk-backend-node/pushAPI/chat.ts create mode 100644 packages/examples/sdk-backend-node/pushAPI/encryption.ts create mode 100644 packages/examples/sdk-backend-node/pushAPI/notification.ts create mode 100644 packages/examples/sdk-backend-node/pushAPI/profile.ts create mode 100644 packages/examples/sdk-backend-node/pushAPI/stream.ts create mode 100644 packages/uiweb/src/lib/components/chat/ChatProfile/MemberProfileCard.tsx delete mode 100644 packages/uiweb/src/lib/components/chat/ChatProfile/ProfileCard.tsx create mode 100644 packages/uiweb/src/lib/components/chat/CreateGroup/AddButtons.tsx create mode 100644 packages/uiweb/src/lib/components/chat/CreateGroup/AddCriteria.tsx create mode 100644 packages/uiweb/src/lib/components/chat/CreateGroup/ConditionsComponent.tsx create mode 100644 packages/uiweb/src/lib/components/chat/CreateGroup/CreateGroupModal.tsx create mode 100644 packages/uiweb/src/lib/components/chat/CreateGroup/CreateGroupType.tsx create mode 100644 packages/uiweb/src/lib/components/chat/CreateGroup/DefineCondition.tsx create mode 100644 packages/uiweb/src/lib/components/chat/CreateGroup/OperatorContainer.tsx create mode 100644 packages/uiweb/src/lib/components/chat/CreateGroup/index.ts delete mode 100644 packages/uiweb/src/lib/components/chat/MessageInput/VerificationFailed.tsx delete mode 100644 packages/uiweb/src/lib/components/chat/ProfileHeader/AddWalletContent.tsx create mode 100644 packages/uiweb/src/lib/components/chat/constants/index.ts delete mode 100644 packages/uiweb/src/lib/components/chat/helpers/Modal.tsx delete mode 100644 packages/uiweb/src/lib/components/chat/helpers/Toast.tsx create mode 100644 packages/uiweb/src/lib/components/chat/helpers/group.ts create mode 100644 packages/uiweb/src/lib/components/chat/helpers/tokenGatedGroup.ts create mode 100644 packages/uiweb/src/lib/components/chat/reusables/Button.tsx create mode 100644 packages/uiweb/src/lib/components/chat/reusables/Checkbox.tsx rename packages/uiweb/src/lib/components/chat/{ChatProfile => reusables}/DropDown.tsx (66%) create mode 100644 packages/uiweb/src/lib/components/chat/reusables/DropDownInput.tsx create mode 100644 packages/uiweb/src/lib/components/chat/reusables/Modal.tsx rename packages/uiweb/src/lib/components/chat/{helpers => reusables}/NewToast.tsx (83%) create mode 100644 packages/uiweb/src/lib/components/chat/reusables/OptionButtons.tsx create mode 100644 packages/uiweb/src/lib/components/chat/reusables/ProfileContainer.tsx create mode 100644 packages/uiweb/src/lib/components/chat/reusables/QuantityInput.tsx create mode 100644 packages/uiweb/src/lib/components/chat/reusables/TextArea.tsx create mode 100644 packages/uiweb/src/lib/components/chat/reusables/TextInput.tsx create mode 100644 packages/uiweb/src/lib/components/chat/reusables/ToggleInput.tsx create mode 100644 packages/uiweb/src/lib/components/chat/types/index.ts create mode 100644 packages/uiweb/src/lib/components/chat/types/tokenGatedGroupCreationType.ts create mode 100644 packages/uiweb/src/lib/hooks/chat/useAccount.ts create mode 100644 packages/uiweb/src/lib/hooks/chat/useCreateGatedGroup.ts create mode 100644 packages/uiweb/src/lib/hooks/chat/useCriteriaState.ts create mode 100644 packages/uiweb/src/lib/hooks/chat/useUpdateGroup.ts create mode 100644 packages/uiweb/src/lib/hooks/useCreateChatProfile.ts create mode 100644 packages/uiweb/src/lib/hooks/useDecryptPGPKey.ts create mode 100644 packages/uiweb/src/lib/icons/Arbitrum.tsx create mode 100644 packages/uiweb/src/lib/icons/Bsc.tsx create mode 100644 packages/uiweb/src/lib/icons/EditSvg.svg create mode 100644 packages/uiweb/src/lib/icons/Ethereum.tsx create mode 100644 packages/uiweb/src/lib/icons/PolygonIcon.tsx create mode 100644 packages/uiweb/src/lib/icons/RemoveSvg.svg create mode 100644 packages/uiweb/src/lib/icons/SpamIcon.tsx create mode 100644 packages/uiweb/src/lib/icons/TokenGatedIcon.svg create mode 100644 packages/uiweb/src/lib/icons/bsc.svg create mode 100644 packages/uiweb/src/lib/icons/ethereum.svg create mode 100644 packages/uiweb/src/lib/icons/optimisim.svg create mode 100644 packages/uiweb/src/lib/icons/polygon.svg diff --git a/packages/examples/sdk-backend-node/pushAPI/channel.ts b/packages/examples/sdk-backend-node/pushAPI/channel.ts new file mode 100644 index 000000000..7e6b236a0 --- /dev/null +++ b/packages/examples/sdk-backend-node/pushAPI/channel.ts @@ -0,0 +1,192 @@ +import { PushAPI } from '@pushprotocol/restapi'; +import { config } from '../config'; +import { ethers } from 'ethers'; + +// CONFIGS +const { env, showAPIResponse } = config; + +export const runPushAPIChannelCases = async (): Promise => { + if (!process.env.WALLET_PRIVATE_KEY) { + console.log( + 'skipping PushAPI.channel examples, no private key passed in .env' + ); + return; + } + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + // Signer Generation + const provider = new ethers.providers.JsonRpcProvider( + 'https://goerli.blockpi.network/v1/rpc/public' // Goerli Provider + ); + const signer = new ethers.Wallet( + `0x${process.env.WALLET_PRIVATE_KEY}`, + provider + ); + const randomWallet1 = ethers.Wallet.createRandom().address; + const randomWallet2 = ethers.Wallet.createRandom().address; + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + const userAlice = await PushAPI.initialize(signer, { env }); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.info'); + const channelInfo = await userAlice.channel.info(); + if (showAPIResponse) { + console.log(channelInfo); + } + console.log('PushAPI.channel.info | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.search'); + const searchedChannels = await userAlice.channel.search( + 'push' // search by name or address + ); + if (showAPIResponse) { + console.log(searchedChannels); + } + console.log('PushAPI.channel.search | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.subscribers'); + const channelSubscribers = await userAlice.channel.subscribers(); + if (showAPIResponse) { + console.log(channelSubscribers); + } + console.log('PushAPI.channel.subscribers | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.send'); + if (channelInfo) { + const broadcastNotif = await userAlice.channel.send(['*'], { + notification: { + title: 'test', + body: 'test', + }, + }); + const targetedNotif = await userAlice.channel.send([randomWallet1], { + notification: { + title: 'test', + body: 'test', + }, + }); + const subsetNotif = await userAlice.channel.send( + [randomWallet1, randomWallet2], + { + notification: { + title: 'test', + body: 'test', + }, + } + ); + if (showAPIResponse) { + console.log(broadcastNotif, targetedNotif, subsetNotif); + } + console.log('PushAPI.channel.send | Response - 200 OK\n\n'); + } else { + console.log( + 'skipping PushAPI.channel.send as no channel exists with the signer\n\n' + ); + } + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + // These Examples requires wallet to hold some ETH & PUSH + const balance = await provider.getBalance(signer.address); + if (parseFloat(ethers.utils.formatEther(balance)) < 0.001) { + console.log( + 'skipping PushAPI.channel examples, wallet does not have enough balance to pay fee' + ); + } + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.create'); + if (channelInfo) { + console.log('skipping PushAPI.channel.create as it already exists\n\n'); + } else { + const createdChannel = await userAlice.channel.create({ + name: 'Test Channel', + description: 'Test Description', + icon: '', + url: 'https://push.org', + }); + if (showAPIResponse) { + console.log(createdChannel); + } + console.log('PushAPI.channel.create | Response - 200 OK\n\n'); + } + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.update'); + const updatedChannel = await userAlice.channel.update({ + name: 'Updated Name', + description: 'Testing new description', + url: 'https://google.com', + icon: '', + }); + if (showAPIResponse) { + console.log(updatedChannel); + } + console.log('PushAPI.channel.update | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.verify'); + // only verified channels can verify other channels (otherwise this action is skipped by sdk) + if (channelInfo.verified_status) { + const verifiedTrx = await userAlice.channel.verify( + '0x35B84d6848D16415177c64D64504663b998A6ab4' + ); + if (showAPIResponse) { + console.log(verifiedTrx); + } + } + console.log('PushAPI.channel.verify | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.setting'); + const channelSettingTrx = await userAlice.channel.setting([ + { type: 0, default: 1, description: 'My Notif Settings' }, + ]); + if (showAPIResponse) { + console.log(channelSettingTrx); + } + console.log('PushAPI.channel.setting | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.delegate.add'); + const addedDelegate = await userAlice.channel.delegate.add( + `eip155:5:${randomWallet1}` + ); + + if (showAPIResponse) { + console.log(addedDelegate); + } + console.log('PushAPI.channel.delegate.add | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.delegate.get'); + const delegates = await userAlice.channel.delegate.get(); + if (showAPIResponse) { + console.log(delegates); + } + console.log('PushAPI.channel.delegate.get | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.delegate.remove'); + const removedDelegate = await userAlice.channel.delegate.remove( + `eip155:5:${randomWallet1}` + ); + if (showAPIResponse) { + console.log(removedDelegate); + } + console.log('PushAPI.channel.delegate.remove | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.channel.alias.info'); + const aliasInfo = await userAlice.channel.alias.info({ + alias: '0x35B84d6848D16415177c64D64504663b998A6ab4', + aliasChain: 'POLYGON', + }); + if (showAPIResponse) { + console.log(aliasInfo); + } + console.log('PushAPI.channel.alias.info | Response - 200 OK\n\n'); +}; diff --git a/packages/examples/sdk-backend-node/pushAPI/chat.ts b/packages/examples/sdk-backend-node/pushAPI/chat.ts new file mode 100644 index 000000000..f34973ec1 --- /dev/null +++ b/packages/examples/sdk-backend-node/pushAPI/chat.ts @@ -0,0 +1,227 @@ +import { PushAPI } from '@pushprotocol/restapi'; +import { + adjectives, + animals, + colors, + uniqueNamesGenerator, +} from 'unique-names-generator'; +import { config } from '../config'; +import { generatePrivateKey, privateKeyToAccount } from 'viem/accounts'; +import { createWalletClient, http } from 'viem'; +import { goerli } from 'viem/chains'; + +// CONFIGS +const { env, showAPIResponse } = config; + +/***************** SAMPLE SIGNER GENERATION *********************/ +// Uing VIEM +// Random Wallet Signers +const signer = createWalletClient({ + account: privateKeyToAccount(generatePrivateKey()), + chain: goerli, + transport: http(), +}); +const signerAddress = signer.account.address; +const secondSigner = createWalletClient({ + account: privateKeyToAccount(generatePrivateKey()), + chain: goerli, + transport: http(), +}); +const secondSignerAddress = secondSigner.account.address; +const thirdSigner = createWalletClient({ + account: privateKeyToAccount(generatePrivateKey()), + chain: goerli, + transport: http(), +}); +const thirdSignerAddress = thirdSigner.account.address; + +// Dummy Wallet Addresses +const randomWallet1 = privateKeyToAccount(generatePrivateKey()).address; +const randomWallet2 = privateKeyToAccount(generatePrivateKey()).address; +const randomWallet3 = privateKeyToAccount(generatePrivateKey()).address; +/****************************************************************/ + +/***************** SAMPLE GROUP DATA ****************************/ +const groupName = uniqueNamesGenerator({ + dictionaries: [adjectives, colors, animals], +}); +const groupDescription = uniqueNamesGenerator({ + dictionaries: [adjectives, colors, animals], +}); +const groupImage = + ''; +/***************** SAMPLE GROUP DATA ****************************/ + +export const runPushAPIChatCases = async (): Promise => { + const userAlice = await PushAPI.initialize(signer, { env }); + const userBob = await PushAPI.initialize(secondSigner, { env }); + const tempUser = await PushAPI.initialize(thirdSigner, { env }); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.chat.list'); + const aliceChats = await userAlice.chat.list('CHATS'); + const aliceRequests = await userAlice.chat.list('REQUESTS'); + if (showAPIResponse) { + console.log(aliceChats); + console.log(aliceRequests); + } + console.log('PushAPI.chat.list | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.chat.latest'); + const aliceLatestChatWithBob = await userAlice.chat.latest( + secondSignerAddress + ); + if (showAPIResponse) { + console.log(aliceLatestChatWithBob); + } + console.log('PushAPI.chat.latest | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.chat.history'); + const aliceChatHistoryWithBob = await userAlice.chat.history( + secondSignerAddress + ); + if (showAPIResponse) { + console.log(aliceChatHistoryWithBob); + } + console.log('PushAPI.chat.history | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.chat.send'); + const aliceMessagesBob = await userAlice.chat.send(secondSignerAddress, { + content: 'Hello Bob!', + type: 'Text', + }); + if (showAPIResponse) { + console.log(aliceMessagesBob); + } + console.log('PushAPI.chat.send | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.chat.accept'); + const bobAcceptsRequest = await userBob.chat.accept(signerAddress); + if (showAPIResponse) { + console.log(bobAcceptsRequest); + } + console.log('PushAPI.chat.accept | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.chat.reject'); + await tempUser.chat.send(secondSignerAddress, { + content: 'Sending malicious message', + type: 'Text', + }); + const bobRejectsRequest = await userBob.chat.reject(thirdSignerAddress); + if (showAPIResponse) { + console.log(bobRejectsRequest); + } + console.log('PushAPI.chat.reject | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.chat.block'); + const AliceBlocksBob = await userAlice.chat.block([secondSignerAddress]); + if (showAPIResponse) { + console.log(AliceBlocksBob); + } + console.log('PushAPI.chat.block | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.chat.unblock'); + const AliceUnblocksBob = await userAlice.chat.unblock([secondSignerAddress]); + if (showAPIResponse) { + console.log(AliceUnblocksBob); + } + console.log('PushAPI.chat.unblock | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.group.create'); + const createdGroup = await userAlice.chat.group.create(groupName, { + description: groupDescription, + image: groupImage, + members: [randomWallet1, randomWallet2], + admins: [], + private: false, + }); + const groupChatId = createdGroup.chatId; // to be used in other examples + if (showAPIResponse) { + console.log(createdGroup); + } + console.log('PushAPI.group.create | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.group.permissions'); + const grouppermissions = await userAlice.chat.group.permissions(groupChatId); + if (showAPIResponse) { + console.log(grouppermissions); + } + console.log('PushAPI.group.permissions | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.group.info'); + const groupInfo = await userAlice.chat.group.info(groupChatId); + if (showAPIResponse) { + console.log(groupInfo); + } + console.log('PushAPI.group.info | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.group.update'); + const updatedGroup = await userAlice.chat.group.update(groupChatId, { + description: 'Updated Description', + }); + if (showAPIResponse) { + console.log(updatedGroup); + } + console.log('PushAPI.group.update | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.group.add'); + const addMember = await userAlice.chat.group.add(groupChatId, { + role: 'MEMBER', + accounts: [randomWallet3], + }); + if (showAPIResponse) { + console.log(addMember); + } + console.log('PushAPI.group.add | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.group.remove'); + const removeMember = await userAlice.chat.group.remove(groupChatId, { + role: 'MEMBER', + accounts: [randomWallet3], + }); + if (showAPIResponse) { + console.log(removeMember); + } + console.log('PushAPI.group.remove | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.group.join'); + const joinGrp = await userBob.chat.group.join(groupChatId); + if (showAPIResponse) { + console.log(joinGrp); + } + console.log('PushAPI.group.join | Response - 200 OK\n\n'); + //------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.group.leave'); + const leaveGrp = await userBob.chat.group.leave(groupChatId); + if (showAPIResponse) { + console.log(leaveGrp); + } + console.log('PushAPI.group.leave | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.group.reject'); + const sampleGrp = await userAlice.chat.group.create('Sample Grp', { + description: groupDescription, + image: groupImage, + members: [secondSignerAddress], // invite bob + admins: [], + private: true, + }); + await userBob.chat.group.reject(sampleGrp.chatId); + console.log('PushAPI.group.reject | Response - 200 OK\n\n'); +}; diff --git a/packages/examples/sdk-backend-node/pushAPI/encryption.ts b/packages/examples/sdk-backend-node/pushAPI/encryption.ts new file mode 100644 index 000000000..67300c65f --- /dev/null +++ b/packages/examples/sdk-backend-node/pushAPI/encryption.ts @@ -0,0 +1,38 @@ +import { PushAPI } from '@pushprotocol/restapi'; +import { config } from '../config'; +import { generatePrivateKey, privateKeyToAccount } from 'viem/accounts'; +import { createWalletClient, http } from 'viem'; +import { goerli } from 'viem/chains'; + +// CONFIGS +const { env, showAPIResponse } = config; + +/***************** SAMPLE SIGNER GENERATION *********************/ +// Uing VIEM +// Random Wallet Signers +const signer = createWalletClient({ + account: privateKeyToAccount(generatePrivateKey()), + chain: goerli, + transport: http(), +}); + +export const runPushAPIEncryptionCases = async (): Promise => { + const userAlice = await PushAPI.initialize(signer, { env }); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.encryption.info'); + const encryptionInfo = await userAlice.encryption.info(); + if (showAPIResponse) { + console.log(encryptionInfo); + } + console.log('PushAPI.encryption.info | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.encryption.update'); + const PGP_V3 = 'eip191-aes256-gcm-hkdf-sha256'; + const encryptionUpdate = await userAlice.encryption.update(PGP_V3 as any); + if (showAPIResponse) { + console.log(encryptionUpdate); + } + console.log('PushAPI.encryption.update | Response - 200 OK\n\n'); +}; diff --git a/packages/examples/sdk-backend-node/pushAPI/notification.ts b/packages/examples/sdk-backend-node/pushAPI/notification.ts new file mode 100644 index 000000000..579ea1e2b --- /dev/null +++ b/packages/examples/sdk-backend-node/pushAPI/notification.ts @@ -0,0 +1,58 @@ +import { PushAPI } from '@pushprotocol/restapi'; +import { config } from '../config'; +import { generatePrivateKey, privateKeyToAccount } from 'viem/accounts'; +import { createWalletClient, http } from 'viem'; +import { goerli } from 'viem/chains'; + +// CONFIGS +const { env, showAPIResponse } = config; + +/***************** SAMPLE SIGNER GENERATION *********************/ +// Uing VIEM +// Random Wallet Signers +const signer = createWalletClient({ + account: privateKeyToAccount(generatePrivateKey()), + chain: goerli, + transport: http(), +}); + +export const runPushAPINotificationCases = async (): Promise => { + const userAlice = await PushAPI.initialize(signer, { env }); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.notification.list'); + const inboxNotifications = await userAlice.notification.list('INBOX'); + const spamNotifications = await userAlice.notification.list('SPAM'); + if (showAPIResponse) { + console.log(inboxNotifications, spamNotifications); + } + console.log('PushAPI.notification.list | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.notification.subscribe'); + const subscribeResponse = await userAlice.notification.subscribe( + 'eip155:5:0xD8634C39BBFd4033c0d3289C4515275102423681' // channel to subscribe + ); + if (showAPIResponse) { + console.log(subscribeResponse); + } + console.log('PushAPI.notification.subscribe | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.notification.subscriptions'); + const aliceSubscriptions = await userAlice.notification.subscriptions(); + if (showAPIResponse) { + console.log(aliceSubscriptions); + } + console.log('PushAPI.notification.subscriptions | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.notification.unsubscribe'); + const unsubscribeResponse = await userAlice.notification.unsubscribe( + 'eip155:5:0xD8634C39BBFd4033c0d3289C4515275102423681' // channel to unsubscribe + ); + if (showAPIResponse) { + console.log(unsubscribeResponse); + } + console.log('PushAPI.notification.unsubscribe | Response - 200 OK\n\n'); +}; diff --git a/packages/examples/sdk-backend-node/pushAPI/profile.ts b/packages/examples/sdk-backend-node/pushAPI/profile.ts new file mode 100644 index 000000000..f72d93f1b --- /dev/null +++ b/packages/examples/sdk-backend-node/pushAPI/profile.ts @@ -0,0 +1,38 @@ +import { PushAPI } from '@pushprotocol/restapi'; +import { config } from '../config'; +import { generatePrivateKey, privateKeyToAccount } from 'viem/accounts'; +import { createWalletClient, http } from 'viem'; +import { goerli } from 'viem/chains'; + +// CONFIGS +const { env, showAPIResponse } = config; + +/***************** SAMPLE SIGNER GENERATION *********************/ +// Uing VIEM +// Random Wallet Signers +const signer = createWalletClient({ + account: privateKeyToAccount(generatePrivateKey()), + chain: goerli, + transport: http(), +}); + +export const runPushAPIProfileCases = async (): Promise => { + const userAlice = await PushAPI.initialize(signer, { env }); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.profile.info'); + const userAliceProfileInfo = await userAlice.profile.info(); + if (showAPIResponse) { + console.log(userAliceProfileInfo); + } + console.log('PushAPI.profile.info | Response - 200 OK\n\n'); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('PushAPI.profile.update'); + const updatedName = 'Bob The Builder'; + const response = await userAlice.profile.update({ name: updatedName }); + if (showAPIResponse) { + console.log(response); + } + console.log('PushAPI.profile.update | Response - 200 OK\n\n'); +}; diff --git a/packages/examples/sdk-backend-node/pushAPI/stream.ts b/packages/examples/sdk-backend-node/pushAPI/stream.ts new file mode 100644 index 000000000..e10c599c8 --- /dev/null +++ b/packages/examples/sdk-backend-node/pushAPI/stream.ts @@ -0,0 +1,202 @@ +import { PushAPI } from '@pushprotocol/restapi'; +import { config } from '../config'; +import { generatePrivateKey, privateKeyToAccount } from 'viem/accounts'; +import { createWalletClient, http } from 'viem'; +import { goerli } from 'viem/chains'; +import { STREAM } from '@pushprotocol/restapi/src/lib/pushstream/pushStreamTypes'; + +// CONFIGS +const { env, showAPIResponse } = config; + +/***************** SAMPLE SIGNER GENERATION *********************/ +// Uing VIEM +// Random Wallet Signers +const signer = createWalletClient({ + account: privateKeyToAccount(generatePrivateKey()), + chain: goerli, + transport: http(), +}); +const signerAddress = signer.account.address; +const secondSigner = createWalletClient({ + account: privateKeyToAccount(generatePrivateKey()), + chain: goerli, + transport: http(), +}); +const secondSignerAddress = secondSigner.account.address; +const thirdSigner = createWalletClient({ + account: privateKeyToAccount(generatePrivateKey()), + chain: goerli, + transport: http(), +}); +const thirdSignerAddress = thirdSigner.account.address; +// Dummy Wallet Addresses +const randomWallet1 = privateKeyToAccount(generatePrivateKey()).address; + +const eventlistener = async ( + pushAPI: PushAPI, + eventName: string +): Promise => { + pushAPI.stream.on(eventName, (data: any) => { + if (showAPIResponse) { + console.log(data); + } + }); +}; + +const delay = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms)); + +export const runPushAPIStreamCases = async (): Promise => { + const userAlice = await PushAPI.initialize(signer, { env }); + const userBob = await PushAPI.initialize(secondSigner, { env }); + const userKate = await PushAPI.initialize(thirdSigner, { env }); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log(`Listening ${STREAM.PROFILE} Events`); + eventlistener(userAlice, STREAM.PROFILE); + console.log(`Listening ${STREAM.ENCRYPTION} Events`); + eventlistener(userAlice, STREAM.ENCRYPTION); + console.log(`Listening ${STREAM.NOTIF} Events`); + eventlistener(userAlice, STREAM.NOTIF); + console.log(`Listening ${STREAM.NOTIF_OPS} Events`); + eventlistener(userAlice, STREAM.NOTIF_OPS); + console.log(`Listening ${STREAM.CHAT} Events`); + eventlistener(userAlice, STREAM.CHAT); + console.log(`Listening ${STREAM.CHAT_OPS} Events`); + eventlistener(userAlice, STREAM.CHAT_OPS); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('\n\nNew Chat Request, Expected Events:\n1. chat.request'); + await userAlice.chat.send(secondSignerAddress, { + content: 'Hello Bob! from Alice', + }); + await delay(3000); // Delay added to log the events in order + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('\n\nNew Chat Request, Expected Events:\n1. chat.request'); + await userAlice.chat.send(thirdSignerAddress, { + content: 'Hello Kate! from Alice', + }); + await delay(3000); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('\n\nChat Request Accept, Expected Events:\n1. chat.accept'); + await userBob.chat.accept(signerAddress); + await delay(3000); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('\n\nChat Request Reject, Expected Events:\n1. chat.reject'); + await userKate.chat.reject(signerAddress); + await delay(3000); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('\n\nCreate Chat Group, Expected Events:\n1. chat.group.create'); + const groupChatId = ( + await userAlice.chat.group.create('Test Grp', { + description: 'Test Desc', + image: + '', + members: [secondSignerAddress, thirdSignerAddress], + admins: [], + private: false, + }) + ).chatId; + await delay(3000); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('\n\nUpdate Chat Group, Expected Events:\n1. chat.group.update'); + await userAlice.chat.group.update(groupChatId, { + description: 'Updated Test Desc', + image: + '', + }); + await delay(3000); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('\n\nAdd member to Group, Expected Events:\n1. chat.request'); + await userAlice.chat.group.add(groupChatId, { + role: 'MEMBER', + accounts: [randomWallet1], + }); + await delay(3000); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log( + '\n\nRemove member from Group, Expected Events:\n1. chat.group.participant.remove' + ); + await userAlice.chat.group.remove(groupChatId, { + role: 'MEMBER', + accounts: [randomWallet1], + }); + await delay(3000); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('\n\nAdd Admin to Group, Expected Events:\n1. chat.request'); + await userAlice.chat.group.add(groupChatId, { + role: 'ADMIN', + accounts: [randomWallet1], + }); + await delay(3000); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log( + '\n\nRemove Admin from Group, Expected Events:\n1. chat.group.participant.remove' + ); + await userAlice.chat.group.remove(groupChatId, { + role: 'ADMIN', + accounts: [randomWallet1], + }); + await delay(3000); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log('\n\nJoin Group, Expected Events:\n1. chat.accept'); + await userBob.chat.group.join(groupChatId); + await delay(3000); + //------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log( + '\n\nLeave Group, Expected Events:\n1. chat.group.participant.leave' + ); + await userBob.chat.group.leave(groupChatId); + await delay(3000); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + console.log( + '\n\nReject Group Joining Request, Expected Events:\n1. chat.reject' + ); + await userKate.chat.group.reject(groupChatId); + await delay(3000); + // ------------------------------------------------------------------- + // ------------------------------------------------------------------- + if (process.env.WALLET_PRIVATE_KEY) { + // create signer + const channelSigner = createWalletClient({ + account: privateKeyToAccount(`0x${process.env.WALLET_PRIVATE_KEY}`), + chain: goerli, + transport: http(), + }); + + await userAlice.notification.subscribe( + `eip155:5:${channelSigner.account.address}` // channel to subscribe + ); + + const channelUser = await PushAPI.initialize(channelSigner, { env }); + console.log( + '\n\nSend channel notification, Expected Events:\n1. notif.send' + ); + await channelUser.channel.send(['*'], { + notification: { + title: 'test', + body: 'test', + }, + }); + await delay(3000); + + await userAlice.notification.unsubscribe( + `eip155:5:${channelSigner.account.address}` // channel to subscribe + ); + } else { + console.log( + 'Skipping channel notification streams, as WALLET_PRIVATE_KEY is not present in .env' + ); + } +}; diff --git a/packages/examples/sdk-frontend-react/src/app/ChatUITest/ChatProfile.tsx b/packages/examples/sdk-frontend-react/src/app/ChatUITest/ChatProfile.tsx index b889f7daf..fc8c7eefb 100644 --- a/packages/examples/sdk-frontend-react/src/app/ChatUITest/ChatProfile.tsx +++ b/packages/examples/sdk-frontend-react/src/app/ChatUITest/ChatProfile.tsx @@ -6,7 +6,7 @@ export const ChatProfileTest = () => {
diff --git a/packages/examples/sdk-frontend-react/src/app/ChatUITest/ChatViewComponent.tsx b/packages/examples/sdk-frontend-react/src/app/ChatUITest/ChatViewComponent.tsx index ba0f92847..2537443d6 100644 --- a/packages/examples/sdk-frontend-react/src/app/ChatUITest/ChatViewComponent.tsx +++ b/packages/examples/sdk-frontend-react/src/app/ChatUITest/ChatViewComponent.tsx @@ -1,7 +1,7 @@ import styled from 'styled-components'; import { Section } from '../components/StyledComponents'; - +import { CreateGroupModal } from "@pushprotocol/uiweb"; import { ChatViewComponent } from '@pushprotocol/uiweb'; const ChatViewComponentTest = () => { @@ -13,11 +13,9 @@ const ChatViewComponentTest = () => { return (

Chat UI Test page

+ {console.log('in close')}} /> - - console.log("BOIIII RETURNNNSSSSS")} chatId='b8e068e02fe12d7136bc2f24408835573f30c6fbf0b65ea26ab4c7055a2c85f1' limit={10} isConnected={true} autoConnect={false}/> - - + console.log("BOIIII RETURNNNSSSSS")} chatId='4ac5ab85c9c3d57adbdf2dba79357e56b2f9ef0256befe750d9f93af78d2ca68' limit={10} isConnected={true} />
); @@ -29,3 +27,4 @@ const ChatViewComponentCard = styled(Section)` height: 60vh; `; //c2d544ad9d1efd5c5a593b143bf8232875c926cf28015564e70ad078b95f807e +//4ac5ab85c9c3d57adbdf2dba79357e56b2f9ef0256befe750d9f93af78d2ca68 diff --git a/packages/examples/sdk-frontend-react/src/app/app.tsx b/packages/examples/sdk-frontend-react/src/app/app.tsx index 7f1db9f4c..98a5bd2c7 100644 --- a/packages/examples/sdk-frontend-react/src/app/app.tsx +++ b/packages/examples/sdk-frontend-react/src/app/app.tsx @@ -313,7 +313,7 @@ export function App() { - + void ,onClose: ()=> void, handlePrevious: ()=> void, memberList: any, handleMemberList: any, title: string, groupMembers: any, isLoading?: boolean }) => { - const theme = useContext(ThemeContext); - - const [searchedUser, setSearchedUser] = useState(''); - const [filteredUserData, setFilteredUserData] = useState(null); - const [isInValidAddress, setIsInvalidAddress] = useState(false); - const [isLoadingSearch, setIsLoadingSearch] = useState(false); - const { account, env } = useChatData(); - const isMobile = useMediaQuery(device.mobileL); - const groupInfoToast = useToast(); - - - useEffect(() => { - if (isInValidAddress) { - groupInfoToast.showMessageToast({ - toastTitle: 'Error', - toastMessage: 'Invalid Address', - toastType: 'ERROR', - getToastIcon: (size) => ( - - ), - }); - } - }, [isInValidAddress]); - - const onChangeSearchBox = (e: any) => { - setSearchedUser(e.target.value); - }; - - const handleUserSearch = async (userSearchData: string): Promise => { - try{ - const caip10 = walletToPCAIP10(userSearchData); - let filteredData: User; - - if (userSearchData.length) { - filteredData = await PushAPI.user.get({ - account: caip10, - env: env - }); - - if (filteredData !== null) { - setFilteredUserData(filteredData); - } - // User is not in the protocol. Create new user - else { - if (ethers.utils.isAddress(userSearchData)) { - const displayUser = displayDefaultUser({ caip10 }); - setFilteredUserData(displayUser); - } else { - setIsInvalidAddress(true); - setFilteredUserData(null); - } - } - } else { - setFilteredUserData(null); - } - setIsLoadingSearch(false); - } - catch(error){ - groupInfoToast.showMessageToast({ - toastTitle: 'Error', - toastMessage: 'Unsuccessful search, Try again', - toastType: 'ERROR', - getToastIcon: (size) => ( - - ), - }); - } - }; - - const handleSearch = async (e: any): Promise => { - setIsLoadingSearch(true); - setIsInvalidAddress(false); - e.preventDefault(); - if (!ethers.utils.isAddress(searchedUser)) { - let address: string; - try { - address = await getAddress(searchedUser, env) as string; - // if (!address) { - // address = await library.resolveName(searchedUser); - // } - // this ensures address are checksummed - address = ethers.utils.getAddress(address?.toLowerCase()); - if (address) { - handleUserSearch(address); - } else { - setIsInvalidAddress(true); - setFilteredUserData(null); - } - } catch (err) { - setIsInvalidAddress(true); - setFilteredUserData(null); - } finally { - setIsLoadingSearch(false); - } - } else { - handleUserSearch(searchedUser); - } - }; - - const clearInput = () => { - setSearchedUser(''); - setFilteredUserData(null); - setIsLoadingSearch(false); - }; +import { MemberListContainer } from './MemberListContainer'; +import useMediaQuery from '../../../hooks/useMediaQuery'; +import useToast from '../reusables/NewToast'; + +import { + getNewChatUser, +} from '../../../helpers'; +import { ModalButtonProps, User } from '../exportedTypes'; +import { addWalletValidation } from '../helpers/helper'; +import { device } from '../../../config'; +import CloseIcon from '../../../icons/close.svg'; +import { ChatSearchInput, CustomStyleParamsType, ModalHeader } from '../reusables'; +import useGetChatProfile from '../../../hooks/useGetChatProfile'; +import { BackIcon } from '../../../icons/Back'; + + +type AddWalletContentProps = { + onSubmit: () => void; + onClose: () => void; + handlePrevious: () => void; + memberList: any; + handleMemberList: any; + groupMembers: any; + isLoading?: boolean; + modalHeader: string; +}; +export const AddWalletContent = ({ + onSubmit, + handlePrevious, + onClose, + memberList, + handleMemberList, + groupMembers, + isLoading, + modalHeader, +}: AddWalletContentProps) => { + const theme = useContext(ThemeContext); + + const [filteredUserData, setFilteredUserData] = useState(null); + const { account, env } = useChatData(); + const isMobile = useMediaQuery(device.mobileL); + const {fetchChatProfile} = useGetChatProfile(); + const groupInfoToast = useToast(); + const customSearchStyle:CustomStyleParamsType = { + background:theme.backgroundColor?.modalInputBackground, + border:theme.border?.modalInnerComponents, + placeholderColor:theme.textColor?.modalSubHeadingText, + fontSize:'15px', + fontWeight:'400' + }; + + const handleSearch = async ({searchedText}:{searchedText:string}): Promise => { + //fix ens search + const newChatUser = await getNewChatUser({ + searchText: searchedText, + fetchChatProfile, + env, + }); + if(newChatUser){ + setFilteredUserData(newChatUser); + } + else{ + groupInfoToast.showMessageToast({ + toastTitle: 'Error', + toastMessage: 'Invalid Address', + toastType: 'ERROR', + getToastIcon: (size) => , + }); + } + }; + + const clearInput = () => { + setFilteredUserData(null); + }; const addMemberToList = (member: User) => { let errorMessage = ''; - errorMessage = addWalletValidation(member, memberList, groupMembers, account); + errorMessage = addWalletValidation( + member, + memberList, + groupMembers, + account + ); if (errorMessage) { groupInfoToast.showMessageToast({ toastTitle: 'Error', toastMessage: errorMessage, toastType: 'ERROR', - getToastIcon: (size) => ( - - ), + getToastIcon: (size) => , }); - } else { - handleMemberList((prev: any) => [...prev, { ...member, isAdmin: false }]); - } - - setFilteredUserData(''); - clearInput(); - }; - - const removeMemberFromList = (member: User) => { - const filteredMembers = memberList?.filter((user: any) => user.wallets !== member.wallets); - handleMemberList(filteredMembers); - }; - - return ( -
-
- - {/* handlePrevious()} cursor='pointer' /> */} - - Add Wallets - - onClose()} cursor='pointer' /> -
- -
- Add Wallets - - - {groupMembers - ? `0${memberList?.length + groupMembers?.length} / 09 Members` - : `0${memberList?.length} / 09 Members`} - -
- -
- - -
- {searchedUser.length > 0 && ( - clearInput()} cursor='pointer' /> - )} - {searchedUser.length == 0 && !filteredUserData && -
- -
- } -
-
-
- - {filteredUserData ? ( - - } - darkIcon={} - /> - - ) : isLoadingSearch ? ( -
- -
- ) : null} - - - {memberList?.map((member: any, index: any) => ( - } - darkIcon={} - /> - ))} - - -
- onSubmit()} - isLoading={isLoading} - memberListCount={memberList?.length > 0} - theme={theme} - > - {!isLoading && groupMembers ? 'Add To Group' : ''} - {isLoading && } - -
- -
- ) -} - -const SearchBarContent = styled.form` - position: relative; - display: flex; - flex: 1; -`; + } else { + handleMemberList((prev: any) => [...prev, { ...member, isAdmin: false }]); + } + + setFilteredUserData(''); + clearInput(); + }; + + const removeMemberFromList = (member: User) => { + const filteredMembers = memberList?.filter( + (user: any) => user.wallets.toLowerCase() !== member.wallets.toLowerCase() + ); + handleMemberList(filteredMembers); + }; + + return ( +
+ + +
+ + Add Wallets + + + + {groupMembers + ? `0${memberList?.length + groupMembers?.length} / 09 Members` + : `0${memberList?.length} / 09 Members`} + +
+ +
+ +
+ + {filteredUserData && ( + + } + /> + + ) } + + + {memberList?.map((member: any, index: any) => ( + } + /> + ))} + + +
+ onSubmit()} + isLoading={isLoading} + memberListCount={memberList?.length > 0} + theme={theme} + > + {!isLoading && groupMembers ? 'Add To Group' : ''} + {isLoading && } + +
+
+ ); +}; + + -const Input = styled.input` - box-sizing: border-box; - display: flex; - flex: 1; -// min-width: 445px; - height: 48px; - padding: 0px 50px 0px 16px; - margin: 10px 0px 0px; - border-radius: 99px; - border: 1px solid; - border-color: ${(props) => props.theme.modalSearchBarBorderColor}; - background: ${(props) => props.theme.modalSearchBarBackground}; - color: ${(props) => props.color || '#000'}; - &:focus { - outline: none; - background-image: linear-gradient( - ${(props) => props.theme.snapFocusBg}, - ${(props) => props.theme.snapFocusBg} - ), - linear-gradient( - to right, - rgba(182, 160, 245, 1), - rgba(244, 110, 246, 1), - rgba(255, 222, 211, 1), - rgba(255, 207, 197, 1) - ); - background-origin: border; - border: 1px solid transparent !important; - background-clip: padding-box, border-box; - } - &::placeholder { - color: #657795; - } - @media ${device.mobileL} { - min-width: 100%; - } -`; const MemberList = styled.div` - // justify-content: flex-start; - // padding: 0px 2px; - // margin: 0 0 34px 0; - flex: 1; - // background: red; - width: 100%; + flex: 1; + width: 100%; + margin-bottom:40px; `; const MultipleMemberList = styled.div` - // overflow-y: auto; height: fit-content; max-height: 216px; padding: 0px 2px; - // overflow-x: hidden; width: 100%; &::-webkit-scrollbar-track { - background-color: ${(props) => props.theme.scrollBg}; + background-color: ${(props) => props.theme.scrollbarColor}; } &::-webkit-scrollbar { - background-color: ${(props) => props.theme.scrollBg}; + background-color: ${(props) => props.theme.scrollbarColor}; width: 6px; } @@ -362,19 +242,24 @@ const MultipleMemberList = styled.div` `; const ModalConfirmButton = styled.button` - margin: 60px 0 0 0; - background: ${(props) => props.memberListCount ? '#CF1C84' : props.theme.groupButtonBackgroundColor}; - color: ${(props) => props.memberListCount ? '#fff' : props.theme.groupButtonTextColor}; - border: ${(props) => props.memberListCount ? 'none' : props.theme.modalConfirmButtonBorder}; - min-width: 50%; - box-sizing: border-box; - cursor: pointer; - border-radius: 15px; - padding: 16px; - font-size: 1.125rem; - font-weight: 500; - display: flex; - align-items: center; - justify-content: center; - box-shadow: none; -`; \ No newline at end of file + margin: 60px 0 0 0; + width: 197px; + background: ${(props) => + props.memberListCount ? props.theme.backgroundColor.buttonBackground : props.theme.backgroundColor.buttonDisableBackground}; + color: ${(props) => + props.memberListCount ? props.theme.textColor.buttonText : props.theme.textColor.buttonDisableText}; + border: ${(props) => + props.memberListCount ? 'none' : props.theme.border.modal}; + min-width: 50%; + box-sizing: border-box; + cursor: pointer; + border-radius: 12px; + padding: 16px; + font-size: 16px; + font-weight: 500; + display: flex; + align-items: center; + justify-content: center; + box-shadow: none; + height: 48px; +`; diff --git a/packages/uiweb/src/lib/components/chat/ChatProfile/ChatProfile.tsx b/packages/uiweb/src/lib/components/chat/ChatProfile/ChatProfile.tsx index b9267b17b..fa6d4e27b 100644 --- a/packages/uiweb/src/lib/components/chat/ChatProfile/ChatProfile.tsx +++ b/packages/uiweb/src/lib/components/chat/ChatProfile/ChatProfile.tsx @@ -1,188 +1,230 @@ // @typescript-eslint/no-non-null-asserted-optional-chain -import { useContext, useEffect, useRef, useState } from "react"; -import { Image, Section, Span } from "../../reusables"; -import styled from "styled-components"; -import TokenGatedIcon from '../../../icons/Token-Gated.svg'; +import { useContext, useEffect, useRef, useState } from 'react'; + +import styled from 'styled-components'; +import type { IUser } from '@pushprotocol/restapi'; +import { ethers } from 'ethers'; +import { ToastContainer } from 'react-toastify'; + +import { Image, Section, Span } from '../../reusables'; +import { useChatData, useClickAway } from '../../../hooks'; +import { ThemeContext } from '../theme/ThemeProvider'; +import useGetGroupByID from '../../../hooks/chat/useGetGroupByID'; +import useChatProfile from '../../../hooks/chat/useChatProfile'; +import { GroupInfoModal } from './GroupInfoModal'; +import useMediaQuery from '../../../hooks/useMediaQuery'; +import { createBlockie } from '../../space/helpers/blockies'; +import { ProfileContainer } from '../reusables'; +import 'react-toastify/dist/ReactToastify.min.css'; + +import { IGroup } from '../../../types'; +import { isValidETHAddress } from '../helpers/helper'; +import { IChatProfile, IChatTheme } from '../exportedTypes'; +import { InfuraAPIKey, allowedNetworks, device } from '../../../config'; +import { resolveNewEns, shortenText } from '../../../helpers'; +import TokenGatedIcon from '../../../icons/TokenGatedIcon.svg'; import PublicChatIcon from '../../../icons/Public-Chat.svg'; -import VideoChatIcon from '../../../icons/VideoCallIcon.svg'; import GreyImage from '../../../icons/greyImage.png'; import InfoIcon from '../../../icons/infodark.svg'; import VerticalEllipsisIcon from '../../../icons/VerticalEllipsis.svg'; -import type { IUser } from '@pushprotocol/restapi'; -import { useChatData, useClickAway, useDeviceWidthCheck } from "../../../hooks"; -import { ThemeContext } from "../theme/ThemeProvider"; -import { IChatTheme } from "../theme"; -import { pCAIP10ToWallet, resolveEns, resolveNewEns, shortenText } from "../../../helpers"; -import useGetGroupByID from "../../../hooks/chat/useGetGroupByID"; -import useChatProfile from "../../../hooks/chat/useChatProfile"; -import { IGroup } from "../../../types"; -import { GroupInfoModal } from "./GroupInfoModal"; -import { isValidETHAddress } from "../helpers/helper"; -import { ethers } from "ethers"; -import { IChatProfile, IToast, OptionProps } from "../exportedTypes"; -import { InfuraAPIKey, allowedNetworks, device } from "../../../config"; -import Toast from "../helpers/Toast"; -import useMediaQuery from "../../../hooks/useMediaQuery"; -import { createBlockie } from "../../space/helpers/blockies"; -// import { NewToast } from "../helpers/NewToast"; -import { ToastContainer, toast } from 'react-toastify'; -import 'react-toastify/dist/ReactToastify.min.css'; - -const Options = ({ options, setOptions, isGroup, chatInfo, groupInfo, setGroupInfo, theme }: OptionProps) => { - const DropdownRef = useRef(null); - const [modal, setModal] = useState(false); - - useClickAway(DropdownRef, () => { - setOptions(false); - }); - - const ShowModal = () => { - setModal(true); - } - console.log(groupInfo?.rules?.chat, "groupInfooo") - - if (groupInfo && isGroup) { - return ( -
- { - (groupInfo?.rules?.chat?.conditions || groupInfo?.rules?.entry?.conditions) && ( - - ) - } - - {groupInfo?.isPublic && - ()} - - setOptions(true)}> - - - {options && - ( - - - - - Group Info - - - )} - - {modal && - ()} - -
- ) - } else { - return null - } +type OptionProps = { + options: boolean; + setOptions: React.Dispatch>; + isGroup: boolean; + groupInfo: IGroup | null | undefined; + setGroupInfo: React.Dispatch>; + theme: IChatTheme; }; +const Options = ({ + options, + setOptions, + isGroup, + groupInfo, + setGroupInfo, + theme, +}: OptionProps) => { + const DropdownRef = useRef(null); + const [modal, setModal] = useState(false); + + useClickAway(DropdownRef, () => { + setOptions(false); + }); + + const ShowModal = () => { + setModal(true); + }; + + if (groupInfo && isGroup) { + return ( +
+ {/* {(groupInfo?.rules?.chat?.conditions || groupInfo.rules?.entry?.conditions) && ( + + )} */} + + + setOptions(true)}> + + + {options && ( + + + + Group Info + + + )} + + {modal && ( + + )} + +
+ ); + } else { + return null; + } +}; - - -export const ChatProfile: React.FC = ({ chatId, style }: { chatId: string, style: "Info" | "Preview" }) => { - const theme = useContext(ThemeContext); - const { account, env } = useChatData(); - const { getGroupByID } = useGetGroupByID(); - const { fetchUserChatProfile } = useChatProfile(); - - const [isGroup, setIsGroup] = useState(false); - const [options, setOptions] = useState(false); - const [chatInfo, setChatInfo] = useState(); - const [groupInfo, setGroupInfo] = useState(); - const [ensName, setEnsName] = useState(''); - const isMobile = useMediaQuery(device.tablet); - const l1ChainId = allowedNetworks[env].includes(1) ? 1 : 5; - const provider = new ethers.providers.InfuraProvider(l1ChainId, InfuraAPIKey); - - - - const fetchProfileData = async () => { - if (isValidETHAddress(chatId)) { - const ChatProfile = await fetchUserChatProfile({ profileId: chatId }); - setChatInfo(ChatProfile); - setGroupInfo(null); - setIsGroup(false); - } else { - const GroupProfile = await getGroupByID({ groupId: chatId }) - setGroupInfo(GroupProfile); - setChatInfo(null); - setIsGroup(true); - } +export const ChatProfile: React.FC = ({ + chatId, + style, +}: { + chatId: string; + style: 'Info' | 'Preview'; +}) => { + const theme = useContext(ThemeContext); + const { account, env } = useChatData(); + const { getGroupByID } = useGetGroupByID(); + const { fetchUserChatProfile } = useChatProfile(); + + const [isGroup, setIsGroup] = useState(false); + const [options, setOptions] = useState(false); + const [chatInfo, setChatInfo] = useState(); + const [groupInfo, setGroupInfo] = useState(); + const [ensName, setEnsName] = useState(''); + const isMobile = useMediaQuery(device.tablet); + const l1ChainId = allowedNetworks[env].includes(1) ? 1 : 5; + const provider = new ethers.providers.InfuraProvider(l1ChainId, InfuraAPIKey); + + const fetchProfileData = async () => { + if (isValidETHAddress(chatId)) { + const ChatProfile = await fetchUserChatProfile({ profileId: chatId }); + const result = await resolveNewEns(chatId, provider); + setEnsName(result); + setChatInfo(ChatProfile); + setGroupInfo(null); + setIsGroup(false); + } else { + const GroupProfile = await getGroupByID({ groupId: chatId }); + setGroupInfo(GroupProfile); + setChatInfo(null); + setIsGroup(true); } - - const getName = async (chatId: string) => { - if (isValidETHAddress(chatId)) { - const result = await resolveNewEns(chatId, provider); - // if(result) - setEnsName(result); - } + }; + + const getImage = () => { + if (chatInfo || groupInfo) { + return isGroup + ? groupInfo?.groupImage ?? GreyImage + : chatInfo?.profile?.picture ?? + createBlockie?.(chatId)?.toDataURL()?.toString(); + } else { + return createBlockie?.(chatId)?.toDataURL()?.toString(); } - - - useEffect(() => { - if (!chatId) return; - fetchProfileData(); - getName(chatId); - }, [chatId, account, env]) - - if (chatId && style === 'Info') { - return ( - - {chatInfo || groupInfo ? ( - - ) : ()} - - - - {isGroup ? groupInfo?.groupName : ensName ? `${ensName} (${isMobile ? shortenText(chatInfo?.did?.split(':')[1] ?? '', 4, true) : chatId})` : chatInfo ? shortenText(chatInfo.did?.split(':')[1] ?? '', 6, true) : shortenText(chatId, 6, true)} - - - - - - {/* {!isGroup && + }; + + const getProfileName = () => { + return isGroup + ? groupInfo?.groupName + : ensName + ? `${ensName} (${isMobile + ? shortenText(chatInfo?.did?.split(':')[1] ?? '', 4, true) + : chatId + })` + : chatInfo + ? shortenText(chatInfo.did?.split(':')[1] ?? '', 6, true) + : shortenText(chatId, 6, true); + }; + + useEffect(() => { + if (!chatId) return; + fetchProfileData(); + }, [chatId, account, env]); + + if (chatId && style === 'Info') { + return ( + + + + + + {/* {!isGroup && } */} - - - - - ) - } else { - return null; - } -} - + + + ); + } else { + return null; + } +}; const Container = styled.div` width: 100%; background: ${(props) => props.theme.backgroundColor.chatProfileBackground}; - border:${(props) => props.theme.border?.chatProfile}; - border-radius:${(props) => props.theme.borderRadius?.chatProfile}; + border: ${(props) => props.theme.border?.chatProfile}; + border-radius: ${(props) => props.theme.borderRadius?.chatProfile}; display: flex; flex-direction: row; align-items: center; @@ -195,28 +237,21 @@ const ImageItem = styled.div` position: relative; `; -const DummyImage = styled.div` - height: 48px; - width: 48px; - border-radius: 100%; - background: #ccc; -`; - const DropDownBar = styled.div` - position: absolute; - top: 30px; - left: -130px; - display: block; - min-width: 140px; - color: rgb(101, 119, 149); - border: ${(props) => `1px solid ${props.theme.defaultBorder}`}; - background: ${(props) => props.theme.backgroundColor.chatReceivedBubbleBackground}; - z-index: 10; - border-radius: 16px; + position: absolute; + top: 30px; + left: -130px; + display: block; + min-width: 140px; + color: rgb(101, 119, 149); + background: ${(props) => props.theme.backgroundColor.modalBackground}; + border: ${(props) => props.theme.border.modalInnerComponents}; + z-index: 10; + border-radius: ${(props) => props.theme.borderRadius.modalInnerComponents}; `; const VideoChatSection = styled.div` - margin: 0 25px 0 auto; + margin: 0 25px 0 auto; `; const DropDownItem = styled(Span)` @@ -231,17 +266,8 @@ const DropDownItem = styled(Span)` `; const TextItem = styled(Span)` - white-space: nowrap; - overflow: hidden; + white-space: nowrap; + overflow: hidden; `; - - - - - - - - - - +//auto update members when an user accepts not done diff --git a/packages/uiweb/src/lib/components/chat/ChatProfile/GroupInfoModal.tsx b/packages/uiweb/src/lib/components/chat/ChatProfile/GroupInfoModal.tsx index e684cef63..aca69ae8f 100644 --- a/packages/uiweb/src/lib/components/chat/ChatProfile/GroupInfoModal.tsx +++ b/packages/uiweb/src/lib/components/chat/ChatProfile/GroupInfoModal.tsx @@ -1,635 +1,969 @@ -import { useRef, useState } from "react"; -import styled from "styled-components"; -import { useChatData, useClickAway } from "../../../hooks"; -import { IGroup } from "../../../types"; -import { IChatTheme } from "../theme"; -import * as PushAPI from '@pushprotocol/restapi'; -import { IToast, ShadowedProps, UpdateGroupType } from "../exportedTypes"; -import { convertToWalletAddressList, getAdminList, getUpdatedAdminList, getUpdatedMemberList } from "../helpers/helper"; -import { DropdownValueType } from "./DropDown"; +import { useEffect, useRef, useState } from 'react'; + +import styled from 'styled-components'; +import { MdCheckCircle, MdError } from 'react-icons/md'; + +import { useChatData, useClickAway } from '../../../hooks'; +import { DropdownValueType } from '../reusables/DropDown'; +import { Section, Span, Image, Div } from '../../reusables/sharedStyling'; +import { AddWalletContent } from './AddWalletContent'; +import { Modal, ModalHeader } from '../reusables'; +import useMediaQuery from '../../../hooks/useMediaQuery'; +import useToast from '../reusables/NewToast'; +import useUpdateGroup from '../../../hooks/chat/useUpdateGroup'; +import { MemberProfileCard } from './MemberProfileCard'; +import { ProfileContainer } from '../reusables'; + +import { IGroup } from '../../../types'; +import { IChatTheme } from '../theme'; +import { device } from '../../../config'; +import { + convertToWalletAddressList, + getAdminList, + getUpdatedAdminList, + getUpdatedMemberList, + isAccountOwnerAdmin, +} from '../helpers/group'; +import LockIcon from '../../../icons/Lock.png'; +import LockSlashIcon from '../../../icons/LockSlash.png'; +import ArrowIcon from '../../../icons/CaretUp.svg'; +import addIcon from '../../../icons/addicon.svg'; import DismissAdmin from '../../../icons/dismissadmin.svg'; import AddAdmin from '../../../icons/addadmin.svg'; import Remove from '../../../icons/remove.svg'; -import { Section, Span, Image } from "../../reusables/sharedStyling"; -import CloseIcon from '../../../icons/close.svg'; -import { ProfileCard } from "./ProfileCard"; -import addIcon from '../../../icons/addicon.svg'; -import { pCAIP10ToWallet, shortenText } from "../../../helpers"; -import LockIcon from '../../../icons/Lock.png' -import LockSlashIcon from '../../../icons/LockSlash.png' -import { AddWalletContent } from './AddWalletContent' -import ArrowIcon from '../../../icons/CaretDown.svg' -import { Modal } from "../helpers/Modal"; -import { device } from "../../../config"; -import useMediaQuery from "../../../hooks/useMediaQuery"; -import useToast from "../helpers/NewToast"; -import { MdCheckCircle, MdError } from "react-icons/md"; - - +import { shortenText } from '../../../helpers'; +import TokenGatedIcon from '../../../icons/TokenGatedIcon.svg'; +import ConditionsComponent from '../CreateGroup/ConditionsComponent'; +import { ConditionArray } from '../exportedTypes'; +import { ACCESS_TYPE_TITLE } from '../constants'; + +const UPDATE_KEYS = { + REMOVE_MEMBER: 'REMOVE_MEMBER', + ADD_MEMBER: 'ADD_MEMBER', + REMOVE_ADMIN: 'REMOVE_ADMIN', + ADD_ADMIN: 'ADD_ADMIN', +} as const; + +type UpdateKeys = typeof UPDATE_KEYS[keyof typeof UPDATE_KEYS]; +const SUCCESS_MESSAGE = { + REMOVE_MEMBER: 'Removed Member successfully', + ADD_MEMBER: 'Group Invitation sent', + REMOVE_ADMIN: 'Admin added successfully', + ADD_ADMIN: 'Removed added successfully', +}; + +type PendingMembersProps = { + groupInfo?: IGroup | null; + setShowPendingRequests: React.Dispatch>; + showPendingRequests: boolean; + theme: IChatTheme; +}; + +interface ShadowedProps { + setPosition: boolean; +} -const PendingMembers = ({ groupInfo, setShowPendingRequests, showPendingRequests, theme }: {groupInfo?: IGroup | null, setShowPendingRequests: React.Dispatch>, showPendingRequests: boolean, theme: IChatTheme }) => { - if(groupInfo){ +const PendingMembers = ({ + groupInfo, + setShowPendingRequests, + showPendingRequests, + theme, +}: PendingMembersProps) => { + if (groupInfo) { return ( - + setShowPendingRequests(!showPendingRequests)} > - setShowPendingRequests(!showPendingRequests)}> - Pending Requests - {groupInfo?.pendingMembers?.length} - - - {/* */} + + Pending Requests + + {groupInfo?.pendingMembers?.length} + + - {showPendingRequests && ( -
- {groupInfo?.pendingMembers && groupInfo?.pendingMembers?.length > 0 && groupInfo?.pendingMembers.map((item) => ( - - - - - {shortenText(item?.wallet?.split(':')[1] ?? '', 6, true)} - + {showPendingRequests && ( +
+ {groupInfo?.pendingMembers && + groupInfo?.pendingMembers?.length > 0 && + groupInfo?.pendingMembers.map((item) => ( + + + + ))} +
+ )} + + ); + } else { + return null; + } +}; + +const dummyConditonsData: ConditionArray[] = [ + [{ operator: 'any' }], + [ + { + type: 'PUSH', + category: 'ERC20', + subcategory: 'holder', + data: { + contract: 'eip155:1:0xf418588522d5dd018b425E472991E52EBBeEEEEE', + amount: 1, + decimals: 18, + }, + }, + ], + [ + { operator: 'all' }, + { + type: 'PUSH', + category: 'ERC20', + subcategory: 'holder', + data: { + contract: 'eip155:137:0x58001cC1A9E17A20935079aB40B1B8f4Fc19EFd1', + amount: 1, + decimals: 18, + }, + }, + { + type: 'PUSH', + category: 'ERC721', + subcategory: 'holder', + data: { + contract: 'eip155:137:0x58001cC1A9E17A20935079aB40B1B8f4Fc19EFd1', + amount: 1, + decimals: 18, + }, + }, + { + type: 'GUILD', + category: 'ROLES', + subcategory: 'DEFAULT', + data: { + id: '1', + role: '346243', + comparison: 'all', + }, + }, + ], + // [ + // { operator: 'any' }, + // { + // type: 'PUSH', + // category: 'INVITE', + // subcategory: 'DEFAULT', + // data: { + // inviterRoles: 'ADMIN', + // }, + // }, + // { + // type: 'PUSH', + // category: 'INVITE', + // subcategory: 'DEFAULT', + // data: { + // inviterRoles: 'OWNER', + // }, + // }, + // ], +]; + +const dummySingleCondtionData: ConditionArray[] = dummyConditonsData[2].map( + (criteria) => [criteria] +); + +interface ConditionsInformationProps { + theme: IChatTheme; + groupInfo?: IGroup | null; +} -
- ))} +export const ConditionsInformation = ({ + theme, + groupInfo, +}: ConditionsInformationProps) => { + return ( +
+ {(groupInfo?.rules?.chat?.conditions || + groupInfo?.rules?.entry?.conditions) && ( + + )} + {Object.keys(ACCESS_TYPE_TITLE).map((key) => ( + <> + + { ACCESS_TYPE_TITLE[key as keyof typeof ACCESS_TYPE_TITLE]?.heading} + + + + + + ))}
- )} - - ) - } else {return null } + ); +}; + +interface GroupTypeProps { + theme: IChatTheme; + icon: string; + header: string; + subheader: string; + cursor?: string; + handleNextInformation?: () => void; } -export const GroupInfoModal = ({ theme, modal, setModal, groupInfo, setGroupInfo }: { theme: IChatTheme, modal: boolean, setModal: React.Dispatch>, groupInfo: IGroup, setGroupInfo: React.Dispatch> }) => { - const { account, env, pgpPrivateKey } = useChatData(); - const [showAddMoreWalletModal, setShowAddMoreWalletModal] = useState(false); - const [showPendingRequests, setShowPendingRequests] = useState(false); - const [memberList, setMemberList] = useState([]); - const [isLoading, setIsLoading] = useState(false); - const [selectedMemberAddress, setSelectedMemberAddress] = useState(null); - - const handleClose = () => onClose(); - const dropdownRef = useRef(null); - useClickAway(dropdownRef, () => setSelectedMemberAddress(null)); - const groupInfoToast = useToast(); - - - const groupCreator = groupInfo?.groupCreator; - const membersExceptGroupCreator = groupInfo?.members?.filter((x) => x.wallet?.toLowerCase() !== groupCreator?.toLowerCase()); - const groupMembers = [...membersExceptGroupCreator, ...groupInfo.pendingMembers]; - - - const updateGroup = async (options:UpdateGroupType) => { - const { groupInfo, connectedUser,adminList,memberList } = options; - const updateResponse = await PushAPI.chat.updateGroup({ - chatId: groupInfo?.chatId, - groupName: groupInfo?.groupName, - groupDescription: groupInfo?.groupDescription ?? '', - groupImage: groupInfo?.groupImage, - members: memberList, - admins: adminList, - account: connectedUser?.wallets, - pgpPrivateKey: pgpPrivateKey, - env: env, +const GroupTypeBadge = ({ + theme, + icon, + header, + subheader, + handleNextInformation, + cursor, +}: GroupTypeProps) => { + return ( +
+ + + +
+ + {header} + + + {subheader} + +
+
+
+ ); +}; + +type GroupSectionProps = GroupInfoModalProps & { + handleNextInformation: () => void; + handlePreviousInformation?: () => void; +}; + +type GroupInfoModalProps = { + theme: IChatTheme; + setModal: React.Dispatch>; + groupInfo: IGroup; + setGroupInfo: React.Dispatch>; +}; + +export const GROUPINFO_STEPS = { + GROUP_INFO: 1, + CRITERIA: 2, +} as const; + +export type GROUP_INFO_TYPE = + typeof GROUPINFO_STEPS[keyof typeof GROUPINFO_STEPS]; + +const GroupInformation = ({ + theme, + setModal, + groupInfo, + setGroupInfo, + handleNextInformation, +}: GroupSectionProps) => { + const { account } = useChatData(); + const [showAddMoreWalletModal, setShowAddMoreWalletModal] = + useState(false); + const [showPendingRequests, setShowPendingRequests] = + useState(false); + const [memberList, setMemberList] = useState([]); + const [isLoading, setIsLoading] = useState(false); + const [checkedValue, setchecked] = useState(false); + const [selectedMemberAddress, setSelectedMemberAddress] = useState< + string | null + >(null); + const { updateGroup } = useUpdateGroup(); + const isMobile = useMediaQuery(device.mobileL); + + const handleClose = () => onClose(); + const dropdownRef = useRef(null); + useClickAway(dropdownRef, () => setSelectedMemberAddress(null)); + const groupInfoToast = useToast(); + + const groupCreator = groupInfo?.groupCreator; + const membersExceptGroupCreator = groupInfo?.members?.filter( + (x) => x.wallet?.toLowerCase() !== groupCreator?.toLowerCase() + ); + + const groupMembers = [ + ...membersExceptGroupCreator, + ...groupInfo.pendingMembers, + ]; + + type UpdateGroupType = { + adminList: Array; + memberList: Array; + }; + + const handleUpdateGroup = async (options: UpdateGroupType) => { + const { adminList, memberList } = options || {}; + const updateResponse = await updateGroup({ + groupInfo, + memberList, + adminList, + }); + return { updateResponse }; + }; + + const handleAddRemove = async ( + options: UpdateGroupType & { updateKey: UpdateKeys } + ) => { + const { adminList, memberList, updateKey } = options || {}; + + try { + setIsLoading(true); + const { updateResponse } = await handleUpdateGroup({ + adminList, + memberList, + }); + + if (typeof updateResponse !== 'string') { + setGroupInfo(updateResponse); + + groupInfoToast.showMessageToast({ + toastTitle: 'Success', + toastMessage: SUCCESS_MESSAGE[updateKey], + toastType: 'SUCCESS', + getToastIcon: (size) => , }); - let updatedCurrentChat = null; - if(typeof updateResponse !== 'string') - { - updatedCurrentChat = groupInfo; - updatedCurrentChat = updateResponse; - } - return {updateResponse,updatedCurrentChat}; + } else { + groupInfoToast.showMessageToast({ + toastTitle: 'Error', + toastMessage: updateResponse, + toastType: 'ERROR', + getToastIcon: (size) => , + }); + } + } catch (error) { + console.error('Error', error); + groupInfoToast.showMessageToast({ + toastTitle: 'Error', + toastMessage: 'Please, try again', + toastType: 'ERROR', + getToastIcon: (size) => , + }); + } finally { + if (updateKey === UPDATE_KEYS.ADD_MEMBER) handleClose(); + setIsLoading(false); + setSelectedMemberAddress(null); } - - const addMembers = async () => { - //Already Present Members and PendingMembers - const groupMemberList = convertToWalletAddressList([ - ...groupInfo.members, - ...groupInfo.pendingMembers, - ]); - - //Newly Added Members and alreadyPresent Members in the groupchat - const newMembersToAdd = memberList.map((member: any) => member.wallets); - const members = [...groupMemberList, ...newMembersToAdd]; - - //Admins wallet address from both members and pendingMembers - const adminList = getAdminList?.(groupInfo); - - - try { - setIsLoading(true); - const connectedUser = await PushAPI.user.get({ account: account as string, env }); - const { updateResponse, updatedCurrentChat } = await updateGroup({ - groupInfo, - connectedUser, - adminList, - memberList: members, - }); - - if (typeof updateResponse !== 'string') { - setSelectedMemberAddress(null); - setGroupInfo(updateResponse); - } else { - groupInfoToast.showMessageToast({ - toastTitle: 'Error', - toastMessage: updateResponse, - toastType: 'ERROR', - getToastIcon: (size) => ( - - ), - }); - setSelectedMemberAddress(null); - } - setIsLoading(false); - groupInfoToast.showMessageToast({ - toastTitle: 'Success', - toastMessage: 'Group Invitation sent', - toastType: 'SUCCESS', - getToastIcon: (size) => ( - - ), - }); - handleClose(); - } catch (error) { - setIsLoading(false); - console.log('Error', error); - groupInfoToast.showMessageToast({ - toastTitle: 'Error', - toastMessage: 'Please, try again', - toastType: 'ERROR', - getToastIcon: (size) => ( - - ), - }); - } - }; - - - const makeGroupAdmin = async () => { - const groupMemberList = convertToWalletAddressList([ - ...groupInfo.members, - ...groupInfo.pendingMembers, - ]); - const newAdminList = getUpdatedAdminList(groupInfo, selectedMemberAddress, false); - try { - const connectedUser = await PushAPI.user.get({ account: account as string, env }); - const { updateResponse, updatedCurrentChat } = await updateGroup({ - groupInfo, - connectedUser, - adminList: newAdminList, - memberList: groupMemberList, - }); - if (typeof updateResponse !== 'string') { - setSelectedMemberAddress(null); - setGroupInfo(updateResponse); - - groupInfoToast.showMessageToast({ - toastTitle: 'Success', - toastMessage: 'Admin added successfully', - toastType: 'SUCCESS', - getToastIcon: (size) => ( - ), - }); - } else { - groupInfoToast.showMessageToast({ - toastTitle: 'Error', - toastMessage: updateResponse, - toastType: 'ERROR', - getToastIcon: (size) => ( - - ), - }); - setSelectedMemberAddress(null); - } - } catch (e) { - console.error('Error while adding admin', e); - groupInfoToast.showMessageToast({ - toastTitle: 'Error', - toastMessage: 'Error', - toastType: 'ERROR', - getToastIcon: (size) => ( - - ), - }); - } - setSelectedMemberAddress(null); - }; - - const dismissGroupAdmin = async () => { - const groupMemberList = convertToWalletAddressList([ - ...groupInfo.members, - ...groupInfo.pendingMembers, - ]); - const newAdminList = getUpdatedAdminList(groupInfo, selectedMemberAddress, true); - try { - const connectedUser = await PushAPI.user.get({ account: account as string, env }); - const { updateResponse, updatedCurrentChat } = await updateGroup({ - groupInfo, - connectedUser, - adminList: newAdminList, - memberList: groupMemberList, - }); - if (typeof updateResponse !== 'string') { - setSelectedMemberAddress(null); - setGroupInfo(updateResponse); - - groupInfoToast.showMessageToast({ - toastTitle: 'Success', - toastMessage: 'Admin removed successfully', - toastType: 'SUCCESS', - getToastIcon: (size) => ( - ), - }); - - } else { - groupInfoToast.showMessageToast({ - toastTitle: 'Error', - toastMessage: updateResponse, - toastType: 'ERROR', - getToastIcon: (size) => ( - - ), - }); - setSelectedMemberAddress(null); - } - } catch (e) { - console.error('Error while dismissing admin', e); - groupInfoToast.showMessageToast({ - toastTitle: 'Error', - toastMessage: 'Please, try again', - toastType: 'ERROR', - getToastIcon: (size) => ( - - ), - }); - } - setSelectedMemberAddress(null); - }; - - const removeMember = async () => { - const updatedMemberList = getUpdatedMemberList(groupInfo, selectedMemberAddress!); - const adminList = getUpdatedAdminList(groupInfo, selectedMemberAddress, true); - try { - const connectedUser = await PushAPI.user.get({ account: account as string, env }); - const { updateResponse, updatedCurrentChat } = await updateGroup({ - groupInfo, - connectedUser, - adminList, - memberList: updatedMemberList, - }); - - if (typeof updateResponse !== 'string') { - setSelectedMemberAddress(null); - setGroupInfo(updateResponse); - - groupInfoToast.showMessageToast({ - toastTitle: 'Success', - toastMessage: 'Removed Member successfully', - toastType: 'SUCCESS', - getToastIcon: (size) => ( - - ), - }); - } else { - groupInfoToast.showMessageToast({ - toastTitle: 'Error', - toastMessage: updateResponse, - toastType: 'ERROR', - getToastIcon: (size) => ( - - ), - }); - setSelectedMemberAddress(null); - } - } catch (error) { - console.error('Error in removing member', error); - groupInfoToast.showMessageToast({ - toastTitle: 'Error', - toastMessage: 'Please, try again', - toastType: 'ERROR', - getToastIcon: (size) => ( - - ), - }); + }; + const removeMember = async () => { + const updatedMemberList = getUpdatedMemberList( + groupInfo, + selectedMemberAddress! + ); + const adminList = getUpdatedAdminList( + groupInfo, + selectedMemberAddress, + true + ); + await handleAddRemove({ + memberList: updatedMemberList, + adminList, + updateKey: UPDATE_KEYS.REMOVE_MEMBER, + }); + }; + + const addMembers = async () => { + //Already Present Members and PendingMembers + const groupMemberList = convertToWalletAddressList([ + ...groupInfo.members, + ...groupInfo.pendingMembers, + ]); + + //Newly Added Members and alreadyPresent Members in the groupchat + const newMembersToAdd = memberList.map((member: any) => member.wallets); + const members = [...groupMemberList, ...newMembersToAdd]; + + //Admins wallet address from both members and pendingMembers + const adminList = getAdminList?.(groupInfo); + + await handleAddRemove({ + memberList: members, + adminList, + updateKey: UPDATE_KEYS.ADD_MEMBER, + }); + }; + + const updateGroupAdmin = async (updateKey: UpdateKeys) => { + const groupMemberList = convertToWalletAddressList([ + ...groupInfo.members, + ...groupInfo.pendingMembers, + ]); + const newAdminList = getUpdatedAdminList( + groupInfo, + selectedMemberAddress, + !(updateKey === UPDATE_KEYS.ADD_ADMIN) + ); + await handleAddRemove({ + memberList: groupMemberList, + adminList: newAdminList, + updateKey, + }); + }; + + // const messageUserDropdown: DropdownValueType = { + // id: 'message_user', + // title: 'Message user', + // icon: Message, + // function: () => messageUser(), + // }; + + const removeAdminDropdown: DropdownValueType = { + id: 'dismiss_admin', + title: 'Dismiss as admin', + icon: DismissAdmin, + function: () => updateGroupAdmin(UPDATE_KEYS.REMOVE_ADMIN), + }; + const addAdminDropdown: DropdownValueType = { + id: 'add_admin', + title: 'Make group admin', + icon: AddAdmin, + function: () => updateGroupAdmin(UPDATE_KEYS.ADD_ADMIN), + }; + const removeMemberDropdown: DropdownValueType = { + id: 'remove_member', + title: 'Remove', + icon: Remove, + function: () => removeMember(), + textColor: '#ED5858', + }; + + //remove all testing things + + const a1: DropdownValueType = { + id: 'dismiss_admin', + title: 'Dismiss as admin', + function: () => updateGroupAdmin(UPDATE_KEYS.REMOVE_ADMIN), + }; + const a2: DropdownValueType = { + id: 'add_admin', + title: 'Make group admin', + function: () => updateGroupAdmin(UPDATE_KEYS.ADD_ADMIN), + }; + + const handlePrevious = () => { + setShowAddMoreWalletModal(false); + }; + + const onClose = (): void => { + setModal(false); + }; + + return ( +
+ + + Group Description + + + {groupInfo?.groupDescription} + + + messageUser(), - // }; - const removeAdminDropdown: DropdownValueType = { - id: 'dismiss_admin', - title: 'Dismiss as admin', - icon: DismissAdmin, - function: () => dismissGroupAdmin(), - }; - const addAdminDropdown: DropdownValueType = { - id: 'add_admin', - title: 'Make group admin', - icon: AddAdmin, - function: () => makeGroupAdmin(), - }; - const removeMemberDropdown: DropdownValueType = { - id: 'remove_member', - title: 'Remove', - icon: Remove, - function: () => removeMember(), - textColor: '#ED5858', - }; - - - - const isAccountOwnerAdmin = groupInfo?.members?.some( - (member) => pCAIP10ToWallet(member?.wallet)?.toLowerCase() === account?.toLowerCase() && member?.isAdmin - ); - - const handlePrevious = () => { - setShowAddMoreWalletModal(false); - }; - - const onClose = () => { - setModal(false); - } - - const isMobile = useMediaQuery(device.mobileL); - if(groupInfo){ - return( - - {!showAddMoreWalletModal && (
-
- -
- - Group Info - - onClose()} cursor='pointer' /> -
- - - - -
- {groupInfo?.groupName} - {groupInfo?.members?.length} Members -
-
- - - Group Description - {groupInfo?.groupDescription} - - - - - -
- {groupInfo?.isPublic ? 'Public' : 'Private'} - {groupInfo?.isPublic ? 'Chats are not encrypted' : 'Chats are encrypted'} -
-
+ /> + {/* {(groupInfo.rules?.chat?.conditions || + groupInfo.rules?.entry?.conditions) && ( + + )} */} + + {isAccountOwnerAdmin(groupInfo, account!) && + groupInfo?.members && + groupInfo?.members?.length < 10 && ( + setShowAddMoreWalletModal(true)} + > + - {isAccountOwnerAdmin && groupInfo?.members && groupInfo?.members?.length < 10 && ( - setShowAddMoreWalletModal(true)} - > - - - - Add more wallets - - )} - -
+ + Add more wallets + + + )} + +
{groupInfo?.pendingMembers?.length > 0 && ( - + )} +
+ +
+ {groupInfo?.members && + groupInfo?.members?.length > 0 && + groupInfo?.members.map((item, index) => ( + - )} + ))} +
+ {showAddMoreWalletModal && ( + + )}
+ ); +}; + +export const GroupInfoModal = ({ + theme, + setModal, + groupInfo, + setGroupInfo, +}: GroupInfoModalProps) => { + const [activeComponent, setActiveComponent] = useState( + GROUPINFO_STEPS.GROUP_INFO + ); + const handleNextInfo = () => { + console.log('criteria'); + setActiveComponent((activeComponent + 1) as GROUP_INFO_TYPE); + console.log(activeComponent); + }; + + const handlePreviousInfo = () => { + setActiveComponent((activeComponent - 1) as GROUP_INFO_TYPE); + }; + + const renderComponent = () => { + switch (activeComponent) { + case GROUPINFO_STEPS.GROUP_INFO: + return ( + + ); + case GROUPINFO_STEPS.CRITERIA: + return ; + + default: + return ( + + ); + } + }; + const [showAddMoreWalletModal, setShowAddMoreWalletModal] = + useState(false); + useState(false); + const [memberList, setMemberList] = useState([]); + const [isLoading, setIsLoading] = useState(false); + const [selectedMemberAddress, setSelectedMemberAddress] = useState< + string | null + >(null); + const { updateGroup } = useUpdateGroup(); + const isMobile = useMediaQuery(device.mobileL); + + const handleClose = () => onClose(); + const dropdownRef = useRef(null); + useClickAway(dropdownRef, () => setSelectedMemberAddress(null)); + const groupInfoToast = useToast(); + + const groupCreator = groupInfo?.groupCreator; + const membersExceptGroupCreator = groupInfo?.members?.filter( + (x) => x.wallet?.toLowerCase() !== groupCreator?.toLowerCase() + ); + + type UpdateGroupType = { + adminList: Array; + memberList: Array; + }; + + const handleUpdateGroup = async (options: UpdateGroupType) => { + const { adminList, memberList } = options || {}; + const updateResponse = await updateGroup({ + groupInfo, + memberList, + adminList, + }); + return { updateResponse }; + }; + + const handleAddRemove = async ( + options: UpdateGroupType & { updateKey: UpdateKeys } + ) => { + const { adminList, memberList, updateKey } = options || {}; + + try { + setIsLoading(true); + const { updateResponse } = await handleUpdateGroup({ + adminList, + memberList, + }); + + if (typeof updateResponse !== 'string') { + setGroupInfo(updateResponse); + + groupInfoToast.showMessageToast({ + toastTitle: 'Success', + toastMessage: SUCCESS_MESSAGE[updateKey], + toastType: 'SUCCESS', + getToastIcon: (size) => , + }); + } else { + groupInfoToast.showMessageToast({ + toastTitle: 'Error', + toastMessage: updateResponse, + toastType: 'ERROR', + getToastIcon: (size) => , + }); + } + } catch (error) { + console.error('Error', error); + groupInfoToast.showMessageToast({ + toastTitle: 'Error', + toastMessage: 'Please, try again', + toastType: 'ERROR', + getToastIcon: (size) => , + }); + } finally { + if (updateKey === UPDATE_KEYS.ADD_MEMBER) handleClose(); + setIsLoading(false); + setSelectedMemberAddress(null); + } + }; + const removeMember = async () => { + const updatedMemberList = getUpdatedMemberList( + groupInfo, + selectedMemberAddress! + ); + const adminList = getUpdatedAdminList( + groupInfo, + selectedMemberAddress, + true + ); + await handleAddRemove({ + memberList: updatedMemberList, + adminList, + updateKey: UPDATE_KEYS.REMOVE_MEMBER, + }); + }; + + const addMembers = async () => { + //Already Present Members and PendingMembers + const groupMemberList = convertToWalletAddressList([ + ...groupInfo.members, + ...groupInfo.pendingMembers, + ]); + + //Newly Added Members and alreadyPresent Members in the groupchat + const newMembersToAdd = memberList.map((member: any) => member.wallets); + const members = [...groupMemberList, ...newMembersToAdd]; + + //Admins wallet address from both members and pendingMembers + const adminList = getAdminList?.(groupInfo); + + await handleAddRemove({ + memberList: members, + adminList, + updateKey: UPDATE_KEYS.ADD_MEMBER, + }); + }; + + const updateGroupAdmin = async (updateKey: UpdateKeys) => { + const groupMemberList = convertToWalletAddressList([ + ...groupInfo.members, + ...groupInfo.pendingMembers, + ]); + const newAdminList = getUpdatedAdminList( + groupInfo, + selectedMemberAddress, + !(updateKey === UPDATE_KEYS.ADD_ADMIN) + ); + await handleAddRemove({ + memberList: groupMemberList, + adminList: newAdminList, + updateKey, + }); + }; + + const onClose = (): void => { + setModal(false); + }; + + if (groupInfo) { + return ( + + {!showAddMoreWalletModal && ( +
+ -
- {groupInfo?.members && groupInfo?.members?.length > 0 && groupInfo?.members.map((item, index) => ( - - ))} -
- -
)} - - - - {showAddMoreWalletModal && ( - - )} -
- ) -} else { return null } - -} + + -const ProfileDiv = styled.div<{minHeight?: number}>` - display: flex; - flex-direction: column; - justify-content: flex-start; - padding-right: 3px; - align-items: center; - min-width: 445px; - min-height: 72px; - max-height: 216px; - min-height: ${(props) => `${props.minHeight}px`}; - overflow-y: auto; - overflow-x: hidden; - &&::-webkit-scrollbar { - width: 4px; +
+ + {groupInfo?.groupName} + + + {groupInfo?.members?.length} Members + +
+
+ {renderComponent()} +
+ )} +
+ ); + } else { + return null; } - &&::-webkit-scrollbar-thumb { - background: #cf1c84; - border-radius: 10px; - } - @media (max-width: 480px) { - min-width: 300px; - } -`; +}; +//styles const GroupHeader = styled.div` - margin-top: 34px; - display: flex; - flex-direction: row; - width: 100%; - gap: 19px; + margin-top: 34px; + display: flex; + flex-direction: row; + width: 100%; + gap: 19px; `; const GroupDescription = styled.div` - margin-top: 34px; - display: flex; - flex-direction: column; - width: 100%; - align-items: flex-start; - gap: 5px; + margin-top: 34px; + display: flex; + flex-direction: column; + width: 100%; + align-items: flex-start; + gap: 5px; `; - const PublicEncrypted = styled.div` - margin-top: 20px; - display: flex; - flex-direction: row; - width: 100%; - gap: 19px; - align-items: center; - border: ${(props) => `1px solid ${props.theme.defaultBorder}`}; - border-radius: 16px; - padding: 16px; - box-sizing: border-box; -`; - -const GroupMembers = styled.div` - margin-top: 20px; - display: flex; - flex-direction: row; - width: 100%; - align-items: center; -`; - -const AdminItem = styled.div` - background: rgb(244, 220, 234); - color: rgb(213, 58, 148); - margin-left: auto; - font-size: 10px; - padding: 6px; - border-radius: 8px; + margin-top: 20px; + display: flex; + flex-direction: row; + width: 100%; + gap: 19px; + align-items: center; + border: ${(props) => props.theme.border.modalInnerComponents}; + border-radius: ${(props) => props.theme.borderRadius.modalInnerComponents}; + padding: 16px; + box-sizing: border-box; + background: ${(props) => props.theme.backgroundColor.modalHoverBackground}; `; const AddWalletContainer = styled.div` - margin-top: 20px; - border: ${(props) => `1px solid ${props.theme.defaultBorder}`}; - border-radius: 16px; - width: 100%; - padding: 20px 16px; - box-sizing: border-box; - display: flex; - flex-direction: row; - justify-content: center; - cursor: pointer; - align-items: center; + margin-top: 20px; + border: ${(props) => props.theme.border.modalInnerComponents}; + border-radius: ${(props) => props.theme.borderRadius.modalInnerComponents}; + width: 100%; + padding: 20px 16px; + box-sizing: border-box; + display: flex; + flex-direction: row; + justify-content: center; + cursor: pointer; + align-items: center; `; const GroupPendingMembers = styled.div` - margin-top: 3px; - display: flex; - flex-direction: row; - width: 100%; - align-items: center; - background: ${(props) => props.theme.pendingCardBackground}; - padding: 10px 15px; - box-sizing: border-box; - - &:last-child { - border-radius: 0px 0px 16px 16px; - } -`; + margin-top: 3px; + display: flex; + flex-direction: row; + width: 100%; + align-items: center; + background: ${(props) => props.theme.backgroundColor.modalHoverBackground}; + padding: 10px 15px; + box-sizing: border-box; + &:last-child { + border-radius: 0px 0px 16px 16px; + } +`; const PendingRequestWrapper = styled.div` - width: 100%; - margin-top: 20px; - border: ${(props) => `1px solid ${props.theme.defaultBorder}`}; - border-radius: 16px; - padding: 0px 0px; - box-sizing: border-box; + width: 100%; + margin-top: 20px; + border: ${(props) => props.theme.border.modalInnerComponents}; + border-radius: ${(props) => props.theme.borderRadius.modalInnerComponents}; + padding: 0px 0px; + box-sizing: border-box; `; const PendingSection = styled.div` - width: 100%; - display: flex; - flex-direction: row; - align-items: center; - flex: 1; - cursor: pointer; - padding: 15px 20px; - box-sizing: border-box; + width: 100%; + display: flex; + flex-direction: row; + align-items: center; + flex: 1; + cursor: pointer; + padding: 15px 20px; + box-sizing: border-box; `; const ArrowImage = styled(Image)` - margin-left: auto; - transform: ${(props) => props?.setPosition ? 'rotate(0)' : 'rotate(180deg)'}; + margin-left: auto; + transform: ${(props) => + props?.setPosition ? 'rotate(0)' : 'rotate(180deg)'}; `; - const Badge = styled.div` - margin: 0 0 0 5px; - font-size: 13px; - background: rgb(207, 28, 132); - padding: 4px 8px; - border-radius: 7px; - color: white; - font-weight: 700; + margin: 0 0 0 5px; + font-size: 13px; + background: rgb(207, 28, 132); + padding: 4px 8px; + border-radius: 7px; + color: white; + font-weight: 700; `; + +const ConditionSection = styled(Section)<{ theme: IChatTheme }>` + &::-webkit-scrollbar-thumb { + background: ${(props) => props.theme.scrollbarColor}; + border-radius: 10px; + } + &::-webkit-scrollbar-button { + height: 20px; + } + &::-webkit-scrollbar { + width: 4px; + } +`; + +//auto update members when an user accepts not done diff --git a/packages/uiweb/src/lib/components/chat/ChatProfile/MemberListContainer.tsx b/packages/uiweb/src/lib/components/chat/ChatProfile/MemberListContainer.tsx index c92529fef..52885bec5 100644 --- a/packages/uiweb/src/lib/components/chat/ChatProfile/MemberListContainer.tsx +++ b/packages/uiweb/src/lib/components/chat/ChatProfile/MemberListContainer.tsx @@ -1,183 +1,201 @@ -import { useContext, useRef, useState } from "react"; -import { ThemeContext } from "../theme/ThemeProvider"; -import { useClickAway } from "../../../hooks"; -import Dropdown, { DropdownValueType } from "./DropDown"; +import { useContext, useRef, useState } from 'react'; + +import styled from 'styled-components'; + +import { ThemeContext } from '../theme/ThemeProvider'; +import { useClickAway } from '../../../hooks'; +import Dropdown, { DropdownValueType } from '../reusables/DropDown'; +import { Section, Span } from '../../reusables/sharedStyling'; + import DismissAdmin from '../../../icons/dismissadmin.svg'; import AddAdmin from '../../../icons/addadmin.svg'; import Remove from '../../../icons/remove.svg'; -import styled from "styled-components"; -import { Section, Image, Span } from "../../reusables/sharedStyling"; -import { MemberListContainerType, WalletProfileContainerProps } from "../exportedTypes"; -import { findObject } from "../helpers/helper"; -import { device } from "../../../config"; -import { shortenText } from "../../../helpers"; - -export const MemberListContainer = ({ key, memberData, handleMembers, handleMemberList, lightIcon, darkIcon, memberList }: MemberListContainerType) => { - const theme = useContext(ThemeContext); - const [selectedWallet, setSelectedWallet] = useState(null); +import { IChatTheme, User } from '../exportedTypes'; +import { findObject } from '../helpers/helper'; +import { device } from '../../../config'; +import { shortenText } from '../../../helpers'; +import { ProfileContainer } from '../reusables'; + + +type MemberListContainerType = { + key?: number; + memberData: User; + handleMemberList: (member: User) => void; + handleMembers?: (value: User[]) => void; + darkIcon: any; + memberList?: any; +}; + +export interface WalletProfileContainerProps { + id?: any; + background?: any; + border?: string; +}; + +export const MemberListContainer = ({ + key, + memberData, + handleMembers, + handleMemberList, + darkIcon, + memberList, +}: MemberListContainerType) => { + const theme = useContext(ThemeContext); + const [selectedWallet, setSelectedWallet] = useState(null); const [dropdownHeight, setDropdownHeight] = useState(0); const dropdownRef = useRef(null); - useClickAway(dropdownRef, () => setSelectedWallet(null)); - const removeAdminDropdown: DropdownValueType = - { id: 'dismiss_admin', title: 'Dismiss as admin', icon: DismissAdmin, function: () => dismissGroupAdmin() } + const removeAdminDropdown: DropdownValueType = { + id: 'dismiss_admin', + title: 'Dismiss as admin', + icon: DismissAdmin, + function: () => dismissGroupAdmin(), + }; - const addAdminDropdown: DropdownValueType = - { id: 'dismiss_admin', title: 'Make group admin', icon: AddAdmin, function: () => makeGroupAdmin() } + const addAdminDropdown: DropdownValueType = { + id: 'dismiss_admin', + title: 'Make group admin', + icon: AddAdmin, + function: () => makeGroupAdmin(), + }; - const removeUserDropdown: DropdownValueType = - { id: 'remove_user', title: 'Remove', icon: Remove, function: () => removeUser() } + const removeUserDropdown: DropdownValueType = { + id: 'remove_user', + title: 'Remove', + icon: Remove, + function: () => removeUser(), + }; const dismissGroupAdmin = () => { - const updatedMembers = memberList.map((member:any) => member?.wallets?.toLowerCase() == memberData?.wallets?.toLowerCase() ? ({ ...member, isAdmin: false }) : member) - handleMembers?.(updatedMembers) - setSelectedWallet(null) - } + const updatedMembers = memberList.map((member: any) => + member?.wallets?.toLowerCase() == memberData?.wallets?.toLowerCase() + ? { ...member, isAdmin: false } + : member + ); + handleMembers?.(updatedMembers); + setSelectedWallet(null); + }; const makeGroupAdmin = () => { - const updatedMembers = memberList.map((member: any) => member?.wallets?.toLowerCase() == memberData?.wallets?.toLowerCase() ? ({ ...member, isAdmin: true }) : member) - handleMembers?.(updatedMembers) - setSelectedWallet(null) - } + const updatedMembers = memberList.map((member: any) => + member?.wallets?.toLowerCase() == memberData?.wallets?.toLowerCase() + ? { ...member, isAdmin: true } + : member + ); + handleMembers?.(updatedMembers); + setSelectedWallet(null); + }; const removeUser = () => { - handleMemberList(memberData) - setSelectedWallet(null) - } - + handleMemberList(memberData); + setSelectedWallet(null); + }; const handleHeight = (id: any) => { - const containerHeight = document.getElementById(id)?.getBoundingClientRect(); + const containerHeight = document + .getElementById(id) + ?.getBoundingClientRect(); setDropdownHeight(containerHeight?.top); }; - - - return ( - - -
- -
- - {shortenText(memberData?.wallets?.split(':')[1], 8, true)} -
+ -
+
{memberData?.isAdmin && ( - + Admin - + )} -
{ - handleHeight(memberData?.wallets); - setSelectedWallet(null) - memberList - ? findObject(memberData, memberList, 'wallets') - ? setSelectedWallet(memberData?.wallets) - : handleMemberList(memberData) - : handleMemberList(memberData) - }} - > - {/* {theme === 'light' ? lightIcon : darkIcon} */} - {darkIcon} -
+
{ + handleHeight(memberData?.wallets); + setSelectedWallet(null); + memberList + ? findObject(memberData, memberList, 'wallets') + ? setSelectedWallet(memberData?.wallets) + : handleMemberList(memberData) + : handleMemberList(memberData); + }} + > + {darkIcon} +
{selectedWallet?.toLowerCase() == memberData?.wallets?.toLowerCase() && ( - 500 ? '30%' : "45%" }} ref={dropdownRef} theme={theme}> + 500 ? '30%' : '45%' }} + ref={dropdownRef} + theme={theme} + > )} - - - ) -} - + + ); +}; const WalletProfileContainer = styled(Section)` - // position: relative; - // padding: 5px 16px; - // margin: 8px 0px; - // justify-content: space-between; - // // min-width: 450px; - // min-width: 100%; - // box-sizing: border-box; - // align-items: center; - // border-radius: 16px; - - // @media (max-width: 480px) { - // // min-width: 300px; - // } - - justify-content: space-between; - padding: 8px 16px; - border-radius: 16px; - position: relative; - box-sizing: border-box; - width: 100%; - // background-color: ${(props) => props.theme.snapFocusBg}; - max-height: 64px; - align-self: stretch; - display: flex; - height: auto; - z-index: auto; - flex: 1; - @media (max-width: 480px) { - max-width: 100%; - } - -`; - -const WalletProfile = styled(Section)` - justify-content: flex-start; + justify-content: space-between; + padding: 8px 16px; + border: ${(props) => props.border}; + position: relative; + box-sizing: border-box; + width: 100%; + max-height: 64px; + align-self: stretch; + display: flex; + height: auto; + z-index: auto; + flex: 1; + @media (max-width: 480px) { + max-width: 100%; + } `; const DropdownContainer = styled.div` - // position: absolute; - // left: 48%; - // border-radius: 16px; - // padding: 14px 8px; - // background: ${(props) => props.theme.modalContentBackground}; - // border: 1px solid ${(props) => props.theme.modalBorderColor}; - // z-index: 400; - // @media ${device.mobileL} { - // left: 27%; - // } - // @media (min-width: 426px) and (max-width: 1150px) { - // left: 47%; - // } position: absolute; left: 48%; top: 69%; - border-radius: 16px; + border-radius: ${(props) => props.theme.borderRadius.modalInnerComponents}; padding: 14px 8px; z-index: 999999999999 !important; display: flex; flex-direction: column !important; - background: ${(props) => props.theme.modalContentBackground}; - border: 1px solid ${(props) => props.theme.modalBorderColor}; + background: ${(props) => props.theme.backgroundColor.modalBackground}; + border: ${(props) => props.theme.border.modalInnerComponents}; @media ${device.mobileL} { left: 27%; @@ -185,7 +203,7 @@ const DropdownContainer = styled.div` @media (min-width: 426px) and (max-width: 1150px) { left: 48%; } - @media (max-width: 480px){ + @media (max-width: 480px) { left: 25%; } -`; \ No newline at end of file +`; diff --git a/packages/uiweb/src/lib/components/chat/ChatProfile/MemberProfileCard.tsx b/packages/uiweb/src/lib/components/chat/ChatProfile/MemberProfileCard.tsx new file mode 100644 index 000000000..80b6bfa9f --- /dev/null +++ b/packages/uiweb/src/lib/components/chat/ChatProfile/MemberProfileCard.tsx @@ -0,0 +1,154 @@ +// React + Web3 Essentials +import React, { useContext, useState } from 'react'; + +// External Packages +import styled from 'styled-components'; + +// Internal Components +import { MoreLightIcon } from '../../../icons/MoreLight'; +import { shortenText } from '../../../helpers'; +import { ThemeContext } from '../theme/ThemeProvider'; +import { useChatData } from '../../../hooks'; +import { Section, Span } from '../../reusables'; +import Dropdown from '../reusables/DropDown'; +import { pCAIP10ToWallet } from '../../../helpers'; +import { device } from '../../../config'; +import { IChatTheme } from '../theme'; +import { ProfileContainer } from '../reusables'; + +type MemberProfileCardProps = { + key?: number | string; + member?: any; + dropdownValues?: any; + selectedMemberAddress?: any; + setSelectedMemberAddress?: any; + dropdownRef?: any; +}; + + +export const MemberProfileCard = ({ + key, + member, + dropdownValues, + selectedMemberAddress, + setSelectedMemberAddress, + dropdownRef, +}: MemberProfileCardProps) => { + const theme = useContext(ThemeContext); + const { account } = useChatData(); + + const [dropdownHeight, setDropdownHeight] = useState(0); + + const handleHeight = (id: any) => { + const containerHeight = document + .getElementById(id) + ?.getBoundingClientRect(); + setDropdownHeight(containerHeight?.top); + }; + + + return ( + + +
+ {member?.isAdmin && ( + + Admin + + )} + {pCAIP10ToWallet(member?.wallet)?.toLowerCase() !== + account?.toLowerCase() && + dropdownValues.length > 0 && ( +
{ + handleHeight(member.wallet); + setSelectedMemberAddress(member?.wallet); + }} + style={{ cursor: 'pointer' }} + > + +
+ )} +
+ {selectedMemberAddress?.toLowerCase() == + member?.wallet?.toLowerCase() && ( + 570 ? '30%' : '40%' }} + theme={theme} + ref={dropdownRef} + > + + + )} +
+ ); +}; + +//styles +const ProfileCardItem = styled(Section)<{ id: any; key: any; background: any }>` + justify-content: space-between; + padding: 8px 8px; + border-radius: 16px; + position: relative; + box-sizing: border-box; + width: 100%; + max-height: 64px; + align-self: stretch; + display: flex; + height: auto; + z-index: auto; + flex: 1; + @media (max-width: 480px) { + max-width: 100%; + } +`; + +const DropdownContainer = styled(Section)` + position: absolute; + left: 48%; + top: 69%; + border-radius: ${(props) => props.theme.borderRadius.modalInnerComponents}; + padding: 14px 8px; + z-index: 999999999999 !important; + display: flex; + flex-direction: column !important; + background: ${(props) => props.theme.backgroundColor.modalBackground}; + border: ${(props) => props.theme.border.modal}; + + @media ${device.mobileL} { + left: 27%; + } + @media (min-width: 426px) and (max-width: 1150px) { + left: 48%; + } + @media (max-width: 480px) { + left: 25%; + } +`; diff --git a/packages/uiweb/src/lib/components/chat/ChatProfile/ProfileCard.tsx b/packages/uiweb/src/lib/components/chat/ChatProfile/ProfileCard.tsx deleted file mode 100644 index 8244a3b24..000000000 --- a/packages/uiweb/src/lib/components/chat/ChatProfile/ProfileCard.tsx +++ /dev/null @@ -1,158 +0,0 @@ -// React + Web3 Essentials -import React, { useContext ,useState } from 'react'; - -// External Packages -import styled from 'styled-components'; -import { ethers } from 'ethers'; - -// Internal Components -import { MoreLightIcon } from '../../../icons/MoreLight'; -import { MoreDarkIcon } from '../../../icons/MoreDark'; -import { shortenText } from "../../../helpers"; -import { ThemeContext } from "../theme/ThemeProvider"; -import { useChatData, useClickAway} from "../../../hooks"; -import { Image, Section, Span } from "../../reusables"; -import Dropdown from './DropDown'; -import { pCAIP10ToWallet } from '../../../helpers'; -import { device } from "../../../config"; - - -type ProfileCardProps = { - key?: number | string, - member?: any, - dropdownValues?: any; - selectedMemberAddress?: any; - setSelectedMemberAddress?: any; - dropdownRef?: any; -} - -export const ProfileCard = ({ - key, - member, - dropdownValues, - selectedMemberAddress, - setSelectedMemberAddress, - dropdownRef, -}: ProfileCardProps) => { - const theme = useContext(ThemeContext); - const { account } = useChatData(); - - const [dropdownHeight, setDropdownHeight] = useState(0); - - const handleHeight = (id: any) => { - const containerHeight = document.getElementById(id)?.getBoundingClientRect(); - setDropdownHeight(containerHeight?.top); - }; - - return ( - -
-
- -
- - {shortenText(member?.wallet?.split(':')[1], 6, true)} - -
-
- {member?.isAdmin && ( - - Admin - - )} - {pCAIP10ToWallet(member?.wallet)?.toLowerCase() !== account?.toLowerCase() && dropdownValues.length > 0 && ( -
{ - handleHeight(member.wallet); - setSelectedMemberAddress(member?.wallet) - }} - style={{ cursor: 'pointer' }} - > - -
- )} -
- {selectedMemberAddress?.toLowerCase() == member?.wallet?.toLowerCase() && ( - 570 ? '30%' : '40%' }} - theme={theme} - ref={dropdownRef}> - - - )} -
- ); -}; - -const ProfileCardItem = styled(Section)<{id: any, key: any, background: any}>` - justify-content: space-between; - padding: 8px 16px; - border-radius: 16px; - position: relative; - box-sizing: border-box; - width: 100%; - // background-color: ${(props) => props.theme.snapFocusBg}; - max-height: 64px; - align-self: stretch; - display: flex; - height: auto; - z-index: auto; - flex: 1; - @media (max-width: 480px) { - max-width: 100%; - } -`; - -const DropdownContainer = styled(Section)` - position: absolute; - left: 48%; - top: 69%; - border-radius: 16px; - padding: 14px 8px; - z-index: 999999999999 !important; - display: flex; - flex-direction: column !important; - background: ${(props) => props.theme.modalContentBackground}; - border: 1px solid ${(props) => props.theme.modalBorderColor}; - - @media ${device.mobileL} { - left: 27%; - } - @media (min-width: 426px) and (max-width: 1150px) { - left: 48%; - } - @media (max-width: 480px){ - left: 25%; - } -`; diff --git a/packages/uiweb/src/lib/components/chat/ChatViewBubble/ChatViewBubble.tsx b/packages/uiweb/src/lib/components/chat/ChatViewBubble/ChatViewBubble.tsx index f527c883b..1b388c4ed 100644 --- a/packages/uiweb/src/lib/components/chat/ChatViewBubble/ChatViewBubble.tsx +++ b/packages/uiweb/src/lib/components/chat/ChatViewBubble/ChatViewBubble.tsx @@ -1,8 +1,23 @@ -import { useContext, useEffect, useState } from 'react'; -import { Section, Span, Image } from '../../reusables'; +import { + ReactElement, + ReactNode, + useContext, + useEffect, + useState, +} from 'react'; + import moment from 'moment'; import styled from 'styled-components'; +import { TwitterTweetEmbed } from 'react-twitter-embed'; + +import { Section, Span, Image } from '../../reusables'; +import { checkTwitterUrl } from '../helpers/twitter'; +import { ChatDataContext } from '../../../context'; +import { useChatData } from '../../../hooks'; +import { ThemeContext } from '../theme/ThemeProvider'; + import { FileMessageContent } from '../../../types'; +import { IMessagePayload, TwitterFeedReturnType } from '../exportedTypes'; import { FILE_ICON } from '../../../config'; import { formatFileSize, @@ -10,12 +25,6 @@ import { pCAIP10ToWallet, shortenText, } from '../../../helpers'; -import { checkTwitterUrl } from '../helpers/twitter'; -import { IMessagePayload, TwitterFeedReturnType } from '../exportedTypes'; -import { TwitterTweetEmbed } from 'react-twitter-embed'; -import { ChatDataContext } from '../../../context'; -import { useChatData } from '../../../hooks'; -import { ThemeContext } from '../theme/ThemeProvider'; const SenderMessageAddress = ({ chat }: { chat: IMessagePayload }) => { const { account } = useContext(ChatDataContext); @@ -73,19 +82,18 @@ const SenderMessageProfilePicture = ({ chat }: { chat: IMessagePayload }) => { ); }; - -//can create a wrapper for till the senderMessageAddress and use it for all cards(types of messages) -const MessageCard = ({ +const MessageWrapper = ({ chat, - position, + children, isGroup, + maxWidth, }: { chat: IMessagePayload; - position: number; + children: ReactNode; isGroup: boolean; + maxWidth?: string; }) => { const theme = useContext(ThemeContext); - const time = moment(chat.timestamp).format('hh:mm a'); return (
{isGroup && }
{isGroup && } -
+ +
+ ); +}; + +const MessageCard = ({ + chat, + position, + isGroup, +}: { + chat: IMessagePayload; + position: number; + isGroup: boolean; +}) => { + const theme = useContext(ThemeContext); + const time = moment(chat.timestamp).format('hh:mm a'); + return ( + +
+ {' '} +
+ {chat.messageContent.split('\n').map((str) => ( + + {str} + + ))} +
+ - {' '} -
- {chat.messageContent.split('\n').map((str) => ( - - {str} - - ))} -
- - {time} - -
+ {time} +
-
+ ); }; @@ -189,51 +216,42 @@ const FileCard = ({ const size = fileContent.size; return ( -
- {isGroup && } -
- {isGroup && } -
- extension icon -
- - {shortenText(name, 11)} - - - {formatFileSize(size)} - -
- - + +
+ extension icon +
+ + {shortenText(name, 11)} + + + {formatFileSize(size)} +
+ +
-
+ ); }; @@ -247,27 +265,21 @@ const ImageCard = ({ isGroup: boolean; }) => { return ( -
- {isGroup && } -
- {isGroup && } -
- -
+ +
+
-
+ ); }; @@ -281,32 +293,21 @@ const GIFCard = ({ isGroup: boolean; }) => { return ( -
- {isGroup && } -
- {isGroup && } -
- -
+ +
+
-
+ ); }; @@ -322,31 +323,21 @@ const TwitterCard = ({ position: number; }) => { return ( -
- {isGroup && } -
- {isGroup && } -
- -
+ +
+
-
+ ); }; export const ChatViewBubble = ({ chat }: { chat: IMessagePayload }) => { - const { account} = - useChatData(); + const { account } = useChatData(); const position = pCAIP10ToWallet(chat.fromDID).toLowerCase() !== account?.toLowerCase() ? 0 @@ -367,8 +358,6 @@ export const ChatViewBubble = ({ chat }: { chat: IMessagePayload }) => { } }, [chat.toDID, isGroup]); - - if (messageType === 'TwitterFeedLink') { chat.messageType = 'TwitterFeedLink'; } diff --git a/packages/uiweb/src/lib/components/chat/ChatViewList/ChatViewList.tsx b/packages/uiweb/src/lib/components/chat/ChatViewList/ChatViewList.tsx index a7011cd74..b63b1ee00 100644 --- a/packages/uiweb/src/lib/components/chat/ChatViewList/ChatViewList.tsx +++ b/packages/uiweb/src/lib/components/chat/ChatViewList/ChatViewList.tsx @@ -76,7 +76,6 @@ export const ChatViewList: React.FC = ( setChatStatusText(''); }, [chatId, account, env]); - //need to do something about fetching connectedUser in every component useEffect(() => { (async () => { if (!connectedProfile && account) { @@ -129,6 +128,7 @@ export const ChatViewList: React.FC = ( setLoading(false); })(); }, [chatId, pgpPrivateKey, account, env]); + //moniters socket changes useEffect(() => { if (checkIfSameChat(messagesSinceLastConnection, account!, chatId)) { @@ -166,22 +166,6 @@ export const ChatViewList: React.FC = ( } } }, [groupInformationSinceLastConnection]); - // useEffect(() => { - // if ( - // Object.keys(messagesSinceLastConnection || {}).length && - // Object.keys(chatFeed || {}).length && - // checkIfSameChat(messagesSinceLastConnection, account!, chatId) - // ) { - - // } - // }, [messagesSinceLastConnection]); - // useEffect(() => { - // (async function () { - // if (!account && !env && !chatId) return; - // const hash = await fetchConversationHash({ conversationId: chatId }); - // setConversationHash(hash?.threadHash); - // })(); - // }, [chatId, account, env, pgpPrivateKey]); useEffect(() => { if (conversationHash) { diff --git a/packages/uiweb/src/lib/components/chat/ConnectButton/ConnectButton.tsx b/packages/uiweb/src/lib/components/chat/ConnectButton/ConnectButton.tsx index f739f6e0a..b0090d41a 100644 --- a/packages/uiweb/src/lib/components/chat/ConnectButton/ConnectButton.tsx +++ b/packages/uiweb/src/lib/components/chat/ConnectButton/ConnectButton.tsx @@ -1,15 +1,17 @@ -import styled from 'styled-components'; -import { IChatTheme } from '../theme'; -import { useChatData } from '../../../hooks'; -import * as PushAPI from '@pushprotocol/restapi'; import { useContext, useEffect, useState } from 'react'; -import { init, useConnectWallet } from "@web3-onboard/react"; -import injectedModule from "@web3-onboard/injected-wallets"; + +import styled from 'styled-components'; import { Signer, ethers } from 'ethers'; +import { useAccount, useChatData } from '../../../hooks'; import { ThemeContext } from '../theme/ThemeProvider'; -import { device } from '../../../config'; +import useGetChatProfile from '../../../hooks/useGetChatProfile'; +import useCreateChatProfile from '../../../hooks/useCreateChatProfile'; +import useDecryptPGPKey from '../../../hooks/useDecryptPGPKey'; + import { getAddressFromSigner } from '../../../helpers'; +import { IChatTheme } from '../theme'; +import { device } from '../../../config'; /** * @interface IThemeProps @@ -18,12 +20,12 @@ import { getAddressFromSigner } from '../../../helpers'; interface IThemeProps { theme?: IChatTheme; } - interface IConnectButtonProps { autoConnect?: boolean; } -export const ConnectButtonSub: React.FC = ({autoConnect = false}) => { - const [{ wallet, connecting }, connect, disconnect] = useConnectWallet(); + +export const ConnectButtonSub = ({autoConnect = false}) => { + const {wallet, connecting , connect, disconnect} = useAccount(); const { signer, @@ -35,11 +37,14 @@ export const ConnectButtonSub: React.FC = ({autoConnect = f setSigner, } = useChatData(); const theme = useContext(ThemeContext); + const {fetchChatProfile} = useGetChatProfile(); + const {creteChatProfile} = useCreateChatProfile(); + const {decryptPGPKey} = useDecryptPGPKey(); - const newFunc = () => { + + const setUserData = () => { if (wallet) { (async () => { - const ethersProvider = new ethers.providers.Web3Provider(wallet.provider, 'any') const signer = ethersProvider.getSigner() const newAdd = await getAddressFromSigner(signer) @@ -52,14 +57,12 @@ export const ConnectButtonSub: React.FC = ({autoConnect = f setPgpPrivateKey(null) } } - useEffect(() => { if(wallet && !autoConnect) disconnect(wallet); - newFunc() + setUserData() }, [wallet]) - useEffect(() => { (async () => { if (account && signer) { @@ -68,24 +71,23 @@ export const ConnectButtonSub: React.FC = ({autoConnect = f })(); }, [account, signer]); + const handleUserCreation = async () => { if (!account && !env) return; try { - let user = await PushAPI.user.get({ account: account!, env: env }); + let user = await fetchChatProfile({ profileId: account! ,env}); if (!user) { if (!signer) return; - user = await PushAPI.user.create({ - signer: signer, - env: env, - }); + user = await creteChatProfile({ signer: signer ,env}); } if (user?.encryptedPrivateKey && !pgpPrivateKey) { - const decryptPgpKey = await PushAPI.chat.decryptPGPKey({ - encryptedPGPPrivateKey: user.encryptedPrivateKey, + const decryptPgpKey = await decryptPGPKey({ + encryptedPrivateKey: user.encryptedPrivateKey, account: account!, signer: signer, env: env, }); + if(decryptPgpKey) setPgpPrivateKey(decryptPgpKey); } } catch (e: any) { @@ -107,10 +109,10 @@ const ConnectButtonDiv = styled.div` button{ background: ${(props) => `${props.theme.backgroundColor.buttonBackground}!important`}; - // color: ${(props) => `${props.theme.backgroundColor.buttonText}!important`}; - color: #fff; + color: ${(props) => `${props.theme.textColor.buttonText}!important`}; text-align:center; font-size: 1em; + cursor:pointer; border-radius: 10px; padding: 10px 20px; outline: none; @@ -126,4 +128,7 @@ const ConnectButtonDiv = styled.div` @media ${device.mobileL} { font-size: 12px; } -`; + body.modal-open { + overflow-y: hidden; + } +`; \ No newline at end of file diff --git a/packages/uiweb/src/lib/components/chat/ConnectButton/index.tsx b/packages/uiweb/src/lib/components/chat/ConnectButton/index.tsx index 6994cbb3c..b6b603fdf 100644 --- a/packages/uiweb/src/lib/components/chat/ConnectButton/index.tsx +++ b/packages/uiweb/src/lib/components/chat/ConnectButton/index.tsx @@ -2,30 +2,30 @@ import { IChatTheme } from '../theme'; import coinbaseWalletModule from '@web3-onboard/coinbase' import { ConnectButtonSub } from './ConnectButton'; -import { InfuraAPIKey } from '../../../config'; +import { BLOCKNATIVE_PROJECT_ID, InfuraAPIKey } from '../../../config'; import { Web3OnboardProvider } from '@web3-onboard/react'; import injectedModule, { ProviderLabel } from '@web3-onboard/injected-wallets'; import walletConnectModule from '@web3-onboard/walletconnect' import init from '@web3-onboard/core'; -import { ethers } from 'ethers'; +const APP_META_DATA = { + name: 'Push Protocol', + icon: 'https://files.slack.com/files-pri/T011WQBLH39-F05QWQA0MSR/pushlogoblocknative.png', + description: 'Example showcasing how to connect a wallet.', -/** - * @interface IThemeProps - * this interface is used for defining the props for styled components - */ -interface IThemeProps { - theme?: IChatTheme; + recommendedInjectedWallets: [ + { name: 'MetaMask', url: 'https://metamask.io' }, + ] } const wcv2InitOptions = { - projectId: '64a44a0fb537407bfe97d24330e4109c', + projectId: BLOCKNATIVE_PROJECT_ID, requiredChains: [1, 56] } const walletConnect = walletConnectModule(wcv2InitOptions) const coinbaseWalletSdk = coinbaseWalletModule({ darkMode: true }) -const chains = [ +const CHAINS = [ { id: '0x1', token: 'ETH', @@ -69,20 +69,11 @@ const wallets = [injectedModule(), walletConnect, coinbaseWalletSdk] -const appMetadata = { - name: 'Push Protocol', - icon: 'https://push.org/static/media/PushLogoTextBlack.fa01629c2ebd2149bab979861756591d.svg', - description: 'Example showcasing how to connect a wallet.', - - recommendedInjectedWallets: [ - { name: 'MetaMask', url: 'https://metamask.io' }, - ] -} const web3OnBoard = init({ wallets, - chains, - appMetadata, + chains:CHAINS, + appMetadata:APP_META_DATA, accountCenter: { desktop: { enabled: false diff --git a/packages/uiweb/src/lib/components/chat/CreateGroup/AddButtons.tsx b/packages/uiweb/src/lib/components/chat/CreateGroup/AddButtons.tsx new file mode 100644 index 000000000..73b56e0c5 --- /dev/null +++ b/packages/uiweb/src/lib/components/chat/CreateGroup/AddButtons.tsx @@ -0,0 +1,27 @@ +import React, { useContext } from 'react' + +import { Button } from '../reusables' +import { ThemeContext } from '../theme/ThemeProvider'; + +interface AddButtonsProps { + title: string; + handleNext?: () => void; +} + +export const AddButtons = ({title, handleNext}: AddButtonsProps) => { + const theme = useContext(ThemeContext); + return ( + + ) +} diff --git a/packages/uiweb/src/lib/components/chat/CreateGroup/AddCriteria.tsx b/packages/uiweb/src/lib/components/chat/CreateGroup/AddCriteria.tsx new file mode 100644 index 000000000..8f9bbd0c2 --- /dev/null +++ b/packages/uiweb/src/lib/components/chat/CreateGroup/AddCriteria.tsx @@ -0,0 +1,557 @@ +import { useContext, useEffect, useState } from 'react'; + +import { + Button, + DropDownInput, + DropdownValueType, + ModalHeader, + TextInput, +} from '../reusables'; +import { Section, } from '../../reusables'; +import useMediaQuery from '../../../hooks/useMediaQuery'; +import { GatingRulesInformation, ModalHeaderProps } from './CreateGroupModal'; +import { useChatData } from '../../../hooks'; +import { QuantityInput } from '../reusables/QuantityInput'; +import { ThemeContext } from '../theme/ThemeProvider'; +import { Checkbox } from '../reusables/Checkbox'; +import OptionButtons from '../reusables/OptionButtons'; + +import EthereumSvg from '../../../icons/ethereum.svg'; +import PolygonSvg from '../../../icons/polygon.svg'; +import BSCSvg from '../../../icons/bsc.svg'; +import OptimismSvg from '../../../icons/optimisim.svg'; +import { BLOCKCHAIN_NETWORK, device } from '../../../config'; +import { GUILD_COMPARISON_OPTIONS, INVITE_CHECKBOX_LABEL } from '../constants'; +import { + CATEGORY, + DropdownCategoryValuesType, + DropdownSubCategoryValuesType, + SUBCATEGORY, + TYPE, + SubCategoryKeys, + TypeKeys, + ReadonlyInputType, +} from '../types'; +import { Data, GuildData, PushData, Rule } from '../types/tokenGatedGroupCreationType'; + + + +const AddCriteria = ({ + handlePrevious, + handleNext, + onClose, + criteriaStateManager +}: ModalHeaderProps) => { + const [selectedTypeValue, setSelectedTypeValue] = useState(0); + const [selectedCategoryValue, setSelectedCategoryValue] = useState(0); + const [selectedSubCategoryValue, setSelectedSubCategoryValue] = + useState(0); + const [guildComparison, setGuildComparison] = useState('') + const [selectedChainValue, setSelectedChainValue] = useState(0); + const [contract, setContract] = useState(''); + const [inviteCheckboxes, setInviteCheckboxes] = useState<{ + admin: boolean; + owner: boolean; + }>({ admin: true, owner: true }); + const [url, setUrl] = useState(''); + const [guildId, setGuildId] = useState(''); + const [specificRoleId, setSpecificRoleId] = useState(''); + + const [quantity, setQuantity] = useState<{ value: number; range: number }>({ + value: 0, + range: 0, + }); + const { env } = useChatData(); + const theme = useContext(ThemeContext); + + const isMobile = useMediaQuery(device.mobileL); + + const dropdownQuantityRangeValues: Array = [ + { + id: 0, + title: 'Greater than', + value: '>', + function: () => setQuantity({ ...quantity, range: 0 }), + }, + { + id: 1, + title: 'Greater or equal to', + value: '>=', + function: () => setQuantity({ ...quantity, range: 1 }), + }, + { + id: 2, + title: 'Less than', + value: '<', + function: () => setQuantity({ ...quantity, range: 2 }), + }, + { + id: 3, + title: 'Less or equal to', + value: '<=', + function: () => setQuantity({ ...quantity, range: 3 }), + }, + { + id: 4, + title: 'Equal to', + value: '==', + function: () => setQuantity({ ...quantity, range: 4 }), + }, + { + id: 5, + title: 'Not equal to', + value: '!=', + function: () => setQuantity({ ...quantity, range: 5 }), + }, + ]; + const dropdownTypeValues: Array = [ + { + id: 0, + title: 'Push protocol', + value: TYPE.PUSH, + function: () => setSelectedTypeValue(0), + }, + { + id: 1, + title: 'Guild', + value: TYPE.GUILD, + function: () => setSelectedTypeValue(1), + }, + ]; + const dropdownCategoryValues: DropdownCategoryValuesType = { + PUSH: [ + { + id: 0, + value: CATEGORY.ERC20, + title: 'Token ERC20', + function: () => setSelectedCategoryValue(0), + }, + { + id: 1, + value: CATEGORY.ERC721, + title: 'NFT ERC721', + function: () => setSelectedCategoryValue(1), + }, + { + id: 2, + value: CATEGORY.INVITE, + title: 'Invite', + function: () => setSelectedCategoryValue(2), + }, + { + id: 3, + value: CATEGORY.CustomEndpoint, + title: 'Custom Endpoint', + function: () => setSelectedCategoryValue(3), + }, + ], + GUILD: { + value: CATEGORY.ROLES, + title: 'Roles', + }, + }; + + const tokenCategoryValues = [ + { + id: 0, + value: SUBCATEGORY.HOLDER, + title: 'Holder', + function: () => setSelectedSubCategoryValue(0), + }, + { + id: 1, + value: SUBCATEGORY.OWENER, + title: 'Owner', + function: () => setSelectedSubCategoryValue(1), + }, + ]; + const dropdownSubCategoryValues: DropdownSubCategoryValuesType = { + ERC20: tokenCategoryValues, + ERC721: tokenCategoryValues, + INVITE: { + value: SUBCATEGORY.DEFAULT, + title: 'Default', + }, + CustomEndpoint: [ + { + id: 0, + value: SUBCATEGORY.GET, + title: 'Get', + function: () => setSelectedSubCategoryValue(0), + }, + ], + ROLES: { + value: SUBCATEGORY.DEFAULT, + title: 'Default', + }, + }; + + const dropdownChainsValues: Array = [ + { + id: 0, + value: BLOCKCHAIN_NETWORK[env].ETHEREUM, + title: 'Ethereum', + icon: EthereumSvg, + function: () => setSelectedChainValue(0), + }, + { + id: 1, + value: BLOCKCHAIN_NETWORK[env].POLYGON, + title: 'Polygon', + icon: PolygonSvg, + function: () => setSelectedChainValue(1), + }, + { + id: 2, + value: BLOCKCHAIN_NETWORK[env].BSC, + title: 'BSC', + icon: BSCSvg, + function: () => setSelectedChainValue(2), + }, + { + id: 3, + value: BLOCKCHAIN_NETWORK[env].OPTIMISM, + title: 'Optimism', + icon: OptimismSvg, + function: () => setSelectedChainValue(3), + }, + ]; + + const getCategoryDropdownValues = () => { + return dropdownCategoryValues[ + dropdownTypeValues[selectedTypeValue].value as TypeKeys + ]; + }; + + const getSelectedCategoryValue = () => { + const category = getCategoryDropdownValues(); + if (Array.isArray(category)) + return (category as DropdownValueType[])[selectedCategoryValue].value!; + else return category.value! as SubCategoryKeys; + }; + + const getSelectedSubCategoryValue = () => { + const subCategory = getSubCategoryDropdownValues(); + if (Array.isArray(subCategory)) + return (subCategory as DropdownValueType[])[selectedCategoryValue].value!; + else return subCategory.value! as SubCategoryKeys; + }; + + const checkIfTokenNFT = () => { + const category = getSelectedCategoryValue(); + if (category === CATEGORY.ERC20 || category === CATEGORY.ERC721) + return true; + + return false; + }; + + const checkIfCustomEndpoint = () => { + const category = getSelectedCategoryValue(); + if (category === CATEGORY.CustomEndpoint) return true; + return false; + }; + + const checkIfPushInvite = () => { + const accessType = dropdownTypeValues[selectedTypeValue].value; + if (accessType === TYPE.PUSH) { + const category = getSelectedCategoryValue(); + if (category === CATEGORY.INVITE) return true; + } + + return false; + }; + + const checkIfGuild = () => { + const accessType = dropdownTypeValues[selectedTypeValue].value; + if (accessType === TYPE.GUILD) { + return true; + } + + return false; + }; + + const getSubCategoryDropdownValues = () => { + const category = getCategoryDropdownValues(); + if (Array.isArray(category)) + return dropdownSubCategoryValues[ + (category as DropdownValueType[])[selectedCategoryValue] + .value as SubCategoryKeys + ]; + else return dropdownSubCategoryValues[category.value as SubCategoryKeys]; + }; + + const onQuantityChange = (e: any) => { + setQuantity({ ...quantity, value: e.target.value }); + }; + + const verifyAndDoNext = ()=>{ + const _type = dropdownTypeValues[selectedTypeValue].value as 'PUSH' | 'GUILD' + const category:string = _type === "PUSH" ? (dropdownCategoryValues[_type] as DropdownValueType[])[ + selectedCategoryValue + ].value || CATEGORY.ERC20 : "ROLES" + + let subCategory = "DEFAULT" + if(_type === "PUSH"){ + if(category === CATEGORY.ERC20 || category === CATEGORY.ERC721){ + subCategory = tokenCategoryValues[selectedSubCategoryValue].value + }else if(category === CATEGORY.CustomEndpoint){ + subCategory = "GET" + } + } + + const getData = (type:string, category:string):Data=>{ + if(type === "PUSH"){ + if(category === CATEGORY.ERC20 || category === CATEGORY.ERC721){ + const selectedChain = dropdownChainsValues[selectedChainValue].value || 'eip155:1'; + return { + contract: `${selectedChain}:${contract}`, + amount: quantity.value, + comparison:dropdownQuantityRangeValues[quantity.range].value, + decimals: 18, + } + }else if(category === CATEGORY.INVITE){ + const _inviteRoles = [] + if(inviteCheckboxes.admin){ + _inviteRoles.push("ADMIN") + } + if(inviteCheckboxes.owner){ + _inviteRoles.push("OWNER") + } + + return{ + inviterRoles: _inviteRoles as ['OWNER' | 'ADMIN'] + } + }else{ + // CATEGORY.CustomEndpoint + // TODO: validate url + return{ + url:url + } + } + }else{ + // GUILD type + return { + id:guildId, + comparison:guildComparison, + role:guildComparison === 'specific' ? specificRoleId : "*", + } + } + } + + const rule:Rule = { + type: _type, + category: category, + subcategory: subCategory, + data: getData(_type, category), + } + + criteriaState.addNewRule(rule) + + if(handlePrevious){ + handlePrevious() + } + + } + + const criteriaState = criteriaStateManager.getSelectedCriteria() + + + // Autofill the form for the update + useEffect(()=>{ + if(criteriaState.isUpdateCriteriaEnabled()){ + //Load the states + const oldValue = criteriaState.selectedRules[criteriaState.updateCriteriaIdx] + + if(oldValue.type === 'PUSH'){ + + // category + setSelectedCategoryValue( + (dropdownCategoryValues.PUSH as DropdownValueType[]).findIndex(obj => obj.value === oldValue.category) + ) + + const pushData = oldValue.data as PushData + + // sub category + if(oldValue.category === CATEGORY.ERC20 || oldValue.category === CATEGORY.ERC721){ + setSelectedSubCategoryValue( + tokenCategoryValues.findIndex(obj => obj.value === oldValue.subcategory) + ) + + const contractAndChain:string[] = (pushData.contract || "eip155:1:0x").split(':') + setSelectedChainValue( + dropdownChainsValues.findIndex( + obj => obj.value === contractAndChain[0]+":"+contractAndChain[1] + ) + ) + setContract(contractAndChain.length === 3 ? contractAndChain[2]: "") + setQuantity({ + value:pushData.amount || 0, + range:dropdownQuantityRangeValues.findIndex( + obj => obj.value === pushData.comparison + ) + }) + }else if(oldValue.category === CATEGORY.INVITE){ + setInviteCheckboxes({ + admin:true, + owner:true, + }) + }else{ + // invite + setUrl(pushData.url || "") + } + }else{ + // guild condition + setGuildId((oldValue.data as GuildData).id) + setSpecificRoleId((oldValue.data as GuildData).role) + setGuildComparison((oldValue.data as GuildData).comparison) + } + + setSelectedTypeValue( + dropdownTypeValues.findIndex(obj => obj.value === oldValue.type) + ) + } + },[]) + + return ( +
+
+ +
+ + {Array.isArray(getCategoryDropdownValues()) ? ( + + ) : ( + + )} + + {Array.isArray(getSubCategoryDropdownValues()) ? ( + + ) : ( + + )} + {/* shift to minor components leave for now*/} + {checkIfTokenNFT() && ( + <> + + setContract(e.target.value)} + placeholder="e.g. 0x123..." + /> + + + )} + + {checkIfCustomEndpoint() && ( + setUrl(e.target.value)} + placeholder="e.g. abc.com" + /> + )} + {checkIfPushInvite() && ( +
+ {Object.keys(INVITE_CHECKBOX_LABEL).map((key) => ( + + setInviteCheckboxes({ + admin:true, + owner:true + }) + } + checked={ + inviteCheckboxes[key as keyof typeof INVITE_CHECKBOX_LABEL] + } + /> + ))} +
+ )} + + {checkIfGuild() && ( + <> + setGuildId(e.target.value)} + placeholder="e.g. 4687" + /> + { + setGuildComparison(newEl)}} + /> + + {guildComparison === "specific" && + setSpecificRoleId(e.target.value)} + placeholder="e.g. 4687" + /> + } + + + + )} + + +
+ ); +}; + +export default AddCriteria; + diff --git a/packages/uiweb/src/lib/components/chat/CreateGroup/ConditionsComponent.tsx b/packages/uiweb/src/lib/components/chat/CreateGroup/ConditionsComponent.tsx new file mode 100644 index 000000000..4efb0d594 --- /dev/null +++ b/packages/uiweb/src/lib/components/chat/CreateGroup/ConditionsComponent.tsx @@ -0,0 +1,322 @@ +import React, { useContext, useRef, useState } from 'react'; + +import styled from 'styled-components'; + +import { Section, Span } from '../../reusables'; +import { MoreDarkIcon } from '../../../icons/MoreDark'; +import { ThemeContext } from '../theme/ThemeProvider'; +import Dropdown, { DropdownValueType } from '../reusables/DropDown'; +import { ConditionArray, ConditionData, IChatTheme } from '../exportedTypes'; +import { useClickAway } from '../../../hooks'; +import { CATEGORY, CRITERIA_TYPE, CriteriaType, TOKEN_NFT_COMPARISION, TokenNftComparision } from '../types'; + +import EditSvg from '../../../icons/EditSvg.svg'; +import RemoveSvg from '../../../icons/RemoveSvg.svg'; +import { shortenText } from '../../../helpers'; +import { GUILD_COMPARISON_OPTIONS } from '../constants'; + +export type CriteraValueType = { + invertedIcon?: any; + id: number; + type?: string; + title?: string; + icon?: string; + function: () => void; +}; + +interface CriteriaProps { + conditionData: ConditionArray[]; + moreOptions?: boolean; + deleteFunction?: (idx: number) => void; + updateFunction?: (idx: number) => void; +} + +interface MoreOptionsContainerProps { + handleMoreOptionsClick: (row: number, col: number) => void; + setSelectedIndex: any; + selectedIndex: Array | null; + row: number; + col: number; + dropDownValues: DropdownValueType[]; +} + +// fix dropdown position mobile view and z index +const MoreOptionsContainer = ({ + handleMoreOptionsClick, + setSelectedIndex, + selectedIndex, + row, + col, + dropDownValues, +}: MoreOptionsContainerProps) => { + const theme = useContext(ThemeContext); + const dropdownRef = useRef(null); + + useClickAway(dropdownRef, () => setSelectedIndex(null)); + return ( +
handleMoreOptionsClick(row, col)} position='static'> + + {selectedIndex?.length && selectedIndex[0] === row && ( + + + + )} +
+ ); +}; + +const CriteriaSection = ({ criteria }: { criteria: ConditionData }) => { + const theme = useContext(ThemeContext); + + const getTokenNftComparisionLabel = () => { + return TOKEN_NFT_COMPARISION[ + criteria?.data?.['comparison'] as TokenNftComparision + ]; + }; + const checkIfNftToken = () => { + if ( + criteria?.category === CATEGORY.ERC721 || + criteria?.category === CATEGORY.ERC20 + ) + return true; + return false; + }; + + const getGuildRole = () =>{ + return (GUILD_COMPARISON_OPTIONS.find(option => option.value === criteria?.data?.['comparison']))?.heading; + + } + return ( +
+ + {CRITERIA_TYPE[criteria.category as CriteriaType]} + + {checkIfNftToken() && ( + + + {getTokenNftComparisionLabel()}{' '} + + {/* need to fetch token symbol */} + {criteria?.data?.['amount']} {criteria.category} + + )} + {criteria.category === CATEGORY.INVITE && ( + + Owner and Admin can invite + + )} + {criteria.category === CATEGORY.CustomEndpoint && ( + + {shortenText(criteria.data?.['url'],30)} + + )} + {criteria.category === CATEGORY.ROLES && ( + + {criteria?.data?.['id']} {' '} + + with {' '} + + {getGuildRole()} role + + )} +
+ ); +}; +// fix dropdown ui +const ConditionsComponent = ({ + conditionData, + deleteFunction, + updateFunction, + moreOptions = true, +}: CriteriaProps) => { + const [selectedIndex, setSelectedIndex] = useState | null>( + null + ); + + const dropdownRef = useRef(null); + + const dropDownValues: DropdownValueType[] = [ + { + id: 0, + value: 'Edit', + title: 'Edit', + icon: EditSvg, + function: () => { + if (updateFunction) { + if (selectedIndex) { + updateFunction(selectedIndex[0]); + setSelectedIndex(null); + } + } + }, + }, + { + id: 1, + value: 'Remove', + title: 'Remove', + icon: RemoveSvg, + function: () => { + if (deleteFunction) { + if (selectedIndex) { + deleteFunction(selectedIndex[0]); + setSelectedIndex(null); + } + } + }, + }, + ]; + const theme = useContext(ThemeContext); + + useClickAway(dropdownRef, () => setSelectedIndex(null)); + + const handleMoreOptionsClick = (row: number, col: number) => { + setSelectedIndex([row, col]); + }; + + return ( +
+ {conditionData && + conditionData.slice(1).map((criteria, row) => ( +
+ {criteria.length <= 2 && + criteria.length >= 1 && + criteria.map((singleCriteria, col) => ( + <> + {singleCriteria.type && ( +
+ + {moreOptions && ( + + )} +
+ )} + + ))} + + {criteria[0]?.operator && criteria.length > 2 && ( + +
+ {criteria.map((singleCriteria, col) => ( + <> + {singleCriteria.type && ( + <> +
+ +
+ + )} + + ))} +
+
+ {criteria.map((singleCriteria) => ( + <> + {criteria.length > 2 && + singleCriteria.operator && + !singleCriteria.type && ( + + {singleCriteria.operator} + + )} + + ))} + {moreOptions && ( + + )} +
+
+ )} + {conditionData && + row < conditionData.length - 2 && + conditionData[0][0]?.operator && ( + // this can be reused + + {conditionData[0][0].operator} + + )} +
+ ))} +
+ ); +}; + +export default ConditionsComponent; + +const DropdownContainer = styled.div` + position: absolute; + // left: 48%; + top: 0; + right: 0; + border-radius: ${(props) => props.theme.borderRadius.modalInnerComponents}; + + padding: 6px 32px 6px 12px; + z-index: 999999999999 !important; + display: flex; + flex-direction: column !important; + background: ${(props) => props.theme.backgroundColor.modalBackground}; + border: ${(props) => props.theme.border.modalInnerComponents}; +`; + +const OperatorSpan = styled(Span)<{ theme: IChatTheme }>` + padding: 4px 8px; + margin: 8px 0; + border-radius: ${(props) => props.theme.borderRadius.modalInnerComponents}; + background: ${(props) => props.theme.backgroundColor.modalHoverBackground}; + color: ${(props) => props.theme.textColor?.modalSubHeadingText}; +`; + +const CriteriaGroup = styled(Section)<{ theme: IChatTheme }>` + border: ${(props) => props.theme.border?.modalInnerComponents}; +`; diff --git a/packages/uiweb/src/lib/components/chat/CreateGroup/CreateGroupModal.tsx b/packages/uiweb/src/lib/components/chat/CreateGroup/CreateGroupModal.tsx new file mode 100644 index 000000000..9b2ad7045 --- /dev/null +++ b/packages/uiweb/src/lib/components/chat/CreateGroup/CreateGroupModal.tsx @@ -0,0 +1,341 @@ +import React, { useContext, useEffect, useRef, useState } from 'react'; + +import styled from 'styled-components'; +import { ToastContainer } from 'react-toastify'; +import { AiTwotoneCamera } from 'react-icons/ai'; +import { MdError } from 'react-icons/md'; + +import { ModalHeader } from '../reusables/Modal'; +import { Modal } from '../reusables/Modal'; +import { TextInput } from '../reusables/TextInput'; +import { TextArea } from '../reusables/TextArea'; +import { Section, Span } from '../../reusables'; +import { Button } from '../reusables'; +import { CreateGroupType } from './CreateGroupType'; +import useToast from '../reusables/NewToast'; +import { CreateGroupModalProps, IChatTheme } from '../exportedTypes'; +import useMediaQuery from '../../../hooks/useMediaQuery'; +import { DefineCondtion } from './DefineCondition'; +import AddCriteria from './AddCriteria'; +import { SpamIcon } from '../../../icons/SpamIcon'; +import { ThemeContext } from '../theme/ThemeProvider'; +import { + CriteriaStateManagerType, + useCriteriaStateManager, +} from '../../../hooks/chat/useCriteriaState'; + +import { Image } from '../../../config/styles'; +import { ProfilePicture, device } from '../../../config'; + +export const CREATE_GROUP_STEP_KEYS = { + INPUT_DETAILS: 1, + GROUP_TYPE: 2, + DEFINITE_CONDITION: 3, + ADD_CRITERIA: 4, +} as const; + +export type CreateGroupStepKeys = + typeof CREATE_GROUP_STEP_KEYS[keyof typeof CREATE_GROUP_STEP_KEYS]; + +interface GroupInputDetailsType { + groupName: string; + groupDescription: string; + groupImage: string; +} + +export const CreateGroupModal: React.FC = ({ + onClose, +}) => { + const [activeComponent, setActiveComponent] = useState( + // replace it with info one + CREATE_GROUP_STEP_KEYS.INPUT_DETAILS + ); + + const handleNext = () => { + setActiveComponent((activeComponent + 1) as CreateGroupStepKeys); + }; + const handlePrevious = () => { + setActiveComponent((activeComponent - 1) as CreateGroupStepKeys); + }; + + const criteriaStateManager = useCriteriaStateManager(); + + useEffect(() => { + // reset update rules + if (activeComponent === 2) { + criteriaStateManager.resetRules(); + } else if (activeComponent === 3) { + criteriaStateManager.resetCriteriaIdx(); + } + }, [activeComponent]); + + const [groupInputDetails, setGroupInputDetails] = + useState({ + groupName: '', + groupDescription: '', + groupImage: '', + }); + + const renderComponent = () => { + switch (activeComponent) { + case CREATE_GROUP_STEP_KEYS.INPUT_DETAILS: + return ( + + ); + case CREATE_GROUP_STEP_KEYS.GROUP_TYPE: + return ( + + ); + case CREATE_GROUP_STEP_KEYS.DEFINITE_CONDITION: + return ( + + ); + case CREATE_GROUP_STEP_KEYS.ADD_CRITERIA: + return ( + + ); + default: + return ( + + ); + } + }; + + return ( + + {renderComponent()} + + ); +}; + +export interface ModalHeaderProps { + handleNext?: () => void; + handlePrevious?: () => void; + onClose: () => void; + criteriaStateManager: CriteriaStateManagerType; +} + +interface GroupDetailState { + groupInputDetails: GroupInputDetailsType; + setGroupInputDetails: React.Dispatch< + React.SetStateAction + >; +} + +export interface GroupTypeState { + groupInputDetails: GroupInputDetailsType; +} + +const CreateGroupDetail = ({ + handleNext, + onClose, + groupInputDetails, + setGroupInputDetails, +}: ModalHeaderProps & GroupDetailState) => { + const groupInfoToast = useToast(); + const { groupName, groupDescription, groupImage } = groupInputDetails; + const theme = useContext(ThemeContext); + + const fileUploadInputRef = useRef(null); + const isMobile = useMediaQuery(device.mobileL); + + const handleChange = (e: Event) => { + if (!(e.target instanceof HTMLInputElement)) { + return; + } + if (!e.target.files) { + return; + } + if ( + (e.target as HTMLInputElement).files && + ((e.target as HTMLInputElement).files as FileList).length + ) { + const reader = new FileReader(); + reader.readAsDataURL(e.target.files[0]); + + reader.onloadend = function () { + setGroupInputDetails({ + groupDescription, + groupName, + groupImage: reader.result as string, + }); + }; + } + }; + + const showError = (errorMessage: string) => { + groupInfoToast.showMessageToast({ + toastTitle: 'Error', + toastMessage: errorMessage, + toastType: 'ERROR', + getToastIcon: (size) => , + }); + }; + + const verifyAndHandelNext = () => { + const skipVerify = false; + + if (!skipVerify) { + // verify name + if (groupName.trim().length === 0) { + showError('Group Name is empty'); + return; + } + + // verify description + if (groupDescription.trim().length === 0) { + showError('Group Description is empty'); + return; + } + + // verify description + // if (!groupImage) { + // showError("Group image can't be empty"); + // return; + // } + } + + if (handleNext) { + handleNext(); + } + }; + + const handleUpload = () => { + if (fileUploadInputRef.current) { + fileUploadInputRef.current.click(); + } + }; + + //groupImage and desccription is optional + return ( +
+ + + + {!groupImage && ( + + + + )} + {groupImage && ( + + group image + + )} + handleChange(e as unknown as Event)} + /> + + + setGroupInputDetails({ + groupDescription, + groupName: e.target.value, + groupImage, + }) + } + /> + +