399 lines
14 KiB
TypeScript
399 lines
14 KiB
TypeScript
import Board from "features/boards/Board.vue";
|
|
import CircleProgress from "features/boards/CircleProgress.vue";
|
|
import SVGNode from "features/boards/SVGNode.vue";
|
|
import SquareProgress from "features/boards/SquareProgress.vue";
|
|
import {
|
|
NodePosition,
|
|
placeInAvailableSpace,
|
|
setupActions,
|
|
setupDraggableNode,
|
|
setupSelectable,
|
|
setupUniqueIds
|
|
} from "features/boards/board";
|
|
import { jsx } from "features/feature";
|
|
import type { BaseLayer, GenericLayer } from "game/layers";
|
|
import { createLayer } from "game/layers";
|
|
import { persistent } from "game/persistence";
|
|
import type { Player } from "game/player";
|
|
import { ComponentPublicInstance, computed, ref, watch } from "vue";
|
|
import prestige from "./layers/prestige";
|
|
|
|
type ANode = NodePosition & { id: number; links: number[]; type: "anode" };
|
|
type BNode = NodePosition & { id: number; links: number[]; type: "bnode" };
|
|
type NodeTypes = ANode | BNode;
|
|
|
|
/**
|
|
* @hidden
|
|
*/
|
|
export const main = createLayer("main", function (this: BaseLayer) {
|
|
const board = ref<ComponentPublicInstance<typeof Board>>();
|
|
|
|
const { select, deselect, selected } = setupSelectable<NodeTypes>();
|
|
const {
|
|
select: selectAction,
|
|
deselect: deselectAction,
|
|
selected: selectedAction
|
|
} = setupSelectable<number>();
|
|
|
|
watch(selected, selected => {
|
|
if (selected == null) {
|
|
deselectAction();
|
|
}
|
|
});
|
|
|
|
const {
|
|
startDrag,
|
|
endDrag,
|
|
drag,
|
|
nodeBeingDragged,
|
|
hasDragged,
|
|
receivingNodes,
|
|
receivingNode,
|
|
dragDelta
|
|
} = setupDraggableNode<NodeTypes /* | typeof cNode*/>({
|
|
board,
|
|
isDraggable: function (node) {
|
|
return nodes.value.includes(node);
|
|
}
|
|
});
|
|
|
|
// a nodes can be slotted into b nodes to draw a branch between them, with limited connections
|
|
// a nodes can be selected and have an action to spawn a b node, and vice versa
|
|
// Newly spawned nodes should find a safe spot to spawn, and display a link to their creator
|
|
// a nodes use all the stuff circles used to have, and b diamonds
|
|
// c node also exists but is a single Upgrade element that cannot be selected, but can be dragged
|
|
// d nodes are a performance test - 1000 simple nodes that have no interactions
|
|
// Make all nodes animate in (decorator? `fadeIn(feature)?)
|
|
const nodes = persistent<NodeTypes[]>([{ id: 0, x: 0, y: 0, links: [], type: "anode" }]);
|
|
const nodesById = computed<Record<string, NodeTypes>>(() =>
|
|
nodes.value.reduce((acc, curr) => ({ ...acc, [curr.id]: curr }), {})
|
|
);
|
|
function mouseDownNode(e: MouseEvent | TouchEvent, node: NodeTypes) {
|
|
if (nodeBeingDragged.value == null) {
|
|
startDrag(e, node);
|
|
}
|
|
deselect();
|
|
}
|
|
function mouseUpNode(e: MouseEvent | TouchEvent, node: NodeTypes) {
|
|
if (!hasDragged.value) {
|
|
endDrag();
|
|
select(node);
|
|
e.stopPropagation();
|
|
}
|
|
}
|
|
function getTranslateString(node: NodePosition, overrideSelected?: boolean) {
|
|
const isSelected = overrideSelected == null ? selected.value === node : overrideSelected;
|
|
const isDragging = !isSelected && nodeBeingDragged.value === node;
|
|
let x = node.x;
|
|
let y = node.y;
|
|
if (isDragging) {
|
|
x += dragDelta.value.x;
|
|
y += dragDelta.value.y;
|
|
}
|
|
return ` translate(${x}px,${y}px)`;
|
|
}
|
|
function getRotateString(rotation: number) {
|
|
return ` rotate(${rotation}deg) `;
|
|
}
|
|
function getScaleString(node: NodePosition, overrideSelected?: boolean) {
|
|
const isSelected = overrideSelected == null ? selected.value === node : overrideSelected;
|
|
return isSelected ? " scale(1.2)" : "";
|
|
}
|
|
function getOpacityString(node: NodePosition, overrideSelected?: boolean) {
|
|
const isSelected = overrideSelected == null ? selected.value === node : overrideSelected;
|
|
const isDragging = !isSelected && nodeBeingDragged.value === node;
|
|
if (isDragging) {
|
|
return "; opacity: 0.5;";
|
|
}
|
|
return "";
|
|
}
|
|
|
|
const renderANode = function (node: ANode) {
|
|
return (
|
|
<SVGNode
|
|
style={`transform: ${getTranslateString(node)}${getOpacityString(node)}`}
|
|
onMouseDown={e => mouseDownNode(e, node)}
|
|
onMouseUp={e => mouseUpNode(e, node)}
|
|
>
|
|
<g style={`transform: ${getScaleString(node)}; transition-duration: 0s`}>
|
|
{receivingNodes.value.includes(node) && (
|
|
<circle
|
|
r="58"
|
|
fill="var(--background)"
|
|
stroke={receivingNode.value === node ? "#0F0" : "#0F03"}
|
|
stroke-width="2"
|
|
/>
|
|
)}
|
|
<CircleProgress r={54.5} progress={0.5} stroke="var(--accent2)" />
|
|
<circle
|
|
r="50"
|
|
fill="var(--raised-background)"
|
|
stroke="var(--outline)"
|
|
stroke-width="4"
|
|
/>
|
|
</g>
|
|
{selected.value === node && selectedAction.value === 0 && (
|
|
<text y="140" fill="var(--foreground)" class="node-text">
|
|
Spawn B Node
|
|
</text>
|
|
)}
|
|
<text fill="var(--foreground)" class="node-text">
|
|
A
|
|
</text>
|
|
</SVGNode>
|
|
);
|
|
};
|
|
const aActions = setupActions({
|
|
node: selected,
|
|
shouldShowActions: () => selected.value?.type === "anode",
|
|
actions(node) {
|
|
return [
|
|
p => (
|
|
<g
|
|
style={`transform: ${getTranslateString(
|
|
p,
|
|
selectedAction.value === 0
|
|
)}${getScaleString(p, selectedAction.value === 0)}`}
|
|
onClick={() => {
|
|
if (selectedAction.value === 0) {
|
|
spawnBNode(node);
|
|
} else {
|
|
selectAction(0);
|
|
}
|
|
}}
|
|
>
|
|
<circle fill="black" r="20"></circle>
|
|
<text fill="white" class="material-icons" x="-12" y="12">
|
|
add
|
|
</text>
|
|
</g>
|
|
)
|
|
];
|
|
},
|
|
distance: 100
|
|
});
|
|
const sqrtTwo = Math.sqrt(2);
|
|
const renderBNode = function (node: BNode) {
|
|
return (
|
|
<SVGNode
|
|
style={`transform: ${getTranslateString(node)}${getOpacityString(node)}`}
|
|
onMouseDown={e => mouseDownNode(e, node)}
|
|
onMouseUp={e => mouseUpNode(e, node)}
|
|
>
|
|
<g
|
|
style={`transform: ${getScaleString(node)}${getRotateString(
|
|
45
|
|
)}; transition-duration: 0s`}
|
|
>
|
|
{receivingNodes.value.includes(node) && (
|
|
<rect
|
|
width={50 * sqrtTwo + 16}
|
|
height={50 * sqrtTwo + 16}
|
|
style={`translate(${(-50 * sqrtTwo + 16) / 2}, ${
|
|
(-50 * sqrtTwo + 16) / 2
|
|
})`}
|
|
fill="var(--background)"
|
|
stroke={receivingNode.value === node ? "#0F0" : "#0F03"}
|
|
stroke-width="2"
|
|
/>
|
|
)}
|
|
<SquareProgress
|
|
size={50 * sqrtTwo + 9}
|
|
progress={0.5}
|
|
stroke="var(--accent2)"
|
|
/>
|
|
<rect
|
|
width={50 * sqrtTwo}
|
|
height={50 * sqrtTwo}
|
|
style={`transform: translate(${(-50 * sqrtTwo) / 2}px, ${
|
|
(-50 * sqrtTwo) / 2
|
|
}px)`}
|
|
fill="var(--raised-background)"
|
|
stroke="var(--outline)"
|
|
stroke-width="4"
|
|
/>
|
|
</g>
|
|
{selected.value === node && selectedAction.value === 0 && (
|
|
<text y="140" fill="var(--foreground)" class="node-text">
|
|
Spawn A Node
|
|
</text>
|
|
)}
|
|
<text fill="var(--foreground)" class="node-text">
|
|
B
|
|
</text>
|
|
</SVGNode>
|
|
);
|
|
};
|
|
const bActions = setupActions({
|
|
node: selected,
|
|
shouldShowActions: () => selected.value?.type === "bnode",
|
|
actions(node) {
|
|
return [
|
|
p => (
|
|
<g
|
|
style={`transform: ${getTranslateString(
|
|
p,
|
|
selectedAction.value === 0
|
|
)}${getScaleString(p, selectedAction.value === 0)}`}
|
|
onClick={() => {
|
|
if (selectedAction.value === 0) {
|
|
spawnANode(node);
|
|
} else {
|
|
selectAction(0);
|
|
}
|
|
}}
|
|
>
|
|
<circle fill="white" r="20"></circle>
|
|
<text fill="black" class="material-icons" x="-12" y="12">
|
|
add
|
|
</text>
|
|
</g>
|
|
)
|
|
];
|
|
},
|
|
distance: 100
|
|
});
|
|
function spawnANode(parent: NodeTypes) {
|
|
const node: ANode = {
|
|
x: parent.x,
|
|
y: parent.y,
|
|
type: "anode",
|
|
links: [parent.id],
|
|
id: nextId.value
|
|
};
|
|
placeInAvailableSpace(node, nodes.value);
|
|
nodes.value.push(node);
|
|
}
|
|
function spawnBNode(parent: NodeTypes) {
|
|
const node: BNode = {
|
|
x: parent.x,
|
|
y: parent.y,
|
|
type: "bnode",
|
|
links: [parent.id],
|
|
id: nextId.value
|
|
};
|
|
placeInAvailableSpace(node, nodes.value);
|
|
nodes.value.push(node);
|
|
}
|
|
|
|
// const cNode = createUpgrade(() => ({
|
|
// requirements: createCostRequirement(() => ({ cost: 10, resource: points })),
|
|
// style: {
|
|
// x: "100px",
|
|
// y: "100px"
|
|
// }
|
|
// }));
|
|
// makeDraggable(cNode); // TODO make decorator
|
|
|
|
// const dNodes;
|
|
|
|
const links = jsx(() => (
|
|
<>
|
|
{nodes.value
|
|
.reduce(
|
|
(acc, curr) => [
|
|
...acc,
|
|
...curr.links.map(l => ({ from: curr, to: nodesById.value[l] }))
|
|
],
|
|
[] as { from: NodeTypes; to: NodeTypes }[]
|
|
)
|
|
.map(link => (
|
|
<line
|
|
stroke="white"
|
|
stroke-width={4}
|
|
x1={
|
|
nodeBeingDragged.value === link.from
|
|
? dragDelta.value.x + link.from.x
|
|
: link.from.x
|
|
}
|
|
y1={
|
|
nodeBeingDragged.value === link.from
|
|
? dragDelta.value.y + link.from.y
|
|
: link.from.y
|
|
}
|
|
x2={
|
|
nodeBeingDragged.value === link.to
|
|
? dragDelta.value.x + link.to.x
|
|
: link.to.x
|
|
}
|
|
y2={
|
|
nodeBeingDragged.value === link.to
|
|
? dragDelta.value.y + link.to.y
|
|
: link.to.y
|
|
}
|
|
/>
|
|
))}
|
|
</>
|
|
));
|
|
|
|
const nextId = setupUniqueIds(() => nodes.value);
|
|
|
|
function filterNodes(n: NodeTypes) {
|
|
return n !== nodeBeingDragged.value && n !== selected.value;
|
|
}
|
|
|
|
function renderNode(node: NodeTypes | undefined) {
|
|
if (node == undefined) {
|
|
return undefined;
|
|
} else if (node.type === "anode") {
|
|
return renderANode(node);
|
|
} else if (node.type === "bnode") {
|
|
return renderBNode(node);
|
|
}
|
|
}
|
|
|
|
return {
|
|
name: "Tree",
|
|
display: jsx(() => (
|
|
<>
|
|
<Board
|
|
onDrag={drag}
|
|
onMouseDown={deselect}
|
|
onMouseUp={endDrag}
|
|
onMouseLeave={endDrag}
|
|
ref={board}
|
|
>
|
|
<SVGNode>{links()}</SVGNode>
|
|
{nodes.value.filter(filterNodes).map(renderNode)}
|
|
<SVGNode>
|
|
{aActions()}
|
|
{bActions()}
|
|
</SVGNode>
|
|
{renderNode(selected.value)}
|
|
{renderNode(nodeBeingDragged.value)}
|
|
</Board>
|
|
</>
|
|
)),
|
|
boardNodes: nodes
|
|
// cNode
|
|
};
|
|
});
|
|
|
|
/**
|
|
* Given a player save data object being loaded, return a list of layers that should currently be enabled.
|
|
* If your project does not use dynamic layers, this should just return all layers.
|
|
*/
|
|
export const getInitialLayers = (
|
|
/* eslint-disable-next-line @typescript-eslint/no-unused-vars */
|
|
player: Partial<Player>
|
|
): Array<GenericLayer> => [main, prestige];
|
|
|
|
/**
|
|
* A computed ref whose value is true whenever the game is over.
|
|
*/
|
|
export const hasWon = computed(() => {
|
|
return false;
|
|
});
|
|
|
|
/**
|
|
* Given a player save data object being loaded with a different version, update the save data object to match the structure of the current version.
|
|
* @param oldVersion The version of the save being loaded in
|
|
* @param player The save data being loaded in
|
|
*/
|
|
/* eslint-disable @typescript-eslint/no-unused-vars */
|
|
export function fixOldSave(
|
|
oldVersion: string | undefined,
|
|
player: Partial<Player>
|
|
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
|
): void {}
|
|
/* eslint-enable @typescript-eslint/no-unused-vars */
|