diff --git a/src/helpers/environment/Environment/Environment.ts b/src/helpers/environment/Environment/Environment.ts
index e34213489..b088bcea8 100644
--- a/src/helpers/environment/Environment/Environment.ts
+++ b/src/helpers/environment/Environment/Environment.ts
@@ -68,6 +68,15 @@ export class Environment {
}),
);
+ public static getGovernanceSubgraphUrl = (): string => {
+ const subgraphApiKey = this.getSubgraphApiKey();
+ return this._get({
+ first: true,
+ key: "VITE_GOVERNANCE_SUBGRAPH_URL",
+ fallback: `https://gateway.thegraph.com/api/${subgraphApiKey}/subgraphs/id/AQoLCXebY1Ga7DrqVaVQ85KMwS7iFof73tv9XMVGRtyJ`,
+ });
+ };
+
public static getNodeUrls = (networkId: NetworkId) => {
switch (networkId) {
case NetworkId.MAINNET:
diff --git a/src/views/Governance/Components/Status.tsx b/src/views/Governance/Components/Status.tsx
index 73b771191..b30e30c10 100644
--- a/src/views/Governance/Components/Status.tsx
+++ b/src/views/Governance/Components/Status.tsx
@@ -3,12 +3,12 @@ import { Paper } from "@olympusdao/component-library";
import { useGetCanceledTime } from "src/views/Governance/hooks/useGetCanceledTime";
import { useGetExecutedTime } from "src/views/Governance/hooks/useGetExecutedTime";
import { useGetProposalDetails } from "src/views/Governance/hooks/useGetProposalDetails";
-import { useGetProposal } from "src/views/Governance/hooks/useGetProposals";
+import { useGetProposalFromSubgraph } from "src/views/Governance/hooks/useGetProposalFromSubgraph";
import { useGetQueuedTime } from "src/views/Governance/hooks/useGetQueuedTime";
import { useGetVetoedTime } from "src/views/Governance/hooks/useGetVetoedTime";
export const Status = ({ proposalId }: { proposalId: number }) => {
- const { data: proposal } = useGetProposal({ proposalId });
+ const { data: proposal } = useGetProposalFromSubgraph({ proposalId: proposalId.toString() });
const { data: proposalDetails } = useGetProposalDetails({ proposalId });
const { data: queueTime } = useGetQueuedTime({ proposalId });
const { data: executedTime } = useGetExecutedTime({ proposalId, status: proposalDetails?.status });
diff --git a/src/views/Governance/Proposals/VoteDetails.tsx b/src/views/Governance/Proposals/VoteDetails.tsx
new file mode 100644
index 000000000..e6c546e1d
--- /dev/null
+++ b/src/views/Governance/Proposals/VoteDetails.tsx
@@ -0,0 +1,104 @@
+import {
+ Box,
+ Paper,
+ Tab,
+ Table,
+ TableBody,
+ TableCell,
+ TableContainer,
+ TableHead,
+ TableRow,
+ Tabs,
+ Typography,
+} from "@mui/material";
+import React, { useState } from "react";
+import { useParams } from "react-router-dom";
+import { useGetVotes } from "src/views/Governance/hooks/useGetVotes";
+import { VoteRow } from "src/views/Governance/Proposals/VoteRow";
+
+// Data for the tables, including a 'reason' field for each row
+const tablesData = [
+ {
+ id: "For",
+ contractValue: 1,
+ },
+ {
+ id: "Against",
+ contractValue: 0,
+ },
+ {
+ id: "Abstain",
+ contractValue: 2,
+ },
+];
+
+const TabPanel = (props: { children: React.ReactNode; value: number; index: number }) => {
+ const { children, value, index, ...other } = props;
+ return (
+
+ {value === index && {children}}
+
+ );
+};
+
+export default function GovernanceTable() {
+ const { id } = useParams();
+ const [tabIndex, setTabIndex] = useState(0);
+ const [supportValue, setSupportValue] = useState(1);
+ const { data: voteData } = useGetVotes({ proposalId: id, support: supportValue });
+
+ const handleTabChange = (event: React.SyntheticEvent, newIndex: number) => {
+ setSupportValue(tablesData[newIndex].contractValue);
+ setTabIndex(newIndex);
+ };
+
+ return (
+
+ is loading
+ TabIndicatorProps={{ style: { display: "none" } }}
+ centered
+ >
+ {" "}
+ {tablesData.map((table, index) => (
+
+ ))}
+
+
+ {tablesData.map((table, index) => (
+
+
+
+
+
+ {tablesData[tabIndex].id}
+ Votes
+
+
+
+ {voteData?.length ? (
+ voteData.map((row, i) => (
+
+
+
+ ))
+ ) : (
+
+
+ No votes yet
+
+
+ )}
+
+
+
+
+ ))}
+
+ );
+}
diff --git a/src/views/Governance/Proposals/VoteRow.tsx b/src/views/Governance/Proposals/VoteRow.tsx
new file mode 100644
index 000000000..771fa809a
--- /dev/null
+++ b/src/views/Governance/Proposals/VoteRow.tsx
@@ -0,0 +1,37 @@
+import { Box, Link, TableCell, Tooltip, Typography } from "@mui/material";
+import { formatEther } from "ethers/lib/utils.js";
+import { abbreviatedNumber } from "src/helpers";
+import { truncateEthereumAddress } from "src/helpers/truncateAddress";
+import { useEnsName } from "wagmi";
+
+export const VoteRow = ({
+ voter,
+ reason,
+ votes,
+ tx,
+}: {
+ voter: string;
+ reason?: string;
+ votes: string;
+ tx: string;
+}) => {
+ const { data: ensName } = useEnsName({ address: voter as `0x${string}` });
+ return (
+ <>
+
+
+
+ {ensName || truncateEthereumAddress(voter)}
+
+
+ {/* Render the reason if provided, and style it as a comment */}
+ {reason && (
+
+ "{reason}"
+
+ )}
+
+ {abbreviatedNumber.format(Number(formatEther(votes) || 0))} gOHM
+ >
+ );
+};
diff --git a/src/views/Governance/Proposals/index.tsx b/src/views/Governance/Proposals/index.tsx
index d4cffa21d..25976eea3 100644
--- a/src/views/Governance/Proposals/index.tsx
+++ b/src/views/Governance/Proposals/index.tsx
@@ -20,13 +20,14 @@ import { useActivateProposal } from "src/views/Governance/hooks/useActivatePropo
import { useExecuteProposal } from "src/views/Governance/hooks/useExecuteProposal";
import { useGetCurrentBlockTime } from "src/views/Governance/hooks/useGetCurrentBlockTime";
import { useGetProposalDetails } from "src/views/Governance/hooks/useGetProposalDetails";
-import { useGetProposal } from "src/views/Governance/hooks/useGetProposals";
+import { useGetProposalFromSubgraph } from "src/views/Governance/hooks/useGetProposalFromSubgraph";
import { useQueueProposal } from "src/views/Governance/hooks/useQueueProposal";
+import VoteDetails from "src/views/Governance/Proposals/VoteDetails";
import { useEnsName, useNetwork, useSwitchNetwork } from "wagmi";
export const ProposalPage = () => {
const { id } = useParams();
- const { data: proposal } = useGetProposal({ proposalId: Number(id) });
+ const { data: proposal } = useGetProposalFromSubgraph({ proposalId: id });
const { data: proposalDetails } = useGetProposalDetails({ proposalId: Number(id) });
const { data: ensAddress } = useEnsName({ address: proposalDetails?.proposer as `0x${string}` });
const [voteModalOpen, setVoteModalOpen] = useState(false);
@@ -160,7 +161,7 @@ export const ProposalPage = () => {
>
- {/* */}
+
@@ -197,11 +198,8 @@ export const ProposalPage = () => {
)}
{tabIndex === 2 && (
<>
-
- Comments
-
- No comments yet
+
>
)}
diff --git a/src/views/Governance/helpers/normalizeProposal.ts b/src/views/Governance/helpers/normalizeProposal.ts
new file mode 100644
index 000000000..a0cb77109
--- /dev/null
+++ b/src/views/Governance/helpers/normalizeProposal.ts
@@ -0,0 +1,20 @@
+import { Proposal } from "src/views/Governance/hooks/useGetProposalFromSubgraph";
+
+// Normalizes the proposal data to match the onchain format
+export const normalizeProposal = (proposal: Proposal) => {
+ return {
+ createdAtBlock: new Date(Number(proposal.blockTimestamp) * 1000),
+ details: {
+ id: proposal.proposalId,
+ proposer: proposal.proposer,
+ targets: proposal.targets,
+ values: proposal.values,
+ signatures: proposal.signatures,
+ calldatas: proposal.calldatas,
+ startBlock: proposal.startBlock,
+ description: proposal.description,
+ },
+ title: proposal.description.split(/#+\s|\n/g)[1] || `${proposal.description.slice(0, 20)}...`,
+ txHash: proposal.transactionHash,
+ };
+};
diff --git a/src/views/Governance/hooks/useGetProposalFromSubgraph.tsx b/src/views/Governance/hooks/useGetProposalFromSubgraph.tsx
new file mode 100644
index 000000000..0038bb14b
--- /dev/null
+++ b/src/views/Governance/hooks/useGetProposalFromSubgraph.tsx
@@ -0,0 +1,60 @@
+import { useQuery } from "@tanstack/react-query";
+import request, { gql } from "graphql-request";
+import { Environment } from "src/helpers/environment/Environment/Environment";
+import { normalizeProposal } from "src/views/Governance/helpers/normalizeProposal";
+
+export type Proposal = {
+ proposalId: string;
+ proposer: string;
+ targets: string[];
+ signatures: string[];
+ calldatas: string[];
+ transactionHash: string;
+ description: string;
+ blockTimestamp: string;
+ blockNumber: string;
+ startBlock: string;
+ values: string[];
+};
+
+type ProposalResponse = {
+ proposalCreated: Proposal;
+};
+
+export const useGetProposalFromSubgraph = ({ proposalId }: { proposalId?: string }) => {
+ const query = gql`
+ query {
+ proposalCreated(id: ${proposalId}) {
+ proposalId
+ proposer
+ targets
+ signatures
+ calldatas
+ transactionHash
+ description
+ blockTimestamp
+ blockNumber
+ startBlock
+ values
+ }
+ }
+`;
+
+ return useQuery(
+ ["getProposal", proposalId],
+ async () => {
+ try {
+ const subgraphUrl = Environment.getGovernanceSubgraphUrl();
+ const response = await request(subgraphUrl, query);
+ if (!response.proposalCreated) {
+ return null;
+ }
+ return normalizeProposal(response.proposalCreated);
+ } catch (error) {
+ console.error("useGetProposalFromSubgraph", error);
+ return null;
+ }
+ },
+ { enabled: !!proposalId },
+ );
+};
diff --git a/src/views/Governance/hooks/useGetProposals.tsx b/src/views/Governance/hooks/useGetProposals.tsx
deleted file mode 100644
index 6ccffe0f3..000000000
--- a/src/views/Governance/hooks/useGetProposals.tsx
+++ /dev/null
@@ -1,87 +0,0 @@
-import { useQuery } from "@tanstack/react-query";
-import { GOVERNANCE_CONTRACT } from "src/constants/contracts";
-import { Environment } from "src/helpers/environment/Environment/Environment";
-import { Providers } from "src/helpers/providers/Providers/Providers";
-import { NetworkId } from "src/networkDetails";
-import { OlympusGovernorBravo__factory } from "src/typechain";
-import { ProposalCreatedEventObject } from "src/typechain/OlympusGovernorBravo";
-
-export const useGetProposals = () => {
- const archiveProvider = Providers.getArchiveStaticProvider(NetworkId.MAINNET);
- // const contract = GOVERNANCE_CONTRACT.getEthersContract(NetworkId.MAINNET);
- const contractAddress = GOVERNANCE_CONTRACT.addresses[NetworkId.MAINNET];
- const contract = OlympusGovernorBravo__factory.connect(contractAddress, archiveProvider);
-
- return useQuery(
- ["getProposals", NetworkId.MAINNET],
- async () => {
- const proposals: { createdAtBlock: Date; details: ProposalCreatedEventObject; title: string; txHash: string }[] =
- [];
- let blockNumber = await archiveProvider.getBlockNumber();
- const startBlock = Environment.getGovernanceStartBlock();
- const chunkSize = 10000; //RPC limit
- // Fetch the last proposal ID
- const lastProposalId = (await contract.proposalCount()).toNumber();
-
- // If there are no proposals, return an empty array
- if (lastProposalId === 0) {
- console.log("No proposals found.");
- return proposals;
- }
- let foundProposalId1 = false;
-
- while (blockNumber > startBlock && !foundProposalId1) {
- const fromBlock = Math.max(blockNumber - chunkSize, startBlock);
-
- const proposalCreatedEvents = await contract.queryFilter(
- contract.filters.ProposalCreated(),
- fromBlock,
- blockNumber,
- );
-
- console.log("ProposalCreated events found:", proposalCreatedEvents.length);
-
- // Process each event
- for (const item of proposalCreatedEvents) {
- const timestamp = (await archiveProvider.getBlock(item.blockNumber)).timestamp;
-
- if (item.decode) {
- const details = { ...item.decode(item.data), values: item.args[3] } as ProposalCreatedEventObject;
-
- proposals.push({
- createdAtBlock: new Date(timestamp * 1000),
- details,
- title: details.description.split(/#+\s|\n/g)[1] || `${details.description.slice(0, 20)}...`,
- txHash: item.transactionHash,
- });
-
- // Stop if we hit proposal ID 1
- if (Number(details.id) === 1) {
- foundProposalId1 = true;
- break;
- }
- }
- }
-
- // Move to the next block range (going backwards)
- blockNumber = fromBlock - 1;
- }
- console.log("Proposals found:", proposals.length);
-
- return proposals;
- },
-
- { enabled: !!archiveProvider && !!contract },
- );
-};
-
-export const useGetProposal = ({ proposalId }: { proposalId: number }) => {
- const proposals = useGetProposals();
- return useQuery(
- ["getProposal", NetworkId.MAINNET, proposalId],
- async () => {
- return proposals.data?.find(item => Number(item?.details.id) === proposalId);
- },
- { enabled: !!proposals.data },
- );
-};
diff --git a/src/views/Governance/hooks/useGetProposalsFromSubgraph.tsx b/src/views/Governance/hooks/useGetProposalsFromSubgraph.tsx
new file mode 100644
index 000000000..dea244136
--- /dev/null
+++ b/src/views/Governance/hooks/useGetProposalsFromSubgraph.tsx
@@ -0,0 +1,41 @@
+import { useQuery } from "@tanstack/react-query";
+import request, { gql } from "graphql-request";
+import { Environment } from "src/helpers/environment/Environment/Environment";
+import { normalizeProposal } from "src/views/Governance/helpers/normalizeProposal";
+import { Proposal } from "src/views/Governance/hooks/useGetProposalFromSubgraph";
+
+export const useGetProposalsFromSubgraph = () => {
+ const query = gql`
+ query {
+ proposalCreateds(orderBy: proposalId, orderDirection: desc) {
+ proposalId
+ proposer
+ targets
+ signatures
+ calldatas
+ transactionHash
+ description
+ blockTimestamp
+ blockNumber
+ startBlock
+ values
+ }
+ }
+ `;
+
+ type Proposals = {
+ proposalCreateds: Proposal[];
+ };
+
+ return useQuery(["getProposals"], async () => {
+ try {
+ const subgraphUrl = Environment.getGovernanceSubgraphUrl();
+ const response = await request(subgraphUrl, query);
+
+ return response.proposalCreateds.map(normalizeProposal);
+ } catch (error) {
+ console.error("useGetProposalsFromSubgraph", error);
+ return [];
+ }
+ });
+};
diff --git a/src/views/Governance/hooks/useGetVotes.tsx b/src/views/Governance/hooks/useGetVotes.tsx
new file mode 100644
index 000000000..d2e848e5b
--- /dev/null
+++ b/src/views/Governance/hooks/useGetVotes.tsx
@@ -0,0 +1,38 @@
+import { useQuery } from "@tanstack/react-query";
+import request, { gql } from "graphql-request";
+import { Environment } from "src/helpers/environment/Environment/Environment";
+
+export const useGetVotes = ({ proposalId, support }: { proposalId?: string; support: number }) => {
+ return useQuery(
+ ["getVotes", proposalId, support],
+ async () => {
+ const query = gql`
+ query MyQuery {
+ voteCasts(orderBy: votes, orderDirection: desc, where: {proposalId: ${proposalId}, support: ${support} }) {
+ votes
+ voter
+ reason
+ support
+ transactionHash
+ }
+ }
+ `;
+
+ type votesResponse = {
+ voteCasts: {
+ votes: string;
+ voter: string;
+ reason: string;
+ support: number;
+ transactionHash: string;
+ }[];
+ };
+
+ const subgraphUrl = Environment.getGovernanceSubgraphUrl();
+ const response = await request(subgraphUrl, query);
+
+ return response.voteCasts || [];
+ },
+ { enabled: !!proposalId && !!support },
+ );
+};
diff --git a/src/views/Governance/index.tsx b/src/views/Governance/index.tsx
index f1958cfad..24f9dabec 100644
--- a/src/views/Governance/index.tsx
+++ b/src/views/Governance/index.tsx
@@ -19,11 +19,11 @@ import { ContractParameters } from "src/views/Governance/Components/ContractPara
import { ProposalContainer } from "src/views/Governance/Components/ProposalContainer";
import { DelegationMessage } from "src/views/Governance/Delegation/DelegationMessage";
import { GovernanceDevTools } from "src/views/Governance/hooks/dev/GovernanceDevTools";
-import { useGetProposals } from "src/views/Governance/hooks/useGetProposals";
+import { useGetProposalsFromSubgraph } from "src/views/Governance/hooks/useGetProposalsFromSubgraph";
import { useGetVotingWeight } from "src/views/Governance/hooks/useGetVotingWeight";
export const Governance = () => {
- const { data: proposals, isFetching } = useGetProposals();
+ const { data: proposals, isFetching } = useGetProposalsFromSubgraph();
const { data: currentVotingWeight } = useGetVotingWeight({});
const theme = useTheme();
const [activeProposals, setActiveProposals] = useState([]);