Flask+LayUI开发手记(八):通用封面缩略图上传实现

news2025/7/27 10:43:36

       前一节做了头像上传的程序,应该说,这个程序编写和操作都相当繁琐,实际上,头像这种缩略图在很多功能中都会用到,屏幕界面有限,绝不会给那么大空间摆开那么大一个界面,更可能的处理,就是显示一个缩略图,然后点击图像,即弹出文件选择框,选择文件后,直接上传,并且将当前缩略图改为选后的图像。

        这一节就来展示这种实现过程。当然,这个实现中的缩略图更新实际上是form表单编辑的一部分,本节只展示缩略图相关的部分。主要也包括Html、JS和后台实现三个部分。

        html页面实现如下:

<form class="layui-form"  id="tbedit-form" lay-filter="tbedit-form" action="" tbd-operate="list" style="width:98%;display:none">
   <div class="layui-form-item">
        <label class="layui-form-label">文章封面</label>
        <div class="layui-input-block">
            <input type="text" class="layui-input layui-hide" id="upload" name="picture" opt-type="thumb" url-upload="/srv_thumbnail/article/" value="" />
            <img id="upload-thumb" src="/static/images/article.png" onerror="this.src='/static/images/article.png'" title="点击图片更换封面" width="100px" height="100px">
        </div>
    </div>
</form>

       在这个表单输入条目里,包括两个域,一个是input域,用于存储缩略图的文件路径,这个域是hide的,在界面上不会展示出来。新增记录时为空,编辑记录时以表单初始化方式将值初始化进来。

       另一个域是缩略图显示域,src会随input域的值更新,从而显示出图像来。src属性设置了一个缺省图像,不过设不设都不会有用,因为表单初始化时,编辑时会更新为后台存储的缩略图文件名,在新增操作时,input域的值会置为空,此时图像域的文件名也是空,展示会出错。所以要设置上onerror属性,保证在出错时可以显示缺省的缩略图。

       html的input值输入域和img图像显示域的id属性有一个隐含约定,即img图像域的id是值输入域的id加后缀‘-thumb',同时,这两个域必须是同层兄弟节点。这样的约定最大的好处,就是在同屏可以有多个表单中拥有多个同名id的缩略图。界面显示效果如下。

        增强渲染,可以做一个组件库来实现,并且通过在当前的标准输入域里配置增加属性来实现参数配置。像上面的html的input域中就多出两项属性来:opt-type="thumb" url-upload="/srv_thumbnail/article/"。这第一个opt-type就是用来配置增强组件类型为thumbnail的,第一项url-upload配置上传服务的url。

       应该说,Html页面其实就是一个空架子,用于配置必要的参数。要起作用,需要通过JS渲染。在系统实现时,layui的标准表单输入域是不够的,需要做很多增强组件,比如树型编辑域、日期域、富文本编辑域还有这个缩略图,都要进行增强渲染。缩略图组件thumb的JavaScript渲染代码如下:

//对富文本编辑域进行渲染
function render_thumb(options) {
    let iform = options.formId;
    let keyw_name = `#${iform} input[opt-type=thumb]`;
    $(keyw_name).each(function() {
	    elem_render_thumb($(this));
    });    
},

//渲染输入组件缩略图
funtion elem_render_thumb(i_elem) {
    render_upload(i_elem);
    
    function render_upload(i_elem) {
	let n_elem = i_elem.attr('id');
	let n_thumb = n_elem + '-thumb';
	let i_thumb = i_elem.siblings('#' + n_thumb);

	let url_upld = i_elem.attr('url-upload');
	if (url_upld==null) {
	    url_upld = '/srv_thumbnail/article';
	}
	let uploadInst = upload.render({
	    elem: '#' + n_thumb,
	    url: url_upld, 
	    size : 2000,
	    acceptMime: 'image/*',
	    before: function(obj){
		    obj.preview(function(index, file, result){
		        i_thumb.attr('src', result); // 图片链接(base64)
		    });
		    layer.msg('上传中', {icon: 16, time: 0});
	    },
	    done: function(rs){
		    if(rs.success == 0){
		        return layer.msg(rs.msg);
		    }
		    layer.msg(rs.msg);
		    let src_file = rs.thumbfile + '?time=' + new Date().getTime();
		    i_thumb.attr('src',src_file);
		    i_elem.val(rs.thumbfile);
	    },
	    error: function(){
		    layer.msg("上传失败");
	    }
	});

	i_elem.on('change',function(e) {
	    let img_file = i_elem.val() + '?time=' + new Date().getTime();
	    i_thumb.attr('src',img_file);
	});

    }
}

       第一段程序render_thumb是form级的,查找到指定form里所有拥有属性opt-type="thumb"的元素,然后逐个对其进行渲染,渲染元素的函数为elem_render_thumb。这段程序为了展示方便进行了修改,实际的组件库是采用layui.define方式定义的,分为在form和elem两个库中。

       文件上传采用layui上传组件upload,将其与缩略图展示域绑定,点击图片自然就弹 出选择窗开始执行了。与上一节不同的是,这次的设置是选择文件后自动上传,而不是分两阶段提交,也没有进度条效果,失败后不再支持重传(这些复杂的实现其实没必要)。

       然后,是python服务端的程序:

import logging,json,time
from PIL import Image

SRVDATA_UPLOAD = 'srvdata/uploads/'

# srvdata缩略图上传服务
@app.route('/srv_thumbnail/<path:filepath>',methods=['POST'])
def srvdata_thumbnail(filepath):
    logging.debug('POST thumbnail srvdata....')
    if request.method == 'POST':
        avt = request.files.get('file')
        sour_file_name = avt.filename
        extname = sour_file_name.split('.')[1]

        dest_filename = 'thumb_' + str(time.time()) +  '.' + extname
        save_path = SRVDATA_UPLOAD + filepath  + dest_filename
        #avt.save(save_path)
        img = Image.open(avt)
        rimg = img.resize((128,128))
        rimg.save(save_path)

        rs_data = {
            'success':1,
            'msg':'更新文章缩略图成功',
            'thumbfile': '/' + save_path,
            'code':0
        }
        return json.dumps(rs_data)

       服务端程序的主体流程就是上传文件并且将其缩小为128*128的缩略图文件并且存储。要注意在路由定义时,在路由定义后跟了文件存储的路径名。这个路由是在前端html的url-upload中定义的"/srv_thumbnail/article/",也就是说文件存储的路径为'/srvdata/uploads/article/’下。这样,一个可以在各处中使用的缩略图上传组件就完成了。

        最后,强调一下,要保证文件名输入域的值和缩略图的src联动,那么在更改值时,就必须手动触发。由于表单初始化值都是统一的,并不会单独就某个域初始化。所以,初始化值后,需要加上如下代码(新增清空时也要加),这样处理后,上面缩略图渲染程序中的i_elem.on('change',function(e) {}就可以起作用了。

//form.val(iform,rsdata);
let keyw_name = `#${iform} input,#${iform} select,#${iform} textarea`;
$(keyw_name).trigger('change');

        最后还是展示一下界面效果图:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2406045.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

低代码采购系统搭建:鲸采云+能源行业订单管理自动化案例

在能源行业数字化转型浪潮下&#xff0c;某大型能源集团通过鲸采云低代码平台&#xff0c;仅用3周时间就完成了采购订单管理系统的定制化搭建。本文将揭秘这一成功案例的实施路径与关键成效。 项目背景与挑战 该企业面临&#xff1a; 供应商分散&#xff1a;200供应商使用不同…

android关于pthread的使用过程

文章目录 简介代码流程pthread使用hello_test.cppAndroid.bp 编译过程报错处理验证过程 简介 android开发经常需要使用pthread来编写代码实现相关的业务需求 代码流程 pthread使用 需要查询某个linux函数的方法使用&#xff0c;可以使用man 函数名 // $ man pthread_crea…

如何用 HTML 展示计算机代码

原文&#xff1a;如何用 HTML 展示计算机代码 | w3cschool笔记 &#xff08;请勿将文章标记为付费&#xff01;&#xff01;&#xff01;&#xff01;&#xff09; 在编程学习和文档编写过程中&#xff0c;清晰地展示代码是一项关键技能。HTML 作为网页开发的基础语言&#x…

2025年ESWA SCI1区TOP,自适应学习粒子群算法AEPSO+动态周期调节灰色模型,深度解析+性能实测

目录 1.摘要2.粒子群算法PSO原理3.改进策略4.结果展示5.参考文献6.代码获取7.算法辅导应用定制读者交流 1.摘要 能源数据的科学预测对于能源行业决策和国家经济发展具有重要意义&#xff0c;尤其是短期能源预测&#xff0c;其精度直接影响经济运行效率。为了更好地提高预测模型…

LeetCode - 53. 最大子数组和

目录 题目 Kadane 算法核心思想 Kadane 算法的步骤分析 读者可能的错误写法 正确的写法 题目 53. 最大子数组和 - 力扣&#xff08;LeetCode&#xff09; Kadane 算法核心思想 定义状态变量: currentSum: 表示以当前元素为结束的子数组的最大和。 maxSum: 记录全局最大…

【读代码】从预训练到后训练:解锁语言模型推理潜能——Xiaomi MiMo项目深度解析

项目开源地址:https://github.com/XiaomiMiMo/MiMo 一、基本介绍 Xiaomi MiMo是小米公司开源的7B参数规模语言模型系列,专为复杂推理任务设计。项目包含基础模型(MiMo-7B-Base)、监督微调模型(MiMo-7B-SFT)和强化学习模型(MiMo-7B-RL)等多个版本。其核心创新在于通过…

DROPP算法详解:专为时间序列和空间数据优化的PCA降维方案

DROPP (Dimensionality Reduction for Ordered Points via PCA) 是一种专门针对有序数据的降维方法。本文将详细介绍该算法的理论基础、实现步骤以及在降维任务中的具体应用。 在现代数据分析中&#xff0c;高维数据集普遍存在特征数量庞大的问题。这种高维特性不仅增加了计算…

MTK-Android12-13 Camera2 设置默认视频画质功能实现

MTK-Android12-13 Camera2 设置默认视频画质功能实现 场景&#xff1a;部分客户使用自己的mipi相机安装到我们主板上&#xff0c;最大分辨率为1280720&#xff0c;但是视频画质默认的是640480。实际场景中&#xff0c;在默认视频分辨率情况下拍出来的视频比较模糊、预览也不清晰…

Linux知识回顾总结----进程状态

本章将会介绍进程的一些概念&#xff1a;冯诺伊曼体系结构、进程是什么&#xff0c;怎么用、怎么表现得、进程空间地址、物理地址、虚拟地址、为什么存在进程空间地址、如何感性得去理解进程空间地址、环境变量是如何使用的。 目录 1. 冯诺伊曼体系结构 1.1 是什么 1.2 结论 …

Linux 进程管理学习指南:架构、计划与关键问题全解

Linux 进程管理学习指南&#xff1a;架构、计划与关键问题全解 本文面向初学者&#xff0c;旨在帮助你从架构视角理解 Linux 进程管理子系统&#xff0c;构建系统化学习路径&#xff0c;并通过结构化笔记方法与典型问题总结&#xff0c;夯实基础、明确方向&#xff0c;逐步掌握…

Git 使用大全:从入门到精通

Git 是目前最流行的分布式版本控制系统&#xff0c;被广泛应用于软件开发中。本文将全面介绍 Git 的各种功能和使用方法&#xff0c;包含大量代码示例和实践建议。 文章目录 Git 基础概念版本控制系统Git 的特点Git 的三个区域Git 文件状态 Git 安装与配置安装 GitLinuxmacOSWi…

奈飞工厂官网,国内Netflix影视在线看|中文网页电脑版入口

奈飞工厂是一个专注于提供免费Netflix影视资源的在线播放平台&#xff0c;致力于为国内用户提供的Netflix热门影视内容。该平台的资源与Netflix官网基本同步&#xff0c;涵盖电影、电视剧、动漫和综艺等多个领域。奈飞工厂的界面简洁流畅&#xff0c;资源分类清晰&#xff0c;方…

Python基于蒙特卡罗方法实现投资组合风险管理的VaR与ES模型项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档&#xff09;&#xff0c;如需数据代码文档可以直接到文章最后关注获取。 1.项目背景 在金融投资中&#xff0c;风险管理是确保资产安全和实现稳健收益的关键环节。随着市场波动性的增加&#xff0c;传统…

【学习记录】Office 和 WPS 文档密码破解实战

文章目录 &#x1f4cc; 引言&#x1f4c1; Office 与 WPS 支持的常见文件格式Microsoft Office 格式WPS Office 格式 &#x1f6e0; 所需工具下载地址&#xff08;Windows 官方编译版&#xff09;&#x1f510; 破解流程详解步骤 1&#xff1a;提取文档的加密哈希值步骤 2&…

AGV|无人叉车工业语音播报器|预警提示器LBE-LEX系列性能与接线说明

LBE-LEX系列AGV|无人叉车工业语音播报器|预警提示器&#xff0c;涵盖LBE-LEI-M-00、LBE-LESM-00、LBE-LES-M-01、LBE-LEC-M-00、LBE-KEI-M-00、LBE-KES-M-00、LBE-KES-M-01、LBE-KEC-M-00等型号&#xff0c;适用于各种需要语音提示的场景&#xff0c;主要有AGV、AMR机器人、无人…

【电路笔记】-变压器电压调节

变压器电压调节 文章目录 变压器电压调节1、概述2、变压器电压调节3、变压器电压调节示例14、变压器电压调节示例25、变压器电压调节示例36、总结变压器电压调节是变压器输出端电压因连接负载电流的变化而从其空载值向上或向下变化的比率或百分比值。 1、概述 电压调节是衡量变…

多层PCB技术解析:从材料选型到制造工艺的深度实践

在电子设备集成度与信号传输要求不断提升的背景下&#xff0c;多层PCB凭借分层布局优势&#xff0c;成为高速通信、汽车电子、工业控制等领域的核心载体。其通过导电层、绝缘层的交替堆叠&#xff0c;实现复杂电路的立体化设计&#xff0c;显著提升空间利用率与信号完整性。 一…

(33)课54:3 张表的 join-on 连接举例,多表查询总结。数据库编程补述及游标综合例题。静态 sqL与动态sqL(可带参数)

&#xff08;112&#xff09;3 张表的 join-on 连接举例 &#xff1a; &#xff08;113&#xff09; 多表查询总结 &#xff1a; &#xff08;114&#xff09;数据库编程补述 &#xff1a; 综合例题 &#xff1a; 以上没有动手练习&#xff0c;不知道这样的语法是否…

centos挂载目录满但实际未满引发系统宕机

测试服务器应用系统突然挂了&#xff0c;经过排查发现是因为磁盘“满了”导致的&#xff0c;使用df -h查看磁盘使用情况/home目录使用率已经到了100%,但使用du -sh /home查看发现实际磁盘使用还不到1G&#xff0c;推测有进程正在写入或占用已删除的大文件&#xff08;Linux 系统…

KKCMS部署

目录 账号 网站目录 快看CMS使用手册 http://10.141.19.241/kkcms/install/ 常规思路&#xff1a;页面点点观察url变化&#xff0c;参数 常规思路&#xff1a;点一个功能模块抓包看什么东西&#xff0c;正确是什么样&#xff0c;错误的是什么样&#xff0c;构造参数。 账号…