apply plugin: "java" sourceCompatibility = 1.8 sourceSets.main.java.srcDirs = ["src/"] import com.badlogic.gdx.graphics.Color import com.badlogic.gdx.tools.texturepacker.TexturePacker import com.badlogic.gdx.utils.IntArray import com.badlogic.gdx.utils.IntIntMap import com.badlogic.gdx.utils.IntMap import javax.imageio.ImageIO import java.awt.* import java.awt.image.BufferedImage import java.util.List import java.util.concurrent.ExecutorService import java.util.concurrent.Executors import java.util.concurrent.TimeUnit def genFolder = "../core/assets-raw/sprites_out/generated/" def doAntialias = !project.hasProperty("disableAntialias") def colorMap = new IntMap>(), colorIndexMap = new IntIntMap() def transformColors = { List> list -> list.each{ colors -> def newColors = [] colors.each{ hexc -> newColors += Color.valueOf(hexc) } newColors.each{ color -> colorMap.put(Color.argb8888(color), newColors) colorIndexMap.put(Color.argb8888(color), newColors.indexOf(color)) } } } transformColors([["6e7080", "989aa4", "b0bac0"], ["bc5452", "ea8878", "feb380"], ["dea158", "f8c266", "ffe18f"], ["feb380", "ea8878", "bc5452"]]) def antialias = { File file -> if(!doAntialias || file.lastModified() <= 1000) return def image = ImageIO.read(file) def out = ImageIO.read(file) def getRGB = { int ix, int iy -> //if(ix <= 0 || iy <= 0 || ix >= image.width || iy >= image.height) return 0 return image.getRGB(Math.max(Math.min(ix, image.width - 1), 0), Math.max(Math.min(iy, image.height - 1), 0)) } def color = new Color() def sum = new Color() def suma = new Color() int[] p = new int[9] for(int x = 0; x < image.getWidth(); x++){ for(int y = 0; y < image.getHeight(); y++){ int A = getRGB(x - 1, y + 1), B = getRGB(x, y + 1), C = getRGB(x + 1, y + 1), D = getRGB(x - 1, y), E = getRGB(x, y), F = getRGB(x + 1, y), G = getRGB(x - 1, y - 1), H = getRGB(x, y - 1), I = getRGB(x + 1, y - 1) Arrays.fill(p, E); if(D == B && D != H && B != F) p[0] = D if((D == B && D != H && B != F && E != C) || (B == F && B != D && F != H && E != A)) p[1] = B if(B == F && B != D && F != H) p[2] = F if((H == D && H != F && D != B && E != A) || (D == B && D != H && B != F && E != G)) p[3] = D if((B == F && B != D && F != H && E != I) || (F == H && F != B && H != D && E != C)) p[5] = F if(H == D && H != F && D != B) p[6] = D if((F == H && F != B && H != D && E != G) || (H == D && H != F && D != B && E != I)) p[7] = H if(F == H && F != B && H != D) p[8] = F suma.set(0) for(int val : p){ Color.argb8888ToColor(color, val) suma.r += color.r * color.a suma.g += color.g * color.a suma.b += color.b * color.a suma.a += color.a } suma.mul(suma.a <= 0.001f ? 0f : (float) (1f / suma.a)) float total = 0 sum.set(0) for(int val : p){ Color.argb8888ToColor(color, val) float a = color.a color.lerp(suma, (float) (1f - a)) sum.r += color.r sum.g += color.g sum.b += color.b sum.a += a total += 1f } sum.mul((float) (1f / total)) int result = Color.argb8888(sum) out.setRGB(x, y, result) sum.set(0) } } ImageIO.write(out, "png", file) } def medianBlur = { File file -> def image = ImageIO.read(file) def result = new BufferedImage(image.width, image.height, BufferedImage.TYPE_INT_ARGB) def radius = 4 IntArray array = new IntArray() def getRGB = { int ix, int iy -> return image.getRGB(Math.max(Math.min(ix, image.width - 1), 0), Math.max(Math.min(iy, image.height - 1), 0)) } for(int x = 0; x < image.width; x++){ for(int y = 0; y < image.height; y++){ array.clear() for(int dx = -radius; dx <= radius; dx ++){ for(int dy = -radius; dy <= radius; dy ++){ if(dx*dx + dy*dy <= radius*radius){ array.add(getRGB(x + dx, y + dy)) } } } array.sort() result.setRGB(x, y, array.get((int)(array.size / 2))) } } ImageIO.write(result, "png", file) } def scaleImage = { File file -> def image = ImageIO.read(file) for(int iteration in 0..1){ def scaled = new BufferedImage(image.width * 2, image.height * 2, BufferedImage.TYPE_INT_ARGB) def getRGB = { int ix, int iy -> return image.getRGB(Math.max(Math.min(ix, image.width - 1), 0), Math.max(Math.min(iy, image.height - 1), 0)) } for(int x = 0; x < image.width; x++){ for(int y = 0; y < image.height; y++){ int p = image.getRGB(x, y) int p1 = p, p2 = p, p3 = p, p4 = p int A = getRGB(x - 1, y + 1), B = getRGB(x, y + 1), C = getRGB(x + 1, y + 1), D = getRGB(x - 1, y), E = getRGB(x, y), F = getRGB(x + 1, y), G = getRGB(x - 1, y - 1), H = getRGB(x, y - 1), I = getRGB(x + 1, y - 1), J = getRGB(x, y + 2), K = getRGB(x - 2, y), L = getRGB(x + 2, y), M = getRGB(x, y - 2) if(B == D && B != F && D != H && (E != A || E == C || E == G || A == J || A == K)) p1 = B if(B == F & B != D & F != H && (E != C || E == A || E == I || C == J || C == L)) p2 = F if(D == H & B != D & F != H && (E != G || E == A || E == I || G == K || G == M)) p3 = D if(F == H & B != F & D != H && (E != I || E == C || E == G || I == L || I == M)) p4 = H scaled.setRGB(x * 2, y * 2 + 1, p1) scaled.setRGB(x * 2 + 1, y * 2 + 1, p2) scaled.setRGB(x * 2, y * 2, p3) scaled.setRGB(x * 2 + 1, y * 2, p4) } } image = scaled } ImageIO.write(image, "png", file) } def tileImage = { File file -> def image = ImageIO.read(file) def result = new BufferedImage(image.width * 2, image.height * 2, image.getType()) Graphics2D graphics = result.createGraphics() graphics.drawImage(image, image.width, 0, -image.width, image.height, null) graphics.drawImage(image, image.width, 0, image.width, image.height, null) graphics.drawImage(image, image.width, image.height*2, -image.width, -image.height, null) graphics.drawImage(image, image.width, image.height*2, image.width, -image.height, null) for(int x = 0; x < result.width; x++){ for(int y = 0; y < result.height; y++){ int p = result.getRGB(x, y) if(x <= y){ List list = colorMap.get(p) int index = colorIndexMap.get(p, -1) if(index != -1){ int resultIndex = (x == y ? 1 : index == 2 ? 0 : index == 0 ? 2 : 1); result.setRGB(x, y, Color.argb8888(list[resultIndex])) } } } } ImageIO.write(result, "png", file) } task swapColors(){ doLast{ if(project.hasProperty("colors")){ def carr = new File(getProperty("colors")).text.split("\n") def map = [:] def swaps = 0 carr.each{ str -> map[Color.argb8888(Color.valueOf(str.split("=")[0]))] = Color.argb8888(Color.valueOf(str.split("=")[1])) } def tmpc = new Color() fileTree(dir: '../core/assets-raw/sprites', include: "**/*.png").visit{ file -> if(file.isDirectory()) return swaps++ def img = ImageIO.read(file.file) for(x in (0..img.getWidth() - 1)){ for(y in (0..img.getHeight() - 1)){ def c = img.getRGB(x, y) Color.argb8888ToColor(tmpc, c) if(tmpc.a < 0.1f) continue if(map.containsKey(c)){ img.setRGB(x, y, (int)map.get(c)) } } } ImageIO.write(img, "png", file.file) } println "Swapped $swaps images." }else{ throw new InvalidUserDataException("No replacement colors specified. Use -Pcolors=\"\"") } } } task antialiasImages(){ doLast{ for(def img : project.getProperty("images").split(",")){ println(project.getProperty("startdir") + "/" + img) antialias(new File(project.getProperty("startdir") + "/" + img)) } } } task scaleImages(){ doLast{ for(def img : project.getProperty("images").split(",")){ println(project.getProperty("startdir") + "/" + img) scaleImage(new File(project.getProperty("startdir") + "/" + img)) } } } task tileImages(){ doLast{ for(def img : project.getProperty("images").split(",")){ println(project.getProperty("startdir") + "/" + img) tileImage(new File(project.getProperty("startdir") + "/" + img)) } } } task pack(dependsOn: classes){ doLast{ //cleanup old sprites delete{ delete "../core/assets-raw/sprites_out/" } //copy in new sprites copy{ from "../core/assets-raw/sprites/" into "../core/assets-raw/sprites_out/" } //run generation task; generate all needed sprites file(genFolder).mkdirs() javaexec{ main = "mindustry.tools.ImagePacker" classpath = sourceSets.main.runtimeClasspath standardInput = System.in workingDir = genFolder } //upscale icon sprites using different method, which requires an OpenGL context javaexec{ if(System.getProperty("os.name").toLowerCase().contains("mac")){ jvmArgs "-XstartOnFirstThread" } jvmArgs("-Djava.awt.headless=true") main = "mindustry.tools.Upscaler" classpath = sourceSets.main.runtimeClasspath standardInput = System.in workingDir = "../core/assets-raw/sprites_out/ui/icons" } copy{ from "../core/assets-raw/sprites_out/ui/icons" into "../core/assets-raw/sprites_out/ui/" } delete{ delete "../core/assets-raw/sprites_out/ui/icons" } ExecutorService executor = Executors.newFixedThreadPool(16) //antialias everything except UI elements fileTree(dir: '../core/assets-raw/sprites_out/', include: "**/*.png").visit{ file -> if(file.isDirectory() || file.toString().replace("\\", "/").contains("zones/") || (file.toString().replace("\\", "/").contains("/ui/") && file.toString().startsWith("icon-"))) return executor.submit{ antialias(file.file) } } executor.shutdown(); try{ executor.awaitTermination(Long.MAX_VALUE, TimeUnit.NANOSECONDS); }catch(InterruptedException e){ e.printStackTrace() } //pack normal sprites TexturePacker.process("core/assets-raw/sprites_out/", "core/assets/sprites/", "sprites.atlas") } } task genSprites(dependsOn: classes, type: JavaExec){ finalizedBy 'antialiasGen' main = "mindustry.tools.ImagePacker" classpath = sourceSets.main.runtimeClasspath jvmArgs("-Djava.awt.headless=true") standardInput = System.in workingDir = genFolder } task updateBundles(dependsOn: classes, type: JavaExec){ file(genFolder).mkdirs() main = "mindustry.tools.BundleLauncher" classpath = sourceSets.main.runtimeClasspath standardInput = System.in workingDir = "../core/assets/bundles/" }