
darwin.jopenctm.compression.MG1Encoder Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of JOpenCTM Show documentation
Show all versions of JOpenCTM Show documentation
Java implementation of the openCTM Mesh compression file format
The newest version!
/*
* Copyright (C) 2012 Daniel Heinrich
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* (version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this library. If not, see
* or write to the Free Software Foundation, Inc., 51 Franklin Street,
* Fifth Floor, Boston, MA 02110-1301 USA.
*/
package darwin.jopenctm.compression;
import java.io.IOException;
import java.util.Arrays;
import darwin.annotations.ServiceProvider;
import darwin.jopenctm.data.Triangle;
import darwin.jopenctm.io.CtmOutputStream;
/**
*
* @author daniel
*/
@ServiceProvider(MeshEncoder.class)
public class MG1Encoder extends RawEncoder {
@Override
public int getTag() {
return MG1Decoder.MG1_TAG;
}
@Override
protected void writeFloatArray(float[] array, CtmOutputStream out,
int count, int size) throws IOException {
out.writePackedFloats(array, count, size);
}
@Override
protected void writeIndices(int[] indices, CtmOutputStream out) throws IOException {
int[] tmp = new int[indices.length];
System.arraycopy(indices, 0, tmp, 0, tmp.length);
rearrangeTriangles(tmp);
makeIndexDeltas(tmp);
out.writePackedInts(tmp, tmp.length / 3, 3, false);
}
/**
* Re-arrange all triangles for optimal compression.
*
* @param indices index data to reorder in place
*/
public void rearrangeTriangles(int[] indices) {
assert indices.length % 3 == 0;
// Step 1: Make sure that the first index of each triangle is the smallest
// one (rotate triangle nodes if necessary)
for (int off = 0; off < indices.length; off += 3) {
if ((indices[off + 1] < indices[off]) && (indices[off + 1] < indices[off + 2])) {
int tmp = indices[off];
indices[off] = indices[off + 1];
indices[off + 1] = indices[off + 2];
indices[off + 2] = tmp;
} else if ((indices[off + 2] < indices[off]) && (indices[off + 2] < indices[off + 1])) {
int tmp = indices[off];
indices[off] = indices[off + 2];
indices[off + 2] = indices[off + 1];
indices[off + 1] = tmp;
}
}
// Step 2: Sort the triangles based on the first triangle index
Triangle[] tris = new Triangle[indices.length / 3];
for (int i = 0; i < tris.length; i++) {
int off = i * 3;
tris[i] = new Triangle(indices, off);
}
Arrays.sort(tris);
for (int i = 0; i < tris.length; i++) {
int off = i * 3;
tris[i].copyBack(indices, off);
}
}
/**
* Calculate various forms of derivatives in order to reduce data entropy.
*
* @param indices index data to encode in place
*/
public void makeIndexDeltas(int[] indices) {
assert indices.length % 3 == 0;
for (int i = indices.length / 3 - 1; i >= 0; --i) {
// Step 1: Calculate delta from second triangle index to the previous
// second triangle index, if the previous triangle shares the same first
// index, otherwise calculate the delta to the first triangle index
if ((i >= 1) && (indices[i * 3] == indices[(i - 1) * 3])) {
indices[i * 3 + 1] -= indices[(i - 1) * 3 + 1];
} else {
indices[i * 3 + 1] -= indices[i * 3];
}
// Step 2: Calculate delta from third triangle index to the first triangle
// index
indices[i * 3 + 2] -= indices[i * 3];
// Step 3: Calculate derivative of the first triangle index
if (i >= 1) {
indices[i * 3] -= indices[(i - 1) * 3];
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy