-
Notifications
You must be signed in to change notification settings - Fork 89
/
04_functor.ts
103 lines (83 loc) · 3.22 KB
/
04_functor.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
// Adapted from https://adrian-salajan.github.io/blog/2021/01/25/images-functor
// run `npm run functor` to execute
import { Endomorphism } from 'fp-ts/function'
import * as R from 'fp-ts/Reader'
// -------------------------------------------------------------------------------------
// model
// -------------------------------------------------------------------------------------
type Color = {
readonly red: number
readonly green: number
readonly blue: number
}
type Point = {
readonly x: number
readonly y: number
}
type Image<A> = R.Reader<Point, A>
// -------------------------------------------------------------------------------------
// constructors
// -------------------------------------------------------------------------------------
const color = (red: number, green: number, blue: number): Color => ({
red,
green,
blue
})
const BLACK: Color = color(0, 0, 0)
const WHITE: Color = color(255, 255, 255)
// -------------------------------------------------------------------------------------
// combinators
// -------------------------------------------------------------------------------------
const brightness = (color: Color): number =>
(color.red + color.green + color.blue) / 3
export const grayscale = (c: Color): Color => {
const n = brightness(c)
return color(n, n, n)
}
export const invert = (c: Color): Color =>
color(255 - c.red, 255 - c.green, 255 - c.red)
// if brightness over some value V then put White else put Black
export const threshold = (c: Color): Color =>
brightness(c) < 100 ? BLACK : WHITE
// -------------------------------------------------------------------------------------
// main
// -------------------------------------------------------------------------------------
// `main` must be called by passing a transformation function, that is, an `Endomorphism<Image<RGB>>`
main(R.map((c: Color) => c))
// main(R.map(grayscale))
// main(R.map(invert))
// main(R.map(threshold))
// -------------------------------------------------------------------------------------
// utils
// -------------------------------------------------------------------------------------
function main(f: Endomorphism<Image<Color>>) {
const canvas: HTMLCanvasElement = document.getElementById('canvas') as any
const ctx: CanvasRenderingContext2D = canvas.getContext('2d') as any
const bird: HTMLImageElement = document.getElementById('bird') as any
bird.onload = function () {
console.log('hello')
function getImage(imageData: ImageData): Image<Color> {
const data = imageData.data
return (loc) => {
const pos = loc.x * 4 + loc.y * 632 * 4
return color(data[pos], data[pos + 1], data[pos + 2])
}
}
function setImage(imageData: ImageData, image: Image<Color>): void {
const data = imageData.data
for (let x = 0; x < 632; x++) {
for (let y = 0; y < 421; y++) {
const pos = x * 4 + y * 632 * 4
const { red, green, blue } = image({ x, y })
data[pos] = red
data[pos + 1] = green
data[pos + 2] = blue
}
}
ctx.putImageData(imageData, 0, 0)
}
ctx.drawImage(bird, 0, 0)
const imageData = ctx.getImageData(0, 0, 632, 421)
setImage(imageData, f(getImage(imageData)))
}
}