[d565449] | 1 | import Konva from "konva";
|
---|
| 2 | import Factory from "../util/Factory.js";
|
---|
| 3 | import HttpService from "../net/HttpService.js";
|
---|
| 4 | import {zoomStage} from "../util/zoomStage.js";
|
---|
[0c6b92a] | 5 | import {addEventHandling} from "../util/addEventHandling.js";
|
---|
[79a0317] | 6 | import ShapeRegistry from "../util/ShapeRegistry.js";
|
---|
[0c6b92a] | 7 | import triggerNavigate from "../util/triggerNavigate.js";
|
---|
| 8 | import config from "../net/netconfig.js";
|
---|
[79a0317] | 9 | import {dispatchCustomEvent} from "../util/dispatchCustomEvent.js";
|
---|
| 10 | import { jsPDF } from "jspdf";
|
---|
| 11 |
|
---|
[0c6b92a] | 12 |
|
---|
[d565449] | 13 | export class MapDisplay {
|
---|
[0c6b92a] | 14 | constructor(containerId, floorNum) {
|
---|
| 15 | this.container = document.getElementById(containerId);
|
---|
| 16 | this.containerId = containerId;
|
---|
| 17 | this.stage = new Konva.Stage({
|
---|
| 18 | container: containerId,
|
---|
| 19 | width: window.innerWidth,
|
---|
| 20 | height: window.innerHeight,
|
---|
| 21 | draggable: true,
|
---|
| 22 | });
|
---|
[d565449] | 23 |
|
---|
[79a0317] | 24 | this.selectedRoom = {
|
---|
| 25 | id: 1,
|
---|
| 26 | name: "223",
|
---|
| 27 | type: "Laboratory",
|
---|
| 28 | image:
|
---|
| 29 | "",
|
---|
| 30 | description: "223 is a laboratory with a capacity of 40 people",
|
---|
| 31 | };
|
---|
| 32 | this.toggleSearch = false;
|
---|
| 33 |
|
---|
[0c6b92a] | 34 | this.shapes = [];
|
---|
| 35 | this.roomTypes = [];
|
---|
| 36 | this.loaded = false;
|
---|
| 37 | this.mainLayer = new Konva.Layer();
|
---|
| 38 | this.routeLayer = new Konva.Layer();
|
---|
| 39 | this.textLayer = new Konva.Layer();
|
---|
| 40 | this.stage.add(this.mainLayer);
|
---|
| 41 | this.stage.add(this.routeLayer);
|
---|
| 42 | this.stage.add(this.textLayer);
|
---|
| 43 |
|
---|
[79a0317] | 44 |
|
---|
| 45 | this.cachedCanvases = [];
|
---|
| 46 |
|
---|
[0c6b92a] | 47 | this.floorNum = floorNum;
|
---|
| 48 |
|
---|
| 49 | this.navArrow = new Konva.Arrow({
|
---|
| 50 | stroke: "#bb0000",
|
---|
| 51 | strokeWidth: 3,
|
---|
| 52 | dash: [12, 7],
|
---|
| 53 | lineCap: "round",
|
---|
| 54 | tension: 10,
|
---|
| 55 | pointerLength: 2,
|
---|
| 56 | pointerWidth: 3,
|
---|
| 57 | });
|
---|
[d565449] | 58 |
|
---|
[0c6b92a] | 59 | this.navArrow.cache();
|
---|
[d565449] | 60 |
|
---|
[0c6b92a] | 61 | this.stage.on("resize", () => {
|
---|
| 62 | this.stage.width = window.innerWidth;
|
---|
| 63 | this.stage.height = window.innerHeight;
|
---|
| 64 | });
|
---|
| 65 |
|
---|
| 66 | this.stage.on("wheel", (e) => {
|
---|
| 67 | zoomStage(e, this.stage);
|
---|
| 68 | });
|
---|
| 69 | }
|
---|
| 70 |
|
---|
| 71 | clearMap() {
|
---|
| 72 | this.mainLayer.removeChildren();
|
---|
| 73 | this.shapes.forEach(shape => shape.clearText())
|
---|
| 74 | this.shapes = [];
|
---|
| 75 | }
|
---|
| 76 |
|
---|
| 77 | deserializeMap(data) {
|
---|
| 78 | this.clearMap();
|
---|
| 79 |
|
---|
| 80 | let dsrData = JSON.parse(data);
|
---|
| 81 | dsrData.forEach((shape) => {
|
---|
| 82 | if (shape.className !== "InfoPin") {
|
---|
[79a0317] | 83 | console.log("SHAPE ATTRS CREATE: " + JSON.stringify(shape.attrs));
|
---|
[0c6b92a] | 84 | const renderedShape = Factory.createRenderedShape(shape.className, shape.attrs);
|
---|
| 85 | addEventHandling(renderedShape, this, "click");
|
---|
| 86 | this.shapes.push(renderedShape);
|
---|
[79a0317] | 87 | console.log(renderedShape)
|
---|
[0c6b92a] | 88 | }
|
---|
| 89 | });
|
---|
| 90 | }
|
---|
| 91 |
|
---|
| 92 | displayRoomNames() {
|
---|
| 93 | this.shapes.forEach((shape) => {
|
---|
| 94 | shape.displayName(this.textLayer);
|
---|
| 95 | });
|
---|
| 96 |
|
---|
| 97 | }
|
---|
| 98 |
|
---|
| 99 | loadMapN(floorData) {
|
---|
| 100 | if (floorData == null || floorData === "") return;
|
---|
| 101 |
|
---|
| 102 | this.deserializeMap(floorData);
|
---|
| 103 | this.shapes.forEach((shape) => {
|
---|
| 104 | this.mainLayer.add(shape);
|
---|
| 105 | });
|
---|
| 106 | this.displayRoomNames();
|
---|
| 107 | this.initializeRoomTypes();
|
---|
| 108 |
|
---|
| 109 | }
|
---|
| 110 |
|
---|
| 111 | clearRoute() {
|
---|
| 112 | this.routeLayer.removeChildren();
|
---|
| 113 | }
|
---|
| 114 |
|
---|
| 115 |
|
---|
[79a0317] | 116 |
|
---|
[0c6b92a] | 117 | drawRouteNEW(nodes, offset = 0) {
|
---|
| 118 |
|
---|
| 119 | this.clearRoute();
|
---|
| 120 | console.log("====PATH====");
|
---|
| 121 | nodes.forEach((node) => console.log("NODE", node));
|
---|
| 122 |
|
---|
| 123 | let idx = offset;
|
---|
| 124 | let buff = [nodes[idx].coordinates.x, nodes[idx].coordinates.y];
|
---|
| 125 |
|
---|
| 126 | ++idx;
|
---|
| 127 |
|
---|
| 128 | console.log("INIT BUFFER", buff);
|
---|
| 129 | console.log("INIT IDX", idx);
|
---|
| 130 |
|
---|
| 131 | const drawNextSegment = () => {
|
---|
| 132 |
|
---|
| 133 | if (idx >= nodes.length){
|
---|
[79a0317] | 134 | console.log("FLOOR NUM DISPLAY: " + this.floorNum)
|
---|
| 135 | let stageInfo = {stage:this.stage.clone(),floor:this.floorNum};
|
---|
| 136 | this.cachedCanvases.push(stageInfo);
|
---|
| 137 | dispatchCustomEvent("navend",this.downloadURI)
|
---|
[0c6b92a] | 138 | return;
|
---|
| 139 | }
|
---|
| 140 |
|
---|
| 141 | const currentNode = nodes[idx - 1];
|
---|
| 142 | const nextNode = nodes[idx];
|
---|
| 143 |
|
---|
| 144 | if (nextNode.floorNumber !== currentNode.floorNumber) {
|
---|
[79a0317] | 145 | let stageInfo = {stage:this.stage.clone(),floor:this.floorNum};
|
---|
| 146 | this.cachedCanvases.push(stageInfo);
|
---|
[0c6b92a] | 147 | triggerNavigate(nodes, idx, nextNode.floorNumber, nextNode);
|
---|
| 148 | return;
|
---|
| 149 | }
|
---|
[d565449] | 150 |
|
---|
[0c6b92a] | 151 | const startX = currentNode.coordinates.x;
|
---|
| 152 | const startY = currentNode.coordinates.y;
|
---|
| 153 | const endX = nextNode.coordinates.x;
|
---|
| 154 | const endY = nextNode.coordinates.y;
|
---|
| 155 |
|
---|
[79a0317] | 156 | const numSegments = 10;
|
---|
[0c6b92a] | 157 |
|
---|
| 158 | const deltaX = (endX - startX) / numSegments;
|
---|
| 159 | const deltaY = (endY - startY) / numSegments;
|
---|
| 160 |
|
---|
| 161 | const drawSegment = (i) => {
|
---|
| 162 | const segmentX = startX + deltaX * i;
|
---|
| 163 | const segmentY = startY + deltaY * i;
|
---|
| 164 |
|
---|
| 165 | buff.push(segmentX, segmentY);
|
---|
| 166 |
|
---|
| 167 | let line = this.navArrow.clone({ points: [...buff] });
|
---|
| 168 | this.routeLayer.add(line);
|
---|
| 169 | this.routeLayer.draw();
|
---|
| 170 |
|
---|
| 171 | buff = [segmentX, segmentY];
|
---|
| 172 | };
|
---|
| 173 |
|
---|
| 174 | let segmentIdx = 1;
|
---|
| 175 | const interval = setInterval(() => {
|
---|
| 176 | drawSegment(segmentIdx);
|
---|
| 177 | segmentIdx++;
|
---|
| 178 |
|
---|
| 179 | if (segmentIdx > numSegments) {
|
---|
| 180 | clearInterval(interval);
|
---|
| 181 | idx++;
|
---|
[79a0317] | 182 | setTimeout(drawNextSegment, 60);
|
---|
[0c6b92a] | 183 | }
|
---|
[79a0317] | 184 | }, 30);
|
---|
[0c6b92a] | 185 | };
|
---|
| 186 |
|
---|
| 187 | drawNextSegment();
|
---|
| 188 | }
|
---|
| 189 |
|
---|
| 190 | initializeRoomTypes() {
|
---|
| 191 | this.roomTypes = this.shapes
|
---|
| 192 | .filter((shape) => shape.class === "Room" && shape.info.type !== "")
|
---|
| 193 | .map((shape) => shape.info.type);
|
---|
| 194 | }
|
---|
| 195 |
|
---|
| 196 | getShapeByType(type) {
|
---|
| 197 | return this.shapes.filter((shape) => shape.class === type)
|
---|
| 198 | }
|
---|
| 199 |
|
---|
| 200 | //ova e loso ne trebit vaka
|
---|
| 201 | highlightShape(roomName) {
|
---|
| 202 | let foundShape = this.shapes.filter((shape) => shape.info.name === roomName)[0];
|
---|
| 203 | foundShape.highlight();
|
---|
| 204 | }
|
---|
| 205 |
|
---|
[79a0317] | 206 | downloadURI(uri, name) {
|
---|
| 207 | let link = document.createElement('a');
|
---|
| 208 | link.download = name;
|
---|
| 209 | link.href = uri;
|
---|
| 210 | document.body.appendChild(link);
|
---|
| 211 | link.click();
|
---|
| 212 | document.body.removeChild(link);
|
---|
[0c6b92a] | 213 | }
|
---|
| 214 |
|
---|
[79a0317] | 215 |
|
---|
| 216 | getRouteImages(mapDetails = { mapName: "mapName", from: "from", to: "to" }) {
|
---|
| 217 |
|
---|
| 218 | const pdf = new jsPDF("p", "mm", "a4");
|
---|
| 219 | const pageWidth = pdf.internal.pageSize.getWidth();
|
---|
| 220 | const pageHeight = pdf.internal.pageSize.getHeight();
|
---|
| 221 | const margin = 10;
|
---|
| 222 | let yOffset = margin;
|
---|
| 223 |
|
---|
| 224 | const imagesData = [];
|
---|
| 225 |
|
---|
| 226 | let minWidth = 800;
|
---|
| 227 |
|
---|
| 228 | this.cachedCanvases.forEach((canvas, index) => {
|
---|
| 229 | console.log("CANVASL " + JSON.stringify(canvas.stage))
|
---|
| 230 | let dsrStage = Konva.Node.create(canvas.stage, document.createElement("div"));
|
---|
| 231 |
|
---|
| 232 | let minX = Infinity, minY = Infinity, maxX = -Infinity, maxY = -Infinity;
|
---|
| 233 |
|
---|
| 234 | dsrStage.getLayers().forEach(layer => {
|
---|
| 235 | layer.getChildren().forEach(shape => {
|
---|
| 236 | const shapeX = shape.x();
|
---|
| 237 | const shapeY = shape.y();
|
---|
| 238 | const width = shape.width() * shape.scaleX();
|
---|
| 239 | const height = shape.height() * shape.scaleY();
|
---|
| 240 |
|
---|
| 241 | minX = Math.min(minX, shapeX - shape.offsetX() * shape.scaleX());
|
---|
| 242 | minY = Math.min(minY, shapeY - shape.offsetY() * shape.scaleY());
|
---|
| 243 | maxX = Math.max(maxX, shapeX + width - shape.offsetX() * shape.scaleX());
|
---|
| 244 | maxY = Math.max(maxY, shapeY + height - shape.offsetY() * shape.scaleY());
|
---|
| 245 | });
|
---|
| 246 | });
|
---|
| 247 |
|
---|
| 248 | let padding = 50;
|
---|
| 249 |
|
---|
| 250 | const virtualWidth = maxX - minX + padding;
|
---|
| 251 | const virtualHeight = maxY - minY + padding;
|
---|
| 252 |
|
---|
| 253 | const scaleDownFactor = Math.min(pageWidth / virtualWidth, pageHeight / virtualHeight);
|
---|
| 254 |
|
---|
| 255 | dsrStage.size({ width: virtualWidth, height: virtualHeight });
|
---|
| 256 | dsrStage.position({ x: -minX, y: -minY });
|
---|
| 257 | dsrStage.batchDraw();
|
---|
| 258 |
|
---|
| 259 | let canvasImageURI = dsrStage.toDataURL({
|
---|
| 260 | pixelRatio: 1
|
---|
| 261 | });
|
---|
| 262 |
|
---|
| 263 | imagesData.push(canvasImageURI);
|
---|
| 264 | console.log(`Generated Image ${index + 1}`);
|
---|
| 265 | });
|
---|
| 266 |
|
---|
| 267 |
|
---|
| 268 |
|
---|
| 269 | imagesData.forEach((imageURI, index) => {
|
---|
| 270 | const text = `Floor: ${this.cachedCanvases[index].floor}`;
|
---|
| 271 | pdf.text(text, margin, yOffset + 5);
|
---|
| 272 |
|
---|
| 273 | const img = new Image();
|
---|
| 274 | img.src = imageURI;
|
---|
| 275 |
|
---|
| 276 | const maxWidth = pageWidth - 2 * margin;
|
---|
| 277 | const maxHeight = pageHeight - 2 * margin;
|
---|
| 278 | let imgWidth = maxWidth;
|
---|
| 279 | let imgHeight = maxHeight / 2
|
---|
| 280 |
|
---|
| 281 | if (imgHeight > maxHeight) {
|
---|
| 282 | imgHeight = maxHeight;
|
---|
| 283 | imgWidth = (pageWidth * maxHeight) / pageHeight;
|
---|
| 284 | }
|
---|
| 285 |
|
---|
| 286 | if (yOffset + imgHeight > pageHeight - margin) {
|
---|
| 287 | pdf.addPage();
|
---|
| 288 | yOffset = margin;
|
---|
| 289 | }
|
---|
| 290 |
|
---|
| 291 | pdf.addImage(imageURI, "PNG", margin, yOffset, imgWidth, imgHeight);
|
---|
| 292 | yOffset += imgHeight + 10;
|
---|
| 293 | });
|
---|
| 294 |
|
---|
| 295 | pdf.save(`${mapDetails.mapName}-${mapDetails.from}-->${mapDetails.to}.route.pdf`);
|
---|
| 296 | }
|
---|
| 297 |
|
---|
| 298 |
|
---|
| 299 |
|
---|
| 300 | setFilter(filter) {
|
---|
[0c6b92a] | 301 | let rooms = this.getShapeByType("Room")
|
---|
| 302 | if (filter === "All") {
|
---|
| 303 | rooms.forEach((shape) => {
|
---|
| 304 | shape.unHighlight()
|
---|
| 305 | })
|
---|
| 306 | } else {
|
---|
| 307 | rooms.filter((shape) => shape.info.type === filter).forEach((shape) => {
|
---|
| 308 | shape.highlight()
|
---|
| 309 | })
|
---|
| 310 | rooms.filter((shape) => shape.info.type !== filter).forEach((shape) => {
|
---|
| 311 | shape.unHighlight()
|
---|
| 312 | })
|
---|
| 313 | }
|
---|
[d565449] | 314 |
|
---|
[0c6b92a] | 315 | }
|
---|
[d565449] | 316 | }
|
---|