From 03e6430641fb1e85598f7380f7d29b2793ea7dcb Mon Sep 17 00:00:00 2001 From: Gauri Rajesh Date: Fri, 28 Nov 2025 21:32:58 -0500 Subject: [PATCH 1/2] growing goal component new branch --- .../GrowingGoal/GrowingGoal.module.css | 158 +++++++++++++ .../components/GrowingGoal/GrowingGoal.tsx | 145 ++++++++++++ .../src/components/GrowingGoal/Plant.tsx | 223 ++++++++++++++++++ 3 files changed, 526 insertions(+) create mode 100644 apps/frontend/src/components/GrowingGoal/GrowingGoal.module.css create mode 100644 apps/frontend/src/components/GrowingGoal/GrowingGoal.tsx create mode 100644 apps/frontend/src/components/GrowingGoal/Plant.tsx diff --git a/apps/frontend/src/components/GrowingGoal/GrowingGoal.module.css b/apps/frontend/src/components/GrowingGoal/GrowingGoal.module.css new file mode 100644 index 0000000..f7598b3 --- /dev/null +++ b/apps/frontend/src/components/GrowingGoal/GrowingGoal.module.css @@ -0,0 +1,158 @@ +.goal-container { + aspect-ratio: 7 / 10; + overflow: hidden; + display: flex; + flex-direction: column; + border-radius: 3%; + border: 2% solid #cecece; + background: #f2f2f2; + text-align: center; + justify-content: start; + container-type: inline-size; + gap: 4%; +} + +.description-label { + border-top-left-radius: 5%; + border-top-right-radius: 5%; + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + flex-shrink: 0; + color: #fff; + font-family: Helvetica; + font-weight: 700; + white-space: nowrap; + height: 12%; + padding: 2%; + background-color: #127a64; + font-size: 5cqw; +} + +.growth-container { + display: flex; + align-items: center; + justify-content: center; + flex-shrink: 0; + position: relative; +} + +.growth-container-solid-grey { + position: relative; + width: 73%; + aspect-ratio: 1 / 1; + border-radius: 50%; + display: flex; + justify-content: center; + align-items: center; + background: #cecece; +} + +.total-donation-label { + color: #000; + text-align: center; + font-family: Helvetica; + font-size: 5cqw; + font-weight: 400; +} + +.sample-donor-container { + overflow: hidden; + display: flex; + align-items: center; + justify-content: center; +} + +.sample-donor-label { + width: 65%; + aspect-ratio: 9 / 2; + overflow: hidden; + border-radius: 4px; + padding: 2%; + background: #127a64; + display: flex; + align-items: center; + justify-content: center; +} + +.sample-donor-amount { + color: #fff; + font-family: Helvetica; + font-size: 4cqw; + overflow: hidden; +} + +.growth-container-solid-white { + z-index: 10; + display: flex; + justify-content: center; + align-items: center; + width: 90%; + aspect-ratio: 1 / 1; + border-radius: 50%; + background: #ffffff; +} + +.growth-container-solid-grey-inner { + z-index: 11; + display: flex; + justify-content: center; + align-items: center; + width: 90%; + aspect-ratio: 1 / 1; + border-radius: 50%; + background: #55565a; +} + +.growth-container-solid-teal { + position: relative; + aspect-ratio: 1 / 1; + width: 90%; + border-radius: 50%; + background: #0c7962; +} + +.growth-container-gradient { + position: absolute; + display: flex; + justify-content: center; + align-items: center; + width: 100%; + height: 100%; + border-radius: 50%; + background: conic-gradient( + from 180deg at 50% 50%, + #c6be3b 0deg, + #863b27 90deg, + #650d77 180deg, + #0c7962 270deg, + #c6be3b 360deg, + transparent 180deg 360deg + ); +} + +.soil-base { + background-color: #211504; + width: 100%; + aspect-ratio: 1 / 1; + border-radius: 50%; +} + +.soil-top { + position: absolute; + background-color: #332308; + width: 90%; + left: 5%; + height: 18%; + border-radius: 50%; + bottom: 20%; +} + +.progress-bar-handle { + position: absolute; + width: 5%; + aspect-ratio: 1 / 1; + border-radius: 50%; + background-color: #ffffff; +} diff --git a/apps/frontend/src/components/GrowingGoal/GrowingGoal.tsx b/apps/frontend/src/components/GrowingGoal/GrowingGoal.tsx new file mode 100644 index 0000000..608c59e --- /dev/null +++ b/apps/frontend/src/components/GrowingGoal/GrowingGoal.tsx @@ -0,0 +1,145 @@ +import { useEffect, useRef, useState } from 'react'; +import styles from './GrowingGoal.module.css'; +import Plant from './Plant'; + +export type SampleDonation = { + name: string; + amount: number; +}; + +export type GrowingGoalProps = { + message: string; + total: number; + goal: number; + sampleDonation?: SampleDonation; +}; + +export const GrowingGoal = (props: GrowingGoalProps) => { + const { message, total, goal } = props; + const growthContainerRef = useRef(null); + const [radius, setRadius] = useState(0); + const [endHandle, setEndHandle] = useState({ + top: 0, + left: 0, + }); + const progress = Math.floor((total / goal) * 360); + + // calculate gradient color of growth container handles + const getGradientColor = (degree: number): string => { + const gradientAngle = (degree + 180) % 360; + + const stops = [ + { angle: 0, color: { r: 198, g: 190, b: 59 } }, + { angle: 90, color: { r: 134, g: 59, b: 39 } }, + { angle: 180, color: { r: 101, g: 13, b: 119 } }, + { angle: 270, color: { r: 12, g: 121, b: 98 } }, + { angle: 360, color: { r: 198, g: 190, b: 59 } }, + ]; + + let startStop = stops[stops.length - 1]; + let endStop = stops[0]; + + for (let i = 0; i < stops.length - 1; i++) { + if ( + gradientAngle >= stops[i].angle && + gradientAngle <= stops[i + 1].angle + ) { + startStop = stops[i]; + endStop = stops[i + 1]; + break; + } + } + const range = endStop.angle - startStop.angle; + const factor = (gradientAngle - startStop.angle) / range; + const r = Math.round( + startStop.color.r + (endStop.color.r - startStop.color.r) * factor, + ); + const g = Math.round( + startStop.color.g + (endStop.color.g - startStop.color.g) * factor, + ); + const b = Math.round( + startStop.color.b + (endStop.color.b - startStop.color.b) * factor, + ); + + return `rgb(${r}, ${g}, ${b})`; + }; + + // calculate growth container end handle position + useEffect(() => { + if (growthContainerRef.current) { + setRadius(growthContainerRef.current.offsetWidth / 2); + const handleSize = radius * 0.05; + const progressRadians = ((90 - progress) * Math.PI) / 180; + const topOffset = (radius - handleSize) * Math.sin(progressRadians); + const leftOffset = (radius - handleSize) * Math.cos(progressRadians); + + const top = radius - topOffset - handleSize; + const left = radius + leftOffset - handleSize; + + setEndHandle({ + top: top, + left: left, + }); + } + }, [growthContainerRef, progress, radius]); + + const startHandleStyle: React.CSSProperties = { + top: '0%', + left: `47.5%`, + }; + + const endHandleStyle: React.CSSProperties = { + top: `${endHandle.top}px`, + left: `${endHandle.left}px`, + }; + + return ( +
+
{message}
+
+
+
+ +
+
+
+
+
+ ${total.toFixed(0)} raised of{' '} + ${goal.toFixed(0)} +
+
+ {props.sampleDonation && ( +
+
+ + {props.sampleDonation.name.length > 10 + ? props.sampleDonation.name.slice(0, 10) + '...' + : props.sampleDonation.name} + + {' donated $'} + {props.sampleDonation.amount.toFixed(2)}! +
+
+ )} +
+
+ ); +}; \ No newline at end of file diff --git a/apps/frontend/src/components/GrowingGoal/Plant.tsx b/apps/frontend/src/components/GrowingGoal/Plant.tsx new file mode 100644 index 0000000..29ee89b --- /dev/null +++ b/apps/frontend/src/components/GrowingGoal/Plant.tsx @@ -0,0 +1,223 @@ +import styles from './GrowingGoal.module.css'; +import React from 'react'; + +const Plant: React.FC = () => { + return ( +
+
+
+
+
+
+
+ + +
+
+
+ ); +}; + +const Shine: React.FC = () => { + return ( +
+ + + + + + + + + + + +
+ ); +}; + +const Sprout: React.FC = () => { + return ( +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ ); +}; + +export default Plant; \ No newline at end of file From 239b5ebe347c73252b867fd67b3e864f2c00def8 Mon Sep 17 00:00:00 2001 From: Gauri Rajesh Date: Tue, 2 Dec 2025 16:39:29 -0500 Subject: [PATCH 2/2] fix design to match hifi --- .../GrowingGoal/GrowingGoal.module.css | 28 ++++++++++++---- .../components/GrowingGoal/GrowingGoal.tsx | 22 ++++++++++--- apps/frontend/src/containers/root.tsx | 32 +++++++++++++------ 3 files changed, 60 insertions(+), 22 deletions(-) diff --git a/apps/frontend/src/components/GrowingGoal/GrowingGoal.module.css b/apps/frontend/src/components/GrowingGoal/GrowingGoal.module.css index f7598b3..78db2ab 100644 --- a/apps/frontend/src/components/GrowingGoal/GrowingGoal.module.css +++ b/apps/frontend/src/components/GrowingGoal/GrowingGoal.module.css @@ -26,7 +26,7 @@ white-space: nowrap; height: 12%; padding: 2%; - background-color: #127a64; + background-color: #3d3e6e; font-size: 5cqw; } @@ -40,7 +40,7 @@ .growth-container-solid-grey { position: relative; - width: 73%; + width: 70%; aspect-ratio: 1 / 1; border-radius: 50%; display: flex; @@ -53,27 +53,41 @@ color: #000; text-align: center; font-family: Helvetica; - font-size: 5cqw; + font-size: 6cqw; font-weight: 400; } .sample-donor-container { + width: 100%; + aspect-ratio: 5 / 1; overflow: hidden; display: flex; align-items: center; justify-content: center; + align-self: center; +} + +.sample-donor-profile { + display: flex; + width: 12%; + border-radius: 50%; + aspect-ratio: 1 / 1; + justify-content: center; + align-items: center; + background: #d9d9d9; } .sample-donor-label { - width: 65%; - aspect-ratio: 9 / 2; + width: 70%; overflow: hidden; border-radius: 4px; padding: 2%; - background: #127a64; + background: #3d3e6e; display: flex; align-items: center; - justify-content: center; + justify-content: start; + gap: 6%; + padding: 4%; } .sample-donor-amount { diff --git a/apps/frontend/src/components/GrowingGoal/GrowingGoal.tsx b/apps/frontend/src/components/GrowingGoal/GrowingGoal.tsx index 608c59e..b3156d4 100644 --- a/apps/frontend/src/components/GrowingGoal/GrowingGoal.tsx +++ b/apps/frontend/src/components/GrowingGoal/GrowingGoal.tsx @@ -5,6 +5,7 @@ import Plant from './Plant'; export type SampleDonation = { name: string; amount: number; + profile?: string; //base64 string }; export type GrowingGoalProps = { @@ -122,16 +123,27 @@ export const GrowingGoal = (props: GrowingGoalProps) => {
- ${total.toFixed(0)} raised of{' '} - ${goal.toFixed(0)} + ${total.toFixed(0)}{' '} + raised of ${goal.toFixed(0)}
{props.sampleDonation && (
+ {props.sampleDonation.profile ? ( +
+ ) : ( +
+ )}
- {props.sampleDonation.name.length > 10 - ? props.sampleDonation.name.slice(0, 10) + '...' + {props.sampleDonation.name.length > 8 + ? props.sampleDonation.name.slice(0,8) + '...' : props.sampleDonation.name} {' donated $'} @@ -142,4 +154,4 @@ export const GrowingGoal = (props: GrowingGoalProps) => {
); -}; \ No newline at end of file +}; diff --git a/apps/frontend/src/containers/root.tsx b/apps/frontend/src/containers/root.tsx index 1f62113..70845d3 100644 --- a/apps/frontend/src/containers/root.tsx +++ b/apps/frontend/src/containers/root.tsx @@ -1,18 +1,30 @@ import { DonationSummary } from '@containers/donations/DonationSummary'; import { DonationForm } from './donations/DonationForm'; +import { GrowingGoal } from '@components/GrowingGoal/GrowingGoal'; const Root: React.FC = () => { + const base64Profile = + ''; + return ( - <> - - +
+
+ +
+
+ +
+
); };