use of com.loohp.interactivechatdiscordsrvaddon.resources.models.TextureUV in project InteractiveChat-DiscordSRV-Addon by LOOHP.
the class ModelRenderer method generateStandardRenderModel.
@SuppressWarnings("SuspiciousNameCombination")
private Model generateStandardRenderModel(BlockModel blockModel, ResourceManager manager, Map<String, TextureResource> providedTextures, TintIndexData tintIndexData, boolean enchanted, boolean skin) {
Map<String, BufferedImage> cachedResize = new ConcurrentHashMap<>();
List<ModelElement> elements = new ArrayList<>(blockModel.getElements());
List<Hexahedron> hexahedrons = new ArrayList<>(elements.size());
Queue<Future<Hexahedron>> tasks = new ConcurrentLinkedQueue<>();
Iterator<ModelElement> itr = elements.iterator();
while (itr.hasNext()) {
ModelElement element = itr.next();
tasks.add(modelResolvingService.submit(() -> {
ModelElementRotation rotation = element.getRotation();
boolean ignoreZFight = false;
if (rotation != null) {
ignoreZFight = rotation.isRescale();
}
Hexahedron hexahedron = Hexahedron.fromCorners(new Point3D(element.getFrom().getX(), element.getFrom().getY(), element.getFrom().getZ()), new Point3D(element.getTo().getX(), element.getTo().getY(), element.getTo().getZ()), ignoreZFight, new BufferedImage[6]);
BufferedImage[] images = new BufferedImage[6];
BufferedImage[] overlayImages = new BufferedImage[6];
int i = 0;
for (ModelFaceSide side : ModelFaceSide.values()) {
ModelFace faceData = element.getFace(side);
if (faceData == null) {
images[i] = null;
} else {
TextureUV uv = faceData.getUV();
TextureResource resource = providedTextures.get(faceData.getTexture());
if (resource == null) {
resource = manager.getTextureManager().getTexture(faceData.getTexture(), false);
}
if (resource == null || !resource.isTexture()) {
images[i] = null;
} else if (uv != null && (uv.getXDiff() == 0 || uv.getYDiff() == 0)) {
images[i] = null;
} else {
BufferedImage cached = cachedResize.get(faceData.getTexture());
if (cached == null) {
cached = resource.getTexture();
if (resource.hasTextureMeta()) {
TextureMeta meta = resource.getTextureMeta();
if (meta.hasProperties()) {
TextureProperties properties = meta.getProperties();
if (properties.isBlur()) {
cached = ImageUtils.applyGaussianBlur(cached);
}
}
if (meta.hasAnimation()) {
TextureAnimation animation = meta.getAnimation();
if (animation.hasWidth() && animation.hasHeight()) {
cached = ImageUtils.copyAndGetSubImage(cached, 0, 0, animation.getWidth(), animation.getHeight());
} else {
cached = ImageUtils.copyAndGetSubImage(cached, 0, 0, cached.getWidth(), cached.getWidth());
}
}
}
if (cached.getWidth() > cached.getHeight()) {
cached = ImageUtils.resizeImageFillWidth(cached, skin ? SKIN_RESOLUTION : TEXTURE_RESOLUTION);
} else {
cached = ImageUtils.resizeImageFillHeight(cached, skin ? SKIN_RESOLUTION : TEXTURE_RESOLUTION);
}
cachedResize.put(faceData.getTexture(), cached);
}
images[i] = ImageUtils.copyImage(cached);
if (uv == null) {
Point3D[] points;
switch(side) {
case DOWN:
points = hexahedron.getDownFace().getPoints();
break;
case EAST:
points = hexahedron.getEastFace().getPoints();
break;
case NORTH:
points = hexahedron.getNorthFace().getPoints();
break;
case SOUTH:
points = hexahedron.getSouthFace().getPoints();
break;
case UP:
points = hexahedron.getUpFace().getPoints();
break;
case WEST:
default:
points = hexahedron.getWestFace().getPoints();
break;
}
double x1;
double y1;
double x2;
double y2;
if (points[0].x == points[2].x) {
x1 = points[0].y;
y1 = points[0].z;
x2 = points[2].y;
y2 = points[2].z;
} else if (points[0].y == points[2].y) {
x1 = points[0].z;
y1 = points[0].x;
x2 = points[2].z;
y2 = points[2].x;
} else {
x1 = points[0].y;
y1 = points[0].x;
x2 = points[2].y;
y2 = points[2].x;
}
uv = new TextureUV(Math.min(x1, x2), Math.min(y1, y2), Math.max(x1, x2), Math.max(y1, y2));
}
uv = uv.getScaled(1, (double) images[i].getHeight() / (double) images[i].getWidth());
uv = uv.getScaled((double) images[i].getWidth() / 16.0);
int x1 = (int) Math.ceil(uv.getX1());
int y1 = (int) Math.ceil(uv.getY1());
int dX = Math.abs((int) Math.floor(uv.getX2()) - x1);
int dY = Math.abs((int) Math.floor(uv.getY2()) - y1);
if (uv.isVerticallyFlipped()) {
images[i] = ImageUtils.flipVertically(images[i]);
y1 = images[i].getHeight() - y1;
}
if (uv.isHorizontallyFlipped()) {
images[i] = ImageUtils.flipHorizontal(images[i]);
x1 = images[i].getWidth() - x1;
}
images[i] = ImageUtils.copyAndGetSubImage(images[i], x1, y1, Math.max(1, dX), Math.max(1, dY));
int rotationAngle = faceData.getRotation();
if (rotationAngle % 360 != 0) {
images[i] = ImageUtils.rotateImageByDegrees(images[i], rotationAngle);
}
images[i] = tintIndexData.applyTint(images[i], faceData.getTintindex());
if (enchanted) {
overlayImages[i] = rawEnchantmentGlintProvider.apply(images[i]);
}
}
}
i++;
}
hexahedron.setImage(images);
hexahedron.setOverlay(overlayImages);
hexahedron.setOverlayAdditionFactor(OVERLAY_ADDITION_FACTORS);
if (rotation != null) {
hexahedron.translate(-rotation.getOrigin().getX(), -rotation.getOrigin().getY(), -rotation.getOrigin().getZ());
switch(rotation.getAxis()) {
case X:
hexahedron.rotate(rotation.getAngle(), 0, 0, false);
break;
case Y:
hexahedron.rotate(0, rotation.getAngle(), 0, false);
break;
case Z:
default:
hexahedron.rotate(0, 0, rotation.getAngle(), false);
break;
}
hexahedron.translate(rotation.getOrigin().getX(), rotation.getOrigin().getY(), rotation.getOrigin().getZ());
}
return hexahedron;
}));
itr.remove();
}
while (!tasks.isEmpty() || !elements.isEmpty()) {
Future<Hexahedron> task = tasks.poll();
if (task == null) {
try {
TimeUnit.MILLISECONDS.sleep(1);
} catch (InterruptedException e) {
}
} else {
try {
hexahedrons.add(task.get());
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
}
}
return new Model(hexahedrons);
}
Aggregations