1
0
mirror of https://github.com/tobspr/shapez.io.git synced 2025-06-13 13:04:03 +00:00

Allow for both text and shapes in a waypoint, fix minor waypoint bugs

Waypoints get an additional array property of the label broken up into
text and shape keys, obtained using a new function splitLabel().
When displaying, each item is rendered or appended in sequence using a for loop.

Because of the variable number of elements, waypoints in the list are now flexboxes,
and an extra element is added to align the edit icon to the right side.

When calculating the length of a label, the function getLabelLength() is used.
Each shape is treated as a set number of characters, using a new constant SHAPE_TEXT_LENGTH.

Other fixes/changes:

- getWaypointCanvas() uses a shape key instead of a waypoint.

- getWaypointScreenParams() returns an array of label parts,
which include the text and whether the part is a shape key.

- Text and shape width for marker boxes now correctly accounts for scale.
getWaypointScreenParams() now determines shape width using the scale,
and getTextWidth also keeps track of scale in its cache.

- Waypoint sorting no longer right-pads labels with 0's.
This commit is contained in:
EmeraldBlock 2020-10-25 20:51:57 -05:00
parent 0146aa91bb
commit 331e7bad0f
2 changed files with 155 additions and 72 deletions

View File

@ -41,6 +41,7 @@
@include S(padding-right, 5px); @include S(padding-right, 5px);
@include S(padding-bottom, 5px); @include S(padding-bottom, 5px);
@include S(padding-top, 5px); @include S(padding-top, 5px);
@include S(padding-left, 5px);
// Scrollbar // Scrollbar
&::-webkit-scrollbar { &::-webkit-scrollbar {
@ -54,8 +55,7 @@
cursor: pointer; cursor: pointer;
color: #333438; color: #333438;
@include S(padding-left, 11px); @include S(padding-left, 11px);
display: grid; display: flex;
grid-template-columns: 1fr auto;
align-items: center; align-items: center;
& { & {
/* @load-async */ /* @load-async */
@ -64,11 +64,16 @@
opacity: 0.7; opacity: 0.7;
@include S(margin-bottom, 1px); @include S(margin-bottom, 1px);
font-weight: bold; font-weight: bold;
white-space: pre-wrap;
&:hover { &:hover {
opacity: 0.8; opacity: 0.8;
} }
.editMargin {
margin: auto;
}
.editButton { .editButton {
@include S(width, 10px); @include S(width, 10px);
@include S(height, 10px); @include S(height, 10px);
@ -92,7 +97,6 @@
// Transform because there is a canvas before // Transform because there is a canvas before
@include S(margin-left, -2px); @include S(margin-left, -2px);
grid-template-columns: auto 1fr;
background: none !important; background: none !important;
@include S(padding-left, 0); @include S(padding-left, 0);
canvas { canvas {

View File

@ -25,14 +25,13 @@ import { enumNotificationType } from "./notifications";
/** @typedef {{ /** @typedef {{
* label: string | null, * label: string | null,
* parts: Array<string> | null,
* center: { x: number, y: number }, * center: { x: number, y: number },
* zoomLevel: number * zoomLevel: number
* }} Waypoint */ * }} Waypoint */
/** const MAX_LABEL_LENGTH = 70;
* Used when a shape icon is rendered instead const SHAPE_TEXT_LENGTH = 2;
*/
const MAX_LABEL_LENGTH = 71;
export class HUDWaypoints extends BaseHUDPart { export class HUDWaypoints extends BaseHUDPart {
/** /**
@ -96,6 +95,7 @@ export class HUDWaypoints extends BaseHUDPart {
this.waypoints = [ this.waypoints = [
{ {
label: null, label: null,
parts: null,
center: { x: 0, y: 0 }, center: { x: 0, y: 0 },
zoomLevel: 3, zoomLevel: 3,
}, },
@ -140,7 +140,7 @@ export class HUDWaypoints extends BaseHUDPart {
/** /**
* Store cached text widths * Store cached text widths
* @type {Object<string, number>} * @type {Object<string, Object<string, number>>}
*/ */
this.cachedTextWidths = {}; this.cachedTextWidths = {};
@ -154,12 +154,16 @@ export class HUDWaypoints extends BaseHUDPart {
* @returns {number} * @returns {number}
*/ */
getTextWidth(text) { getTextWidth(text) {
if (this.cachedTextWidths[text]) { const scale = this.getTextScale();
return this.cachedTextWidths[text]; if (!this.cachedTextWidths[scale]) {
this.cachedTextWidths[scale] = {};
}
if (this.cachedTextWidths[scale][text]) {
return this.cachedTextWidths[scale][text];
} }
this.dummyBuffer.font = "bold " + this.getTextScale() + "px GameFont"; this.dummyBuffer.font = "bold " + scale + "px GameFont";
return (this.cachedTextWidths[text] = this.dummyBuffer.measureText(text).width); return (this.cachedTextWidths[scale][text] = this.dummyBuffer.measureText(text).width);
} }
/** /**
@ -185,28 +189,32 @@ export class HUDWaypoints extends BaseHUDPart {
for (let i = 0; i < this.waypoints.length; ++i) { for (let i = 0; i < this.waypoints.length; ++i) {
const waypoint = this.waypoints[i]; const waypoint = this.waypoints[i];
const label = this.getWaypointLabel(waypoint); const parts = this.getWaypointParts(waypoint);
const element = makeDiv(this.waypointsListElement, null, ["waypoint"]); const element = makeDiv(this.waypointsListElement, null, ["waypoint"]);
if (ShapeDefinition.isValidShortKey(label)) { for (let j = 0; j < parts.length; ++j) {
const canvas = this.getWaypointCanvas(waypoint); const part = parts[j];
/** if (ShapeDefinition.isValidShortKey(part)) {
* Create a clone of the cached canvas, as calling appendElement when a canvas is const canvas = this.getWaypointCanvas(part);
* already in the document will move the existing canvas to the new position. /**
*/ * Create a clone of the cached canvas, as calling appendElement when a canvas is
const [newCanvas, context] = makeOffscreenBuffer(48, 48, { * already in the document will move the existing canvas to the new position.
smooth: true, */
label: label + "-waypoint-" + i, const [newCanvas, context] = makeOffscreenBuffer(48, 48, {
}); smooth: true,
context.drawImage(canvas, 0, 0); label: part + "-waypoint-" + i,
element.appendChild(newCanvas); });
element.classList.add("shapeIcon"); context.drawImage(canvas, 0, 0);
} else { element.appendChild(newCanvas);
element.innerText = label; element.classList.add("shapeIcon");
} else {
element.appendChild(document.createTextNode(part));
}
} }
if (this.isWaypointDeletable(waypoint)) { if (this.isWaypointDeletable(waypoint)) {
makeDiv(element, null, ["editMargin"]);
const editButton = makeDiv(element, null, ["editButton"]); const editButton = makeDiv(element, null, ["editButton"]);
this.trackClicks(editButton, () => this.requestSaveMarker({ waypoint })); this.trackClicks(editButton, () => this.requestSaveMarker({ waypoint }));
} }
@ -242,12 +250,11 @@ export class HUDWaypoints extends BaseHUDPart {
} }
/** /**
* Gets the canvas for a given waypoint * Gets the canvas for a given waypoint key
* @param {Waypoint} waypoint * @param {string} key
* @returns {HTMLCanvasElement} * @returns {HTMLCanvasElement}
*/ */
getWaypointCanvas(waypoint) { getWaypointCanvas(key) {
const key = waypoint.label;
if (this.cachedKeyToCanvas[key]) { if (this.cachedKeyToCanvas[key]) {
return this.cachedKeyToCanvas[key]; return this.cachedKeyToCanvas[key];
} }
@ -272,8 +279,7 @@ export class HUDWaypoints extends BaseHUDPart {
label: null, label: null,
placeholder: "", placeholder: "",
defaultValue: waypoint ? waypoint.label : "", defaultValue: waypoint ? waypoint.label : "",
validator: val => validator: val => val.length > 0 && this.getLabelLength(val) <= MAX_LABEL_LENGTH,
val.length > 0 && (val.length < MAX_LABEL_LENGTH || ShapeDefinition.isValidShortKey(val)),
}); });
const dialog = new DialogWithForm({ const dialog = new DialogWithForm({
app: this.root.app, app: this.root.app,
@ -322,8 +328,10 @@ export class HUDWaypoints extends BaseHUDPart {
* @param {Vector} position * @param {Vector} position
*/ */
addWaypoint(label, position) { addWaypoint(label, position) {
const parts = this.splitLabel(label);
this.waypoints.push({ this.waypoints.push({
label, label,
parts,
center: { x: position.x, y: position.y }, center: { x: position.x, y: position.y },
zoomLevel: this.root.camera.zoomLevel, zoomLevel: this.root.camera.zoomLevel,
}); });
@ -347,6 +355,7 @@ export class HUDWaypoints extends BaseHUDPart {
*/ */
renameWaypoint(waypoint, label) { renameWaypoint(waypoint, label) {
waypoint.label = label; waypoint.label = label;
waypoint.parts = this.splitLabel(waypoint.label);
this.sortWaypoints(); this.sortWaypoints();
@ -360,6 +369,57 @@ export class HUDWaypoints extends BaseHUDPart {
this.rerenderWaypointList(); this.rerenderWaypointList();
} }
/**
* Splits a label into shortkeys and text
* @param {string} label
* @returns {Array<string>}
*/
splitLabel(label) {
const words = label.split(" ");
let part = null;
let parts = [];
for (let i = 0; i < words.length; ++i) {
const word = words[i];
if (ShapeDefinition.isValidShortKey(word)) {
if (part !== null) {
parts.push(part);
part = null;
}
parts.push(word);
} else {
if (part !== null) {
part += " " + word;
} else {
part = word;
}
}
}
if (part !== null) {
parts.push(part);
}
return parts;
}
/**
* Returns the character length of a label,
* treating shapes as a constant number of characters
* @param {string} label
* @returns {number}
*/
getLabelLength(label) {
const parts = this.splitLabel(label);
let length = 0;
for (let i = 0; i < parts.length; ++i) {
const part = parts[i];
if (ShapeDefinition.isValidShortKey(part)) {
length += SHAPE_TEXT_LENGTH;
} else {
length += part.length;
}
}
return length;
}
/** /**
* Called every frame to update stuff * Called every frame to update stuff
*/ */
@ -380,9 +440,7 @@ export class HUDWaypoints extends BaseHUDPart {
if (!b.label) { if (!b.label) {
return 1; return 1;
} }
return this.getWaypointLabel(a) return this.getWaypointLabel(a).localeCompare(this.getWaypointLabel(b));
.padEnd(MAX_LABEL_LENGTH, "0")
.localeCompare(this.getWaypointLabel(b).padEnd(MAX_LABEL_LENGTH, "0"));
}); });
} }
@ -395,6 +453,15 @@ export class HUDWaypoints extends BaseHUDPart {
return waypoint.label || T.ingame.waypoints.hub; return waypoint.label || T.ingame.waypoints.hub;
} }
/**
* Returns the parts of the label for a given waypoint
* @param {Waypoint} waypoint
* @returns {Array<string>}
*/
getWaypointParts(waypoint) {
return waypoint.parts || [T.ingame.waypoints.hub];
}
/** /**
* Returns if a waypoint is deletable * Returns if a waypoint is deletable
* @param {Waypoint} waypoint * @param {Waypoint} waypoint
@ -410,8 +477,10 @@ export class HUDWaypoints extends BaseHUDPart {
* @param {Waypoint} waypoint * @param {Waypoint} waypoint
* @return {{ * @return {{
* screenBounds: Rectangle * screenBounds: Rectangle
* item: BaseItem|null, * parts: Array<{
* item: boolean,
* text: string * text: string
* }>
* }} * }}
*/ */
getWaypointScreenParams(waypoint) { getWaypointScreenParams(waypoint) {
@ -424,17 +493,23 @@ export class HUDWaypoints extends BaseHUDPart {
const screenPos = this.root.camera.worldToScreen(new Vector(waypoint.center.x, waypoint.center.y)); const screenPos = this.root.camera.worldToScreen(new Vector(waypoint.center.x, waypoint.center.y));
// Distinguish between text and item waypoints -> Figure out parameters // Distinguish between text and item waypoints -> Figure out parameters
const originalLabel = this.getWaypointLabel(waypoint); const originalParts = this.getWaypointParts(waypoint);
let text, item, textWidth; let parts = [];
let textWidth = 0;
if (ShapeDefinition.isValidShortKey(originalLabel)) { for (let i = 0; i < originalParts.length; ++i) {
// If the label is actually a key, render the shape icon const originalPart = originalParts[i];
item = this.root.shapeDefinitionMgr.getShapeItemFromShortKey(originalLabel); let item = false;
textWidth = 40; const text = originalPart;
} else { if (ShapeDefinition.isValidShortKey(originalPart)) {
// Otherwise render a regular waypoint // If the label is actually a key, render the shape icon
text = originalLabel; item = true;
textWidth = this.getTextWidth(text); textWidth += 14 * scale;
} else {
// Otherwise render a regular waypoint
textWidth += this.getTextWidth(text);
}
parts.push({ item, text });
} }
return { return {
@ -444,8 +519,7 @@ export class HUDWaypoints extends BaseHUDPart {
15 * scale + textWidth, 15 * scale + textWidth,
15 * scale 15 * scale
), ),
item, parts,
text,
}; };
} }
@ -582,6 +656,7 @@ export class HUDWaypoints extends BaseHUDPart {
} }
const bounds = waypointData.screenBounds; const bounds = waypointData.screenBounds;
const parts = waypointData.parts;
const contentPaddingX = 7 * scale; const contentPaddingX = 7 * scale;
const isSelected = mousePos && bounds.containsPoint(mousePos.x, mousePos.y); const isSelected = mousePos && bounds.containsPoint(mousePos.x, mousePos.y);
@ -591,28 +666,32 @@ export class HUDWaypoints extends BaseHUDPart {
parameters.context.fillRect(bounds.x, bounds.y, bounds.w, bounds.h); parameters.context.fillRect(bounds.x, bounds.y, bounds.w, bounds.h);
// Render the text // Render the text
if (waypointData.item) { let textWidth = 0;
const canvas = this.getWaypointCanvas(waypoint); for (let j = 0; j < parts.length; ++j) {
const itemSize = 14 * scale; const part = parts[j];
parameters.context.drawImage( if (part.item) {
canvas, const canvas = this.getWaypointCanvas(part.text);
bounds.x + contentPaddingX + 6 * scale, const itemSize = 14 * scale;
bounds.y + bounds.h / 2 - itemSize / 2, parameters.context.drawImage(
itemSize, canvas,
itemSize bounds.x + contentPaddingX + 6 * scale + textWidth,
); bounds.y + bounds.h / 2 - itemSize / 2,
} else if (waypointData.text) { itemSize,
// Render the text itemSize
parameters.context.fillStyle = "#000"; );
parameters.context.textBaseline = "middle"; textWidth += 14 * scale;
parameters.context.fillText( } else {
waypointData.text, // Render the text
bounds.x + contentPaddingX + 6 * scale, parameters.context.fillStyle = "#000";
bounds.y + bounds.h / 2 parameters.context.textBaseline = "middle";
); parameters.context.fillText(
parameters.context.textBaseline = "alphabetic"; part.text,
} else { bounds.x + contentPaddingX + 6 * scale + textWidth,
assertAlways(false, "Waypoint has no item and text"); bounds.y + bounds.h / 2
);
parameters.context.textBaseline = "alphabetic";
textWidth += this.getTextWidth(part.text);
}
} }
// Render the small icon on the left // Render the small icon on the left