2021-07-08 19:02:42 +00:00
|
|
|
import React, { useState, useContext } from "react";
|
2021-11-25 09:28:45 +00:00
|
|
|
import { NavLink, useNavigate } from "react-router-dom";
|
2021-08-08 22:49:45 +00:00
|
|
|
import { commify } from "@ethersproject/units";
|
2021-07-30 08:18:28 +00:00
|
|
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
2021-08-08 06:51:21 +00:00
|
|
|
import { faBurn } from "@fortawesome/free-solid-svg-icons/faBurn";
|
2021-09-01 20:40:42 +00:00
|
|
|
import { faQrcode } from "@fortawesome/free-solid-svg-icons/faQrcode";
|
2021-07-01 18:21:40 +00:00
|
|
|
import Logo from "./Logo";
|
|
|
|
import Timestamp from "./components/Timestamp";
|
2021-07-09 05:07:20 +00:00
|
|
|
import { RuntimeContext } from "./useRuntime";
|
2021-07-04 01:42:06 +00:00
|
|
|
import { useLatestBlock } from "./useLatestBlock";
|
2021-07-23 22:46:21 +00:00
|
|
|
import { blockURL } from "./url";
|
2021-11-25 09:28:45 +00:00
|
|
|
import { search } from "./search/search";
|
2021-07-01 18:21:40 +00:00
|
|
|
|
2021-10-21 07:10:49 +00:00
|
|
|
const CameraScanner = React.lazy(() => import("./search/CameraScanner"));
|
|
|
|
|
2021-07-01 18:21:40 +00:00
|
|
|
const Home: React.FC = () => {
|
2021-07-09 05:07:20 +00:00
|
|
|
const { provider } = useContext(RuntimeContext);
|
2021-11-25 09:28:45 +00:00
|
|
|
const [searchString, setSearchString] = useState<string>("");
|
2021-07-01 18:21:40 +00:00
|
|
|
const [canSubmit, setCanSubmit] = useState<boolean>(false);
|
2021-11-25 09:28:45 +00:00
|
|
|
const navigate = useNavigate();
|
2021-07-01 18:21:40 +00:00
|
|
|
|
|
|
|
const handleChange: React.ChangeEventHandler<HTMLInputElement> = (e) => {
|
2021-11-25 09:28:45 +00:00
|
|
|
const searchTerm = e.target.value.trim();
|
|
|
|
setCanSubmit(searchTerm.length > 0);
|
|
|
|
setSearchString(searchTerm);
|
2021-07-01 18:21:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const handleSubmit: React.FormEventHandler<HTMLFormElement> = (e) => {
|
|
|
|
e.preventDefault();
|
|
|
|
if (!canSubmit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-11-25 09:28:45 +00:00
|
|
|
search(searchString, navigate);
|
2021-07-01 18:21:40 +00:00
|
|
|
};
|
|
|
|
|
2021-07-08 19:02:42 +00:00
|
|
|
const latestBlock = useLatestBlock(provider);
|
2021-09-01 20:40:42 +00:00
|
|
|
const [isScanning, setScanning] = useState<boolean>(false);
|
2021-07-01 18:21:40 +00:00
|
|
|
|
|
|
|
document.title = "Home | Otterscan";
|
|
|
|
|
|
|
|
return (
|
2021-10-24 04:25:57 +00:00
|
|
|
<div className="mx-auto flex flex-col flex-grow pb-5">
|
2021-09-01 20:40:42 +00:00
|
|
|
{isScanning && <CameraScanner turnOffScan={() => setScanning(false)} />}
|
2021-10-24 04:25:57 +00:00
|
|
|
<div className="m-5 mb-10 flex items-end flex-grow max-h-64">
|
2021-10-22 12:29:54 +00:00
|
|
|
<Logo />
|
|
|
|
</div>
|
2021-07-10 06:17:07 +00:00
|
|
|
<form
|
|
|
|
className="flex flex-col"
|
|
|
|
onSubmit={handleSubmit}
|
|
|
|
autoComplete="off"
|
|
|
|
spellCheck={false}
|
|
|
|
>
|
2021-09-01 20:40:42 +00:00
|
|
|
<div className="flex mb-10">
|
|
|
|
<input
|
|
|
|
className="w-full border-l border-t border-b rounded-l focus:outline-none px-2 py-1"
|
|
|
|
type="text"
|
|
|
|
size={50}
|
|
|
|
placeholder="Search by address / txn hash / block number / ENS name"
|
|
|
|
onChange={handleChange}
|
|
|
|
autoFocus
|
|
|
|
/>
|
|
|
|
<button
|
2021-09-01 21:12:35 +00:00
|
|
|
className="border rounded-r bg-skin-button-fill hover:bg-skin-button-hover-fill focus:outline-none px-2 py-1 text-base text-skin-button flex justify-center items-center"
|
2021-09-01 20:40:42 +00:00
|
|
|
type="button"
|
|
|
|
onClick={() => setScanning(true)}
|
|
|
|
title="Scan an ETH address using your camera"
|
|
|
|
>
|
|
|
|
<FontAwesomeIcon icon={faQrcode} />
|
|
|
|
</button>
|
|
|
|
</div>
|
2021-07-10 06:17:07 +00:00
|
|
|
<button
|
2021-09-01 21:12:35 +00:00
|
|
|
className="mx-auto px-3 py-1 mb-10 rounded bg-skin-button-fill hover:bg-skin-button-hover-fill focus:outline-none"
|
2021-07-10 06:17:07 +00:00
|
|
|
type="submit"
|
2021-07-01 18:21:40 +00:00
|
|
|
>
|
2021-07-10 06:17:07 +00:00
|
|
|
Search
|
|
|
|
</button>
|
2021-10-22 12:29:54 +00:00
|
|
|
</form>
|
|
|
|
<div className="mx-auto h-32">
|
|
|
|
<div className="text-lg text-link-blue hover:text-link-blue-hover font-bold">
|
2021-07-29 04:24:39 +00:00
|
|
|
<NavLink to="/special/london">
|
2021-07-30 08:18:28 +00:00
|
|
|
<div className="flex space-x-2 items-baseline text-orange-500 hover:text-orange-700 hover:underline">
|
|
|
|
<span>
|
|
|
|
<FontAwesomeIcon icon={faBurn} />
|
|
|
|
</span>
|
2021-07-30 21:02:11 +00:00
|
|
|
<span>Check out the special dashboard for EIP-1559</span>
|
2021-07-30 08:18:28 +00:00
|
|
|
<span>
|
|
|
|
<FontAwesomeIcon icon={faBurn} />
|
|
|
|
</span>
|
|
|
|
</div>
|
2021-07-29 04:24:39 +00:00
|
|
|
</NavLink>
|
|
|
|
</div>
|
2021-07-10 06:17:07 +00:00
|
|
|
{latestBlock && (
|
|
|
|
<NavLink
|
2021-10-22 12:29:54 +00:00
|
|
|
className="flex flex-col items-center space-y-1 mt-5 text-sm text-gray-500 hover:text-link-blue"
|
2021-07-23 22:46:21 +00:00
|
|
|
to={blockURL(latestBlock.number)}
|
2021-07-01 18:21:40 +00:00
|
|
|
>
|
2021-08-08 22:49:45 +00:00
|
|
|
<div>Latest block: {commify(latestBlock.number)}</div>
|
2021-07-10 06:17:07 +00:00
|
|
|
<Timestamp value={latestBlock.timestamp} />
|
|
|
|
</NavLink>
|
|
|
|
)}
|
2021-10-22 12:29:54 +00:00
|
|
|
</div>
|
2021-07-01 18:21:40 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default React.memo(Home);
|