用Python给二维码图片添加提示文字

一、需求:

判断当前浏览器是否为微信,是的话挑起微信支付,不是的话,显示二维码图片并提示用户到微信中打开

二、效果图:

三、代码实现:

1. 判断是否微信

# toolbox.py
from typing import Any

class UserAgent:
    def __init__(self, user_agent: str = '', request: Any = None):
        if request is not None:
            try:
                user_agent = request.headers.get('user-agent', '')  # For Sanic
            except AttributeError:
                user_agent = request.META.get('HTTP_USER_AGENT', '')  # Django
        self.user_agent = user_agent

    @property
    def is_alipay(self) -> bool:
        return "AlipayClient/" in self.user_agent

    @property
    def is_wechat(self) -> bool:
        return "MicroMessenger/" in self.user_agent

    @property
    def is_qq(self) -> bool:
        return " QQ/" in self.user_agent

    @property
    def scan_type(self) -> str:
        if self.is_wechat or self.is_qq:
            return "wechat"
        if self.is_alipay:
            return "alipay"
        return "unknown"

2. 给图片加文字 (参考了这篇文章并做了一些修改:https://www.jb51.net/article/175078.htm)

# image_text.py
"""
给图片(如二维码)添上文字
Usage::
    >>> from xxx import deco_image
    >>> deco_image(image_path, text)  # 替换旧图片
    >>> deco_image(image_path, text, new_path, color='red')  # 保留旧图片并指定文字颜色
"""
from pathlib import Path
from typing import Optional, Tuple, Union

from PIL import Image, ImageDraw, ImageFont  # pip install pillow

TIP = "请用微信扫码支付\n或分享到微信中打开"

# 获取图片宽度
def get_img_width(fname) -> int:
    return Image.open(fname).size[0]

# 获取图片高度
def get_img_height(fname) -> int:
    return Image.open(fname).size[1]

# 给图片加文字
# 生成blank_img空白图片,加上文字之后生成新图片或覆盖旧图, 宽度为origin_img原始图片的宽度

MARGIN_LEFT, MARGIN_TOP = 50, 15
FONT_SIZE = 22
FONT_COLOR = "red"

def gen_text_img(
    origin_img: Union[Path, str],
    text: str,
    img_path=None,
    color=FONT_COLOR,
    font_size: int = FONT_SIZE,
    margin_left: int = MARGIN_LEFT,
    margin_top: int = MARGIN_TOP,
    blank_img=None,
    font_path: Optional[str] = None,
    show_img: bool = False,
) -> Union[Path, str]:
    width = get_img_width(origin_img)
    if blank_img is None:
        blank_img = Path(f"/tmp/blank-{width}.png")
    elif isinstance(blank_img, str):
        blank_img = Path(blank_img)
    if not blank_img.exists():
        Image.new("RGB", (width, 70), (255, 255, 255)).save(blank_img)
    im = Image.open(blank_img)
    draw = ImageDraw.Draw(im)
    if font_path is None:
        # font_path = r"C:\Windows\Fonts\simsun.ttc"
        # font_path = "/System/Library/Fonts/Supplemental/Songti.ttc"
        font_path = "/usr/share/fonts/truetype/windows-font/Songti.ttc"
    fnt = ImageFont.truetype(font_path, font_size)
    draw.text((margin_left, margin_top), text, fill=color, font=fnt)
    if img_path is None:
        img_path = Path(origin_img)
        img_path = img_path.with_name(f"{img_path.stem}-{len(text)}{img_path.suffix}")
    im.save(img_path)
    if show_img:
        im.show()
    return img_path

# 拼接图片,把上面生成的文字图片拼接到原图上面
# 生成一张宽度一致,高度为两张图片之和的空白长图
# 分别打开图片进行粘贴到空白长图里面

def join_imgs(text_img, origin_img, new_path=None) -> None:
    w = get_img_width(text_img)
    fh = get_img_height(text_img)
    oh = get_img_height(origin_img)

    blank_long_img = Image.new("RGBA", (w, fh + oh))  # 空白长图

    font_img = Image.open(text_img).resize((w, fh), Image.ANTIALIAS)
    blank_long_img.paste(font_img, (0, 0))

    img1 = Image.open(origin_img).resize((w, oh), Image.ANTIALIAS)
    blank_long_img.paste(img1, (0, fh))
    if new_path is None:
        new_path = origin_img
    blank_long_img.save(new_path)
    blank_long_img.show()

def deco_image(
    fpath: Union[Path, str],  # 图片路径
    text: str = TIP,  # 要添加的文字
    new_path: Union[Path, str, None] = None,  # 新图片要保存的路径(默认覆盖原图)
    color: Union[str, Tuple[int, int, int]] = FONT_COLOR,  # 文字颜色
    font_size: int = FONT_SIZE,  # 文字高度
    margin_left: int = MARGIN_LEFT,
    margin_top: int = MARGIN_TOP,
) -> None:
    text_img = gen_text_img(
        fpath,
        text,
        color=color,
        font_size=font_size,
        margin_left=margin_left,
        margin_top=margin_top,
    )
    join_imgs(text_img, fpath)

3. 如果系统缺字体,那么需要去下载

sudo mkdir /usr/share/fonts/truetype/windows-font
sudo chmod 777 /usr/share/fonts/truetype/windows-font
cd /usr/share/fonts/truetype/windows-font
wget https://gitee.com/waketzheng/carstino/attach_files/703450/download/Songti.ttc  # 该文件比较大,有66.9MB

4. 调起支付或生成图片

from pathlib import Path
from hashlib import md5

import qrcode  # pip install qrcode
from sanic import Blueprint
from sanic.log import logger
from sanic.request import Request
from sanic.response import json

from .models import Order
from .image_text import deco_image
from .toolbox import UserAgent
from .utils import async_http_post, get_host
from .consts import URL_PREFIX, WX_PAY_URL

bp = Blueprint("epay", url_prefix=URL_PREFIX)

async def get_qf_mch(community):
    pass

@bp.route("/pay-link", methods=["POST"])
async def pay_link(request: Request):
    requires, data = ["bills", "total", "next"], request.json
    logger.info(f"{request.url = } ;  {request.json = }")
    # 已经1分钟内生成过对应订单的,直接去支付
    content = request.body + f"{datetime.now():%y%m%d%H%M%S}".encode()
    body = md5(content).hexdigest()
    if not (order := await Order.filter(body=body).first()):
        order = await new_pay_order(origin, data, request, body)
    mchid, mch_name = await get_qf_mch(order.community)
    if mchid:
        host = get_host(request.headers)
        if not UserAgent(request=request).is_wechat:
            # 故判断当前是否在微信里,如果不是就直接生成二维码
            frontend_url = data["next"]
            fpath = "payit/" + md5(frontend_url.encode()).hexdigest() + ".png"
            if not (p := BASE_DIR / "media" / fpath).parent.exists():
                p.parent.mkdir(parents=True)
            qrcode.make(frontend_url).save(p)
            deco_image(p)
            img_url = host + URL_PREFIX + "/media/" + fpath
            return json({"payUrl": img_url})
        return json(qf_pay_it(mchid, mch_name, order, host=host))
    url = WX_PAY_URL
    if not (request_data := order.post_data).get("mch"):
        request_data.update(mch=1)  # 未配置支付的,先用1
    res = await async_http_post(url, request_data)
    try:
        res_json = res.json()
    except Exception as e:
        logger.error(f"{e = }; {url = }; {order.post_data=}; {res.content = }")
    return json(res_json)

到此这篇关于用Python给二维码图片添加提示文字的文章就介绍到这了,更多相关Python给二维码添加文字内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Python使用MyQR制作专属动态彩色二维码功能

    Python中有一个非常有趣好玩的库MyQR,不仅可以制作各种漂亮的二维码,还可以生成动态彩色二维码. MyQR是一个能够生成自定义二维码的第三方库,你可以根据需要生成普通二维码.带图片的艺术二维码,也可以生成动态二维码. 生成动态二维码 效果图如下: 二维码扫描上图看看 我们首先要安装MyQR库,直接用pip3 install myqr(or MyQR).需要注意的是MyQR依赖于Python3,在Python2的环境下可能无法正常运行. 这个库提供了两种使用方法,一种是直接使用命令行的方式,

  • Python3批量生成带logo的二维码方法

    最近有个需求:批量生成带Logo的二维码 生成二维码比较简单,网上的资源也比较多,不赘述了.自己研究了一下加了logo并且美化了一下(网上的资源直接加Logo特别丑!!!忍不了!!!),直接上代码: def create_qrcode(url, filename): qr = qrcode.QRCode( version=1, #设置容错率为最高 error_correction=qrcode.ERROR_CORRECT_H, box_size=10, border=4, ) qr.add_da

  • 用python生成(动态彩色)二维码的方法(使用myqr库实现)

    最近真的感觉到了python生态的强大(倒吸一口凉气) 现在介绍一个可以生成动态二维码的库(myqr) 效果如图: 第一步要安装myqr库 在cmd中直接用pip安装 pip install myqr 第二步 from MyQR import myqr import os version, level, qr_name = myqr.run( words="https://www.baidu.com", # 可以是字符串,也可以是网址(前面要加http(s)://) version=1

  • 一行Python代码制作动态二维码的实现

    在GitHub上发现了一个比较有意思的项目,只需要一行Python代码就可以快捷方便生成普通二维码.艺术二维码(黑白/彩色)和动态GIF二维码. GitHub网站参加:https://github.com/sylnsfar/qrcode 用法比较简单,直接通过pip安装即可. pip3 install myqr 安装过程如下所示: 安装完成后,就可以基于命令指令生成想要的二维码了,myqr模块参数说明如下: 主要参数翻译如下: -v :定义二维码的大小,范围为 1 ~ 40,默认大小取决于输入的

  • Python qrcode 生成一个二维码的实例详解

    借助第三方库qrcode实现. 二维码图片生成借助pillow qrcode的安装 在命令行中输入 pip install qrcode[pil] 用法: 1.在命令行中输入 qr "Some text" > test.png 2.在python中输入 import qrcode img = qrcode.make('Some data here') 高级用法: 使用QRCode类 import qrcode qr = qrcode.QRCode( version=1, erro

  • Python使用qrcode二维码库生成二维码方法详解

    安装qrcode库 pip install qrcode 声明 import qrcode 使用qrcode QRCode 方法 qrcode.QRCode( version=1, error_correction=qrcode.ERROR_CORRECT_L, box_size=10, border=4, image_factory=None, mask_pattern=None ) 参数解释: version:控制二维码的大小,取值范围从1到40.取最小值1时,二维码大小为21*21.取值为

  • python二维码操作:对QRCode和MyQR入门详解

    python是所有编程语言中模块最丰富的 生活中常见的二维码功能在使用python第三方库来生成十分容易 三个大矩形是定位图案,用于标记二维码的大小.这三个定位图案有白边,通过这三个矩形就可以标识一个二维码了. QRCode 生成这个二维码只用一行 import qrcode qrcode.make("不睡觉干嘛呢").get_image().show() #设置URL必须添加http:// 安装导入QRCode pip install qrcode #方法多,体量小 安装导入MyQR

  • 使用python写的opencv实时监测和解析二维码和条形码

    今天,我实现了一个很有趣的demo,它可以在视频里找到并解析二维码,然后把解析的内容实时在屏幕上显示出来. 然后我们直入主题,首先你得确保你装了opencv,python,zbar等环境.然后这个教程对于学过opencv的人可能更好理解,但是没学过也无妨,到时候也可以直接用. 比如我的电脑上的环境是opencv2.4.x,python2.7,和最新的zbar,在Ubuntu 12.12的系统下运行的 假设你的opencv已经安装好了,那么我们就可以安装zbar 你可以先更新一下 sudo apt

  • Python django框架输入汉字,数字,字符生成二维码实现详解

    这篇文章主要介绍了Python django框架输入汉字,数字,字符转成二维码实现详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 模块必备:Python环境 + pillow + qrcode 模块 核心代码<br>import qrcode qr = qrcode.QRCode( version=2, error_correction=qrcode.constants.ERROR_CORRECT_L, box_size=20, bord

  • python-图片流传输的思路及示例(url转换二维码)

    1.场景 将URL动态生成二维码前端展示(微信支付等,)--> 1.静态文件路径访问 返回URL_name,(a标签,src 静态路由访问) 2.流传输,前端渲染 二进制流返回前端,前端根据二进制流编码类型显示 3.前端js生成 后台获取到微信支付的code_url,前端js将code_url生成二维码,并渲染 实际代码 使用python_web 框架-->tornado manager.py import os import asyncio import tornado.ioloop imp

  • 基于Python生成个性二维码过程详解

    一.问题描述 通过调用MyQR模块来实现生成个人所需二维码. 安装: pip install myqr 二.代码实现 1.普通二维码 from MyQR import myqr # 普通二维码 myqr.run( words='http://www.cnblogs.com/mayi0312', save_name='qrcode.png' ) 效果图: 2.带图片的艺术二维码 光是二维码,太单调了点.我们可以加上我们想要的图片,使二维码更具辨识度! 准备的Logo图片: 当然,也可以选择其他个人

  • Python二维码生成识别实例详解

    前言 在 JavaWeb 开发中,一般使用 Zxing 来生成和识别二维码,但是,Zxing 的识别有点差强人意,不少相对模糊的二维码识别率很低.不过就最新版本的测试来说,识别率有了现显著提高. 对比 在没接触 Python 之前,曾使用 Zbar 的客户端进行识别,测了大概几百张相对模糊的图片,Zbar的识别速度要快很多,识别率也比 Zxing 稍微准确那边一丢丢,但是,稍微模糊一点就无法识别.相比之下,微信和支付宝的识别效果就逆天了. 代码案例 # -*- coding:utf-8 -*-

  • python 使用MyQR和qrcode来制作二维码

    前言   今天下午无聊,突发奇想想玩玩二维码,于是就结合着上学期学的标识技术,用Python制作了一下二维码. 一.二维码   二维码( 2 − D i m e n s i o n a l (2-Dimensional(2−Dimensional B a r c o d e ) Barcode)Barcode),又称二维条形码.它是用某种特定的几何图形按一定规律在平面(二维方向上)分布的黑白相间的图形来记录数据符号信息的:在代码编制上巧妙地利用构成计算机内部逻辑基础的"0"."

  • 通过python扫描二维码/条形码并打印数据

    需提前安装好pyzbar和opencv-python库(博主的电脑安装opencv-python库比较麻烦,但大部分都不会出现该问题) 安装方法:打开命令框输入 pip install pyzbar/opencv- python 接下来介绍代码 #首先导入本次所需要的库,最后一个csv是Python自带的csv表格操作库,这里我们需要把我们扫到的二维码信息都存入csv表格里. import cv2 from pyzbar import pyzbar import csv #然后我们设置一个变量,

随机推荐