LoginSignup
5
9

【笑い男事件2024】OpenCVで顔を検出して笑い男にする

Last updated at Posted at 2024-02-03

笑い男事件発生

Qiitaを利用しているプログラマやエンジニアのほとんどが履修してるであろう作品、「攻殻機動隊」。その中の2002年から2003年に放映されたシリーズ「STAND ALONE COMPLEX」の劇中に出てくる笑い男事件と呼ばれる一連の事件の発端が2024年の2月に起こります。

ついに、時代が追い付きました!まだ電脳化もしてないのに…w。
笑い男事件の詳細を知らない方は、この機会に攻殻機動隊SACを観ましょう。

Twitter(X)の公式アカウントもお祭り騒ぎです。これは攻殻ファンにはたまりませんねぇ。
アニメの限定公開や、特設サイトまでできています。

笑い男といえばやはり、特徴的な笑い男マークですね。彼は自分の顔を知られないように人々の電脳をハッキングしてこのマークを表示させていました。事件発生を記念して、上記の特設サイトでは作中で出てきたものとは少し違った記念版の画像が配布されています。劇中ではマークの縁の文字が「I thought what I'd do was, I'd pretend I was one of those deaf-mutes(僕は耳と目を閉じ、口をつぐんだ人間になろうと考えた)」となっていたところが、「The Laughing Man Incidente 0th Anniversary:February 2024(笑い男事件0周年:2024年2月)」となっていますね。
粋なことするなぁ、公式。
img_mark_01.png

というわけで、この画像を使ってプログラムを作ってみることにしました。目指せ特A級ハッカー。

実装

タイトルにある通り、画像や動画処理用のライブラリであるOpenCV (https://opencv.org/) を使って写真から認識した顔の位置に笑い男マークを張り付けるプログラムを作ります。言語はPythonを使います。今回使ったPythonのバージョンは3.10.6です。

OpenCVをインストールするときはコマンドラインで以下のコマンドを実行します。

pip install opencv-python

OpenCVを使って画像から人の顔を識別するには人の顔を学習させたモデルである「カスケード分類器」と呼ばれるものを使います。今回はこちらから正面からの顔を認識するカスケード分類器「haarcascade_frontalface_default.xml」をダウンロードして使います。
Pythonのパッケージの中にはpip installのときに入れられたものがあるらしいけど、探すのめんどいんでこっちを使いますw。

ソースコードは以下の通り。

thelaughingman.py
import os # ファイル関係いじるので
import cv2 # OpenCVのPython用ライブラリ

# 入力画像のファイルパス
path = 'input_img/(入力画像のファイル名)'

# ファイルがなければプログラム終了
if not os.path.exists(path):
    print(("No File!"))
    exit()

# 入力画像の読み込み
input_img = cv2.imread(path)
# 笑い男マークの読み込み(アルファチャンネルで)
laughingman = cv2.imread('img/img_mark_01.png', cv2.IMREAD_UNCHANGED)

# カスケード分類器の読み込み
cascade = cv2.CascadeClassifier('haarcascade_frontalface_default.xml')
# 顔の検出, パラメータは適当に変えてください
face = cascade.detectMultiScale(input_img, scaleFactor=1.1, minNeighbors=3, minSize=(75, 75))

for (x, y, w, h) in face:
    # 画像合成用にコピー
    laughingman_tmp = laughingman.copy()
    # 顔のサイズに合わせる
    laughingman_tmp = cv2.resize(laughingman_tmp, dsize=(w, h))

    # 入力画像の顔の部分にはめ込む
    for dx in range(0, w):
        for dy in range(0, h):
            # 透過していなければ画像を入れ替える
            if laughingman_tmp[dy, dx, 3] != 0:
                for i in range(0, 3):
                    input_img[y + dy, x + dx, i] = laughingman_tmp[dy, dx, i]

# 画像の出力
cv2.imwrite('output_img/(出力画像のファイル名)', input_img)
cv2.waitKey(0)

ディレクトリ構造は次のようにします。

thelaughingman
├ thelaughingman.py -ソースコード
├ haarcascade_frontalface_default.xml -学習済みカスケード分類器
├ img -笑い男マークの画像を入れるフォルダ
│ └ 笑い男の画像
├ input_img -入力画像を入れるフォルダ
│ └ 入力画像
└ output_img -入力画像を入れるフォルダ
  └ 出力画像

さっそく実行してみます。サンプル画像は神奈川工科大学 情報学部 情報工学科 信号処理応用研究室のページからダウンロードしました。

入力画像
Lenna.png

出力画像
thelaughingman.png

みんな大好きレナさんも笑い男に電脳をハックされしまいました。

OpenCVと学習済みカスケード分類器を使えば簡単に笑い男的なことができました。彼も電脳向けのOpenCV的なライブラリを使ってたんでしょうか?

最後に

現実では全然電脳化も義体化もまだまだ技術的に無理っぽいですね。少佐みたいに全身義体の人が現れるのはいつ頃になるんでしょうか…。はよ電脳化したいなぁ。

ちなみに私事ですが、この記事を書いている現在、大学の卒論の締め切りが1週間を切っています。ヤバいです。本当は笑い男とか言ってる場合じゃないです。でもきっと卒業できると信じています。そう囁くのよ、私のゴーストが…(卒論書け)。

5
9
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
5
9