Image convolution: Difference between revisions

Added Kotlin
mNo edit summary
(Added Kotlin)
Line 906:
save("imagesharper.png", imfilt)
</lang>
 
=={{header|Kotlin}}==
{{trans|Java}}
<lang scala>// version 1.2.10
 
import kotlin.math.round
import java.awt.image.*
import java.io.File
import javax.imageio.*
 
class ArrayData(val width: Int, val height: Int) {
var dataArray = IntArray(width * height)
 
operator fun get(x: Int, y: Int) = dataArray[y * width + x]
 
operator fun set(x: Int, y: Int, value: Int) {
dataArray[y * width + x] = value
}
}
 
fun bound(value: Int, endIndex: Int) = when {
value < 0 -> 0
value < endIndex -> value
else -> endIndex - 1
}
 
fun convolute(
inputData: ArrayData,
kernel: ArrayData,
kernelDivisor: Int
): ArrayData {
val inputWidth = inputData.width
val inputHeight = inputData.height
val kernelWidth = kernel.width
val kernelHeight = kernel.height
if (kernelWidth <= 0 || (kernelWidth and 1) != 1)
throw IllegalArgumentException("Kernel must have odd width")
if (kernelHeight <= 0 || (kernelHeight and 1) != 1)
throw IllegalArgumentException("Kernel must have odd height")
val kernelWidthRadius = kernelWidth ushr 1
val kernelHeightRadius = kernelHeight ushr 1
 
val outputData = ArrayData(inputWidth, inputHeight)
for (i in inputWidth - 1 downTo 0) {
for (j in inputHeight - 1 downTo 0) {
var newValue = 0.0
for (kw in kernelWidth - 1 downTo 0) {
for (kh in kernelHeight - 1 downTo 0) {
newValue += kernel[kw, kh] * inputData[
bound(i + kw - kernelWidthRadius, inputWidth),
bound(j + kh - kernelHeightRadius, inputHeight)
].toDouble()
outputData[i, j] = round(newValue / kernelDivisor).toInt()
}
}
}
}
return outputData
}
 
fun getArrayDatasFromImage(filename: String): Array<ArrayData> {
val inputImage = ImageIO.read(File(filename))
val width = inputImage.width
val height = inputImage.height
val rgbData = inputImage.getRGB(0, 0, width, height, null, 0, width)
val reds = ArrayData(width, height)
val greens = ArrayData(width, height)
val blues = ArrayData(width, height)
for (y in 0 until height) {
for (x in 0 until width) {
val rgbValue = rgbData[y * width + x]
reds[x, y] = (rgbValue ushr 16) and 0xFF
greens[x,y] = (rgbValue ushr 8) and 0xFF
blues[x, y] = rgbValue and 0xFF
}
}
return arrayOf(reds, greens, blues)
}
 
fun writeOutputImage(filename: String, redGreenBlue: Array<ArrayData>) {
val (reds, greens, blues) = redGreenBlue
val outputImage = BufferedImage(
reds.width, reds.height, BufferedImage.TYPE_INT_ARGB
)
for (y in 0 until reds.height) {
for (x in 0 until reds.width) {
val red = bound(reds[x , y], 256)
val green = bound(greens[x , y], 256)
val blue = bound(blues[x, y], 256)
outputImage.setRGB(
x, y, (red shl 16) or (green shl 8) or blue or -0x01000000
)
}
}
ImageIO.write(outputImage, "PNG", File(filename))
}
 
fun main(args: Array<String>) {
val kernelWidth = args[2].toInt()
val kernelHeight = args[3].toInt()
val kernelDivisor = args[4].toInt()
println("Kernel size: $kernelWidth x $kernelHeight, divisor = $kernelDivisor")
var y = 5
val kernel = ArrayData(kernelWidth, kernelHeight)
for (i in 0 until kernelHeight) {
print("[")
for (j in 0 until kernelWidth) {
kernel[j, i] = args[y++].toInt()
print(" ${kernel[j, i]} ")
}
println("]")
}
 
val dataArrays = getArrayDatasFromImage(args[0])
for (i in 0 until dataArrays.size) {
dataArrays[i] = convolute(dataArrays[i], kernel, kernelDivisor)
}
writeOutputImage(args[1], dataArrays)
}</lang>
 
{{out}}
<pre>
Same as Java entry when using identical command line arguments
</pre>
 
=={{header|Liberty BASIC}}==
9,485

edits