Codetown ::: a software developer's community
Welcome to Kotlin Thursdays! Last week, we were able to render an image with TornadoFX and even manipulate its pixels. Today, we will go over Pixel Math!
Think of these resources as supplemental if you happen to be more curious. We always encourage looking into documentation for things you use!
Last week, we got the hang of how to grab these pixels and do something with them. Today, we're going to expand by creating our own filters using operational pixel manipulation.
For all practical purposes, we're going to be talking about monochromatic images. If we try to write filters using colored pixels, it will prove a lot more difficult to work with RGB values as opposed to just black or white.
Best we learn to walk before we start trying to fly!
In order to create our own image filters, we need to have a solid understanding of pixel math, or binary operations.
Binary operations are the bread and butter of computers! You can compute operations on binary values 1 and 0.
AND - both inputs must be true for the output to be true
0 && 0 = 0
0 && 1 = 0
1 && 0 = 0
1 && 1 = 1
OR - one or both inputs must be true for the output to be true
0 || 0 = 0
0 || 1 = 1
1 || 0 = 1
1 || 1 = 1
NOT - inverse result
!0 = 1
!1 = 0
!(0 && 0) = 1
!(1 || 1) = 0
Likewise, if we assign the color BLACK to 1 and the color WHITE to 0, we can easily apply binary operations to to the binary values black and white. Working with colors gets significantly more difficult when there are RGB values to consider. There are other binary operations like XANDS, XORS, and XNORS, but for now, let's just focus on the first three.
Now that we understand how OR, AND, and NOT works, let's implement these functions with colors.
fun or (a: Color, b: Color) {
return if (a == Color.BLACK || b == Color.BLACK) {
Color.BLACK
}
else { Color.WHITE
}
fun and (a: Color, b: Color) {
return if (a == Color.BLACK && b == Color.BLACK) {
Color.BLACK
} else {
Color.WHITE
}
}
fun not (color: Color) {
return if (color == Color.BLACK) Color.WHITE else Color.BLACK
}
You'll notice that these functions are for pixel colors only. Next week, we look into higher-order functions in Kotlin to learn how we can pass functions as a parameter - but you'll welcome to check out the video to see how we can apply one of these primitive filters to our images! See you next week :)
Tags:
Codetown is a social network. It's got blogs, forums, groups, personal pages and more! You might think of Codetown as a funky camper van with lots of compartments for your stuff and a great multimedia system, too! Best of all, Codetown has room for all of your friends.
Created by Michael Levin Dec 18, 2008 at 6:56pm. Last updated by Michael Levin May 4, 2018.
Check out the Codetown Jobs group.

Airbnb has redesigned its identity system to support privacy-first social features in Experiences. The platform introduces context-specific profiles that separate global user identity from externally visible profiles, preventing cross-context linkage. The migration leveraged automated auditing, manual validation, and AI-assisted refactoring to enforce correct identity usage across services.
By Leela Kumili
JEP 533, Structured Concurrency, has reached integrated status for JDK 27. It refines exception handling and type safety in its API, particularly focusing on exception flow with a new ExecutionException type. Changes include an updated Joiner interface and a new open overload for easier configuration. The steady evolution signals ongoing development as feedback shapes the API.
By A N M Bazlur Rahman
Paulo Arruda discusses Shopify’s evolution in AI adoption, moving from simple chat tools to a sophisticated swarm of specialized agents. He explains the transition from massive "all-in-one" prompts to lean, narrow-focused agent microservices that slash task times from hours to minutes. He also shares a future-looking hypothesis on using filesystem-based adapters to solve context bloat.
By Paulo Arruda
Backlogs in distributed systems are arithmetic problems, not mysteries. This article provides practical formulas for calculating backlog drain time, sizing consumer headroom, and setting auto-scaling triggers. It covers key failure modes — retry amplification, metastable states, and cascading pipeline bottlenecks — plus when to shed load instead of draining.
By Rajesh Kumar Pandey
Grafana Labs has launched Pyroscope 2.0, a rearchitected open-source continuous profiling database. This version improves storage costs, query performance, and operational complexity. Key changes include single write paths for profiles, stateless query processing, and enhanced capabilities for profiling data. It supports the OpenTelemetry Protocol, aligning with current trends in observability.
By Matt Saunders
© 2026 Created by Michael Levin.
Powered by