153 lines
4.7 KiB
TypeScript
153 lines
4.7 KiB
TypeScript
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>
|
|
);
|
|
};
|