最近、お出かけすることが減って、Pythonを勉強しています。
そのPython言語の、Pillowという画像ライブラリを使ったスクリプトです。
Pythonをインストールして、Pillowのライブラリをpipでインストールしておく必要があります。
記事を書いている時点のPythonは3.8です。
前回のToycamera風に加工するスクリプトは、黒あるいは白で塗りつぶした画像を
丸いマスクでブレンドしていましたが、今回はぼかした画像を
横長な長方形のマスクでブレンドしています。
今回のスクリプトは、おもちゃっぽくするために、ブレンドした後の画像の彩度を上げて、
メディアンフィルターで質感を変えています。
以下にスクリプトのコードを書きます。
"""
ミニチュア風に上下をぼかす
tkinterを使ったGUI版
Usage : コマンドラインから以下のコマンドで起動
python miniture_gui.py
"""
import os
from PIL import Image, ImageDraw, ImageFilter, ImageTk, ImageEnhance
import tkinter as tk
from tkinter import ttk
from tkinter import filedialog, messagebox
class FormClass:
"""
ミニチュア風に加工する画面のフォームを作成するクラス
"""
def __init__(self, root):
# インスタンス変数に初期値設定
self.root = root # Windowオブジェクト
self.filepath = "" # 入力ファイル名
self.readimg = None # 入力画像
self.dst_img = None # 加工結果画像
rowno = 0 # 表示位置(行番号)
# ラベルを作成
tk.Label(root, text='入力ファイルPATH'
).grid(column=0, row=rowno)
# 入力ファイル名
self.src_path = tk.StringVar()
# テキストボックスを作成
tb_widget = tk.Entry(root,
width=120,
textvariable=self.src_path)
tb_widget.grid(column=1, row=rowno)
# 入力ファイル選択ボタンを作成
btn_widget = tk.Button(root,
text='入力ファイル選択',
command=self.select_file)
btn_widget.grid(column=2, row=rowno)
rowno += 1
# 上位置 スケールの作成
tk.Label(root, text='上位置(%)'
).grid(column=0, row=rowno)
self.upper = tk.DoubleVar()
upper_sc = tk.Scale(
root,
variable=self.upper,
orient=tk.HORIZONTAL,
from_=0.0,
to=100,
command=self.draw_effect)
upper_sc.grid(column=1,
row=rowno,
sticky=(tk.N, tk.E, tk.S, tk.W))
self.upper.set(40)
rowno += 1
# 下位置 スケールの作成
tk.Label(root, text='下位置(%)'
).grid(column=0, row=rowno)
self.lower = tk.DoubleVar()
lower_sc = tk.Scale(
root,
variable=self.lower,
orient=tk.HORIZONTAL,
from_=0,
to=100.0,
command=self.draw_effect)
lower_sc.grid(column=1,
row=rowno,
sticky=(tk.N, tk.E, tk.S, tk.W))
self.lower.set(70)
rowno += 1
# 境界ぼかし スケールの作成
tk.Label(root, text='境界ぼかし'
).grid(column=0, row=rowno)
self.blur_radius = tk.IntVar()
blur_sc = tk.Scale(
root,
variable=self.blur_radius,
orient=tk.HORIZONTAL,
from_=0,
to=100,
command=self.draw_effect)
blur_sc.grid(column=1,
row=rowno,
sticky=(tk.N, tk.E, tk.S, tk.W))
self.blur_radius.set(35)
rowno += 1
# 効果の強さ スケールの作成
tk.Label(root, text='効果の強さ'
).grid(column=0, row=rowno)
self.strength = tk.IntVar()
strength_sc = tk.Scale(
root,
variable=self.strength,
orient=tk.HORIZONTAL,
from_=0,
to=50,
command=self.draw_effect)
strength_sc.grid(column=1,
row=rowno,
sticky=(tk.N, tk.E, tk.S, tk.W))
self.strength.set(10)
rowno += 1
# ファイル保存ボタンを作成
savebtn_widget = tk.Button(root,
text='ファイル保存',
command=self.show_save_dialog)
savebtn_widget.grid(column=0, row=rowno,
sticky=(tk.N))
self.CANVAS_WIDTH = 800
self.CANVAS_HEIGHT = 600
# 画像を表示するためのキャンバスの作成
self.canvas = tk.Canvas(self.root,
width=self.CANVAS_WIDTH,
height=self.CANVAS_HEIGHT)
self.canvas.grid(column=1, row=rowno, columnspan=2)
# 入力ファイル選択ボタン押下イベント
def select_file(self):
# ファイル選択ダイアログを表示
ftypes = [("All Files", ".*"),
("JPEG Image Files", ".jpg .jpeg"),
("PNG Image Files", ".png")]
if len(self.filepath) == 0:
idir = os.path.abspath(os.path.dirname(__file__))
else:
idir = os.path.dirname(self.filepath)
self.filepath = filedialog.askopenfilename(filetypes = ftypes,
initialdir = idir)
if len(self.filepath) == 0 :
return
self.src_path.set(self.filepath)
# 画像を読み取り
self.readimg = Image.open(self.filepath)
# 加工した画像を表示
self.draw_effect()
# 加工した画像を表示する
def draw_effect(self,val=0):
if self.readimg is None :
return
# 円を描画する開始位置、終了位置を計算する
w,h = self.readimg.size
upper = h * self.upper.get() / 100.0
lower = h * self.lower.get() / 100.0
h_start = int(min(upper,lower))
h_end = int(max(upper,lower))
# マスク画像を描画する
mask = Image.new("L", (w, h), 0)
draw = ImageDraw.Draw(mask)
draw.rectangle((0, h_start, w, h_end), fill=255)
mask_blur = mask.filter(
ImageFilter.GaussianBlur(
self.blur_radius.get()))
# ぼかし画像を作成する
blur_img = self.readimg.filter(ImageFilter.GaussianBlur(self.strength.get()))
# 元の画像と、塗りつぶし画像を合成する
self.dst_img = Image.composite(self.readimg, blur_img, mask_blur)
# 合成した画像の彩度を上げる
img_enhancer = ImageEnhance.Color(self.dst_img)
self.dst_img = img_enhancer.enhance(1.5)
# medianフィルタを適用する
self.dst_img = self.dst_img.filter(ImageFilter.MedianFilter(3))
# 画像の表示倍率を計算する
rate = min(1.0,
self.CANVAS_WIDTH / self.readimg.width,
self.CANVAS_HEIGHT / self.readimg.height)
# 表示用に画像を縮小する
new_w = int(self.readimg.width * rate)
new_h = int(self.readimg.height * rate)
small_resized = self.dst_img.resize((new_w, new_h))
# キャンバスに画像を表示する。
self.im = ImageTk.PhotoImage(image=small_resized)
self.canvas.create_image(0, 0, anchor=tk.NW, image=self.im)
# ファイル保存ボタンイベント
def show_save_dialog(self):
if self.dst_img is not None :
ftypes = [("All Files", ".*"),
("JPEG Image Files", ".jpg .jpeg"),
("PNG Image Files", ".png")]
ini_fname = os.path.basename(self.filepath)
filename = filedialog.asksaveasfilename(filetypes=ftypes,
initialfile=ini_fname)
if filename:
# 保存先ファイルが指定されたら、加工した画像を保存
self.dst_img.save(filename, quality=100)
else :
tk.messagebox.showerror(title="エラー",
message="入力ファイルを指定してください")
# windowを描画
window = tk.Tk()
# windowサイズを変更
window.geometry("1000x800")
# windowタイトルを設定
window.title("Miniture effect")
# フォームを作成、表示
FormClass(window)
# 画面を操作されるのを待つ
window.mainloop()
上のスクリプトをUTF-8のエンコーディングで、miniture_gui.py というファイルに保存します。
Windowsだとコマンドプロンプト等を開いて、スクリプトを保存したフォルダにcdコマンドで移動して、
python miniture_gui.py
で実行します。 実行すると、
このような画面を表示します。
入力ファイル選択ボタンで、ファイル選択ダイアログを開きます。
そこでファイルを選択すると、
このように画像のプレビューを表示します。
「上位置」を大きくすると、上側のぼける面積が広くなります。
「下位置」を大きくすると、下側のぼける面積が狭くなります。
「境界ぼかし」を大きくすると、ぼける領域とぼけない領域の境がわかりにくくなります。
「ファイル保存ボタン」をクリックすると、保存先選択ダイアログを開きます。
以下、このスクリプトで加工した画像をアップします。
江の島にて
羽村堰の玉川上水
福生の公園
0 件のコメント:
コメントを投稿