All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.pepsoft.worldpainter.painting.DiscreteLayerPaint Maven / Gradle / Ivy

There is a newer version: 2.23.2
Show newest version
/*
 * WorldPainter, a graphical and interactive map generator for Minecraft.
 * Copyright � 2011-2015  pepsoft.org, The Netherlands
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program 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 General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */

package org.pepsoft.worldpainter.painting;

import org.pepsoft.worldpainter.Dimension;
import org.pepsoft.worldpainter.Tile;
import org.pepsoft.worldpainter.layers.Layer;

import static org.pepsoft.worldpainter.Constants.TILE_SIZE_BITS;
import static org.pepsoft.worldpainter.Constants.TILE_SIZE_MASK;

/**
 * Note: does not do any event inhibitation management.
 *
 * 

Created by pepijn on 20-05-15. */ public final class DiscreteLayerPaint extends LayerPaint { public DiscreteLayerPaint(Layer layer, int value) { super(layer); if ((layer.getDataSize() != Layer.DataSize.NIBBLE) && (layer.getDataSize() != Layer.DataSize.BYTE)) { throw new IllegalArgumentException("Layer " + layer + " not a continuous layer"); } this.value = value; defaultValue = layer.getDefaultValue(); } public int getValue() { return value; } @Override public String getId() { return super.getId() + '/' + value; } @Override public void apply(Dimension dimension, int centreX, int centreY, float dynamicLevel) { if (brush.getRadius() == 0) { // Special case: if the radius is 0, assume that the user wants to paint complete pixels instead of trying // to apply the brush applyPixel(dimension, centreX, centreY); return; } final int effectiveRadius = brush.getEffectiveRadius(); final int x1 = centreX - effectiveRadius, y1 = centreY - effectiveRadius, x2 = centreX + effectiveRadius, y2 = centreY + effectiveRadius; final int tileX1 = x1 >> TILE_SIZE_BITS, tileY1 = y1 >> TILE_SIZE_BITS, tileX2 = x2 >> TILE_SIZE_BITS, tileY2 = y2 >> TILE_SIZE_BITS; if ((tileX1 == tileX2) && (tileY1 == tileY2)) { // The bounding box of the brush is entirely on one tile; optimize by painting directly to the tile final Tile tile = dimension.getTileForEditing(tileX1, tileY1); if (tile == null) { return; } final int x1InTile = x1 & TILE_SIZE_MASK, y1InTile = y1 & TILE_SIZE_MASK, x2InTile = x2 & TILE_SIZE_MASK, y2InTile = y2 & TILE_SIZE_MASK; final int tileXInWorld = tileX1 << TILE_SIZE_BITS, tileYInWorld = tileY1 << TILE_SIZE_BITS; if (dither) { for (int y = y1InTile; y <= y2InTile; y++) { for (int x = x1InTile; x <= x2InTile; x++) { final float strength = dynamicLevel * getStrength(centreX, centreY, tileXInWorld + x, tileYInWorld + y); if ((strength > 0.95f) || (Math.random() < strength)) { tile.setLayerValue(layer, x, y, value); } } } } else { for (int y = y1InTile; y <= y2InTile; y++) { for (int x = x1InTile; x <= x2InTile; x++) { final float strength = dynamicLevel * getFullStrength(centreX, centreY, tileXInWorld + x, tileYInWorld + y); if (strength > 0.75f) { tile.setLayerValue(layer, x, y, value); } } } } } else { // The bounding box of the brush straddles more than one tile; paint to the dimension if (dither) { for (int y = y1; y <= y2; y++) { for (int x = x1; x <= x2; x++) { final float strength = dynamicLevel * getStrength(centreX, centreY, x, y); if ((strength > 0.95f) || (Math.random() < strength)) { dimension.setLayerValueAt(layer, x, y, value); } } } } else { for (int y = y1; y <= y2; y++) { for (int x = x1; x <= x2; x++) { final float strength = dynamicLevel * getFullStrength(centreX, centreY, x, y); if (strength > 0.75f) { dimension.setLayerValueAt(layer, x, y, value); } } } } } } @Override public void remove(Dimension dimension, int centreX, int centreY, float dynamicLevel) { if (brush.getRadius() == 0) { // Special case: if the radius is 0, assume that the user wants to remove complete pixels instead of trying // to apply the brush removePixel(dimension, centreX, centreY); return; } final int effectiveRadius = brush.getEffectiveRadius(); final int x1 = centreX - effectiveRadius, y1 = centreY - effectiveRadius, x2 = centreX + effectiveRadius, y2 = centreY + effectiveRadius; final int tileX1 = x1 >> TILE_SIZE_BITS, tileY1 = y1 >> TILE_SIZE_BITS, tileX2 = x2 >> TILE_SIZE_BITS, tileY2 = y2 >> TILE_SIZE_BITS; if ((tileX1 == tileX2) && (tileY1 == tileY2)) { // The bounding box of the brush is entirely on one tile; optimize by painting directly to the tile final Tile tile = dimension.getTileForEditing(tileX1, tileY1); if (tile == null) { return; } final int x1InTile = x1 & TILE_SIZE_MASK, y1InTile = y1 & TILE_SIZE_MASK, x2InTile = x2 & TILE_SIZE_MASK, y2InTile = y2 & TILE_SIZE_MASK; final int tileXInWorld = tileX1 << TILE_SIZE_BITS, tileYInWorld = tileY1 << TILE_SIZE_BITS; if (dither) { for (int y = y1InTile; y <= y2InTile; y++) { for (int x = x1InTile; x <= x2InTile; x++) { final float strength = dynamicLevel * getFullStrength(centreX, centreY, tileXInWorld + x, tileYInWorld + y); if ((strength > 0.95f) || (Math.random() < strength)) { tile.setLayerValue(layer, x, y, defaultValue); } } } } else { for (int y = y1InTile; y <= y2InTile; y++) { for (int x = x1InTile; x <= x2InTile; x++) { final float strength = dynamicLevel * getFullStrength(centreX, centreY, tileXInWorld + x, tileYInWorld + y); if (strength > 0.75f) { tile.setLayerValue(layer, x, y, defaultValue); } } } } } else { // The bounding box of the brush straddles more than one tile; paint to the dimension if (dither) { for (int y = y1; y <= y2; y++) { for (int x = x1; x <= x2; x++) { final float strength = dynamicLevel * getFullStrength(centreX, centreY, x, y); if ((strength > 0.95f) || (Math.random() < strength)) { dimension.setLayerValueAt(layer, x, y, defaultValue); } } } } else { for (int y = y1; y <= y2; y++) { for (int x = x1; x <= x2; x++) { final float strength = dynamicLevel * getFullStrength(centreX, centreY, x, y); if (strength > 0.75f) { dimension.setLayerValueAt(layer, x, y, defaultValue); } } } } } } @Override public void applyPixel(Dimension dimension, int x, int y) { dimension.setLayerValueAt(layer, x, y, value); } @Override public void removePixel(Dimension dimension, int x, int y) { dimension.setLayerValueAt(layer, x, y, defaultValue); } private final int value, defaultValue; }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy