mirror of
https://github.com/status-im/dappconnect-vote-poll-sdk.git
synced 2025-01-12 06:14:23 +00:00
Add mobile components (#63)
This commit is contained in:
parent
605d5e1950
commit
7d97abdf82
@ -15,6 +15,7 @@ export function Proposal({ wakuVoting }: ProposalProps) {
|
|||||||
<ProposalWrapper>
|
<ProposalWrapper>
|
||||||
<ProposalHeader theme={blueTheme} wakuVoting={wakuVoting} />
|
<ProposalHeader theme={blueTheme} wakuVoting={wakuVoting} />
|
||||||
<ProposalList theme={blueTheme} wakuVoting={wakuVoting} />
|
<ProposalList theme={blueTheme} wakuVoting={wakuVoting} />
|
||||||
|
{/* <VotingEmpty theme={blueTheme} /> */}
|
||||||
<NotificationItem text={'Proposal you finalized will be settled after 10 confirmations.'} address={'#'} />
|
<NotificationItem text={'Proposal you finalized will be settled after 10 confirmations.'} address={'#'} />
|
||||||
</ProposalWrapper>
|
</ProposalWrapper>
|
||||||
)
|
)
|
||||||
@ -29,12 +30,4 @@ const ProposalWrapper = styled.div`
|
|||||||
padding: 150px 32px 50px;
|
padding: 150px 32px 50px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
min-height: 100vh;
|
min-height: 100vh;
|
||||||
|
|
||||||
@media (max-width: 600px) {
|
|
||||||
padding: 132px 16px 32px;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 425px) {
|
|
||||||
padding: 64px 16px 84px;
|
|
||||||
}
|
|
||||||
`
|
`
|
||||||
|
@ -1,10 +1,12 @@
|
|||||||
import React from 'react'
|
import React from 'react'
|
||||||
|
import { useHistory } from 'react-router'
|
||||||
import styled from 'styled-components'
|
import styled from 'styled-components'
|
||||||
import { Theme } from '@status-waku-voting/react-components'
|
import { Theme } from '@status-waku-voting/react-components'
|
||||||
import { ProposalInfo } from './ProposalInfo'
|
import { ProposalInfo } from './ProposalInfo'
|
||||||
import { ProposalVote } from './ProposalVoteCard/ProposalVote'
|
import { ProposalVote } from './ProposalVoteCard/ProposalVote'
|
||||||
|
|
||||||
interface ProposalCardProps {
|
interface ProposalCardProps {
|
||||||
|
id: number
|
||||||
theme: Theme
|
theme: Theme
|
||||||
heading: string
|
heading: string
|
||||||
text: string
|
text: string
|
||||||
@ -14,9 +16,11 @@ interface ProposalCardProps {
|
|||||||
hideModalFunction?: (val: boolean) => void
|
hideModalFunction?: (val: boolean) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
export function ProposalCard({ heading, text, address, vote, voteWinner, theme }: ProposalCardProps) {
|
export function ProposalCard({ id, heading, text, address, vote, voteWinner, theme }: ProposalCardProps) {
|
||||||
|
const history = useHistory()
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card>
|
<Card onClick={() => history.push(`/votingRoom/${id.toString}`)}>
|
||||||
<ProposalInfo heading={heading} text={text} address={address} />
|
<ProposalInfo heading={heading} text={text} address={address} />
|
||||||
<ProposalVote vote={vote} voteWinner={voteWinner} address={address} heading={heading} theme={theme} />
|
<ProposalVote vote={vote} voteWinner={voteWinner} address={address} heading={heading} theme={theme} />
|
||||||
</Card>
|
</Card>
|
||||||
|
@ -90,7 +90,7 @@ export function ProposalHeader({ theme, wakuVoting }: ProposalHeaderProps) {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
const Wrapper = styled.div`
|
export const Wrapper = styled.div`
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
@ -6,14 +6,15 @@ type ProposalInfoProps = {
|
|||||||
heading: string
|
heading: string
|
||||||
text: string
|
text: string
|
||||||
address: string
|
address: string
|
||||||
|
mobileMode?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
export function ProposalInfo({ heading, text, address }: ProposalInfoProps) {
|
export function ProposalInfo({ heading, text, address, mobileMode }: ProposalInfoProps) {
|
||||||
return (
|
return (
|
||||||
<Card>
|
<Card>
|
||||||
<CardHeading>{heading}</CardHeading>
|
<CardHeading>{heading}</CardHeading>
|
||||||
<CardText>{text}</CardText>
|
<CardText className={mobileMode ? 'mobile' : ''}>{text}</CardText>
|
||||||
<CardViewLink>
|
<CardViewLink className={mobileMode ? 'mobile' : ''}>
|
||||||
<ViewLink address={address} />
|
<ViewLink address={address} />
|
||||||
</CardViewLink>
|
</CardViewLink>
|
||||||
</Card>
|
</Card>
|
||||||
@ -68,10 +69,27 @@ export const CardText = styled.div`
|
|||||||
-webkit-line-clamp: 3;
|
-webkit-line-clamp: 3;
|
||||||
-webkit-box-orient: vertical;
|
-webkit-box-orient: vertical;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&.mobile {
|
||||||
|
@media (max-width: 600px) {
|
||||||
|
height: 100%;
|
||||||
|
overflow: unset;
|
||||||
|
text-overflow: unset;
|
||||||
|
-webkit-line-clamp: unset;
|
||||||
|
margin-bottom: 24px;
|
||||||
|
}
|
||||||
|
}
|
||||||
`
|
`
|
||||||
|
|
||||||
const CardViewLink = styled.div`
|
const CardViewLink = styled.div`
|
||||||
@media (max-width: 768px) {
|
@media (max-width: 768px) {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&.mobile {
|
||||||
|
@media (max-width: 600px) {
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 37px;
|
||||||
|
}
|
||||||
|
}
|
||||||
`
|
`
|
||||||
|
@ -22,7 +22,7 @@ export function ProposalList({ theme, wakuVoting }: ProposalListProps) {
|
|||||||
return (
|
return (
|
||||||
<List>
|
<List>
|
||||||
{votes.map((vote, idx) => {
|
{votes.map((vote, idx) => {
|
||||||
return <ProposalCard heading={vote[2]} text={vote[3]} address={'#'} theme={theme} key={idx} />
|
return <ProposalCard heading={vote[2]} text={vote[3]} address={'#'} theme={theme} key={idx} id={idx} />
|
||||||
})}
|
})}
|
||||||
{votes && votes?.length === 0 && <VotingEmpty wakuVoting={wakuVoting} theme={theme} />}
|
{votes && votes?.length === 0 && <VotingEmpty wakuVoting={wakuVoting} theme={theme} />}
|
||||||
</List>
|
</List>
|
||||||
|
@ -105,19 +105,19 @@ export const ProposingInfo = styled.div`
|
|||||||
}
|
}
|
||||||
`
|
`
|
||||||
|
|
||||||
const InfoText = styled.div`
|
export const InfoText = styled.div`
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
line-height: 16px;
|
line-height: 16px;
|
||||||
letter-spacing: 0.1px;
|
letter-spacing: 0.1px;
|
||||||
`
|
`
|
||||||
|
|
||||||
const ProposingInput = styled(TextArea)`
|
export const ProposingInput = styled(TextArea)`
|
||||||
height: 68px;
|
height: 68px;
|
||||||
`
|
`
|
||||||
const ProposingTextInput = styled(ProposingInput)`
|
export const ProposingTextInput = styled(ProposingInput)`
|
||||||
height: 222px;
|
height: 222px;
|
||||||
`
|
`
|
||||||
const Label = styled.label`
|
export const Label = styled.label`
|
||||||
width: 100%;
|
width: 100%;
|
||||||
font-size: 15px;
|
font-size: 15px;
|
||||||
line-height: 22px;
|
line-height: 22px;
|
||||||
|
@ -23,7 +23,6 @@ export function VoteAnimatedModal({
|
|||||||
return (
|
return (
|
||||||
<VoteConfirm>
|
<VoteConfirm>
|
||||||
<ConfirmText>Your vote {selectedVote === 0 ? 'against' : 'for'} this proposal has been cast!</ConfirmText>
|
<ConfirmText>Your vote {selectedVote === 0 ? 'against' : 'for'} this proposal has been cast!</ConfirmText>
|
||||||
|
|
||||||
<VoteChart
|
<VoteChart
|
||||||
votesFor={votesFor}
|
votesFor={votesFor}
|
||||||
votesAgainst={votesAgainst}
|
votesAgainst={votesAgainst}
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import React, { useState } from 'react'
|
import React, { useEffect, useState } from 'react'
|
||||||
|
import { useHistory } from 'react-router'
|
||||||
import { useEthers } from '@usedapp/core'
|
import { useEthers } from '@usedapp/core'
|
||||||
import styled from 'styled-components'
|
import styled from 'styled-components'
|
||||||
import { CreateButton, Modal, Networks, Theme } from '@status-waku-voting/react-components'
|
import { CreateButton, Modal, Networks, Theme } from '@status-waku-voting/react-components'
|
||||||
@ -16,14 +17,29 @@ export function VotingEmpty({ wakuVoting, theme }: VotingEmptyProps) {
|
|||||||
const [selectConnect, setSelectConnect] = useState(false)
|
const [selectConnect, setSelectConnect] = useState(false)
|
||||||
const [showProposeModal, setShowProposeModal] = useState(false)
|
const [showProposeModal, setShowProposeModal] = useState(false)
|
||||||
const [showProposeVoteModal, setShowProposeVoteModal] = useState(false)
|
const [showProposeVoteModal, setShowProposeVoteModal] = useState(false)
|
||||||
|
const [mobileVersion, setMobileVersion] = useState(false)
|
||||||
const [title, setTitle] = useState('')
|
const [title, setTitle] = useState('')
|
||||||
const [text, setText] = useState('')
|
const [text, setText] = useState('')
|
||||||
|
const history = useHistory()
|
||||||
|
|
||||||
const setNext = (val: boolean) => {
|
const setNext = (val: boolean) => {
|
||||||
setShowProposeVoteModal(val)
|
setShowProposeVoteModal(val)
|
||||||
setShowProposeModal(false)
|
setShowProposeModal(false)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleResize = () => {
|
||||||
|
if (window.innerWidth < 600) {
|
||||||
|
setMobileVersion(true)
|
||||||
|
} else {
|
||||||
|
setMobileVersion(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
handleResize()
|
||||||
|
window.addEventListener('resize', handleResize)
|
||||||
|
return () => window.removeEventListener('resize', handleResize)
|
||||||
|
}, [])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<VotingEmptyWrap>
|
<VotingEmptyWrap>
|
||||||
<EmptyWrap>
|
<EmptyWrap>
|
||||||
@ -60,9 +76,14 @@ export function VotingEmpty({ wakuVoting, theme }: VotingEmptyProps) {
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{account ? (
|
{account ? (
|
||||||
<CreateButton theme={theme} onClick={() => setShowProposeModal(true)}>
|
<EmptyCreateButton
|
||||||
|
theme={theme}
|
||||||
|
onClick={() => {
|
||||||
|
mobileVersion ? history.push(`/creation`) : setShowProposeModal(true)
|
||||||
|
}}
|
||||||
|
>
|
||||||
Create proposal
|
Create proposal
|
||||||
</CreateButton>
|
</EmptyCreateButton>
|
||||||
) : (
|
) : (
|
||||||
<CreateButton
|
<CreateButton
|
||||||
theme={theme}
|
theme={theme}
|
||||||
@ -138,3 +159,9 @@ const EmptyText = styled.p`
|
|||||||
margin-bottom: 20px;
|
margin-bottom: 20px;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
|
|
||||||
|
const EmptyCreateButton = styled(CreateButton)`
|
||||||
|
@media (max-width: 425px) {
|
||||||
|
position: static;
|
||||||
|
}
|
||||||
|
`
|
||||||
|
@ -0,0 +1,91 @@
|
|||||||
|
import React, { useState } from 'react'
|
||||||
|
import { useHistory } from 'react-router'
|
||||||
|
import styled from 'styled-components'
|
||||||
|
import { useEthers } from '@usedapp/core'
|
||||||
|
import { Modal, Networks, CreateButton } from '@status-waku-voting/react-components'
|
||||||
|
import { Theme } from '@status-waku-voting/react-components/dist/esm/src/style/themes'
|
||||||
|
import { Wrapper } from '../ProposalHeader'
|
||||||
|
|
||||||
|
type ProposalHeaderMobileProps = {
|
||||||
|
theme: Theme
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ProposalHeaderMobile({ theme }: ProposalHeaderMobileProps) {
|
||||||
|
const { activateBrowserWallet, account } = useEthers()
|
||||||
|
const [selectConnect, setSelectConnect] = useState(false)
|
||||||
|
const history = useHistory()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Wrapper>
|
||||||
|
<Header>
|
||||||
|
<Heading>Your voice has real power</Heading>
|
||||||
|
<HeaderText>
|
||||||
|
Take part in a decentralised governance by voting on proposals provided by community or creating your own.
|
||||||
|
</HeaderText>
|
||||||
|
</Header>
|
||||||
|
{account ? (
|
||||||
|
<CreateButton theme={theme} onClick={() => history.push(`/creation`)}>
|
||||||
|
Create proposal
|
||||||
|
</CreateButton>
|
||||||
|
) : (
|
||||||
|
<CreateButton
|
||||||
|
theme={theme}
|
||||||
|
onClick={() => {
|
||||||
|
if ((window as any).ethereum) {
|
||||||
|
activateBrowserWallet()
|
||||||
|
} else setSelectConnect(true)
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Connect to vote
|
||||||
|
</CreateButton>
|
||||||
|
)}
|
||||||
|
{selectConnect && (
|
||||||
|
<Modal heading="Connect" setShowModal={setSelectConnect} theme={theme}>
|
||||||
|
<Networks />
|
||||||
|
</Modal>
|
||||||
|
)}
|
||||||
|
</Wrapper>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const Header = styled.div`
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
text-align: center;
|
||||||
|
max-width: 680px;
|
||||||
|
|
||||||
|
@media (max-width: 425px) {
|
||||||
|
position: fixed;
|
||||||
|
padding: 12px 16px 0;
|
||||||
|
width: 100%;
|
||||||
|
background: #f8faff;
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
|
const Heading = styled.h1`
|
||||||
|
font-weight: bold;
|
||||||
|
font-size: 28px;
|
||||||
|
line-height: 38px;
|
||||||
|
letter-spacing: -0.4px;
|
||||||
|
margin: 0;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
|
||||||
|
@media (max-width: 425px) {
|
||||||
|
font-size: 22px;
|
||||||
|
line-height: 30px;
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
|
const HeaderText = styled.p`
|
||||||
|
font-size: 22px;
|
||||||
|
line-height: 32px;
|
||||||
|
margin: 0;
|
||||||
|
margin-bottom: 24px;
|
||||||
|
|
||||||
|
@media (max-width: 425px) {
|
||||||
|
font-size: 13px;
|
||||||
|
line-height: 18px;
|
||||||
|
margin-bottom: 16px;
|
||||||
|
}
|
||||||
|
`
|
@ -0,0 +1,38 @@
|
|||||||
|
import React from 'react'
|
||||||
|
import { blueTheme } from '@status-waku-voting/react-components/dist/esm/src/style/themes'
|
||||||
|
import { ProposalList } from '../ProposalList'
|
||||||
|
import { VotingEmpty } from '../VotingEmpty'
|
||||||
|
import { NotificationItem } from '../NotificationItem'
|
||||||
|
import { ProposalHeaderMobile } from './ProposalHeaderMobile'
|
||||||
|
import styled from 'styled-components'
|
||||||
|
import { WakuVoting } from '@status-waku-voting/core'
|
||||||
|
|
||||||
|
type ProposalMainMobileProps = {
|
||||||
|
wakuVoting: WakuVoting
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ProposalMainMobile({ wakuVoting }: ProposalMainMobileProps) {
|
||||||
|
return (
|
||||||
|
<ProposalWrapper>
|
||||||
|
<ProposalHeaderMobile theme={blueTheme} />
|
||||||
|
<ProposalList theme={blueTheme} wakuVoting={wakuVoting} />
|
||||||
|
{/* <VotingEmpty theme={blueTheme} /> */}
|
||||||
|
<NotificationItem text={'Proposal you finalized will be settled after 10 confirmations.'} address={'#'} />
|
||||||
|
</ProposalWrapper>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const ProposalWrapper = styled.div`
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
max-width: 1000px;
|
||||||
|
margin: 0 auto;
|
||||||
|
width: 100%;
|
||||||
|
min-height: 100vh;
|
||||||
|
padding: 132px 16px 32px;
|
||||||
|
|
||||||
|
@media (max-width: 425px) {
|
||||||
|
padding: 64px 16px 84px;
|
||||||
|
}
|
||||||
|
`
|
@ -0,0 +1,39 @@
|
|||||||
|
import React from 'react'
|
||||||
|
import { Redirect, Route, Switch } from 'react-router'
|
||||||
|
import { BrowserRouter } from 'react-router-dom'
|
||||||
|
import styled from 'styled-components'
|
||||||
|
import { ProposalVoteMobile } from './ProposalVoteMobile'
|
||||||
|
import { ProposeMobile } from './ProposeMobile'
|
||||||
|
import { ProposalMainMobile } from './ProposalMainMobile'
|
||||||
|
import { WakuVoting } from '@status-waku-voting/core'
|
||||||
|
|
||||||
|
type ProposalMobileProps = {
|
||||||
|
wakuVoting: WakuVoting
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ProposalMobile({ wakuVoting }: ProposalMobileProps) {
|
||||||
|
return (
|
||||||
|
<BrowserRouter>
|
||||||
|
<ProposalWrapper>
|
||||||
|
<Switch>
|
||||||
|
<Route exact path="/" render={() => <Redirect to="/proposal" />} />
|
||||||
|
<Route exact path="/votingRoom/:id" component={ProposalVoteMobile} />
|
||||||
|
<Route exact path="/creation" component={ProposeMobile} />
|
||||||
|
<Route exact path="/proposal">
|
||||||
|
<ProposalMainMobile wakuVoting={wakuVoting} />
|
||||||
|
</Route>
|
||||||
|
</Switch>
|
||||||
|
</ProposalWrapper>
|
||||||
|
</BrowserRouter>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const ProposalWrapper = styled.div`
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
max-width: 1000px;
|
||||||
|
margin: 0 auto;
|
||||||
|
width: 100%;
|
||||||
|
min-height: 100vh;
|
||||||
|
`
|
@ -0,0 +1,118 @@
|
|||||||
|
import React, { useState } from 'react'
|
||||||
|
import { useParams } from 'react-router'
|
||||||
|
import styled from 'styled-components'
|
||||||
|
import { useEthers } from '@usedapp/core'
|
||||||
|
import { FinalBtn, VoteBtnAgainst, VoteBtnFor } from '../Buttons'
|
||||||
|
import { VoteSubmitButton } from '../ProposalVoteCard/VoteSubmitButton'
|
||||||
|
import { VoteChart } from '../ProposalVoteCard/VoteChart'
|
||||||
|
import { ProposalInfo } from '../ProposalInfo'
|
||||||
|
import { VotePropose } from '../VotePropose'
|
||||||
|
import { VotesBtns } from '../ProposalVoteCard/ProposalVote'
|
||||||
|
|
||||||
|
interface ProposalVoteMobileProps {
|
||||||
|
vote?: number
|
||||||
|
voteWinner?: number
|
||||||
|
votesFor: number
|
||||||
|
votesAgainst: number
|
||||||
|
timeLeft: number
|
||||||
|
availableAmount: number
|
||||||
|
heading: string
|
||||||
|
text: string
|
||||||
|
address: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ProposalVoteMobile({
|
||||||
|
votesFor,
|
||||||
|
votesAgainst,
|
||||||
|
timeLeft,
|
||||||
|
vote,
|
||||||
|
voteWinner,
|
||||||
|
address,
|
||||||
|
heading,
|
||||||
|
text,
|
||||||
|
availableAmount,
|
||||||
|
}: ProposalVoteMobileProps) {
|
||||||
|
const { id } = useParams<{ id: string }>()
|
||||||
|
const { account } = useEthers()
|
||||||
|
const [proposingAmount, setProposingAmount] = useState(0)
|
||||||
|
const [selectedVoted, setSelectedVoted] = useState(0)
|
||||||
|
const [mobileVersion, setMobileVersion] = useState(true)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card>
|
||||||
|
<ProposalInfo
|
||||||
|
heading={'This is a very long, explainative and sophisticated title for a proposal.'}
|
||||||
|
text={
|
||||||
|
'This is a longer description of the proposal. Lorem ipsum dolor sit amet, consectetur adipiscing elit. Pellentesque interdum rutrum sodales. Nullam mattis fermentum libero, non volutpat. Lorem ipsum dolor sit amet, consectetur adipiscing elit. Pellentesque interdum rutrum sodales. Nullam mattis fermentum libero. Lorem ipsum dolor sit amet, consectetur adipiscing elit. Pellentesque interdum rutrum sodales. Nullam mattis fermentum libero.'
|
||||||
|
}
|
||||||
|
address={'#'}
|
||||||
|
mobileMode={mobileVersion}
|
||||||
|
/>
|
||||||
|
<VoteChartWrap>
|
||||||
|
<VoteChart votesFor={1865567} votesAgainst={1740235} timeLeft={4855555577} selectedVote={selectedVoted} />
|
||||||
|
</VoteChartWrap>
|
||||||
|
{!voteWinner && (
|
||||||
|
<VotePropose
|
||||||
|
availableAmount={65245346}
|
||||||
|
setProposingAmount={setProposingAmount}
|
||||||
|
proposingAmount={proposingAmount}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<CardButtons>
|
||||||
|
{voteWinner ? (
|
||||||
|
<FinalBtn disabled={!account}>Finalize the vote</FinalBtn>
|
||||||
|
) : (
|
||||||
|
<VotesBtns>
|
||||||
|
<VoteBtnAgainst
|
||||||
|
disabled={!account}
|
||||||
|
onClick={() => {
|
||||||
|
setSelectedVoted(0)
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Vote Against
|
||||||
|
</VoteBtnAgainst>
|
||||||
|
<VoteBtnFor
|
||||||
|
disabled={!account}
|
||||||
|
onClick={() => {
|
||||||
|
setSelectedVoted(1)
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Vote For
|
||||||
|
</VoteBtnFor>
|
||||||
|
</VotesBtns>
|
||||||
|
)}
|
||||||
|
</CardButtons>
|
||||||
|
|
||||||
|
<CardVoteBottom>
|
||||||
|
{' '}
|
||||||
|
<VoteSubmitButton votes={2345678} disabled={!account} />
|
||||||
|
</CardVoteBottom>
|
||||||
|
</Card>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Card = styled.div`
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
padding: 88px 16px 32px;
|
||||||
|
`
|
||||||
|
|
||||||
|
const CardButtons = styled.div`
|
||||||
|
width: 100%;
|
||||||
|
`
|
||||||
|
|
||||||
|
const CardVoteBottom = styled.div`
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
width: 100%;
|
||||||
|
margin-top: 32px;
|
||||||
|
`
|
||||||
|
|
||||||
|
const VoteChartWrap = styled.div`
|
||||||
|
width: 100%;
|
||||||
|
margin-bottom: 32px;
|
||||||
|
`
|
@ -0,0 +1,121 @@
|
|||||||
|
import { blueTheme } from '@status-waku-voting/react-components/dist/esm/src/style/themes'
|
||||||
|
import React, { useState } from 'react'
|
||||||
|
import styled from 'styled-components'
|
||||||
|
import { useHistory } from 'react-router'
|
||||||
|
import { ProposingBtn } from '../Buttons'
|
||||||
|
import { CardHeading, CardText } from '../ProposalInfo'
|
||||||
|
import { InfoText, Label, ProposingData, ProposingInfo, ProposingInput, ProposingTextInput } from '../ProposeModal'
|
||||||
|
import { VotePropose } from '../VotePropose'
|
||||||
|
|
||||||
|
interface ProposeVoteModalProps {
|
||||||
|
availableAmount: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ProposeMobile({ availableAmount }: ProposeVoteModalProps) {
|
||||||
|
const insufficientFunds = availableAmount < 10000
|
||||||
|
const [proposingAmount, setProposingAmount] = useState(0)
|
||||||
|
const [title, setTitle] = useState('')
|
||||||
|
const [text, setText] = useState('')
|
||||||
|
const [customData, setCustomData] = useState(false)
|
||||||
|
const history = useHistory()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ProposingDataMobile>
|
||||||
|
{insufficientFunds && (
|
||||||
|
<ProposingInfo>
|
||||||
|
<span>⚠️</span>
|
||||||
|
<InfoText>You need at least 10,000 ABC to create a proposal!</InfoText>
|
||||||
|
</ProposingInfo>
|
||||||
|
)}
|
||||||
|
<ProposingCardHeading>Create proposal</ProposingCardHeading>
|
||||||
|
|
||||||
|
{!customData && (
|
||||||
|
<ProposingCustomData>
|
||||||
|
<Label>
|
||||||
|
Title
|
||||||
|
<ProposingInput
|
||||||
|
cols={2}
|
||||||
|
maxLength={90}
|
||||||
|
placeholder="E.g. Change the rate of the token issuance"
|
||||||
|
value={title}
|
||||||
|
onInput={(e) => {
|
||||||
|
setTitle(e.currentTarget.value)
|
||||||
|
}}
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</Label>
|
||||||
|
|
||||||
|
<Label>
|
||||||
|
Description
|
||||||
|
<ProposingTextInput
|
||||||
|
maxLength={440}
|
||||||
|
placeholder="Describe your proposal as detailed as you can in 440 characters."
|
||||||
|
value={text}
|
||||||
|
onInput={(e) => {
|
||||||
|
setText(e.currentTarget.value)
|
||||||
|
}}
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</Label>
|
||||||
|
|
||||||
|
<ProposingBtn
|
||||||
|
disabled={!text || !title || insufficientFunds}
|
||||||
|
theme={blueTheme}
|
||||||
|
onClick={() => setCustomData(true)}
|
||||||
|
>
|
||||||
|
Continue
|
||||||
|
</ProposingBtn>
|
||||||
|
</ProposingCustomData>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{customData && (
|
||||||
|
<ProposingCustomData>
|
||||||
|
<CustomProposingHeading>{title}</CustomProposingHeading>
|
||||||
|
<ProposingCardText>{text}</ProposingCardText>
|
||||||
|
|
||||||
|
<VoteProposeWrap>
|
||||||
|
<VotePropose
|
||||||
|
availableAmount={65245555}
|
||||||
|
setProposingAmount={setProposingAmount}
|
||||||
|
proposingAmount={proposingAmount}
|
||||||
|
/>
|
||||||
|
</VoteProposeWrap>
|
||||||
|
|
||||||
|
<ProposingBtn
|
||||||
|
disabled={proposingAmount === 0}
|
||||||
|
onClick={() => {
|
||||||
|
history.push(`/proposal`), setTitle(''), setText('')
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Create proposal
|
||||||
|
</ProposingBtn>
|
||||||
|
</ProposingCustomData>
|
||||||
|
)}
|
||||||
|
</ProposingDataMobile>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const ProposingDataMobile = styled(ProposingData)`
|
||||||
|
padding: 88px 16px 32px;
|
||||||
|
margin-top: 0;
|
||||||
|
`
|
||||||
|
export const VoteProposeWrap = styled.div`
|
||||||
|
margin-bottom: 32px;
|
||||||
|
width: 100%;
|
||||||
|
`
|
||||||
|
|
||||||
|
const ProposingCustomData = styled.div`
|
||||||
|
width: 100%;
|
||||||
|
`
|
||||||
|
|
||||||
|
const ProposingCardHeading = styled(CardHeading)`
|
||||||
|
margin-bottom: 16px;
|
||||||
|
`
|
||||||
|
const CustomProposingHeading = styled(ProposingCardHeading)`
|
||||||
|
font-size: 22px;
|
||||||
|
margin-top: 24px;
|
||||||
|
`
|
||||||
|
|
||||||
|
const ProposingCardText = styled(CardText)`
|
||||||
|
margin-bottom: 0;
|
||||||
|
`
|
@ -1,3 +1,4 @@
|
|||||||
import { Proposal } from './components/Proposal'
|
import { Proposal } from './components/Proposal'
|
||||||
|
import { ProposalMobile } from './components/mobile/ProposalMobile'
|
||||||
|
|
||||||
export { Proposal }
|
export { Proposal, ProposalMobile }
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import React from 'react'
|
import React, { useEffect, useState } from 'react'
|
||||||
import { useWakuProposal } from '@status-waku-voting/proposal-hooks'
|
import { useWakuProposal } from '@status-waku-voting/proposal-hooks'
|
||||||
import { Proposal } from '@status-waku-voting/proposal-components'
|
import { Proposal, ProposalMobile } from '@status-waku-voting/proposal-components'
|
||||||
import { TopBar, GlobalStyle } from '@status-waku-voting/react-components'
|
import { TopBar, GlobalStyle } from '@status-waku-voting/react-components'
|
||||||
import votingIcon from './assets/images/voting.svg'
|
import votingIcon from './assets/images/voting.svg'
|
||||||
import styled from 'styled-components'
|
import styled from 'styled-components'
|
||||||
@ -27,6 +27,20 @@ const config = {
|
|||||||
function Proposals() {
|
function Proposals() {
|
||||||
const { account, library, activateBrowserWallet, deactivate } = useEthers()
|
const { account, library, activateBrowserWallet, deactivate } = useEthers()
|
||||||
const waku = useWakuProposal()
|
const waku = useWakuProposal()
|
||||||
|
const [mobileVersion, setMobileVersion] = useState(false)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleResize = () => {
|
||||||
|
if (window.innerWidth < 600) {
|
||||||
|
setMobileVersion(true)
|
||||||
|
} else {
|
||||||
|
setMobileVersion(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
handleResize()
|
||||||
|
window.addEventListener('resize', handleResize)
|
||||||
|
return () => window.removeEventListener('resize', handleResize)
|
||||||
|
}, [])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Wrapper>
|
<Wrapper>
|
||||||
@ -38,7 +52,7 @@ function Proposals() {
|
|||||||
account={account}
|
account={account}
|
||||||
deactivate={deactivate}
|
deactivate={deactivate}
|
||||||
/>
|
/>
|
||||||
{waku && <Proposal wakuVoting={waku} />}
|
{waku && (mobileVersion ? <ProposalMobile wakuVoting={waku} /> : <Proposal wakuVoting={waku} />)}
|
||||||
</Wrapper>
|
</Wrapper>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user