Initial commit with 🏗️ create-eth @ 2.0.4

This commit is contained in:
han
2026-01-23 20:20:58 +07:00
commit b330aba2b4
185 changed files with 36981 additions and 0 deletions

View File

@@ -0,0 +1,152 @@
import { useEffect, useRef, useState } from "react";
import { HighlightedCell } from "./HighlightedCell";
import { parseEther } from "viem";
import { useWriteContract } from "wagmi";
import { ArrowPathIcon, PencilIcon } from "@heroicons/react/24/outline";
import { SIMPLE_ORACLE_ABI } from "~~/utils/constants";
import { notification } from "~~/utils/scaffold-eth";
type EditableCellProps = {
value: string | number;
address: string;
highlightColor?: string;
};
export const EditableCell = ({ value, address, highlightColor = "" }: EditableCellProps) => {
const [isEditing, setIsEditing] = useState(false);
const [isRefreshing, setIsRefreshing] = useState(false);
const [editValue, setEditValue] = useState(Number(value.toString()) || "");
const inputRef = useRef<HTMLInputElement>(null);
const { writeContractAsync } = useWriteContract();
// Update edit value when prop value changes
useEffect(() => {
if (!isEditing) {
setEditValue(Number(value.toString()) || "");
}
}, [value, isEditing]);
// Focus input when editing starts
useEffect(() => {
if (isEditing && inputRef.current) {
inputRef.current.focus();
inputRef.current.select();
}
}, [isEditing]);
const handleSubmit = async () => {
const parsedValue = Number(editValue);
if (isNaN(parsedValue)) {
notification.error("Invalid number");
return;
}
try {
await writeContractAsync({
abi: SIMPLE_ORACLE_ABI,
address: address,
functionName: "setPrice",
args: [parseEther(parsedValue.toString())],
});
setIsEditing(false);
} catch (error) {
console.error("Submit failed:", error);
}
};
// Resubmits the currently displayed value without entering edit mode
const handleRefresh = async () => {
const parsedValue = Number(value.toString());
if (isNaN(parsedValue)) {
notification.error("Invalid number");
return;
}
try {
await writeContractAsync({
abi: SIMPLE_ORACLE_ABI,
address: address,
functionName: "setPrice",
args: [parseEther(parsedValue.toString())],
});
} catch (error) {
console.error("Refresh failed:", error);
}
};
const handleCancel = () => {
setIsEditing(false);
};
const startEditing = () => {
setIsEditing(true);
};
return (
<HighlightedCell
value={value}
highlightColor={highlightColor}
className={`w-[6rem] max-w-[6rem] whitespace-nowrap overflow-hidden`}
>
<div className="flex w-full items-start">
{/* 70% width for value display/editing */}
<div className="w-[70%]">
{isEditing ? (
<div className="relative px-1">
<input
ref={inputRef}
type={"text"}
value={editValue}
onChange={e => setEditValue(e.target.value)}
className="w-full text-sm bg-secondary rounded-md"
/>
</div>
) : (
<div className="flex items-center gap-2 h-full items-stretch">
<span className="truncate">{value}</span>
<div className="flex items-stretch gap-1">
<button
className="px-2 text-sm bg-primary rounded cursor-pointer"
onClick={startEditing}
title="Edit price"
>
<PencilIcon className="w-2.5 h-2.5" />
</button>
<button
className="px-2 text-sm bg-secondary rounded cursor-pointer disabled:opacity-50"
onClick={() => {
if (isRefreshing) return;
setIsRefreshing(true);
try {
void handleRefresh();
} catch {}
setTimeout(() => setIsRefreshing(false), 3000);
}}
disabled={isRefreshing}
title="Resubmit price"
>
<ArrowPathIcon className={`w-2.5 h-2.5 ${isRefreshing ? "animate-spin" : ""}`} />
</button>
</div>
</div>
)}
</div>
{/* 30% width for action buttons */}
<div className="w-[30%] items-stretch justify-start pl-2">
{isEditing && (
<div className="flex items-stretch gap-1 w-full h-full">
<button onClick={handleSubmit} className="px-2 text-sm bg-primary rounded cursor-pointer">
</button>
<button onClick={handleCancel} className="px-2 text-sm bg-secondary rounded cursor-pointer">
</button>
</div>
)}
</div>
</div>
</HighlightedCell>
);
};