2020-08-28 18:56:02 +00:00
|
|
|
import { globalConfig } from "../core/config";
|
|
|
|
import { DrawParameters } from "../core/draw_parameters";
|
|
|
|
import { getBuildingDataFromCode } from "./building_codes";
|
|
|
|
import { Entity } from "./entity";
|
|
|
|
import { MapChunk } from "./map_chunk";
|
|
|
|
import { GameRoot } from "./root";
|
|
|
|
import { THEME } from "./theme";
|
|
|
|
import { drawSpriteClipped } from "../core/draw_utils";
|
|
|
|
|
|
|
|
export const CHUNK_OVERLAY_RES = 3;
|
|
|
|
|
|
|
|
export class MapChunkView extends MapChunk {
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {GameRoot} root
|
|
|
|
* @param {number} x
|
|
|
|
* @param {number} y
|
|
|
|
*/
|
|
|
|
constructor(root, x, y) {
|
|
|
|
super(root, x, y);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whenever something changes, we increase this number - so we know we need to redraw
|
|
|
|
*/
|
|
|
|
this.renderIteration = 0;
|
|
|
|
|
|
|
|
this.markDirty();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Marks this chunk as dirty, rerendering all caches
|
|
|
|
*/
|
|
|
|
markDirty() {
|
|
|
|
++this.renderIteration;
|
|
|
|
this.renderKey = this.x + "/" + this.y + "@" + this.renderIteration;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Draws the background layer
|
|
|
|
* @param {DrawParameters} parameters
|
|
|
|
*/
|
|
|
|
drawBackgroundLayer(parameters) {
|
|
|
|
const systems = this.root.systemMgr.systems;
|
Puzzle DLC (#1172)
* Puzzle mode (#1135)
* Add mode button to main menu
* [WIP] Add mode menu. Add factory-based gameMode creation
* Add savefile migration, serialize, deserialize
* Add hidden HUD elements, zone, and zoom, boundary constraints
* Clean up lint issues
* Add building, HUD exclusion, building exclusion, and refactor
- [WIP] Add ConstantProducer building that combines ConstantSignal
and ItemProducer functionality. Currently using temp assets.
- Add pre-placement check to the zone
- Use Rectangles for zone and boundary
- Simplify zone drawing
- Account for exclusion in savegame data
- [WIP] Add puzzle play and edit buttons in puzzle mode menu
* [WIP] Add building, component, and systems for producing and
accepting user-specified items and checking goal criteria
* Add ingame puzzle mode UI elements
- Add minimal menus in puzzle mode for back, next navigation
- Add lower menu for changing zone dimenensions
Co-authored-by: Greg Considine <gconsidine@users.noreply.github.com>
* Performance optimizations (#1154)
* 1.3.1 preparations
* Minor fixes, update translations
* Fix achievements not working
* Lots of belt optimizations, ~15% performance boost
* Puzzle mode, part 1
* Puzzle mode, part 2
* Fix missing import
* Puzzle mode, part 3
* Fix typo
* Puzzle mode, part 4
* Puzzle Mode fixes: Correct zone restrictions and more (#1155)
* Hide Puzzle Editor Controls in regular game mode, fix typo
* Disallow shrinking zone if there are buildings
* Fix multi-tile buildings for shrinking
* Puzzle mode, Refactor hud
* Puzzle mode
* Fixed typo in latest puzzle commit (#1156)
* Allow completing puzzles
* Puzzle mode, almost done
* Bump version to 1.4.0
* Fixes
* [puzzle] Prevent pipette cheats (miners, emitters) (#1158)
* Puzzle mode, almost done
* Allow clearing belts with 'B'
* Multiple users for the puzzle dlc
* Bump api key
* Minor adjustments
* Update
* Minor fixes
* Fix throughput
* Fix belts
* Minor puzzle adjustments
* New difficulty
* Minor puzzle improvements
* Fix belt path
* Update translations
* Added a button to return to the menu after a puzzle is completed (#1170)
* added another button to return to the menu
* improved menu return
* fixed continue button to not go back to menu
* [Puzzle] Added ability to lock buildings in the puzzle editor! (#1164)
* initial test
* tried to get it to work
* added icon
* added test exclusion
* reverted css
* completed flow for building locking
* added lock option
* finalized look and changed locked building to same sprite
* removed unused art
* added clearing every goal acceptor on lock to prevent creating impossible puzzles
* heavily improved validation and prevented autocompletion
* validation only checks every 100 ticks to improve performance
* validation only checks every 100 ticks to improve performance
* removed clearing goal acceptors as it isn't needed because of validation
* Add soundtrack, puzzle dlc fixes
Co-authored-by: Greg Considine <gconsidine@users.noreply.github.com>
Co-authored-by: dengr1065 <dengr1065@gmail.com>
Co-authored-by: Sense101 <67970865+Sense101@users.noreply.github.com>
2021-05-23 14:32:05 +00:00
|
|
|
if (systems.zone) {
|
|
|
|
systems.zone.drawChunk(parameters, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.root.gameMode.hasResources()) {
|
|
|
|
systems.mapResources.drawChunk(parameters, this);
|
|
|
|
}
|
|
|
|
|
2020-08-28 18:56:02 +00:00
|
|
|
systems.beltUnderlays.drawChunk(parameters, this);
|
|
|
|
systems.belt.drawChunk(parameters, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Draws the dynamic foreground layer
|
|
|
|
* @param {DrawParameters} parameters
|
|
|
|
*/
|
|
|
|
drawForegroundDynamicLayer(parameters) {
|
|
|
|
const systems = this.root.systemMgr.systems;
|
|
|
|
|
|
|
|
systems.itemEjector.drawChunk(parameters, this);
|
|
|
|
systems.itemAcceptor.drawChunk(parameters, this);
|
|
|
|
systems.miner.drawChunk(parameters, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Draws the static foreground layer
|
|
|
|
* @param {DrawParameters} parameters
|
|
|
|
*/
|
|
|
|
drawForegroundStaticLayer(parameters) {
|
|
|
|
const systems = this.root.systemMgr.systems;
|
|
|
|
|
|
|
|
systems.staticMapEntities.drawChunk(parameters, this);
|
|
|
|
systems.lever.drawChunk(parameters, this);
|
|
|
|
systems.display.drawChunk(parameters, this);
|
|
|
|
systems.storage.drawChunk(parameters, this);
|
Puzzle DLC (#1172)
* Puzzle mode (#1135)
* Add mode button to main menu
* [WIP] Add mode menu. Add factory-based gameMode creation
* Add savefile migration, serialize, deserialize
* Add hidden HUD elements, zone, and zoom, boundary constraints
* Clean up lint issues
* Add building, HUD exclusion, building exclusion, and refactor
- [WIP] Add ConstantProducer building that combines ConstantSignal
and ItemProducer functionality. Currently using temp assets.
- Add pre-placement check to the zone
- Use Rectangles for zone and boundary
- Simplify zone drawing
- Account for exclusion in savegame data
- [WIP] Add puzzle play and edit buttons in puzzle mode menu
* [WIP] Add building, component, and systems for producing and
accepting user-specified items and checking goal criteria
* Add ingame puzzle mode UI elements
- Add minimal menus in puzzle mode for back, next navigation
- Add lower menu for changing zone dimenensions
Co-authored-by: Greg Considine <gconsidine@users.noreply.github.com>
* Performance optimizations (#1154)
* 1.3.1 preparations
* Minor fixes, update translations
* Fix achievements not working
* Lots of belt optimizations, ~15% performance boost
* Puzzle mode, part 1
* Puzzle mode, part 2
* Fix missing import
* Puzzle mode, part 3
* Fix typo
* Puzzle mode, part 4
* Puzzle Mode fixes: Correct zone restrictions and more (#1155)
* Hide Puzzle Editor Controls in regular game mode, fix typo
* Disallow shrinking zone if there are buildings
* Fix multi-tile buildings for shrinking
* Puzzle mode, Refactor hud
* Puzzle mode
* Fixed typo in latest puzzle commit (#1156)
* Allow completing puzzles
* Puzzle mode, almost done
* Bump version to 1.4.0
* Fixes
* [puzzle] Prevent pipette cheats (miners, emitters) (#1158)
* Puzzle mode, almost done
* Allow clearing belts with 'B'
* Multiple users for the puzzle dlc
* Bump api key
* Minor adjustments
* Update
* Minor fixes
* Fix throughput
* Fix belts
* Minor puzzle adjustments
* New difficulty
* Minor puzzle improvements
* Fix belt path
* Update translations
* Added a button to return to the menu after a puzzle is completed (#1170)
* added another button to return to the menu
* improved menu return
* fixed continue button to not go back to menu
* [Puzzle] Added ability to lock buildings in the puzzle editor! (#1164)
* initial test
* tried to get it to work
* added icon
* added test exclusion
* reverted css
* completed flow for building locking
* added lock option
* finalized look and changed locked building to same sprite
* removed unused art
* added clearing every goal acceptor on lock to prevent creating impossible puzzles
* heavily improved validation and prevented autocompletion
* validation only checks every 100 ticks to improve performance
* validation only checks every 100 ticks to improve performance
* removed clearing goal acceptors as it isn't needed because of validation
* Add soundtrack, puzzle dlc fixes
Co-authored-by: Greg Considine <gconsidine@users.noreply.github.com>
Co-authored-by: dengr1065 <dengr1065@gmail.com>
Co-authored-by: Sense101 <67970865+Sense101@users.noreply.github.com>
2021-05-23 14:32:05 +00:00
|
|
|
systems.constantProducer.drawChunk(parameters, this);
|
|
|
|
systems.goalAcceptor.drawChunk(parameters, this);
|
2020-08-29 07:35:14 +00:00
|
|
|
systems.itemProcessorOverlays.drawChunk(parameters, this);
|
2020-08-28 18:56:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Overlay
|
|
|
|
* @param {DrawParameters} parameters
|
|
|
|
*/
|
|
|
|
drawOverlay(parameters) {
|
|
|
|
const overlaySize = globalConfig.mapChunkSize * CHUNK_OVERLAY_RES;
|
|
|
|
const sprite = this.root.buffers.getForKey({
|
|
|
|
key: "chunk@" + this.root.currentLayer,
|
|
|
|
subKey: this.renderKey,
|
|
|
|
w: overlaySize,
|
|
|
|
h: overlaySize,
|
|
|
|
dpi: 1,
|
|
|
|
redrawMethod: this.generateOverlayBuffer.bind(this),
|
|
|
|
});
|
|
|
|
|
|
|
|
const dims = globalConfig.mapChunkWorldSize;
|
2020-09-19 08:34:46 +00:00
|
|
|
const extrude = 0.05;
|
2020-08-28 18:56:02 +00:00
|
|
|
|
|
|
|
// Draw chunk "pixel" art
|
|
|
|
parameters.context.imageSmoothingEnabled = false;
|
|
|
|
drawSpriteClipped({
|
|
|
|
parameters,
|
|
|
|
sprite,
|
2020-09-19 08:34:46 +00:00
|
|
|
x: this.x * dims - extrude,
|
|
|
|
y: this.y * dims - extrude,
|
|
|
|
w: dims + 2 * extrude,
|
|
|
|
h: dims + 2 * extrude,
|
2020-08-28 18:56:02 +00:00
|
|
|
originalW: overlaySize,
|
|
|
|
originalH: overlaySize,
|
|
|
|
});
|
|
|
|
|
|
|
|
parameters.context.imageSmoothingEnabled = true;
|
2020-10-08 07:13:53 +00:00
|
|
|
const resourcesScale = this.root.app.settings.getAllSettings().mapResourcesScale;
|
2020-08-28 18:56:02 +00:00
|
|
|
|
|
|
|
// Draw patch items
|
2020-10-08 07:13:53 +00:00
|
|
|
if (this.root.currentLayer === "regular" && resourcesScale > 0.05) {
|
|
|
|
const diameter = (70 / Math.pow(parameters.zoomLevel, 0.35)) * (0.2 + 2 * resourcesScale);
|
|
|
|
|
2020-08-28 18:56:02 +00:00
|
|
|
for (let i = 0; i < this.patches.length; ++i) {
|
|
|
|
const patch = this.patches[i];
|
2020-09-22 09:21:18 +00:00
|
|
|
if (patch.item.getItemType() === "shape") {
|
|
|
|
const destX = this.x * dims + patch.pos.x * globalConfig.tileSize;
|
|
|
|
const destY = this.y * dims + patch.pos.y * globalConfig.tileSize;
|
|
|
|
patch.item.drawItemCenteredClipped(destX, destY, parameters, diameter);
|
|
|
|
}
|
2020-08-28 18:56:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {HTMLCanvasElement} canvas
|
|
|
|
* @param {CanvasRenderingContext2D} context
|
|
|
|
* @param {number} w
|
|
|
|
* @param {number} h
|
|
|
|
* @param {number} dpi
|
|
|
|
*/
|
|
|
|
generateOverlayBuffer(canvas, context, w, h, dpi) {
|
|
|
|
context.fillStyle =
|
|
|
|
this.containedEntities.length > 0
|
|
|
|
? THEME.map.chunkOverview.filled
|
|
|
|
: THEME.map.chunkOverview.empty;
|
|
|
|
context.fillRect(0, 0, w, h);
|
|
|
|
|
2020-08-29 20:52:52 +00:00
|
|
|
if (this.root.app.settings.getAllSettings().displayChunkBorders) {
|
|
|
|
context.fillStyle = THEME.map.chunkBorders;
|
|
|
|
context.fillRect(0, 0, w, 1);
|
|
|
|
context.fillRect(0, 1, 1, h);
|
|
|
|
}
|
|
|
|
|
2020-08-28 18:56:02 +00:00
|
|
|
for (let x = 0; x < globalConfig.mapChunkSize; ++x) {
|
|
|
|
const lowerArray = this.lowerLayer[x];
|
|
|
|
const upperArray = this.contents[x];
|
|
|
|
for (let y = 0; y < globalConfig.mapChunkSize; ++y) {
|
|
|
|
const upperContent = upperArray[y];
|
|
|
|
if (upperContent) {
|
|
|
|
const staticComp = upperContent.components.StaticMapEntity;
|
|
|
|
const data = getBuildingDataFromCode(staticComp.code);
|
|
|
|
const metaBuilding = data.metaInstance;
|
|
|
|
|
|
|
|
const overlayMatrix = metaBuilding.getSpecialOverlayRenderMatrix(
|
|
|
|
staticComp.rotation,
|
|
|
|
data.rotationVariant,
|
|
|
|
data.variant,
|
|
|
|
upperContent
|
|
|
|
);
|
|
|
|
|
|
|
|
if (overlayMatrix) {
|
|
|
|
// Draw lower content first since it "shines" through
|
|
|
|
const lowerContent = lowerArray[y];
|
|
|
|
if (lowerContent) {
|
|
|
|
context.fillStyle = lowerContent.getBackgroundColorAsResource();
|
|
|
|
context.fillRect(
|
|
|
|
x * CHUNK_OVERLAY_RES,
|
|
|
|
y * CHUNK_OVERLAY_RES,
|
|
|
|
CHUNK_OVERLAY_RES,
|
|
|
|
CHUNK_OVERLAY_RES
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-23 09:14:35 +00:00
|
|
|
context.fillStyle = metaBuilding.getSilhouetteColor(
|
|
|
|
data.variant,
|
|
|
|
data.rotationVariant
|
|
|
|
);
|
2020-08-28 18:56:02 +00:00
|
|
|
for (let dx = 0; dx < 3; ++dx) {
|
|
|
|
for (let dy = 0; dy < 3; ++dy) {
|
|
|
|
const isFilled = overlayMatrix[dx + dy * 3];
|
|
|
|
if (isFilled) {
|
|
|
|
context.fillRect(
|
|
|
|
x * CHUNK_OVERLAY_RES + dx,
|
|
|
|
y * CHUNK_OVERLAY_RES + dy,
|
|
|
|
1,
|
|
|
|
1
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
} else {
|
2020-09-23 09:14:35 +00:00
|
|
|
context.fillStyle = metaBuilding.getSilhouetteColor(
|
|
|
|
data.variant,
|
|
|
|
data.rotationVariant
|
|
|
|
);
|
2020-08-28 18:56:02 +00:00
|
|
|
context.fillRect(
|
|
|
|
x * CHUNK_OVERLAY_RES,
|
|
|
|
y * CHUNK_OVERLAY_RES,
|
|
|
|
CHUNK_OVERLAY_RES,
|
|
|
|
CHUNK_OVERLAY_RES
|
|
|
|
);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const lowerContent = lowerArray[y];
|
|
|
|
if (lowerContent) {
|
|
|
|
context.fillStyle = lowerContent.getBackgroundColorAsResource();
|
|
|
|
context.fillRect(
|
|
|
|
x * CHUNK_OVERLAY_RES,
|
|
|
|
y * CHUNK_OVERLAY_RES,
|
|
|
|
CHUNK_OVERLAY_RES,
|
|
|
|
CHUNK_OVERLAY_RES
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.root.currentLayer === "wires") {
|
|
|
|
// Draw wires overlay
|
|
|
|
|
|
|
|
context.fillStyle = THEME.map.wires.overlayColor;
|
|
|
|
context.fillRect(0, 0, w, h);
|
|
|
|
|
|
|
|
for (let x = 0; x < globalConfig.mapChunkSize; ++x) {
|
|
|
|
const wiresArray = this.wireContents[x];
|
|
|
|
for (let y = 0; y < globalConfig.mapChunkSize; ++y) {
|
|
|
|
const content = wiresArray[y];
|
|
|
|
if (!content) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
MapChunkView.drawSingleWiresOverviewTile({
|
|
|
|
context,
|
|
|
|
x: x * CHUNK_OVERLAY_RES,
|
|
|
|
y: y * CHUNK_OVERLAY_RES,
|
|
|
|
entity: content,
|
|
|
|
tileSizePixels: CHUNK_OVERLAY_RES,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {object} param0
|
|
|
|
* @param {CanvasRenderingContext2D} param0.context
|
|
|
|
* @param {number} param0.x
|
|
|
|
* @param {number} param0.y
|
|
|
|
* @param {Entity} param0.entity
|
|
|
|
* @param {number} param0.tileSizePixels
|
|
|
|
* @param {string=} param0.overrideColor Optionally override the color to be rendered
|
|
|
|
*/
|
|
|
|
static drawSingleWiresOverviewTile({ context, x, y, entity, tileSizePixels, overrideColor = null }) {
|
|
|
|
const staticComp = entity.components.StaticMapEntity;
|
|
|
|
const data = getBuildingDataFromCode(staticComp.code);
|
|
|
|
const metaBuilding = data.metaInstance;
|
|
|
|
const overlayMatrix = metaBuilding.getSpecialOverlayRenderMatrix(
|
|
|
|
staticComp.rotation,
|
|
|
|
data.rotationVariant,
|
|
|
|
data.variant,
|
|
|
|
entity
|
|
|
|
);
|
2020-09-23 09:14:35 +00:00
|
|
|
context.fillStyle =
|
|
|
|
overrideColor || metaBuilding.getSilhouetteColor(data.variant, data.rotationVariant);
|
2020-08-28 18:56:02 +00:00
|
|
|
if (overlayMatrix) {
|
|
|
|
for (let dx = 0; dx < 3; ++dx) {
|
|
|
|
for (let dy = 0; dy < 3; ++dy) {
|
|
|
|
const isFilled = overlayMatrix[dx + dy * 3];
|
|
|
|
if (isFilled) {
|
|
|
|
context.fillRect(
|
|
|
|
x + (dx * tileSizePixels) / CHUNK_OVERLAY_RES,
|
|
|
|
y + (dy * tileSizePixels) / CHUNK_OVERLAY_RES,
|
|
|
|
tileSizePixels / CHUNK_OVERLAY_RES,
|
|
|
|
tileSizePixels / CHUNK_OVERLAY_RES
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
context.fillRect(x, y, tileSizePixels, tileSizePixels);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Draws the wires layer
|
|
|
|
* @param {DrawParameters} parameters
|
|
|
|
*/
|
|
|
|
drawWiresForegroundLayer(parameters) {
|
|
|
|
const systems = this.root.systemMgr.systems;
|
|
|
|
systems.wire.drawChunk(parameters, this);
|
|
|
|
systems.staticMapEntities.drawWiresChunk(parameters, this);
|
|
|
|
systems.wiredPins.drawChunk(parameters, this);
|
|
|
|
}
|
|
|
|
}
|