package com.vitco.export.generic.container; import com.vitco.core.data.Data; import com.vitco.core.data.container.Voxel; import com.vitco.util.graphic.G2DUtil; import com.vitco.util.graphic.ImageComparator; import com.vitco.util.graphic.TextureTools; import com.vitco.util.misc.IntegerTools; import gnu.trove.iterator.TIntObjectIterator; import gnu.trove.map.hash.TIntObjectHashMap; import gnu.trove.procedure.TIntObjectProcedure; import javax.swing.*; import java.awt.image.BufferedImage; import java.util.ArrayList; /** * Represents a texture that belongs to a triangle. */ public class TriTexture { // interpolation value private static final float interp = 0.001f; // --------------- // true if this texture has a corresponding triangle private final boolean hasTriangle; // reference to the the uvs private final TexTriUV[] texTriUVs = new TexTriUV[3]; // reference to the uv points private final double[][] uvPoints = new double[3][2]; // holds the pixels in this triangle, the format is (x, y, color) // Note: Not final since this needs to be nullable private TIntObjectHashMap<int[]> pixels = new TIntObjectHashMap<int[]>(); // size of this texture image public final int width; public final int height; // reference to texture manager private final TriTextureManager textureManager; // used for pixel comparison // Note: Not final since this needs to be nullable private ImageComparator imageComparator; // --------------- // -- parent texture (i.e. this texture is part of the parent texture) private TriTexture parentTexture = null; // left top corner and orientation of this image in its parent image private final int[] leftTop = new int[] {0,0}; private int orientationFlag = 0; // false if the uv of this texture is outdated private TriTexture lastTopTexture = null; // ################################# // get a random rgb from this texture public final int getSampleRGB() { TIntObjectIterator<int[]> it = pixels.iterator(); it.advance(); return it.value()[2]; } // set parent texture for this texture // Note: A parent texture is a texture that contains this texture public final void setParentTexture(TriTexture parentTexture, int[] leftTop, int orientationFlag) { this.parentTexture = parentTexture; // store the uv translation values for this texture to parent if (leftTop != null) { this.leftTop[0] = leftTop[0]; this.leftTop[1] = leftTop[1]; } // Indicates the different orientation changes // 0 - original, 1 - rotated x 1, 2 - rotated x 2, 3 - rotated x 3, // 4 - flipped, 5 - flipped & rotated x 1, 6 - flipped & rotated x 2, 7 - flipped & rotated x 3 // Note: Rotation is clockwise this.orientationFlag = orientationFlag; // free internal memory that is no longer needed pixels = null; imageComparator = null; } // get identifier of this texture // (only assigned when this function is accessed to avoid high ids) public final int getId() { // return parent id if (parentTexture != null) { return parentTexture.getId(); } // else return this id return textureManager.getId(this); } // obtain the pixel count public final int getPixelCount() { return imageComparator.pixelCount; } // obtain the area this texture uses // (this is usually larger than the pixel count) public final int getArea() { return width * height; } // obtain the jaccard distance public final float jaccard(TriTexture other) { return this.imageComparator.jaccard(other.imageComparator); } // retrieve image representation of this texture public final BufferedImage getImage() { // return parent texture image if (parentTexture != null) { return parentTexture.getImage(); } // else compute the image for this texture BufferedImage result = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB); for (TIntObjectIterator<int[]> it = pixels.iterator(); it.hasNext();) { it.advance(); int[] pixel = it.value(); assert pixel[0] < width; assert pixel[1] < height; result.setRGB(pixel[0], pixel[1], pixel[2]); } return result; } // recursively retrieve parent texture // (or the texture itself if no parent is set) public final TriTexture getTopTexture() { if (parentTexture == null) { return this; } else { return parentTexture.getTopTexture(); } } // check if the uv mapping of this texture is valid public final boolean hasValidUV() { // the uv mapping is likely to have changed when the parent texture changed return lastTopTexture == getTopTexture() || !hasTriangle; } // return true if this has a parent texture // (otherwise this is a "top" texture) public final boolean hasParent() { return parentTexture != null; } // make the passed texture a child (if possible) // return true on success public final boolean makeChild(TriTexture child, int pos[]) { // -- can only make child if it has no parent if (child.hasParent()) { return false; } // -- check that we don't create infinite recursion if (this.getTopTexture() == child) { return false; } // -- check for containment position pos = pos == null ? imageComparator.getPosition(child.imageComparator, null) : pos; if (pos != null) { child.setParentTexture(this, new int[] {pos[0], pos[1]}, pos[2]); return true; } // -- no containment position found return false; } // ################################ // helper function - swap x and y values private void swap(double[][] array) { double tmp = array[0][0]; array[0][0] = array[0][1]; array[0][1] = tmp; tmp = array[1][0]; array[1][0] = array[1][1]; array[1][1] = tmp; tmp = array[2][0]; array[2][0] = array[2][1]; array[2][1] = tmp; } // make sure the uv of this texture is validated // against the assigned triangle @SuppressWarnings("SuspiciousNameCombination") public final void validateUVMapping() { if (!hasValidUV()) { // compute uvs (not adjusted yet) double[][] uvs = new double[][]{ new double[]{uvPoints[0][0], uvPoints[0][1]}, new double[]{uvPoints[1][0], uvPoints[1][1]}, new double[]{uvPoints[2][0], uvPoints[2][1]} }; // compute position and size of top parent TriTexture list = this; int x = 0; int y = 0; double width = this.width; double height = this.height; boolean swapped = false; int tmp; while (list != null) { switch (list.orientationFlag) { case 0: // do nothing break; case 4: uvs[0][0] = 1-uvs[0][0]; uvs[1][0] = 1-uvs[1][0]; uvs[2][0] = 1-uvs[2][0]; x = list.width - x - (swapped ? this.height : this.width); break; case 2: uvs[0][0] = 1-uvs[0][0]; uvs[1][0] = 1-uvs[1][0]; uvs[2][0] = 1-uvs[2][0]; uvs[0][1] = 1-uvs[0][1]; uvs[1][1] = 1-uvs[1][1]; uvs[2][1] = 1-uvs[2][1]; x = list.width - x - (swapped ? this.height : this.width); y = list.height - y - (swapped ? this.width : this.height); break; case 6: uvs[0][1] = 1-uvs[0][1]; uvs[1][1] = 1-uvs[1][1]; uvs[2][1] = 1-uvs[2][1]; y = list.height - y - (swapped ? this.width : this.height); break; // ========== case 7: swapped = !swapped; swap(uvs); tmp = x; x = y; y = tmp; break; case 1: swapped = !swapped; swap(uvs); uvs[0][0] = 1-uvs[0][0]; uvs[1][0] = 1-uvs[1][0]; uvs[2][0] = 1-uvs[2][0]; tmp = x; x = list.height - y - (swapped ? this.height : this.width); y = tmp; break; case 3: swapped = !swapped; swap(uvs); uvs[0][1] = 1-uvs[0][1]; uvs[1][1] = 1-uvs[1][1]; uvs[2][1] = 1-uvs[2][1]; tmp = x; x = y; y = list.width - tmp - (swapped ? this.width : this.height); break; default: // case 5 swapped = !swapped; swap(uvs); uvs[0][0] = 1-uvs[0][0]; uvs[1][0] = 1-uvs[1][0]; uvs[2][0] = 1-uvs[2][0]; uvs[0][1] = 1-uvs[0][1]; uvs[1][1] = 1-uvs[1][1]; uvs[2][1] = 1-uvs[2][1]; tmp = x; x = list.height - y - (swapped ? this.height : this.width); y = list.width - tmp - (swapped ? this.width : this.height); break; } // alter left top according to this parent x += list.leftTop[0]; y += list.leftTop[1]; // set to top width and height (maximum) width = list.width; height = list.height; // get next parent list = list.parentTexture; } // ----------- // -- interpolation // compute the center of the uv triangle double[] center = new double[]{ (uvs[0][0] + uvs[1][0] + uvs[2][0]) / 3, (uvs[0][1] + uvs[1][1] + uvs[2][1]) / 3 }; // compute the offsets (the direction we need to interpolate in) double[][] offsets = new double[][]{ new double[]{center[0] - uvs[0][0], center[1] - uvs[0][1]}, new double[]{center[0] - uvs[1][0], center[1] - uvs[1][1]}, new double[]{center[0] - uvs[2][0], center[1] - uvs[2][1]} }; // normalize these offsets using the minimum // Note: The minimum is used to ensure that sides of the uv // triangle that were parallel stay parallel after interpolation double dist = Math.min( Math.min( Math.sqrt(Math.pow(offsets[0][0], 2) + Math.pow(offsets[0][1], 2)), Math.sqrt(Math.pow(offsets[1][0], 2) + Math.pow(offsets[1][1], 2)) ), Math.sqrt(Math.pow(offsets[2][0], 2) + Math.pow(offsets[2][1], 2)) ); offsets[0][0] /= dist * width; offsets[0][1] /= dist * height; offsets[1][0] /= dist * width; offsets[1][1] /= dist * height; offsets[2][0] /= dist * width; offsets[2][1] /= dist * height; // ----------- // scale, shift and apply interpolation double offsetX = x/width; double offsetY = y/height; double scaleX = (swapped ? this.height : this.width)/width; double scaleY = (swapped ? this.width : this.height)/height; uvs[0][0] = offsetX + uvs[0][0] * scaleX + offsets[0][0] * interp; uvs[1][0] = offsetX + uvs[1][0] * scaleX + offsets[1][0] * interp; uvs[2][0] = offsetX + uvs[2][0] * scaleX + offsets[2][0] * interp; uvs[0][1] = offsetY + uvs[0][1] * scaleY + offsets[0][1] * interp; uvs[1][1] = offsetY + uvs[1][1] * scaleY + offsets[1][1] * interp; uvs[2][1] = offsetY + uvs[2][1] * scaleY + offsets[2][1] * interp; // set the uv positions texTriUVs[0].set((float)uvs[0][0], (float)(1 - uvs[0][1])); texTriUVs[1].set((float)uvs[1][0], (float)(1 - uvs[1][1])); texTriUVs[2].set((float)uvs[2][0], (float)(1 - uvs[2][1])); // validate - the uv representation is ok now for this top texture lastTopTexture = getTopTexture(); } } // alternative constructor that forces power of two texture public TriTexture(TriTexture tex, TriTextureManager textureManager) { // set final variables this.textureManager = textureManager; // has no corresponding triangle this.hasTriangle = false; int powOfTwoDim = (int)Math.pow(2, Math.ceil(Math.log(Math.max(tex.width, tex.height) + 2)/Math.log(2))); this.width = powOfTwoDim; this.height = powOfTwoDim; pixels.putAll(tex.pixels); imageComparator = new ImageComparator(pixels.valueCollection()); // make child this.makeChild(tex, new int[] {0, 0, 0}); assert tex.hasParent(); } // action to notify listener of progression public abstract static class TickAction { abstract void onTick(int current, int target); } // alternative constructor that merges multiple textures public TriTexture(ArrayList<TriTexture> textures, TickAction tickAction, TriTextureManager textureManager) { int texCount = textures.size(); assert texCount >= 2; for (TriTexture tex : textures) { assert !tex.hasParent(); } // set final variables this.textureManager = textureManager; // has no corresponding triangle this.hasTriangle = false; TriTexture main = textures.remove(0); this.pixels.putAll(main.pixels); int width = main.width; int height = main.height; this.makeChild(main, new int[] {0, 0, 0}); assert main.hasParent(); while (!textures.isEmpty()) { tickAction.onTick(texCount - textures.size(), texCount); TriTexture sec = textures.remove(0); int[] merge = null; for (int y = 0; y <= height && merge == null; y++) { for (int x = 0; x <= width && merge == null; x++) { // ensure the generate texture is square if (x + sec.width > height && height < width && x > 0) { break; } boolean works = true; // check if it fits for (TIntObjectIterator<int[]> it = sec.pixels.iterator(); it.hasNext() && works; ) { it.advance(); int[] val = it.value(); works = !pixels.containsKey(IntegerTools.makeInt(x + val[0], y + val[1])); } if (works) { merge = new int[]{x, y}; } } } assert merge != null; // merge pixels into this TriTexture for (TIntObjectIterator<int[]> it = sec.pixels.iterator(); it.hasNext(); ) { it.advance(); int[] val = it.value(); this.pixels.put( IntegerTools.makeInt(merge[0] + val[0], merge[1] + val[1]), new int[]{merge[0] + val[0], merge[1] + val[1], val[2]} ); } width = Math.max(width, merge[0] + sec.width); height = Math.max(height, merge[1] + sec.height); // ----------- // make child this.makeChild(sec, new int[] {merge[0], merge[1], 0}); assert sec.hasParent(); } // set the image comparator imageComparator = new ImageComparator(this.pixels.valueCollection()); this.width = width; this.height = height; } // alternative constructor that merges two textures public TriTexture(TriTexture one, TriTexture two, TriTextureManager textureManager) { assert !one.hasParent(); assert !two.hasParent(); // set final variables this.textureManager = textureManager; // has no corresponding triangle this.hasTriangle = false; // obtain merge position and orientation int[] mergePos = ImageComparator.getMergePoint(one.imageComparator, two.imageComparator); // compute pixels int minX = Math.min(0, mergePos[0]); int minY = Math.min(0, mergePos[1]); int maxX = Math.max(0, mergePos[0]); int maxY = Math.max(0, mergePos[1]); for (TIntObjectIterator<int[]> it = one.pixels.iterator(); it.hasNext();) { it.advance(); int[] pixel = it.value(); int x = pixel[0] - minX; int y = pixel[1] - minY; pixels.put(IntegerTools.makeInt(x, y), new int[] {x,y,pixel[2]}); } // rotate second image according to result for (TIntObjectIterator<int[]> it = two.pixels.iterator(); it.hasNext();) { it.advance(); int[] pixel = it.value(); int x; int y; switch (mergePos[2]) { case 1: // 1 : check for "rotation 1" (1) x = (two.height - 1 - pixel[1]); y = pixel[0]; break; case 3: // 3 : check for "rotation 3" (3) x = pixel[1]; y = (two.width - 1 - pixel[0]); break; case 5: // 5 : check for "flipped and rotation 1" (5) x = (two.height - 1 - pixel[1]); y = (two.width - 1 - pixel[0]); break; case 7: // 7 : check for "flipped and rotation 3" (7) x = pixel[1]; y = pixel[0]; break; case 2: // 2 : check for "twice rotated" (2) x = (two.width - 1 - pixel[0]); y = (two.height - 1 - pixel[1]); break; case 4: // 4 : check for "flipped" (4) x = (two.width - 1 - pixel[0]); y = pixel[1]; break; case 6: // 6 : check for "flipped and twice rotated" (6) x = pixel[0]; y = (two.height - 1 - pixel[1]); break; default: // 0 : check for "default orientation" (0) x = pixel[0]; y = pixel[1]; break; } pixels.put(IntegerTools.makeInt(x + maxX, y + maxY), new int[] {x + maxX,y + maxY,pixel[2]}); } // set the image comparator imageComparator = new ImageComparator(pixels.valueCollection()); // compute new dimensions (depending on whether the second // texture was rotated or not) if (mergePos[2]%2 == 0) { this.width = Math.max(two.width + maxX, one.width - minX); this.height = Math.max(two.height + maxY, one.height - minY); } else { this.width = Math.max(two.height + maxX, one.width - minX); this.height = Math.max(two.width + maxY, one.height - minY); } // ----------- // make children this.makeChild(one, new int[] {-minX, -minY, 0}); this.makeChild(two, new int[] {maxX, maxY, mergePos[2]}); assert one.hasParent(); assert two.hasParent(); } // constructor public TriTexture( TexTriUV uv1, int xf1, int yf1, TexTriUV uv2, int xf2, int yf2, TexTriUV uv3, int xf3, int yf3, final int side, int depth, boolean usePadding, TexTriangle texTri, final Data data, TriTextureManager textureManager, boolean exportTexturedVoxels, boolean useSkewedUvs ) { // store variables internally uvPoints[0][0] = xf1; uvPoints[0][1] = yf1; uvPoints[1][0] = xf2; uvPoints[1][1] = yf2; uvPoints[2][0] = xf3; uvPoints[2][1] = yf3; texTriUVs[0] = uv1; texTriUVs[1] = uv2; texTriUVs[2] = uv3; // -- normalize uv points // compute top left point and range double minCornerX = Math.min(Math.min(uvPoints[0][0], uvPoints[1][0]), uvPoints[2][0]); double rangeX = Math.max(Math.max(uvPoints[0][0], uvPoints[1][0]), uvPoints[2][0]) - minCornerX; double minCornerY = Math.min(Math.min(uvPoints[0][1], uvPoints[1][1]), uvPoints[2][1]); double rangeY = Math.max(Math.max(uvPoints[0][1], uvPoints[1][1]), uvPoints[2][1]) - minCornerY; // compute uvs (not shifted yet) uvPoints[0][0] = (uvPoints[0][0] - minCornerX) / rangeX; uvPoints[1][0] = (uvPoints[1][0] - minCornerX) / rangeX; uvPoints[2][0] = (uvPoints[2][0] - minCornerX) / rangeX; uvPoints[0][1] = (uvPoints[0][1] - minCornerY) / rangeY; uvPoints[1][1] = (uvPoints[1][1] - minCornerY) / rangeY; uvPoints[2][1] = (uvPoints[2][1] - minCornerY) / rangeY; // has a corresponding triangle this.hasTriangle = true; // store texture manager reference this.textureManager = textureManager; // compute the voxels that are inside this triangle // (so we're not using any "extra" pixels in the buffered image) int[][] points = G2DUtil.getTriangleGridIntersection( xf1, yf1, xf2, yf2, xf3, yf3 ); // get min/max pixel values (this is different from UV!) int minX = Integer.MAX_VALUE; int maxX = Integer.MIN_VALUE; int minY = Integer.MAX_VALUE; int maxY = Integer.MIN_VALUE; for (int[] p : points) { minX = Math.min(p[0], minX); maxX = Math.max(p[0], maxX); minY = Math.min(p[1], minY); maxY = Math.max(p[1], maxY); } // compute the image size for this texture image int width = maxX - minX + 1; int height = maxY - minY + 1; // get orientation final int axis = texTri.getOrientation()/2; final TIntObjectHashMap<Voxel> texels = new TIntObjectHashMap<Voxel>(); // fetch colors for (int[] point : points) { // set the position (for this color) int x = point[0] - minX; int y = point[1] - minY; int p = IntegerTools.makeInt(x, y); // get the pixel color Voxel voxel = data.searchVoxel(new int[] { axis == 0 ? depth : point[0], axis == 1 ? depth : (axis == 0 ? point[0] : point[1]), axis == 2 ? depth : point[1], }, false); assert voxel != null; if (exportTexturedVoxels && voxel.getTexture() != null) { texels.put(p, voxel); } // add the pixel pixels.put(p, new int[] {x, y, voxel.getColor().getRGB()}); } if (exportTexturedVoxels && texels.size() > 0) { int[][] enlargedPoints = G2DUtil.getTriangleGridIntersection( xf1 * 32, yf1 * 32, xf2 * 32, yf2 * 32, xf3 * 32, yf3 * 32 ); final TIntObjectHashMap<int[]> remapped = new TIntObjectHashMap<int[]>(); // enlarge pixel and fill with textures when appropriate pixels.forEachEntry(new TIntObjectProcedure<int[]>() { @Override public boolean execute(int p, int[] info) { Voxel voxel = texels.get(p); BufferedImage tex = null; int rotation = 0; boolean flip = false; int[] pos = new int[] {0, 0}; if (voxel != null) { //noinspection ConstantConditions ImageIcon imageIcon = data.getTexture(voxel.getTexture()[side]); tex = new BufferedImage(imageIcon.getIconWidth(), imageIcon.getIconHeight(), BufferedImage.TYPE_INT_ARGB); tex.getGraphics().drawImage(imageIcon.getImage(), 0, 0, null); //noinspection ConstantConditions rotation = voxel.getRotation() != null ? voxel.getRotation()[side] : 0; //noinspection ConstantConditions flip = voxel.getFlip() != null && voxel.getFlip()[side]; } int offset; switch (side) { case 0: offset = flip ? 1 : 7; break; case 1: offset = flip ? 7 : 1; break; case 2: offset = flip ? 6 : 2; break; case 3: offset = flip ? 2 : 6; break; case 4: offset = flip ? 0 : 4; break; default: offset = flip ? 4 : 0; break; } offset = (((offset % 4 + (flip ? rotation : 4 - rotation)) % 4) + (offset / 4) * 4) % 8; for (int x = 0; x < 32; x++) { for (int y = 0; y < 32; y++) { if (tex != null) { switch (offset) { // 0 - original, 1 - rotated x 1, 2 - rotated x 2, 3 - rotated x 3, // 4 - flipped, 5 - flipped & rotated x 1, 6 - flipped & rotated x 2, 7 - flipped & rotated x 3 case 0: pos[0] = x; pos[1] = y; break; case 4: pos[0] = 31 - x; pos[1] = y; break; case 2: pos[0] = 31 - x; pos[1] = 31 - y; break; case 6: pos[0] = x; pos[1] = 31 - y; break; case 7: pos[0] = y; pos[1] = x; break; case 1: pos[0] = 31 - y; pos[1] = x; break; case 3: pos[0] = y; pos[1] = 31 - x; break; default: pos[0] = 31 - y; pos[1] = 31 - x; break; // case 5 } } remapped.put( IntegerTools.makeInt(info[0] * 32 + x, info[1] * 32 + y), new int[] {info[0] * 32 + x, info[1] * 32 + y, tex == null ? info[2] : tex.getRGB(pos[0], pos[1])} ); } } return true; } }); pixels.clear(); for (int[] point : enlargedPoints) { int x = point[0] - minX * 32; int y = point[1] - minY * 32; int pos = IntegerTools.makeInt(x, y); assert remapped.containsKey(pos); pixels.put(pos, remapped.get(pos)); } width = width * 32; height = height * 32; } // might now change depending on options int[] newSize = new int[] {width, height}; if (useSkewedUvs) { // compress textures (scale if this can be done loss-less) // Note: Doing this several times should not make any sense newSize = compress(newSize[0], newSize[1], pixels, uvPoints); } // do texture padding (if enabled) if (usePadding) { newSize = pad(newSize[0], newSize[1], pixels, uvPoints); } // set the image comparator imageComparator = new ImageComparator(pixels.valueCollection()); if (useSkewedUvs) { // overwrite uv to prevent unnecessary unique uv coordinates. // Note: this enables better compression for COLLADA if (imageComparator.pixelCount == 1) { uvPoints[0][0] = 0; uvPoints[0][1] = 0; uvPoints[1][0] = 1; uvPoints[1][1] = 0; uvPoints[2][0] = 0; uvPoints[2][1] = 1; } else if (imageComparator.colorCount == 1 && usePadding) { uvPoints[0][0] = 1 / 3f; uvPoints[0][1] = 1 / 3f; uvPoints[1][0] = 2 / 3f; uvPoints[1][1] = 1 / 3f; uvPoints[2][0] = 1 / 3f; uvPoints[2][1] = 2 / 3f; } } // finalize width and height this.width = newSize[0]; this.height = newSize[1]; } // pad textures "surround with same color pixels" protected static int[] pad(int width, int height, TIntObjectHashMap<int[]> pixels, double[][] uvPoints) { int[] newSize = new int[] {width + 2, height + 2}; // -- translate pixels TIntObjectHashMap<int[]> translatedPixels = new TIntObjectHashMap<int[]>(); for (TIntObjectIterator<int[]> it = pixels.iterator(); it.hasNext();) { it.advance(); int[] pixel = it.value(); int point = IntegerTools.makeInt(pixel[0] + 1, pixel[1] + 1); translatedPixels.put(point, new int[] {pixel[0] + 1, pixel[1] + 1, pixel[2]}); } pixels.clear(); pixels.putAll(translatedPixels); // -- fix horizontal padding for (TIntObjectIterator<int[]> it = pixels.iterator(); it.hasNext();) { it.advance(); int[] pixel = it.value(); int point = IntegerTools.makeInt(pixel[0] + 1, pixel[1]); if (!pixels.containsKey(point)) { translatedPixels.put(point, new int[] {pixel[0] + 1, pixel[1], pixel[2]}); } point = IntegerTools.makeInt(pixel[0] - 1, pixel[1]); if (!pixels.containsKey(point)) { translatedPixels.put(point, new int[] {pixel[0] - 1, pixel[1], pixel[2]}); } } pixels.clear(); pixels.putAll(translatedPixels); // -- fix vertical padding for (TIntObjectIterator<int[]> it = pixels.iterator(); it.hasNext();) { it.advance(); int[] pixel = it.value(); int point = IntegerTools.makeInt(pixel[0], pixel[1] + 1); if (!pixels.containsKey(point)) { translatedPixels.put(point, new int[] {pixel[0], pixel[1] + 1, pixel[2]}); } point = IntegerTools.makeInt(pixel[0], pixel[1] - 1); if (!pixels.containsKey(point)) { translatedPixels.put(point, new int[] {pixel[0], pixel[1] - 1, pixel[2]}); } } pixels.clear(); pixels.putAll(translatedPixels); // -- fix uv points uvPoints[0][0] = (uvPoints[0][0] * width + 1) / newSize[0]; uvPoints[0][1] = (uvPoints[0][1] * height + 1) / newSize[1]; uvPoints[1][0] = (uvPoints[1][0] * width + 1) / newSize[0]; uvPoints[1][1] = (uvPoints[1][1] * height + 1) / newSize[1]; uvPoints[2][0] = (uvPoints[2][0] * width + 1) / newSize[0]; uvPoints[2][1] = (uvPoints[2][1] * height + 1) / newSize[1]; return newSize; } // prune unnecessary and add missing pixels from this texture // Note: This should only be needed after compression changed the image and uvs private static int[] repairPixel(int width, int height, TIntObjectHashMap<int[]> pixels, double[][] uvPoints, boolean useHeight) { // will store the minimum and maximum pixel values int minX = Integer.MAX_VALUE; int maxX = Integer.MIN_VALUE; int minY = Integer.MAX_VALUE; int maxY = Integer.MIN_VALUE; // calculate all required points // Note: some old pixels might not be necessary, and there might be // some pixels that are not set as they were shifted by the distortion // (we need to fetch those) ArrayList<int[]> newPixels = new ArrayList<int[]>(); int[][] points = G2DUtil.getTriangleGridIntersection( uvPoints[0][0] * width, uvPoints[0][1] * height, uvPoints[1][0] * width, uvPoints[1][1] * height, uvPoints[2][0] * width, uvPoints[2][1] * height ); // loop over all required pixel positions for (int[] pixel : points) { // verify position (this is just a precaution and shouldn't be necessary) if (pixel[0] > -1 && pixel[1] > -1 && pixel[0] < width && pixel[1] < height) { int point = IntegerTools.makeInt(pixel[0], pixel[1]); int[] data = pixels.get(point); // check if the pixel exists if (data == null) { // -- the pixel does not exists // find other point if (useHeight) { // -- search vertically boolean found = false; // search down for (int y = pixel[1] + 1; y < height; y++) { point = IntegerTools.makeInt(pixel[0], y); data = pixels.get(point); if (data != null) { newPixels.add(new int[]{pixel[0], pixel[1], data[2]}); found = true; break; } } if (!found) { // search up for (int y = pixel[1] - 1; y > -1; y--) { point = IntegerTools.makeInt(pixel[0], y); data = pixels.get(point); if (data != null) { newPixels.add(new int[]{pixel[0], pixel[1], data[2]}); found = true; break; } } } assert found; } else { // -- search horizontal boolean found = false; // search right for (int x = pixel[0] + 1; x < width; x++) { point = IntegerTools.makeInt(x, pixel[1]); data = pixels.get(point); if (data != null) { newPixels.add(new int[]{pixel[0], pixel[1], data[2]}); found = true; break; } } if (!found) { // search left for (int x = pixel[0] - 1; x > -1; x--) { point = IntegerTools.makeInt(x, pixel[1]); data = pixels.get(point); if (data != null) { newPixels.add(new int[]{pixel[0], pixel[1], data[2]}); found = true; break; } } } assert found; } } else { // use the existing pixel newPixels.add(data); } minX = Math.min(minX, pixel[0]); maxX = Math.max(maxX, pixel[0]); minY = Math.min(minY, pixel[1]); maxY = Math.max(maxY, pixel[1]); } } // add the pixels that we have found and add an offset pixels.clear(); for (int[] pixel : newPixels) { int x = pixel[0] - minX; int y = pixel[1] - minY; pixels.put(IntegerTools.makeInt(x, y), new int[] {x, y, pixel[2]}); } // compute the new width and height int newWidth = maxX - minX + 1; int newHeight = maxY - minY + 1; if (newWidth < width || newHeight < height) { // -- fix uv points uvPoints[0][0] = (uvPoints[0][0] * width - minX) / newWidth; uvPoints[0][1] = (uvPoints[0][1] * height - minY) / newHeight; uvPoints[1][0] = (uvPoints[1][0] * width - minX) / newWidth; uvPoints[1][1] = (uvPoints[1][1] * height - minY) / newHeight; uvPoints[2][0] = (uvPoints[2][0] * width - minX) / newWidth; uvPoints[2][1] = (uvPoints[2][1] * height - minY) / newHeight; } return new int[] {newWidth, newHeight}; } // compress the texture and return new size // Note: This changes the pixel array and also the uv positions (!) protected static int[] compress(int width, int height, TIntObjectHashMap<int[]> pixels, double[][] uvPoints) { // size array (that might still change!) int[] size = new int[] {width, height, 1}; // -- compress this texture (scale if this can be done loss-less) // basic compression in x direction size = TextureTools.compress(size[0], size[1], 0, size[0], false, pixels); // basic compression in y direction size = TextureTools.compress(size[0], size[1], 0, size[1], true, pixels); // obtain offset and compress with offsets (X) int[] offsetsX = TextureTools.getOffsets(size[0], size[1], false, pixels); if (offsetsX[0] > 0) { // skip left int[] oldSize = new int[] {size[0], size[1]}; size = TextureTools.compress(size[0], size[1], offsetsX[0], size[0], false, pixels); if (size[2] > 1) { // check if there has been a compression // move uvs for (double[] p : uvPoints) { p[0] *= oldSize[0]; if (p[0] > offsetsX[0]) { p[0] = ((p[0] - offsetsX[0]) / size[2]) + offsetsX[0]; } else { p[0] += (offsetsX[0] - p[0]) * (1 - 1d / size[2]); } p[0] /= size[0]; } // fix the pixels (erase unnecessary and fill in missing) size = repairPixel(size[0], size[1], pixels, uvPoints, false); // update offsets offsetsX = TextureTools.getOffsets(size[0], size[1], false, pixels); } } if (offsetsX[1] < size[0]) { // skip right int[] oldSize = new int[] {size[0], size[1]}; size = TextureTools.compress(size[0], size[1], 0, offsetsX[1], false, pixels); if (size[2] > 1) { // check if there has been a compression // move uvs for (double[] p : uvPoints) { p[0] *= oldSize[0]; if (p[0] < offsetsX[1]) { p[0] = p[0] / size[2]; } else { p[0] = (p[0] - offsetsX[1]) * (1d / size[2]) + offsetsX[1] / size[2]; } p[0] /= size[0]; } // fix the pixels (erase unnecessary and fill in missing) size = repairPixel(size[0], size[1], pixels, uvPoints, false); // update offsets offsetsX = TextureTools.getOffsets(size[0], size[1], false, pixels); } } if (offsetsX[0] > 0 || offsetsX[1] < size[0]) { // skip left and right int[] oldSize = new int[] {size[0], size[1]}; size = TextureTools.compress(size[0], size[1], offsetsX[0], offsetsX[1], false, pixels); if (size[2] > 1) { // check if there has been a compression // move uvs for (double[] p : uvPoints) { p[0] *= oldSize[0]; if (p[0] < offsetsX[0]) { p[0] += (offsetsX[0] - p[0]) * (1 - 1d / size[2]); } else if (p[0] < offsetsX[1]) { p[0] = (p[0] - offsetsX[0]) / size[2] + offsetsX[0]; } else { p[0] = (p[0] - offsetsX[1]) * (1d / size[2]) + offsetsX[0] + (offsetsX[1] - offsetsX[0]) / size[2]; } p[0] /= size[0]; } // fix the pixels (erase unnecessary and fill in missing) size = repairPixel(size[0], size[1], pixels, uvPoints, false); } } // obtain offset and compress with offsets (Y) int[] offsetsY = TextureTools.getOffsets(size[0], size[1], true, pixels); if (offsetsY[0] > 0) { // skip top int[] oldSize = new int[] {size[0], size[1]}; size = TextureTools.compress(size[0], size[1], offsetsY[0], size[1], true, pixels); if (size[2] > 1) { // check if there has been a compression // move uvs for (double[] p : uvPoints) { p[1] *= oldSize[1]; if (p[1] > offsetsY[0]) { p[1] = ((p[1] - offsetsY[0]) / size[2]) + offsetsY[0]; } else { p[1] += (offsetsY[0] - p[1]) * (1 - 1d / size[2]); } p[1] /= size[1]; } // fix the pixels (erase unnecessary and fill in missing) size = repairPixel(size[0], size[1], pixels, uvPoints, true); // update offsets offsetsY = TextureTools.getOffsets(size[0], size[1], true, pixels); } } if (offsetsY[1] < size[1]) { // skip bottom int[] oldSize = new int[] {size[0], size[1]}; size = TextureTools.compress(size[0], size[1], 0, offsetsY[1], true, pixels); if (size[2] > 1) { // check if there has been a compression // move uvs for (double[] p : uvPoints) { p[1] *= oldSize[1]; if (p[1] < offsetsY[1]) { p[1] = p[1] / size[2]; } else { p[1] = (p[1] - offsetsY[1]) * (1d / size[2]) + offsetsY[1] / size[2]; } p[1] /= size[1]; } // fix the pixels (erase unnecessary and fill in missing) size = repairPixel(size[0], size[1], pixels, uvPoints, true); // update offsets offsetsY = TextureTools.getOffsets(size[0], size[1], true, pixels); } } if (offsetsY[0] > 0 || offsetsY[1] < size[1]) { // skip top and bottom int[] oldSize = new int[] {size[0], size[1]}; size = TextureTools.compress(size[0], size[1], offsetsY[0], offsetsY[1], true, pixels); if (size[2] > 1) { // check if there has been a compression // move uvs for (double[] p : uvPoints) { p[1] *= oldSize[1]; if (p[1] < offsetsY[0]) { p[1] += (offsetsY[0] - p[1]) * (1 - 1d / size[2]); } else if (p[1] < offsetsY[1]) { p[1] = (p[1] - offsetsY[0]) / size[2] + offsetsY[0]; } else { p[1] = (p[1] - offsetsY[1]) * (1d / size[2]) + offsetsY[0] + (offsetsY[1] - offsetsY[0]) / size[2]; } p[1] /= size[1]; } // fix the pixels (erase unnecessary and fill in missing) size = repairPixel(size[0], size[1], pixels, uvPoints, true); } } return size; } }