代码拉取完成,页面将自动刷新
同步操作将从 王福迎/asciilize 强制同步,此操作会覆盖自 Fork 仓库以来所做的任何修改,且无法恢复!!!
确定后同步将在后台操作,完成时将刷新页面,请耐心等待。
package main
import (
"flag"
"fmt"
"image/jpeg"
"math"
"os"
)
// max block size = 4096 X 4096
// gray to ascii map
var grayMap = [...]string{"@", "w", "#", "$", "k", "d", "t", "j", "i", ".", " "}
func main() {
ptr_filename := flag.String("f", "./a.jpg", "Define the source jpeg file")
ptr_blockX := flag.Int("bx", 5, "Define the block size in X axis")
ptr_blockY := flag.Int("by", 10, "Define the block size in Y axis")
ptr_step := flag.Int("j", 25, "sampling step")
flag.Parse()
_, asciiData, err := Asciilize(*ptr_filename, *ptr_blockX, *ptr_blockY, *ptr_step)
if nil != err || *ptr_blockX < 1 || *ptr_blockY < 1 {
fmt.Println(err)
flag.Usage()
fmt.Println("Example: ./program -f ./hello.jpg -bx 4 -by 6")
fmt.Println("可以使用-j n参数来控制采用的细腻度,表示每n个点取样一次,如果n等于0,则关闭此功能,n=0等价于n=1")
}
OutputAsciilizedData(asciiData)
}
func Asciilize(jpegFile string, blockSizeX int, blockSizeY int, jump int) (grayData [][]uint8, asciiData [][]string, err error) {
blockX := blockSizeX
blockY := blockSizeY
// open file
m_file, err := os.Open(jpegFile)
defer m_file.Close()
if err != nil {
return nil, nil, err
}
// decode jpeg
m_img, err := jpeg.Decode(m_file)
if err != nil {
return nil, nil, err
}
// pre
m_bounds := m_img.Bounds()
// split blocks
nBlockX := (int)(math.Ceil((float64)(m_bounds.Dx()) / float64(blockX)))
nBlockY := (int)(math.Ceil((float64)(m_bounds.Dy()) / float64(blockY)))
// gen 2d array
result := make([][]uint8, nBlockY)
for row := 0; row < nBlockY; row++ {
subresult := make([]uint8, nBlockX, nBlockX)
for col := 0; col < nBlockX; col++ {
subresult[col] = 0
}
result[row] = subresult
}
// calc grey value
var pickColorCount uint = 0
for x := 0; x < nBlockX; x++ {
for y := 0; y < nBlockY; y++ {
// every block
var graySum uint32 = 0
pixelX := x * blockX
pixelY := y * blockY
breakPixel := 0
stepCount := 0
for i := 0; i < blockX; i++ {
for j := 0; j < blockY; j++ {
if 0 != jump {
stepCount %= jump
if 0 != stepCount {
breakPixel++
stepCount++
continue
} else {
stepCount++
}
}
targetX := pixelX + i
targetY := pixelY + j
if targetX >= m_bounds.Dx() || targetY >= m_bounds.Dy() {
breakPixel++
continue
}
// m_color := m_img.At(targetX, targetY)
// gray := image.NewGray(image.Rect(0, 0, m_bounds.Dx(), m_bounds.Dy())).GrayAt(targetX, targetY)
pickColorCount++
r, g, b, a := m_img.At(targetX, targetY).RGBA()
r /= 256
g /= 256
b /= 256
a /= 256
graySum += (uint32)((float64(r)*float64(0.3) + float64(g)*float64(0.59) + float64(b)*float64(0.11)) * (float64(a) / float64(256.0)))
}
}
graySum /= uint32(blockX*blockY - breakPixel)
result[y][x] = uint8(graySum)
}
}
// output pick time
fmt.Printf("Pick Color %d times\n", pickColorCount)
// to ascii 2d array
step := float64(256) / float64(len(grayMap))
// var ascii [][]string
ascii := make([][]string, nBlockY)
for row := 0; row < nBlockY; row++ {
subascii := make([]string, nBlockX)
for col := 0; col < nBlockX; col++ {
index := int(float64(result[row][col]) / step)
subascii[col] = grayMap[index]
}
ascii[row] = subascii
}
grayData = result
asciiData = ascii
return
}
func OutputAsciilizedData(asciiData [][]string) {
// format output
nBlockY := len(asciiData)
if 0 == nBlockY {
return
}
nBlockX := len(asciiData[0])
for row := 0; row < nBlockY; row++ {
for col := 0; col < nBlockX; col++ {
fmt.Print(asciiData[row][col])
}
fmt.Println()
}
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。