基于Python如何实现简易的动漫图片转换器

2023-06-12 16:41:05 转换器 简易 如何实现

本篇内容介绍了“基于Python如何实现简易的动漫图片转换器”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

效果图

基于Python如何实现简易的动漫图片转换器

UI界面的制作使用的还是pyqt5,因为用习惯了使用起来相当方便,接下来还是先将使用到的python非标准库列举一下。

# PyQt5相关模块
from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *

# 动漫图片制作的业务模块
import cv2
import sys
import os

# 日志模块
from loguru import logger

制作UI界面,创建CartoonUI类,用于初始化图片转换器的应用。CartoonUI类继承自QWidget,因为这里只需要制作一个单页面的应用,因此,这里只继承了QWidget。

class CartoonUI(QWidget):
    def __init__(self):
        '''
        初始化UI界面应用
        '''
        super(CartoonUI, self).__init__()
        self.init_ui()

    def init_ui(self):
        '''
        自定义实现的ui应用函数
        :return:
        '''
        self.setWindowTitle('动漫图片转换器  公众号:Python 集中营')
        self.setWindowIcon(QIcon('ico.png'))
        self.setFixedWidth(500)

        hbox = QHBoxLayout()
        self.input_image_path = QLineEdit()
        self.input_image_path.setPlaceholderText('源图片路径')
        self.input_image_path.setReadOnly(True)

        self.input_image_btn = QPushButton()
        self.input_image_btn.setText('导入源图片')
        self.input_image_btn.clicked.connect(self.input_image_btn_click)

        self.generate_btn = QPushButton()
        self.generate_btn.setText('一键生成动漫图片')
        self.generate_btn.clicked.connect(self.generate_btn_click)

        hbox.addWidget(self.input_image_path)
        hbox.addWidget(self.input_image_btn)
        hbox.addWidget(self.generate_btn)

        self.thread_ = WorkThread(self)
        self.thread_.finished.connect(self.finished)

        self.setLayout(hbox)

    def input_image_btn_click(self):
        '''
        input_image_btn按钮绑定的槽函数,
        用于实现打开文件浏览项
        :return:
        '''
        im_path = QFileDialog.getOpenFileName(self, os.getcwd(), '打开图片',
                                              'Image File(*.jpg);;Image File(*.png);;Image File(*.jpeg)')
        self.input_image_path.setText(im_path[0])

    def generate_btn_click(self):
        '''
        generate_btn按钮绑定的槽函数,
        用于启动业务子线程
        :return:
        '''
        self.thread_.start()
        self.generate_btn.setEnabled(False)

    def finished(self,finished):
        '''
        接收子线程中finished变量,
        判定子线程业务是否执行完成,若执行完成则将按钮状态改变为可点击状态
        :param finished:
        :return:
        '''
        if finished is True:
            self.generate_btn.setEnabled(True)

创建WorkThread类,继承自QThread子线程,子线程这里专门用来实现对普通图片的准换过程。之所以使用子线程来完成业务实现,是因为通常直接在主线程中完成业务会导致主线程异常卡死的情况。

单独使用子线程来完成业务会保证PyQt5的主线程正常的运行,将业务实现和界面应用分离开来。

class WorkThread(QThread):
    finished = pyqtSignal(bool)

    def __init__(self, parent=None):
        '''
        子线程初始化函数
        :param parent:
        '''
        super(WorkThread, self).__init__(parent)
        self.parent = parent
        self.working = True

    def __del__(self):
        '''
        子线程停止函数
        :return:
        '''
        self.working = False
        self.wait()

    def run(self):
        '''
        子线程执行函数
        :return:
        '''
        try:
            input_picture_name = os.path.basename(self.parent.input_image_path.text().strip())
            logger.info(input_picture_name)
            output_picture_name = 'cartoon_' + input_picture_name
            num_down = 2  # 缩减像素采样的数目
            num_bilateral = 7  # 定义双边滤波的数目
            img_rgb = cv2.imread(input_picture_name)  # 读取图片
            # 用高斯金字塔降低取样
            img_color = img_rgb

            for _ in range(num_down):
                img_color = cv2.pyrDown(img_color)

            # 重复使用小的双边滤波代替一个大的滤波
            for _ in range(num_bilateral):
                img_color = cv2.bilateralFilter(img_color, d=9, sigmaColor=9, sigmaSpace=7)

            # 升采样图片到原始大小
            for _ in range(num_down):
                img_color = cv2.pyrUp(img_color)

            # 转换为灰度并且使其产生中等的模糊
            img_gray = cv2.cvtColor(img_color, cv2.COLOR_RGB2GRAY)
            img_blur = cv2.medianBlur(img_gray, 7)
            # 检测到边缘并且增强其效果
            img_edge = cv2.adaptiveThreshold(img_blur, 255,
                                             cv2.ADAPTIVE_THRESH_MEAN_C,
                                             cv2.THRESH_BINARY,
                                             blockSize=9,
                                             C=2)
            # 转换回彩色图像
            img_edge = cv2.cvtColor(img_edge, cv2.COLOR_GRAY2RGB)
            img_cartoon = cv2.bitwise_and(img_color, img_edge)
            # 保存转换后的图片
            cv2.imwrite(output_picture_name, img_cartoon)

            logger.info('动漫图片转换完成!')
            self.finished.emit(True)
        except Exception as e:
            logger.error(e)

实现完成上述所有的业务之后,需要使用main函数将PyQt5应用加入到主体循环中,这样整个UI应用就直接拉起了。

if __name__ == '__main__':
    app = QApplication(sys.argv)
    main = CartoonUI()
    main.show()
    sys.exit(app.exec_())

相关文章