cache: update world map dumper for Areas

This commit is contained in:
Adam
2017-09-29 10:10:11 -04:00
parent a58dca9cd0
commit 9936fe352a
6 changed files with 145 additions and 69 deletions

View File

@@ -0,0 +1,70 @@
/*
* Copyright (c) 2017, Adam <Adam@sigterm.info>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package net.runelite.cache;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import net.runelite.cache.definitions.AreaDefinition;
import net.runelite.cache.definitions.loaders.AreaLoader;
import net.runelite.cache.fs.Archive;
import net.runelite.cache.fs.FSFile;
import net.runelite.cache.fs.Index;
import net.runelite.cache.fs.Store;
public class AreaManager
{
private final Store store;
private final Map<Integer, AreaDefinition> areas = new HashMap<>();
public AreaManager(Store store)
{
this.store = store;
}
public void load()
{
Index index = store.getIndex(IndexType.CONFIGS);
Archive archive = index.getArchive(ConfigType.AREA.getId());
for (FSFile file : archive.getFiles())
{
AreaLoader loader = new AreaLoader();
AreaDefinition area = loader.load(file.getContents(), file.getFileId());
areas.put(area.id, area);
}
}
public Collection<AreaDefinition> getAreas()
{
return Collections.unmodifiableCollection(areas.values());
}
public AreaDefinition getArea(int areaId)
{
return areas.get(areaId);
}
}

View File

@@ -24,8 +24,6 @@
*/ */
package net.runelite.cache; package net.runelite.cache;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.Multimap;
import java.awt.Color; import java.awt.Color;
import java.awt.Graphics2D; import java.awt.Graphics2D;
import java.awt.Image; import java.awt.Image;
@@ -35,6 +33,7 @@ import java.util.ArrayList;
import java.util.HashMap; import java.util.HashMap;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import net.runelite.cache.definitions.AreaDefinition;
import net.runelite.cache.definitions.ObjectDefinition; import net.runelite.cache.definitions.ObjectDefinition;
import net.runelite.cache.definitions.OverlayDefinition; import net.runelite.cache.definitions.OverlayDefinition;
import net.runelite.cache.definitions.SpriteDefinition; import net.runelite.cache.definitions.SpriteDefinition;
@@ -69,13 +68,13 @@ public class MapImageDumper
private final List<UnderlayDefinition> underlays = new ArrayList<>(); private final List<UnderlayDefinition> underlays = new ArrayList<>();
private final List<OverlayDefinition> overlays = new ArrayList<>(); private final List<OverlayDefinition> overlays = new ArrayList<>();
private final List<TextureDefinition> textures = new ArrayList<>(); private final List<TextureDefinition> textures = new ArrayList<>();
private final Multimap<Integer, SpriteDefinition> sprites = HashMultimap.create();
private final Map<SpriteDefinition, Integer> averageColors = new HashMap<>(); private final Map<SpriteDefinition, Integer> averageColors = new HashMap<>();
private final Map<Integer, Image> scaledMapIcons = new HashMap<>(); private final Map<Integer, Image> scaledMapIcons = new HashMap<>();
private final Map<Integer, ObjectDefinition> objects = new HashMap<>(); private final Map<Integer, ObjectDefinition> objects = new HashMap<>();
private final Map<Integer, Image> mapFunctions = new HashMap<>(); // quest, water, etc
private RegionLoader regionLoader; private RegionLoader regionLoader;
private final AreaManager areas;
private final SpriteManager sprites;
private boolean labelRegions; private boolean labelRegions;
private boolean outlineRegions; private boolean outlineRegions;
@@ -83,6 +82,8 @@ public class MapImageDumper
public MapImageDumper(Store store) public MapImageDumper(Store store)
{ {
this.store = store; this.store = store;
this.areas = new AreaManager(store);
this.sprites = new SpriteManager(store);
} }
public void load() throws IOException public void load() throws IOException
@@ -90,10 +91,12 @@ public class MapImageDumper
loadUnderlays(store); loadUnderlays(store);
loadOverlays(store); loadOverlays(store);
loadTextures(store); loadTextures(store);
loadSprites(store);
loadObjects(store); loadObjects(store);
loadRegions(store); loadRegions(store);
areas.load();
sprites.load();
loadSprites();
} }
public BufferedImage drawMap(int z) throws IOException public BufferedImage drawMap(int z) throws IOException
@@ -131,7 +134,7 @@ public class MapImageDumper
int drawBaseY = regionLoader.getHighestY().getBaseY() - baseY; int drawBaseY = regionLoader.getHighestY().getBaseY() - baseY;
Graphics2D graphics = image.createGraphics(); Graphics2D graphics = image.createGraphics();
drawObjects(graphics, region, z, drawBaseX, drawBaseY); drawObjects(graphics, region, z, drawBaseX, drawBaseY);
graphics.dispose(); graphics.dispose();
@@ -172,7 +175,7 @@ public class MapImageDumper
return image; return image;
} }
private void drawUnderlay(BufferedImage image, int z) private void drawUnderlay(BufferedImage image, int z)
{ {
// pass 1 // pass 1
@@ -199,7 +202,7 @@ public class MapImageDumper
int overlayId = region.getOverlayId(z, x, y) - 1; int overlayId = region.getOverlayId(z, x, y) - 1;
int underlayId = region.getUnderlayId(z, x, y) - 1; int underlayId = region.getUnderlayId(z, x, y) - 1;
int rgb = 0; int rgb = 0;
if (overlayId > -1) if (overlayId > -1)
{ {
OverlayDefinition overlay = findOverlay(overlayId); OverlayDefinition overlay = findOverlay(overlayId);
@@ -220,7 +223,7 @@ public class MapImageDumper
} }
} }
} }
private void drawOverlay(BufferedImage image, int z) private void drawOverlay(BufferedImage image, int z)
{ {
for (Region region : regionLoader.getRegions()) for (Region region : regionLoader.getRegions())
@@ -265,7 +268,7 @@ public class MapImageDumper
TextureDefinition texture = findTexture(overlay.getTexture()); TextureDefinition texture = findTexture(overlay.getTexture());
assert texture.getFileIds().length == 1; assert texture.getFileIds().length == 1;
SpriteDefinition sprite = findSprite(texture.getFileIds()[0], 0); SpriteDefinition sprite = sprites.findSprite(texture.getFileIds()[0], 0);
assert sprite != null; assert sprite != null;
rgb = averageColors.get(sprite); rgb = averageColors.get(sprite);
@@ -336,14 +339,22 @@ public class MapImageDumper
int drawX = drawBaseX + localX; int drawX = drawBaseX + localX;
int drawY = drawBaseY + (Region.Y - 1 - localY); int drawY = drawBaseY + (Region.Y - 1 - localY);
if (od.getMapIconID() != -1) if (od.getMapAreaId() != -1)
{ {
Image iconImage = mapFunctions.get(od.getMapIconID()); AreaDefinition area = areas.getArea(od.getMapAreaId());
assert area != null;
int spriteId = area.spriteId;
SpriteDefinition sprite = sprites.findSprite(spriteId, 0);
assert sprite != null;
BufferedImage iconImage = sprites.getSpriteImage(sprite);
graphics.drawImage(iconImage, drawX * MAP_SCALE, drawY * MAP_SCALE, null); graphics.drawImage(iconImage, drawX * MAP_SCALE, drawY * MAP_SCALE, null);
} }
} }
} }
private void loadRegions(Store store) throws IOException private void loadRegions(Store store) throws IOException
{ {
regionLoader = new RegionLoader(store); regionLoader = new RegionLoader(store);
@@ -435,11 +446,10 @@ public class MapImageDumper
return null; return null;
} }
private void loadSprites(Store store) private void loadSprites()
{ {
Index index = store.getIndex(IndexType.SPRITES); Index index = store.getIndex(IndexType.SPRITES);
final int mapsceneHash = Djb2.hash("mapscene"); final int mapsceneHash = Djb2.hash("mapscene");
final int mapfunctionHash = Djb2.hash("mapfunction");
for (Archive a : index.getArchives()) for (Archive a : index.getArchives())
{ {
@@ -460,8 +470,6 @@ public class MapImageDumper
continue; continue;
} }
this.sprites.put(sprite.getId(), sprite);
averageColors.put(sprite, getAverageColor(sprite.getPixels())); averageColors.put(sprite, getAverageColor(sprite.getPixels()));
if (a.getNameHash() == mapsceneHash) if (a.getNameHash() == mapsceneHash)
@@ -475,30 +483,10 @@ public class MapImageDumper
assert scaledMapIcons.containsKey(sprite.getFrame()) == false; assert scaledMapIcons.containsKey(sprite.getFrame()) == false;
scaledMapIcons.put(sprite.getFrame(), scaledImage); scaledMapIcons.put(sprite.getFrame(), scaledImage);
} }
else if (a.getNameHash() == mapfunctionHash)
{
BufferedImage spriteImage = new BufferedImage(sprite.getWidth(), sprite.getHeight(), BufferedImage.TYPE_INT_ARGB);
spriteImage.setRGB(0, 0, sprite.getWidth(), sprite.getHeight(), sprite.getPixels(), 0, sprite.getWidth());
assert mapFunctions.containsKey(sprite.getFrame()) == false;
mapFunctions.put(sprite.getFrame(), spriteImage);
}
} }
} }
} }
private SpriteDefinition findSprite(int id, int frame)
{
for (SpriteDefinition def : sprites.get(id))
{
if (def.getFrame() == frame)
{
return def;
}
}
return null;
}
private int getAverageColor(int[] pixels) private int getAverageColor(int[] pixels)
{ {
int redTotal = 0; int redTotal = 0;

View File

@@ -24,10 +24,13 @@
*/ */
package net.runelite.cache; package net.runelite.cache;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.Multimap;
import java.awt.image.BufferedImage;
import java.io.File; import java.io.File;
import java.io.IOException; import java.io.IOException;
import java.util.ArrayList; import java.util.Collection;
import java.util.Arrays; import java.util.Collections;
import java.util.List; import java.util.List;
import net.runelite.cache.definitions.SpriteDefinition; import net.runelite.cache.definitions.SpriteDefinition;
import net.runelite.cache.definitions.exporters.SpriteExporter; import net.runelite.cache.definitions.exporters.SpriteExporter;
@@ -40,7 +43,7 @@ import net.runelite.cache.fs.Store;
public class SpriteManager public class SpriteManager
{ {
private final Store store; private final Store store;
private final List<SpriteDefinition> sprites = new ArrayList<>(); private final Multimap<Integer, SpriteDefinition> sprites = HashMultimap.create();
public SpriteManager(Store store) public SpriteManager(Store store)
{ {
@@ -61,20 +64,42 @@ public class SpriteManager
byte[] contents = file.getContents(); byte[] contents = file.getContents();
SpriteLoader loader = new SpriteLoader(); SpriteLoader loader = new SpriteLoader();
SpriteDefinition[] sprites = loader.load(a.getArchiveId(), contents); SpriteDefinition[] defs = loader.load(a.getArchiveId(), contents);
this.sprites.addAll(Arrays.asList(sprites)); for (SpriteDefinition sprite : defs)
{
sprites.put(sprite.getId(), sprite);
}
} }
} }
public List<SpriteDefinition> getSprites() public Collection<SpriteDefinition> getSprites()
{ {
return sprites; return Collections.unmodifiableCollection(sprites.values());
}
public SpriteDefinition findSprite(int spriteId, int frameId)
{
for (SpriteDefinition sprite : sprites.get(spriteId))
{
if (sprite.getFrame() == frameId)
{
return sprite;
}
}
return null;
}
public BufferedImage getSpriteImage(SpriteDefinition sprite)
{
BufferedImage image = new BufferedImage(sprite.getWidth(), sprite.getHeight(), BufferedImage.TYPE_INT_ARGB);
image.setRGB(0, 0, sprite.getWidth(), sprite.getHeight(), sprite.getPixels(), 0, sprite.getWidth());
return image;
} }
public void export(File outDir) throws IOException public void export(File outDir) throws IOException
{ {
for (SpriteDefinition sprite : sprites) for (SpriteDefinition sprite : sprites.values())
{ {
// I don't know why this happens // I don't know why this happens
if (sprite.getHeight() <= 0 || sprite.getWidth() <= 0) if (sprite.getHeight() <= 0 || sprite.getWidth() <= 0)

View File

@@ -37,7 +37,7 @@ public class ObjectDefinition
private int[] objectModels; private int[] objectModels;
private int[] objectTypes; private int[] objectTypes;
private short[] recolorToFind; private short[] recolorToFind;
private int mapIconID = -1; private int mapAreaId = -1;
private short[] textureToReplace; private short[] textureToReplace;
private int sizeX = 1; private int sizeX = 1;
private int sizeY = 1; private int sizeY = 1;
@@ -154,14 +154,14 @@ public class ObjectDefinition
this.recolorToFind = recolorToFind; this.recolorToFind = recolorToFind;
} }
public int getMapIconID() public int getMapAreaId()
{ {
return mapIconID; return mapAreaId;
} }
public void setMapIconID(int mapIconID) public void setMapAreaId(int mapAreaId)
{ {
this.mapIconID = mapIconID; this.mapAreaId = mapAreaId;
} }
public short[] getTextureToReplace() public short[] getTextureToReplace()

View File

@@ -59,7 +59,7 @@ public class ObjectLoader
private void processOp(int opcode, ObjectDefinition def, InputStream is) private void processOp(int opcode, ObjectDefinition def, InputStream is)
{ {
if (1 == opcode) if (opcode == 1)
{ {
int length = is.readUnsignedByte(); int length = is.readUnsignedByte();
if (length > 0) if (length > 0)
@@ -101,7 +101,7 @@ public class ObjectLoader
{ {
def.setSizeX(is.readUnsignedByte()); def.setSizeX(is.readUnsignedByte());
} }
else if (15 == opcode) else if (opcode == 15)
{ {
def.setSizeY(is.readUnsignedByte()); def.setSizeY(is.readUnsignedByte());
} }
@@ -122,7 +122,7 @@ public class ObjectLoader
{ {
def.setAnInt2105(0); def.setAnInt2105(0);
} }
else if (22 == opcode) else if (opcode == 22)
{ {
def.setNonFlatShading(false); def.setNonFlatShading(false);
} }
@@ -130,7 +130,7 @@ public class ObjectLoader
{ {
def.setaBool2111(true); def.setaBool2111(true);
} }
else if (24 == opcode) else if (opcode == 24)
{ {
def.setAnimationID(is.readUnsignedShort()); def.setAnimationID(is.readUnsignedShort());
if (def.getAnimationID() == 0xFFFF) if (def.getAnimationID() == 0xFFFF)
@@ -193,7 +193,7 @@ public class ObjectLoader
def.setRetextureToFind(retextureToFind); def.setRetextureToFind(retextureToFind);
def.setTextureToReplace(textureToReplace); def.setTextureToReplace(textureToReplace);
} }
else if (62 == opcode) else if (opcode == 62)
{ {
def.setIsRotated(true); def.setIsRotated(true);
} }
@@ -209,7 +209,7 @@ public class ObjectLoader
{ {
def.setModelSizeHeight(is.readUnsignedShort()); def.setModelSizeHeight(is.readUnsignedShort());
} }
else if (67 == opcode) else if (opcode == 67)
{ {
def.setModelSizeY(is.readUnsignedShort()); def.setModelSizeY(is.readUnsignedShort());
} }
@@ -221,7 +221,7 @@ public class ObjectLoader
{ {
is.readByte(); is.readByte();
} }
else if (70 == opcode) else if (opcode == 70)
{ {
def.setOffsetX(is.readUnsignedShort()); def.setOffsetX(is.readUnsignedShort());
} }
@@ -303,7 +303,7 @@ public class ObjectLoader
} }
else if (opcode == 82) else if (opcode == 82)
{ {
def.setMapIconID(is.readUnsignedShort()); def.setMapAreaId(is.readUnsignedShort());
} }
else if (opcode == 92) else if (opcode == 92)
{ {

View File

@@ -31,10 +31,6 @@ import java.io.File;
import java.io.IOException; import java.io.IOException;
import java.nio.charset.Charset; import java.nio.charset.Charset;
import net.runelite.cache.definitions.AreaDefinition; import net.runelite.cache.definitions.AreaDefinition;
import net.runelite.cache.definitions.loaders.AreaLoader;
import net.runelite.cache.fs.Archive;
import net.runelite.cache.fs.FSFile;
import net.runelite.cache.fs.Index;
import net.runelite.cache.fs.Store; import net.runelite.cache.fs.Store;
import org.junit.Rule; import org.junit.Rule;
import org.junit.Test; import org.junit.Test;
@@ -63,15 +59,12 @@ public class AreaDumper
{ {
store.load(); store.load();
Index index = store.getIndex(IndexType.CONFIGS); AreaManager areaManager = new AreaManager(store);
Archive archive = index.getArchive(ConfigType.AREA.getId()); areaManager.load();
for (FSFile file : archive.getFiles()) for (AreaDefinition area : areaManager.getAreas())
{ {
AreaLoader loader = new AreaLoader(); Files.write(gson.toJson(area), new File(outDir, area.id + ".json"), Charset.defaultCharset());
AreaDefinition area = loader.load(file.getContents(), file.getFileId());
Files.write(gson.toJson(area), new File(outDir, file.getFileId() + ".json"), Charset.defaultCharset());
++count; ++count;
} }
} }