Skip to content

Commit

Permalink
Merge pull request #67 from web3wagers/dev
Browse files Browse the repository at this point in the history
Merge dev features into main
  • Loading branch information
adrianvrj authored Sep 2, 2024
2 parents 1fe5b7f + c12ef87 commit 1e15922
Show file tree
Hide file tree
Showing 11 changed files with 954 additions and 19 deletions.
663 changes: 663 additions & 0 deletions LICENSE

Large diffs are not rendered by default.

4 changes: 4 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,11 @@ If you are interested in contributing to Go Stark Me, please follow these steps:
2. Comment on the issue you want to participate in, detailing how you plan to address it and how long you expect it to take.
3. Wait to be assigned to the issue.

Official Discord Channel:
- [Web3Wagers](https://discord.gg/sEpnC6JB2U)

You can also contact us on Telegram:

- [@adrian_vrj](https://t.me/adrian_vrj)
- [@EmmanuelDevCr](https://t.me/EmmanuelDevCr)

1 change: 1 addition & 0 deletions contracts/.snfoundry_cache/.prev_tests_failed
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
tests::test_fund::test_receive_vote_unsuccessful_wrong_state
1 change: 1 addition & 0 deletions contracts/src/constants/funds.cairo
Original file line number Diff line number Diff line change
@@ -1 +1,2 @@
pub mod state_constants;
pub mod fund_constants;
8 changes: 8 additions & 0 deletions contracts/src/constants/funds/fund_constants.cairo
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
// *************************************************************************
// FUND CONSTANTS
// *************************************************************************
pub mod FundConstants {
pub const UP_VOTES_NEEDED: u32 = 100;
pub const INITIAL_UP_VOTES: u32 = 0;
pub const INITIAL_GOAL: u64 = 0;
}
13 changes: 8 additions & 5 deletions contracts/src/fund.cairo
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
use starknet::ContractAddress;

#[starknet::interface]
trait IFund<TContractState> {
pub trait IFund<TContractState> {
fn getId(self: @TContractState) -> u128;
fn getOwner(self: @TContractState) -> ContractAddress;
fn setName(ref self: TContractState, name: felt252);
Expand All @@ -27,6 +27,8 @@ mod Fund {
use starknet::ContractAddress;
use starknet::get_caller_address;
use gostarkme::constants::{funds::{state_constants::FundStates},};
use gostarkme::constants::{funds::{fund_constants::FundConstants},};


// *************************************************************************
// STORAGE
Expand Down Expand Up @@ -60,9 +62,9 @@ mod Fund {
self.owner.write(owner);
self.name.write(name);
self.reason.write(reason);
self.up_votes.write(1);
self.up_votes.write(FundConstants::INITIAL_UP_VOTES);
self.goal.write(goal);
self.current_goal_state.write(0);
self.current_goal_state.write(FundConstants::INITIAL_GOAL);
self.state.write(FundStates::RECOLLECTING_VOTES);
}

Expand Down Expand Up @@ -98,9 +100,9 @@ mod Fund {
assert(
self.state.read() == FundStates::RECOLLECTING_VOTES, 'Fund not recollecting votes!'
);
self.voters.write(get_caller_address(), self.up_votes.read());
self.up_votes.write(self.up_votes.read() + 1);
if self.up_votes.read() >= 1 {
self.voters.write(get_caller_address(), self.up_votes.read());
if self.up_votes.read() >= FundConstants::UP_VOTES_NEEDED {
self.state.write(FundStates::RECOLLECTING_DONATIONS);
}
}
Expand Down Expand Up @@ -129,6 +131,7 @@ mod Fund {
fn getCurrentGoalState(self: @ContractState) -> u64 {
return self.current_goal_state.read();
}
// TODO: Validate to change method to change setState and getState
fn setIsActive(ref self: ContractState, state: u8) {
self.state.write(state);
}
Expand Down
157 changes: 157 additions & 0 deletions contracts/tests/test_fund.cairo
Original file line number Diff line number Diff line change
@@ -0,0 +1,157 @@
// *************************************************************************
// FUND TEST
// *************************************************************************
use starknet::{ContractAddress, contract_address_const};

use snforge_std::{declare, ContractClassTrait, CheatTarget};

use openzeppelin::utils::serde::SerializedAppend;

use gostarkme::fund::IFundDispatcher;
use gostarkme::fund::IFundDispatcherTrait;

fn ID() -> u128 {
1
}
fn OWNER() -> ContractAddress {
contract_address_const::<'OWNER'>()
}
fn OTHER_USER() -> ContractAddress {
contract_address_const::<'USER'>()
}
fn NAME() -> felt252 {
'NAME_FUND_TEST'
}
fn REASON() -> felt252 {
'REASON_FUND_TEST'
}
fn GOAL() -> u64 {
1000
}
fn __setup__() -> ContractAddress {
let contract = declare("Fund");
let mut calldata: Array<felt252> = array![];
calldata.append_serde(ID());
calldata.append_serde(OWNER());
calldata.append_serde(NAME());
calldata.append_serde(REASON());
calldata.append_serde(GOAL());
contract.deploy(@calldata).unwrap()
}
// *************************************************************************
// TEST
// *************************************************************************
#[test]
fn test_constructor() {
let contract_address = __setup__();
let dispatcher = IFundDispatcher { contract_address };
let id = dispatcher.getId();
let owner = dispatcher.getOwner();
let name = dispatcher.getName();
let reason = dispatcher.getReason();
let up_votes = dispatcher.getUpVotes();
let goal = dispatcher.getGoal();
let current_goal_state = dispatcher.getCurrentGoalState();
let state = dispatcher.getIsActive();
assert(id == ID(), 'Invalid id');
assert(owner == OWNER(), 'Invalid owner');
assert(name == NAME(), 'Invalid name');
assert(reason == REASON(), 'Invalid reason');
assert(up_votes == 0, 'Invalid up votes');
assert(goal == GOAL(), 'Invalid goal');
assert(current_goal_state == 0, 'Invalid current goal state');
assert(state == 1, 'Invalid state');
}

#[test]
fn test_set_name() {
let contract_address = __setup__();
let dispatcher = IFundDispatcher { contract_address };
let name = dispatcher.getName();
assert(name == NAME(), 'Invalid name');
snforge_std::start_prank(CheatTarget::One(contract_address), OWNER());
dispatcher.setName('NEW_NAME');
let new_name = dispatcher.getName();
assert(new_name == 'NEW_NAME', 'Set name method not working')
}

#[test]
fn test_set_reason() {
let contract_address = __setup__();
let dispatcher = IFundDispatcher { contract_address };
let reason = dispatcher.getReason();
assert(reason == REASON(), 'Invalid reason');
snforge_std::start_prank(CheatTarget::One(contract_address), OWNER());
dispatcher.setReason('NEW_REASON');
let new_reason = dispatcher.getReason();
assert(new_reason == 'NEW_REASON', 'Set reason method not working')
}

#[test]
fn test_set_goal() {
let contract_address = __setup__();
let dispatcher = IFundDispatcher { contract_address };
let goal = dispatcher.getGoal();
assert(goal == GOAL(), 'Invalid goal');
snforge_std::start_prank(CheatTarget::One(contract_address), OWNER());
dispatcher.setGoal(123);
let new_goal = dispatcher.getGoal();
assert(new_goal == 123, 'Set goal method not working')
}

#[test]
fn test_receive_vote_successful() {
let contract_address = __setup__();
let dispatcher = IFundDispatcher { contract_address };
dispatcher.receiveVote();
let me = dispatcher.getVoter();
// Owner vote, fund have one vote
assert(me == 1, 'Owner is not in the voters');
let votes = dispatcher.getUpVotes();
assert(votes == 1, 'Vote unuseccessful');
}

#[test]
#[should_panic(expected: ('User already voted!',))]
fn test_receive_vote_unsuccessful_double_vote() {
let contract_address = __setup__();
let dispatcher = IFundDispatcher { contract_address };
dispatcher.receiveVote();
let me = dispatcher.getVoter();
// Owner vote, fund have one vote
assert(me == 1, 'Owner is not in the voters');
let votes = dispatcher.getUpVotes();
assert(votes == 1, 'Vote unuseccessful');
// Owner vote, second time
dispatcher.receiveVote();
}

#[test]
fn test_receive_donation_successful() {
let contract_address = __setup__();
let dispatcher = IFundDispatcher { contract_address };
// Put state as recollecting dons
dispatcher.setIsActive(2);
// Put 10 strks as goal, only owner
snforge_std::start_prank(CheatTarget::One(contract_address), OWNER());
dispatcher.setGoal(10);
// Donate 5 strks
dispatcher.receiveDonation(5);
let current_goal_state = dispatcher.getCurrentGoalState();
assert(current_goal_state == 5, 'Receive donation not working');
// Donate 5 strks, the goal is done
dispatcher.receiveDonation(5);
let state = dispatcher.getIsActive();
assert(state == 3, 'State should be close');
}

#[test]
#[should_panic(expected: ('Fund not recollecting dons!',))]
fn test_receive_donation_unsuccessful_wrong_state() {
let contract_address = __setup__();
let dispatcher = IFundDispatcher { contract_address };
// Put a wrong state to receive donations
dispatcher.setIsActive(1);
// Donate
dispatcher.receiveDonation(5);
}
60 changes: 60 additions & 0 deletions frontend/gostarkme-web/animations/StardustAnimation.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
import React, { useEffect, useRef } from 'react';

export const StardustAnimation = () => {
const canvasRef = useRef<HTMLCanvasElement>(null);

useEffect(() => {
const canvas = canvasRef.current;
if (canvas == null) return;
const ctx = canvas.getContext('2d');
if (ctx == null) return;
let animationFrameId: number;

// Set canvas size
canvas.width = window.innerWidth;
canvas.height = window.innerHeight;

// Create stars
const stars: { x: number; y: number; size: number; speed: number; }[] = [];
for (let i = 0; i < 200; i++) {
stars.push({
x: Math.random() * canvas.width,
y: Math.random() * canvas.height,
size: Math.random() * 2.5,
speed: Math.random() * 0.5 + 0.2,
});
}

// Animation function
const animate = () => {
ctx.clearRect(0, 0, canvas.width, canvas.height);
ctx.fillStyle = 'gray';

stars.forEach((star) => {
ctx.beginPath();
ctx.arc(star.x, star.y, star.size, 0, 2 * Math.PI);
ctx.fill();

// Move star
star.y += star.speed;

// Reset star position if it goes off screen
if (star.y > canvas.height) {
star.y = 0;
star.x = Math.random() * canvas.width;
}
});

animationFrameId = requestAnimationFrame(animate);
};

animate();

// Clean up
return () => {
cancelAnimationFrame(animationFrameId);
};
}, []);

return <canvas ref={canvasRef} className="absolute top-0 left-0 w-full h-full" />;
};
60 changes: 50 additions & 10 deletions frontend/gostarkme-web/app/page.tsx
Original file line number Diff line number Diff line change
@@ -1,29 +1,31 @@
'use client'
import { LinkButton } from "@/components/ui/LinkButton";
import { WelcomeBar } from "@/components/welcomepage/WelcomeBar";
import { WelcomeItems } from "@/components/welcomepage/WelcomeItems";
import Image from "next/image";
import dotenv from 'dotenv';
import { StardustAnimation } from "@/animations/StardustAnimation";

export default function Home() {
dotenv.config();
const ROOT = process.env.ROOT;
const ROOT = process.env.NEXT_PUBLIC_APP_ROOT;

return (
<main className="flex min-h-screen w-full flex-col items-center">
<WelcomeBar />
<section className="w-full max-w-screen-2xl grid grid-cols-1 md:grid-cols-2 p-10">
<div className="justify-self-center flex flex-col justify-center items-center md:items-start gap-4 p-4">
<h1 className="text-4xl font-bold">Upload your cause</h1>
<WelcomeItems text="Give it a name." src={ROOT+"/icons/user.png"} />
<WelcomeItems text="Give a good purpose." src={ROOT+"/icons/target.png"} />
<WelcomeItems text="Recollect Stars." src={ROOT+"/icons/star.png"} />
<WelcomeItems text="Receive donations." src={ROOT+"/icons/starklogo.png"} />
<WelcomeItems text="Give it a name." src={ROOT + "icons/user.png"} />
<WelcomeItems text="Give a good purpose." src={ROOT + "icons/target.png"} />
<WelcomeItems text="Recollect Stars." src={ROOT + "icons/star.png"} />
<WelcomeItems text="Receive donations." src={ROOT + "icons/starklogo.png"} />
</div>

<StardustAnimation />
<Image
src={ROOT +"/images/starcard.png"}
src={ROOT + "images/starcard.png"}
alt="stark logo"
height={771}
width={450}
className="self-center justify-self-center w-2/3 max-w-80"
className="self-center justify-self-center w-2/3 max-w-80 rounded-2xl"
/>
</section>

Expand Down Expand Up @@ -62,6 +64,44 @@ export default function Home() {
/>
</div>
</section>
<section className="my-16 px-4 mx-28">
<h2 className="text-3xl font-bold text-center mb-12">Our Process</h2>

<div className="flex flex-col md:flex-row justify-between items-start gap-8">
{/* Stage 1: Application - Represents submitting a proposal */}
<div className="flex-1 text-center">
<div className="text-5xl mb-4">&#x1F4DD;</div>
<h3 className="text-xl font-semibold mb-2">1. Application</h3>
<p className="text-gray-600 mb-4">Submit your project or cause for voting</p>
<div className="text-sm text-gray-500">
Upload your project or cause with lots of details for people to start noticing your cause, be as detailed as possible, this will
bring you better chances of getting more votes and funds.
</div>
</div>

{/* Stage 2: Verification - Represents the review process */}
<div className="flex-1 text-center">
<div className="text-5xl mb-4">&#x1F50D;</div>
<h3 className="text-xl font-semibold mb-2">2. Verification</h3>
<p className="text-gray-600 mb-4">The community votes if your cause is worthy of funds</p>
<div className="text-sm text-gray-500">
To ensure the project is genuine and worthy of funds, we will verify the project and its purpose, we believe in decentralization
and transparency and that is why the community is the one who decides which projects are worthy of funds.
</div>
</div>

{/* Stage 3: Funding - Represents successful funding */}
<div className="flex-1 text-center">
<div className="text-5xl mb-4">&#x1F4B0;</div>
<h3 className="text-xl font-semibold mb-2">3. Funding</h3>
<p className="text-gray-600 mb-4">Approved projects receive funding and support</p>
<div className="text-sm text-gray-500">
Successful applicants are notified and can start receiving funds. We also provide ongoing support and resources to help ensure
project success. Once you reach your goal, you can withdraw your funds, we will deduct 5% of the funds for the platform fee.
</div>
</div>
</div>
</section>
</main>
);
}
Loading

0 comments on commit 1e15922

Please sign in to comment.