From 2987143b707eebed806c60bbbd9198933575dac1 Mon Sep 17 00:00:00 2001 From: ripe Date: Sat, 18 Nov 2023 13:40:33 -0500 Subject: [PATCH 1/2] fix bug when updating a candidate that is proposal update candidate. use proposalIdToUpdate in candidate --- packages/nouns-webapp/src/pages/EditCandidate/index.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/nouns-webapp/src/pages/EditCandidate/index.tsx b/packages/nouns-webapp/src/pages/EditCandidate/index.tsx index 5812e8873a..961da8f89b 100644 --- a/packages/nouns-webapp/src/pages/EditCandidate/index.tsx +++ b/packages/nouns-webapp/src/pages/EditCandidate/index.tsx @@ -245,7 +245,7 @@ const EditCandidatePage: React.FC = props => { proposalTransactions.map(({ calldata }) => calldata), // Calldatas `# ${titleValue}\n\n${bodyValue}`, // Description candidate.data?.slug, // Slug - 0, // proposalIdToUpdate + candidate.data?.proposalIdToUpdate ? candidate.data?.proposalIdToUpdate : 0, // if candidate is an update to a proposal, use the proposalIdToUpdate number commitMessage, { value: hasVotes ? 0 : updateCandidateCost ?? 0 }, // Fee for non-nouners ); From a7977e4e20f5581dbef5bb978a7117bb02a0da94 Mon Sep 17 00:00:00 2001 From: ripe Date: Sat, 18 Nov 2023 13:50:43 -0500 Subject: [PATCH 2/2] more robust validation on the update proposal edit button, fix reload sometimes not displaying prop content, add update-slug function to create a unique slug for update candidates with the same title --- .../src/pages/EditProposal/index.tsx | 64 ++++++++++++++----- 1 file changed, 48 insertions(+), 16 deletions(-) diff --git a/packages/nouns-webapp/src/pages/EditProposal/index.tsx b/packages/nouns-webapp/src/pages/EditProposal/index.tsx index 8f0fa31e8e..ea18fa7fa8 100644 --- a/packages/nouns-webapp/src/pages/EditProposal/index.tsx +++ b/packages/nouns-webapp/src/pages/EditProposal/index.tsx @@ -81,6 +81,12 @@ const EditProposalPage: React.FC = props => { proposal?.description && removeTitleFromDescription(proposal?.description, titleValue); const isProposedBySigners = proposal?.signers && proposal?.signers?.length > 0 ? true : false; + const candidateUpdateSlug = (slug: string) => { + // add random string to slug to make it unique + const randomString = Math.random().toString(36).substring(7); + return `${slug}-update-${randomString}`; + } + const handleAddProposalAction = useCallback( (transactions: ProposalTransaction | ProposalTransaction[]) => { const transactionsArray = Array.isArray(transactions) ? transactions : [transactions]; @@ -158,19 +164,24 @@ const EditProposalPage: React.FC = props => { const handleTitleInput = useCallback( (title: string) => { setTitleValue(title); - setSlug( - title - .toLowerCase() - .replace(/ /g, '-') - .replace(/[^\w-]+/g, ''), - ); + if (isProposedBySigners && title === proposal?.title) { + // if title is edited, then reset to original, make sure the unique update slug gets passed + setSlug(candidateUpdateSlug(title)); + } else { + setSlug( + title + .toLowerCase() + .replace(/ /g, '-') + .replace(/[^\w-]+/g, ''), + ); + } if (title === proposal?.title) { setIsTitleEdited(false); } else { setIsTitleEdited(true); } }, - [setTitleValue, setSlug, proposal?.title], + [setTitleValue, setSlug, proposal?.title, isProposedBySigners], ); const handleBodyInput = useCallback( @@ -358,6 +369,8 @@ const EditProposalPage: React.FC = props => { } }; + + // set initial values on page load useEffect(() => { if ( @@ -377,13 +390,14 @@ const EditProposalPage: React.FC = props => { signature: txn.functionSig ?? '', }; }); + const slugValue = proposal.title.toLowerCase().replace(/ /g, '-').replace(/[^\w-]+/g, ''); setTitleValue(proposal.title); - setSlug( - proposal.title - .toLowerCase() - .replace(/ /g, '-') - .replace(/[^\w-]+/g, ''), - ); + if (isProposedBySigners) { + // new candidate will need to be created, which needs a unique slug, so add a random string to the slug here + setSlug(candidateUpdateSlug(slugValue)); + } else { + setSlug(slugValue); + } setBodyValue(removeTitleFromDescription(proposal.description, proposal.title)); setProposalTransactions(transactions); setOriginalTitleValue(proposal.title); @@ -391,9 +405,15 @@ const EditProposalPage: React.FC = props => { setOriginalProposalTransactions(proposal.details); } // eslint-disable-next-line react-hooks/exhaustive-deps - }, []); + }, [proposal]); const handleCreateNewCandidate = async () => { + if (!proposalTransactions?.length || + !titleValue || + !bodyValue || + !slug || + !props.match.params.id + ) return; await createProposalCandidate( proposalTransactions.map(({ address }) => address), // Targets proposalTransactions.map(({ value }) => value ?? '0'), // Values @@ -401,7 +421,7 @@ const EditProposalPage: React.FC = props => { proposalTransactions.map(({ calldata }) => calldata), // Calldatas `# ${titleValue}\n\n${bodyValue}`, // Description slug, // Slug - props.match.params.id, // proposalIdToUpdate - use 0 for new proposals + props.match.params.id, // use 0 for new proposals { value: availableVotes! > 0 ? 0 : createCandidateCost }, ); }; @@ -449,6 +469,18 @@ const EditProposalPage: React.FC = props => { // eslint-disable-next-line react-hooks/exhaustive-deps }, [createProposalCandidateState, setModal]); + const isFormInvalid = () => { + if ( + !(isProposalEdited || isTransactionsEdited() || isDescriptionEdited()) || + !proposalTransactions.length || + titleValue === '' || + bodyValue === '' || + slug === '' + ) { + return true; + } + return false; + } if (!isProposer()) { return null; } @@ -535,7 +567,7 @@ const EditProposalPage: React.FC = props => { hasActiveOrPendingProposal={false} // not relevant for edit hasEnoughVote={isProposer() ? true : hasEnoughVote} isFormInvalid={ - isProposalEdited || isTransactionsEdited() || isDescriptionEdited() ? false : true + isFormInvalid() } handleCreateProposal={ isProposedBySigners ? handleCreateNewCandidate : handleUpdateProposal