2019-03-21 09:57:27 +01:00
|
|
|
import { Colors } from '../../utils.js'
|
2019-08-06 15:34:57 +02:00
|
|
|
import { WorkerTileLoader, PIXITileLoader } from './loader.js'
|
2019-03-21 09:57:27 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A layer of tiles that represents a zoom level of a DeepZoomImage as a grid
|
|
|
|
* of sprites.
|
|
|
|
* @constructor
|
|
|
|
* @param {number} level - the zoom level of the tile layer
|
|
|
|
* @param {DeepZoomImage} view - the zoomable image the layer belongs to
|
|
|
|
* @param {number} scale - the scale of the tile layer
|
|
|
|
* @param {number} cols - the number of columns of the layer
|
|
|
|
* @param {number} rows - the number of rows of the layer
|
|
|
|
* @param {number} width - the width of the layer in pixel
|
|
|
|
* @param {number} height - the height of the layer in pixel
|
|
|
|
* @param {number} tileSize - the size of a single tile in pixel
|
|
|
|
* @param {number} overlap - the overlap of the tiles in pixel
|
|
|
|
* @param {number} fadeInTime - time needed to fade in tiles if TweenLite is set
|
|
|
|
**/
|
|
|
|
export class Tiles extends PIXI.Container {
|
2019-08-06 15:34:57 +02:00
|
|
|
constructor(level, view, scale, cols, rows, width, height, tileSize, overlap, fadeInTime = 0.33) {
|
2019-03-21 09:57:27 +01:00
|
|
|
super()
|
|
|
|
this.debug = false
|
|
|
|
this.showGrid = false
|
|
|
|
this.view = view
|
|
|
|
this.level = level
|
|
|
|
this.cols = cols
|
|
|
|
this.rows = rows
|
|
|
|
this.pixelWidth = width
|
|
|
|
this.pixelHeight = height
|
|
|
|
this.tileSize = tileSize
|
|
|
|
this.overlap = overlap
|
|
|
|
this.needed = new Map() // url as key, [col, row] as value
|
|
|
|
this.requested = new Set()
|
|
|
|
this.available = new Map()
|
|
|
|
this.scale.set(scale, scale)
|
|
|
|
this.tileScale = scale
|
|
|
|
this.fadeInTime = fadeInTime
|
|
|
|
this.keep = false
|
2019-07-04 09:17:43 +02:00
|
|
|
if (this.view.useWorker && view.info.compression && view.info.compression.length > 0) {
|
|
|
|
this.loader = new WorkerTileLoader(this, this.view.useWorker)
|
|
|
|
} else {
|
2019-03-21 09:57:27 +01:00
|
|
|
this.loader = new PIXITileLoader(this, view.info.compression)
|
2019-07-04 09:17:43 +02:00
|
|
|
}
|
2019-03-21 09:57:27 +01:00
|
|
|
this.interactive = false
|
|
|
|
this._highlight = null
|
|
|
|
|
|
|
|
this._info = null
|
|
|
|
|
|
|
|
this._centerPoint = null
|
|
|
|
this._boundsRect = null
|
|
|
|
|
|
|
|
this.infoColor = Colors.random()
|
|
|
|
this.pprint()
|
|
|
|
this.destroyed = false
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Tests whether all tiles are loaded. **/
|
|
|
|
isComplete() {
|
|
|
|
return this.cols * this.rows === this.children.length
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Returns the highligh graphics layer for debugging purposes.
|
|
|
|
**/
|
|
|
|
get highlight() {
|
|
|
|
if (this._highlight == null) {
|
|
|
|
let graphics = new PIXI.Graphics()
|
|
|
|
graphics.beginFill(0xffff00, 0.1)
|
|
|
|
graphics.lineStyle(2, 0xffff00)
|
|
|
|
graphics.drawRect(1, 1, this.tileSize - 2, this.tileSize - 2)
|
|
|
|
graphics.endFill()
|
|
|
|
graphics.interactive = false
|
|
|
|
this._highlight = graphics
|
|
|
|
}
|
|
|
|
return this._highlight
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Returns the highligh graphics layer for debugging purposes.
|
|
|
|
**/
|
|
|
|
get info() {
|
|
|
|
if (this._info == null) {
|
|
|
|
let graphics = new PIXI.Graphics()
|
|
|
|
graphics.lineStyle(4, 0xff0000)
|
|
|
|
graphics.interactive = false
|
|
|
|
this._info = graphics
|
|
|
|
this.addChild(this._info)
|
|
|
|
}
|
|
|
|
return this._info
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Helper method pretty printing debug information. **/
|
|
|
|
pprint() {
|
|
|
|
if (this.debug)
|
|
|
|
console.log(
|
|
|
|
'Tiles level: ' +
|
2019-08-06 15:34:57 +02:00
|
|
|
this.level +
|
|
|
|
' scale: ' +
|
|
|
|
this.scale.x +
|
|
|
|
' cols: ' +
|
|
|
|
this.cols +
|
|
|
|
' rows: ' +
|
|
|
|
this.rows +
|
|
|
|
' w: ' +
|
|
|
|
this.pixelWidth +
|
|
|
|
' h: ' +
|
|
|
|
this.pixelHeight +
|
|
|
|
' tsize:' +
|
|
|
|
this.tileSize
|
2019-03-21 09:57:27 +01:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Computes the tile position and obeys the overlap.
|
|
|
|
* @param {number} col - The column of the tile
|
|
|
|
* @param {number} row - The row of the tile
|
|
|
|
* @returns {PIXI.Point} obj
|
|
|
|
**/
|
|
|
|
tilePosition(col, row) {
|
|
|
|
let x = col * this.tileSize
|
|
|
|
let y = row * this.tileSize
|
|
|
|
let overlap = this.overlap
|
|
|
|
if (col != 0) {
|
|
|
|
x -= overlap
|
|
|
|
}
|
|
|
|
if (row != 0) {
|
|
|
|
y -= overlap
|
|
|
|
}
|
|
|
|
return new PIXI.Point(x, y)
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Computes the tile size without overlap
|
|
|
|
* @param {number} col - The column of the tile
|
|
|
|
* @param {number} row - The row of the tile
|
|
|
|
* @returns {PIXI.Point} obj
|
|
|
|
**/
|
|
|
|
tileDimensions(col, row) {
|
|
|
|
let w = this.tileSize
|
|
|
|
let h = this.tileSize
|
|
|
|
let pos = this.tilePosition(col, row)
|
|
|
|
if (col == this.cols - 1) {
|
|
|
|
w = this.pixelWidth - pos.x
|
|
|
|
}
|
|
|
|
if (row == this.rows - 1) {
|
|
|
|
h = this.pixelHeight - pos.y
|
|
|
|
}
|
|
|
|
return new PIXI.Point(w, h)
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Method to support debugging. Highlights the specified tile at col, row **/
|
|
|
|
highlightTile(col, row) {
|
|
|
|
if (col > -1 && row > -1 && col < this.cols && row < this.rows) {
|
|
|
|
let graphics = this.highlight
|
|
|
|
let dim = this.tileDimensions(col, row)
|
|
|
|
graphics.position = this.tilePosition(col, row)
|
|
|
|
graphics.clear()
|
|
|
|
graphics.beginFill(0xff00ff, 0.1)
|
|
|
|
graphics.lineStyle(2, 0xffff00)
|
|
|
|
graphics.drawRect(1, 1, dim.x - 2, dim.y - 2)
|
|
|
|
graphics.endFill()
|
|
|
|
this.addChild(this.highlight)
|
|
|
|
} else {
|
|
|
|
this.removeChild(this.highlight)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Loads the tiles for the given urls and adds the tiles as sprites.
|
|
|
|
* @param {array} urlpos - An array of URL, pos pairs
|
|
|
|
* @param {boolean} onlyone - Loads only on tile at a time if true
|
|
|
|
**/
|
|
|
|
loadTiles(urlpos, onlyone, refCol, refRow) {
|
|
|
|
if (this.showGrid) {
|
|
|
|
this.highlightTile(refCol, refRow)
|
|
|
|
}
|
|
|
|
urlpos.forEach(d => {
|
|
|
|
let [url, col, row] = d
|
|
|
|
if (this.loader.schedule(url, col, row)) {
|
|
|
|
if (onlyone) {
|
|
|
|
return this.loader.loadOneTile()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
this.loader.loadAll()
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Private method: add a red border to a tile for debugging purposes. **/
|
|
|
|
_addTileBorder(tile, col, row) {
|
|
|
|
let dim = this.tileDimensions(col, row)
|
|
|
|
let graphics = new PIXI.Graphics()
|
|
|
|
graphics.beginFill(0, 0)
|
|
|
|
graphics.lineStyle(2, 0xff0000)
|
|
|
|
graphics.drawRect(1, 1, dim.x - 2, dim.y - 2)
|
|
|
|
graphics.endFill()
|
|
|
|
tile.addChild(graphics)
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Adds a tile. **/
|
|
|
|
addTile(tile, col, row, url) {
|
|
|
|
if (this.available.has(url)) {
|
|
|
|
console.warn('Trying to add available tile')
|
|
|
|
return
|
|
|
|
}
|
|
|
|
this.addChildAt(tile, 0)
|
|
|
|
this.available.set(url, tile)
|
|
|
|
if (this.destroyed) {
|
|
|
|
console.warn('Adding to destroyed tiles layer')
|
|
|
|
}
|
|
|
|
// this._addTileBorder(tile, col, row)
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Called by the loader after each successfull loading of a single tile.
|
|
|
|
* Adds the sprite to the tile layer.
|
|
|
|
* @param {Object} tile - the loaded tile sprite
|
|
|
|
* @param {Number} col - the col position
|
|
|
|
* @param {Number} row - the rowposition
|
|
|
|
**/
|
|
|
|
tileAvailable(tile, col, row, url) {
|
|
|
|
let pos = this.tilePosition(col, row)
|
|
|
|
if (this.showGrid) {
|
|
|
|
this._addTileBorder(tile, col, row)
|
|
|
|
}
|
|
|
|
tile.position = pos
|
|
|
|
tile.interactive = false
|
|
|
|
if (TweenLite) {
|
|
|
|
tile.alpha = 0
|
|
|
|
TweenLite.to(tile, this.fadeInTime, { alpha: this.alpha })
|
|
|
|
}
|
|
|
|
this.addTile(tile, col, row, url)
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Destroys the tiles layer and destroys the loader. Async load calls are
|
|
|
|
* cancelled.
|
|
|
|
**/
|
|
|
|
destroy() {
|
|
|
|
this.destroyed = true
|
|
|
|
this.loader.destroy()
|
|
|
|
super.destroy({ children: true }) // Calls destroyChildren
|
|
|
|
this.available.clear()
|
|
|
|
this.requested.clear()
|
|
|
|
this.needed.clear()
|
|
|
|
}
|
|
|
|
|
|
|
|
destroyTile(url, tile) {
|
|
|
|
this.loader.unschedule(url)
|
|
|
|
this.removeChild(tile)
|
|
|
|
tile.destroy()
|
|
|
|
this.available.delete(url)
|
|
|
|
}
|
|
|
|
|
|
|
|
destroyTileByUrl(url) {
|
|
|
|
if (this.available.has(url)) {
|
|
|
|
let tile = this.available.get(url)
|
|
|
|
this.destroyTile(url, tile)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Destroys the tiles which are not with the bounds of the app to free
|
2019-08-06 15:34:57 +02:00
|
|
|
* memory.
|
|
|
|
**/
|
2019-03-21 09:57:27 +01:00
|
|
|
destroyTiles(quadTrees) {
|
|
|
|
let count = 0
|
|
|
|
for (let [url, tile] of this.available.entries()) {
|
|
|
|
if (!quadTrees.has(url)) {
|
|
|
|
this.destroyTile(url, tile)
|
|
|
|
count += 1
|
|
|
|
}
|
|
|
|
}
|
2019-08-06 15:34:57 +02:00
|
|
|
if (count && this.debug) console.log('destroyTiles', this.level, count)
|
2019-03-21 09:57:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
destroyUnneededTiles() {
|
|
|
|
let count = 0
|
|
|
|
for (let [url, tile] of this.available.entries()) {
|
|
|
|
if (!this.needed.has(url)) {
|
|
|
|
this.destroyTile(url, tile)
|
|
|
|
count += 1
|
|
|
|
}
|
|
|
|
}
|
2019-08-06 15:34:57 +02:00
|
|
|
if (count && this.debug) console.log('destroyUnneededTiles', this.level, count)
|
2019-03-21 09:57:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
highlightInfos() {
|
|
|
|
let graphics = this.info
|
|
|
|
let color = this.infoColor
|
|
|
|
graphics.clear()
|
|
|
|
graphics.lineStyle(2, color)
|
|
|
|
for (let [col, row] of this.needed.values()) {
|
|
|
|
let dim = this.tileDimensions(col, row)
|
|
|
|
let pos = this.tilePosition(col, row)
|
|
|
|
graphics.beginFill(color, 0.2)
|
|
|
|
graphics.drawRect(pos.x + 1, pos.y + 1, dim.x - 2, dim.y - 2)
|
|
|
|
graphics.endFill()
|
|
|
|
}
|
|
|
|
let r = this._boundsRect
|
|
|
|
if (r != null) {
|
|
|
|
graphics.lineStyle(20, color)
|
|
|
|
graphics.drawRect(r.x, r.y, r.width, r.height)
|
|
|
|
graphics.moveTo(r.x, r.y)
|
|
|
|
graphics.lineTo(r.x + r.width, r.y + r.height)
|
|
|
|
|
|
|
|
graphics.moveTo(r.x, r.y + r.height)
|
|
|
|
graphics.lineTo(r.x + r.width, r.y)
|
|
|
|
}
|
|
|
|
|
|
|
|
let p = this._centerPoint
|
|
|
|
if (p != null) {
|
|
|
|
graphics.drawCircle(p.x, p.y, 20)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tintTiles(quadTrees) {
|
|
|
|
for (let [url, tile] of this.available.entries()) {
|
|
|
|
if (!quadTrees.has(url)) tile.tint = 0xff0000
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
untintTiles() {
|
|
|
|
for (let [url, tile] of this.available.entries()) {
|
|
|
|
tile.tint = 0xffffff
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|