2022-05-10 06:52:06 -07:00
|
|
|
import { h } from 'preact';
|
|
|
|
import { useCallback, useEffect, useState } from 'preact/hooks';
|
|
|
|
import cs from 'classnames';
|
|
|
|
import Disk2 from '../cards/disk2';
|
|
|
|
import { FileModal } from './FileModal';
|
|
|
|
import { loadJSON, loadHttpFile, getHashParts } from './util/files';
|
2022-05-31 17:41:24 -07:00
|
|
|
import { ErrorModal } from './ErrorModal';
|
|
|
|
import { useHash } from './hooks/useHash';
|
2022-05-10 06:52:06 -07:00
|
|
|
|
2022-06-03 15:30:39 -07:00
|
|
|
import styles from './css/DiskII.module.css';
|
|
|
|
|
2022-05-10 06:52:06 -07:00
|
|
|
/**
|
|
|
|
* Storage structure for Disk II state returned via callbacks.
|
|
|
|
*/
|
|
|
|
export interface DiskIIData {
|
|
|
|
number: 1 | 2;
|
|
|
|
on: boolean;
|
2022-05-11 17:20:49 -07:00
|
|
|
name: string;
|
|
|
|
side?: string | undefined;
|
2022-05-10 06:52:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Interface for Disk II component.
|
|
|
|
*/
|
|
|
|
export interface DiskIIProps extends DiskIIData {
|
|
|
|
disk2: Disk2 | undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Disk II component
|
|
|
|
*
|
|
|
|
* Includes drive light, disk name and side, and UI for loading disks.
|
|
|
|
* Handles initial loading of disks specified in the hash.
|
|
|
|
*
|
|
|
|
* @param disk2 Disk2 object
|
|
|
|
* @param number Drive 1 or 2
|
|
|
|
* @param on Active state
|
|
|
|
* @param name Disk name identifier
|
|
|
|
* @param side Disk side identifier
|
|
|
|
* @returns DiskII component
|
|
|
|
*/
|
|
|
|
export const DiskII = ({ disk2, number, on, name, side }: DiskIIProps) => {
|
|
|
|
const label = side ? `${name} - ${side}` : name;
|
|
|
|
const [modalOpen, setModalOpen] = useState(false);
|
2022-06-01 06:28:05 -07:00
|
|
|
const [error, setError] = useState<unknown>();
|
2022-05-31 17:41:24 -07:00
|
|
|
const [currentHash, setCurrentHash] = useState<string>();
|
|
|
|
|
|
|
|
const hash = useHash();
|
|
|
|
|
2022-05-10 06:52:06 -07:00
|
|
|
useEffect(() => {
|
2022-05-31 17:41:24 -07:00
|
|
|
const hashParts = getHashParts(hash);
|
|
|
|
const newHash = hashParts[number];
|
|
|
|
if (disk2 && newHash) {
|
|
|
|
const hashPart = decodeURIComponent(newHash);
|
|
|
|
if (hashPart !== currentHash) {
|
|
|
|
if (hashPart.match(/^https?:/)) {
|
|
|
|
loadHttpFile(disk2, number, hashPart)
|
2022-06-01 06:28:05 -07:00
|
|
|
.catch((e) => setError(e));
|
2022-05-31 17:41:24 -07:00
|
|
|
} else {
|
|
|
|
const filename = `/json/disks/${hashPart}.json`;
|
|
|
|
loadJSON(disk2, number, filename)
|
2022-06-01 06:28:05 -07:00
|
|
|
.catch((e) => setError(e));
|
2022-05-31 17:41:24 -07:00
|
|
|
}
|
|
|
|
setCurrentHash(hashPart);
|
2022-05-10 06:52:06 -07:00
|
|
|
}
|
|
|
|
}
|
2022-05-31 17:41:24 -07:00
|
|
|
}, [currentHash, disk2, hash, number]);
|
2022-05-10 06:52:06 -07:00
|
|
|
|
|
|
|
const doClose = useCallback(() => {
|
|
|
|
setModalOpen(false);
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
const onOpenModal = useCallback(() => {
|
|
|
|
setModalOpen(true);
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return (
|
2022-06-03 15:30:39 -07:00
|
|
|
<div className={styles.disk}>
|
2022-05-10 06:52:06 -07:00
|
|
|
<FileModal disk2={disk2} number={number} onClose={doClose} isOpen={modalOpen} />
|
2022-05-31 17:41:24 -07:00
|
|
|
<ErrorModal error={error} setError={setError} />
|
2022-06-03 15:30:39 -07:00
|
|
|
<div className={cs(styles.diskLight, { on })} />
|
2022-05-10 06:52:06 -07:00
|
|
|
<button title="Load Disk">
|
2022-05-29 13:48:51 -07:00
|
|
|
<i className="fas fa-folder-open" onClick={onOpenModal} />
|
2022-05-10 06:52:06 -07:00
|
|
|
</button>
|
2022-06-03 15:30:39 -07:00
|
|
|
<div className={styles.diskLabel}>
|
2022-05-10 06:52:06 -07:00
|
|
|
{label}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|