0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

ASCIIアート作るツール作ってみた

Posted at

概要

画像処理をやりたいと考えていたため、シンプルなツールを作ってみた。

処理概要と限界

画像処理初心者なため、シンプルな実装を行っている。
使うライブラリは有名どころのOpenCVである。

また、ASCII_CHARS = "@%#*+=-:. "のような形で輝度によって指定する文字を変えるようにしているが、よく見るようなASCIIアートと比べると文字数が少ないため、表現も限定的になってしまう。

画像の読み込みと前処理

画像を読み込み、グレースケール変換する。通常の画像はBGRの3要素扱う必要があるため大変である。

画像処理で、よくグレースケールが出てくるがあまり知らなかったので、簡単に調べた。

カラー画像は一般的にRGB(Red, Green, Blue)の3つのチャンネルで表されますが、グレースケール画像は各ピクセルが単一の明るさを表す1つのチャンネルで構成されます。

画像を処理する場合に、余計な情報を落として画像を1つのチャンネルで表現できるため、閾値の設定等が容易に可能になるイメージである。

またBGRとは何ぞやとなったので簡単に調べてみたところ以下の記事がヒットした。
どうやら順序が違うだけのようだ。

    # 画像の読み込み
    img = cv2.imread(image_path)
    if img is None:
        raise IOError("画像が見つかりません。パスを確認してください。")

    # グレースケール変換
    gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)

次に画像を指定のサイズにリサイズする。

    # 指定の幅にリサイズして高さはアスペクト比を保つ
    height, width = gray.shape
    aspect_ratio = height / width
    new_height = int(aspect_ratio * new_width)
    gray_resized = cv2.resize(gray, (new_width, new_height))

背景の推定

最初はこの設定をしていなかったが、これがないとすべてを埋めてしまいASCIIアートが埋もれてしまうため設定した。

最頻値の輝度を特定し、最頻値のものは「 」半角スペースで埋めるようにしている。最も多い部分を表現しないことで、ASCIIアートが際立つ。

    # ヒストグラムを計算し、最も多い輝度値(max_bin)を取得
    hist = cv2.calcHist([gray_resized], [0], None, [256], [0, 256])
    max_bin = np.argmax(hist)

どの程度の明るさか判定する

ブロック単位で分割して、平均輝度を求める。ピクセル単位ではASCIIアートにするには細かすぎるため、ブロックに分割して画像のノイズなどに影響されなようにする。

    # ブロック単位で平均輝度を計算し、ASCII文字にマッピング
    ascii_art_lines = []
    h, w = gray_resized.shape
    num_chars = len(ASCII_CHARS)

    for row_start in range(0, h, block_size):
        row_end = min(row_start + block_size, h)  # 画像端を超えないように
        ascii_line = []
        for col_start in range(0, w, block_size):
            col_end = min(col_start + block_size, w)
            block = gray_resized[row_start:row_end, col_start:col_end]

            # ブロック内の平均輝度を求める
            avg_val = np.mean(block)

アスキー文字にマッピングする

アスキー文字の数で割ることで、どの輝度がどのASCII文字にマッピングされるのが適切か判別する。

            # 「最も多い輝度」に近い場合はスペースに置き換える
            if abs(avg_val - max_bin) <= threshold:
                ascii_char = ' '
            else:
                # 通常の方法でマッピング(0~255 -> 0~(num_chars-1))
                index = int(avg_val * (num_chars - 1) / 255)
                ascii_char = ASCII_CHARS[index]

            ascii_line.append(ascii_char)

        ascii_art_lines.append("".join(ascii_line))

全体のコード

import cv2
import numpy as np

ASCII_CHARS = "@%#*+=-:. "

def convert_to_ascii_block_with_space_bg(image_path, block_size=4, new_width=100, threshold=10):
    """
    画像ファイルを読み込み、ブロック単位でASCIIアートに変換して返す。
    画像全体で最頻出の輝度に近いブロックは背景(スペース)に置き換える。
    """
    # 画像の読み込み
    img = cv2.imread(image_path)
    if img is None:
        raise IOError("画像が見つかりません。パスを確認してください。")

    # グレースケール変換
    gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)

    # 指定の幅にリサイズして高さはアスペクト比を保つ
    height, width = gray.shape
    aspect_ratio = height / width
    new_height = int(aspect_ratio * new_width)
    gray_resized = cv2.resize(gray, (new_width, new_height))

    # ヒストグラムを計算し、最も多い輝度値(max_bin)を取得
    # histはshapeが(256, 1)になる(0~255の輝度をカウント)
    hist = cv2.calcHist([gray_resized], [0], None, [256], [0, 256])
    max_bin = np.argmax(hist)  # 輝度値のうち、カウントが最大のもの

    # ブロック単位で平均輝度を計算し、ASCII文字にマッピング
    ascii_art_lines = []
    h, w = gray_resized.shape
    num_chars = len(ASCII_CHARS)

    for row_start in range(0, h, block_size):
        row_end = min(row_start + block_size, h)  # 画像端を超えないように
        ascii_line = []
        for col_start in range(0, w, block_size):
            col_end = min(col_start + block_size, w)
            block = gray_resized[row_start:row_end, col_start:col_end]

            # ブロック内の平均輝度を求める
            avg_val = np.mean(block)

            # 「最も多い輝度」に近い場合はスペースに置き換える
            if abs(avg_val - max_bin) <= threshold:
                ascii_char = ' '
            else:
                # 通常の方法でマッピング(0~255 -> 0~(num_chars-1))
                index = int(avg_val * (num_chars - 1) / 255)
                ascii_char = ASCII_CHARS[index]

            ascii_line.append(ascii_char)

        ascii_art_lines.append("".join(ascii_line))

    return "\n".join(ascii_art_lines)


if __name__ == "__main__":
    import sys

    # 使い方:
    #   python ascii_art_block_bg.py image.png
    # ブロックサイズや幅、thresholdはお好みで調整
    image_path = sys.argv[1] if len(sys.argv) > 1 else "image.png"
    ascii_art = convert_to_ascii_block_with_space_bg(
        image_path, 
        block_size=2, 
        new_width=80,   # 幅を少し狭めにしてみる例
        threshold=10    # 近いとみなすしきい値
    )
    print(ascii_art)

処理前:
image.png

image.png

image.png

画像は以下から拝借

処理後:
image.png

image.png

image.png

まとめ

シンプルな実装ではあるものの、それなりにASCIIアートっぽいものが作れるようになったとは思う。

しかし、人物画などの背景があるものなどはやはり難しいところがあると感じた。
結局ほとんど画像処理という画像処理をやっていないが、今回は良しとしよう、

0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?