js双击修改元素内容并提交到后端封装实现

前面发过一个版本了,后来又追加了些功能。重新发一版。新版支持select和radio。

效果图:

右上角带有绿标的,是可以修改的单元格。如果不喜欢显示绿标,可以传递参数时指定不显示,如果想改为其它颜色,也可以传递参数时指定颜色。如果觉得标记符号太小,可以通过signSize设置其大小。例如 1, 1.5, 2, 3 都可以。必须 为数字。

封装代码:

function wm_click_modify(config) {this.identify = config.identify;this.callback = config.callback;this.srcElement = null;this.id = null;this.field = null;this.input = null;this.signSize = typeof config.signSize == 'undefined' ? 1 : config.signSize;this.sign = typeof config.signObj == 'undefined' ? true : config.signObj;this.signColor = typeof config.signColor == 'undefined' ? '#f00' : config.signColor;this.dataType = null;this.datas = null;this.radio_value = null;this.select_value = null;this.selector = null;
}
wm_click_modify.prototype = {create: function () {var elem = document.querySelectorAll(this.identify)var that = this;for (i = 0; i < elem.length; i++) {if (that.sign == true) {elem[i].style.position = 'relative';elem[i].style.overflow = 'hidden';var div = document.createElement("div");div.style.position = 'absolute';div.style.backgroundColor = this.signColor;div.style.right = this.signSize * -2 + 'px';div.style.top = this.signSize * -2 + 'px';div.title = '双击空白区域可编辑'div.style.width = this.signSize * 4 + 'px';div.style.height = this.signSize * 4 + 'px';div.style.transform = "rotate(45deg)"elem[i].appendChild(div)}elem[i].addEventListener('dblclick', function (e) {that.dblclick_e(e);}, true)}},dblclick_e: function (e) {if (e.srcElement.nodeName.toLowerCase() == 'input' || e.srcElement.nodeName.toLowerCase() == 'textarea' || e.srcElement.nodeName.toLowerCase() == 'radio' || e.srcElement.nodeName.toLowerCase() == 'select') {return;}var input;var that = this;this.srcElement = e.srcElement;var id = e.srcElement.getAttribute('data-id');var field = e.srcElement.getAttribute('data-field');this.id = id;this.field = field;this.dataType = e.srcElement.getAttribute('data-datatype');var ohtml = e.srcElement.innerHTML;var ovalue = e.srcElement.innerText;var container_remove = document.getElementsByClassName('container_wm_click_modify');if (container_remove.length > 0) {for (var i = 0; i < container_remove.length; i++) {container_remove[i].parentNode.innerHTML = container_remove[i].getAttribute('ohtml');}}let offsetWidth = e.srcElement.offsetWidth == 0 ? 40 : e.srcElement.offsetWidth;let offsetHeight = e.srcElement.offsetHeight == 0 ? 100 : e.srcElement.offsetHeight;e.srcElement.innerHTML = '';var container = document.createElement('span')container.className = 'container_wm_click_modify';container.setAttribute('ohtml', ohtml);e.srcElement.appendChild(container);if (this.dataType == null || this.dataType == 'text' || this.dataType == '') {input = document.createElement("textarea");this.input = input;input.id = 'wm_click_create_input_' + field + '_' + id;input.style.borderColor = "#ccc";input.style.borderWidth = "1px"input.style.padding = "5px";//input.style.resize = 'none';input.style.width = offsetWidth + 'px';input.style.height = offsetHeight + 'px';input.setAttribute('ovalue', ovalue);input.setAttribute('ovalue_html', ohtml);input.value = ovalue;e.srcElement.appendChild(input)input.focus();input.onblur = function () {if (this.getAttribute('ovalue') != this.value) {console.log('提交')var newvalue = this.value;if (typeof that.callback != 'undefined') {that.callback({ id: id, field: field, value: newvalue })} else {console.log('未定义callback')e.srcElement.innerHTML = document.getElementById(field + '_' + id).value;}} else {console.log('不提交')e.srcElement.innerHTML = this.getAttribute('ovalue_html');}}} else if (this.dataType == 'radio') {var label, span, that = thisvar data_list = e.srcElement.getAttribute('data-datalist');if (data_list == '' || data_list == null) {return;}this.datas = JSON.parse(data_list);var tmp = document.createElement('input')//tmp.id = this.field + '_' + this.id;tmp.value = ovalue;tmp.type = 'hidden';tmp.setAttribute('ovalue_html', ohtml);tmp.setAttribute('ovalue', ovalue);this.input = tmp;container.appendChild(tmp)for (var k = 0; k < this.datas.length; k++) {label = document.createElement("label");label.style.marginRight = "10px"container.appendChild(label);input = document.createElement("input");input.type = "radio";input.name = field;input.value = this.datas[k].value;input.setAttribute('value_title', this.datas[k].title)if (ovalue == this.datas[k].title) {input.checked = true;input.focus();}input.style.marginRight = "6px";input.style.width = "24px";input.style.height = "24px";input.style.verticalAlign = "middle";span = document.createElement("span");span.innerHTML = this.datas[k].title;label.appendChild(span);label.insertBefore(input, span);var that = this;input.onclick = function (e) {var newvalue = this.getAttribute('value_title');that.radio_value = newvalue;if (newvalue != ovalue) {console.log('提交')tmp.value = newvalue;if (typeof that.callback != 'undefined') {that.callback({ id: id, field: field, value: this.value })} else {console.log('未定义callback')that.srcElement.innerHTML = ohtml;}} else {console.log('不提交')that.srcElement.innerHTML = ohtml;}}input.onblur = function (e) {console.log("on my god")}}} else if (this.dataType == 'select') {var label, span, that = thisvar data_list = e.srcElement.getAttribute('data-datalist');if (data_list == '' || data_list == null) {return;}this.datas = JSON.parse(data_list);var tmp = document.createElement('input')tmp.value = ovalue;tmp.type = 'hidden';tmp.setAttribute('ovalue_html', ohtml);tmp.setAttribute('ovalue', ovalue);this.input = tmp;container.appendChild(tmp)var selector = document.createElement('select');this.selector = selector;container.appendChild(selector)selector.style.padding = "2px";selector.style.fontSize = "18px"for (var k = 0; k < this.datas.length; k++) {selector.options.add(new Option(this.datas[k].title, this.datas[k].value))if (this.datas[k].title == ovalue) {selector.options[k].selected = true;}}selector.focus();selector.onchange = function (e) {var newvalue = selector.options[selector.selectedIndex].text;if (newvalue != ovalue) {that.select_value = newvalue;console.log('提交', newvalue)if (typeof that.callback != 'undefined') {that.callback({ id: id, field: field, value: this.value })} else {console.log('未定义callback')that.srcElement.innerHTML = ohtml;}}}selector.onblur = function (e) {that.srcElement.innerHTML = ohtml;}}},success: function () {try {if (this.dataType == null || this.dataType == 'text' || this.dataType == '') {this.srcElement.innerHTML = document.getElementById('wm_click_create_input_' + this.field + '_' + this.id).value;} else if (this.dataType == 'radio') {this.srcElement.innerHTML = this.radio_value} else if (this.dataType == 'select') {this.selector.onblur = null;this.srcElement.innerHTML = this.select_value;}} catch (e) {console.log('这里不要展示给用户。select在blur时已删除元素,因此会报错。而又必须在blur时干点什么')}if (this.sign == true) {var div = document.createElement("div");div.style.position = 'absolute';div.style.backgroundColor = this.signColor;div.style.right = this.signSize * -2 + 'px';div.style.top = this.signSize * -2 + 'px';div.style.width = this.signSize * 4 + 'px';div.style.height = this.signSize * 4 + 'px';div.style.transform = "rotate(45deg)"this.srcElement.appendChild(div)}},fail: function () {try {this.srcElement.innerHTML = this.input.getAttribute('ovalue_html');} catch (e) {}}
};

1、调用方法:

new wm_click_modify(json格式的参数).create();

可以看到,主要就是json格式的参数这一块怎么写。后面会说,先说说前置工作。

html里的代码也要做些修改。

原来你的代码是这样写的:

<div>sos</div>

假如这个内容是数据库中id为3的记录字段名为name的内容,现在我们想双击这个DIV可以修改,修改完后再提交到后端。我们要这样来修改:

 

<div modify data-id="3" data-field="name">sos</div>

事实上,并不局限于div。在table中的td也可以这样来弄。理论上在哪都行,这取决于选择器能否正确选择要修改的元素。

如果是td那就要这样改:

<td modify data-id="3" data-field="name">sos</td>

 html内容已经准备好了。下面我们要让双击修改效果生效,调用上面封装的代码。以表格元素td为例。js代码如下:

new wm_click_modify({identify:"td[modify]", signObj: true, signSize:1.5, signColor:'green', callback: function(res){var that = this;var loading = msgbox.show('正在提交', 'loading')wm_ajax('/admin/Setting/edit.html',{data: {id:res.id,field:res.field,value:res.value},dataType: 'json',type: 'post',sync: true,success: function(data){console.log(data)if(data.code > 0){msgbox.close(loading)msgbox.show(data.msg, 'right');that.success()} else {msgbox.close(loading)msgbox.show(data.msg, 'error');that.fail();}},error: function(data){msgbox.close(loading)msgbox.show(data.msg, 'error');that.fail();}});}}).create();

第一个参数是 identify,即指定选择器的选择。凡是能被querySelector 选择的写法均受支持。示例代码中写的是td[modify],即带modify属性的td元素。必须传递。

第二个参数是signObj,即指定是否显示可编辑指示器。就是那个小绿点。如果未指定这个参数,则默认显示。

第三个参数signSize,即指定指示器大小。取值可以为1,1.5, 2 等这些数字。如果未指定这个参数,默认为1

第四个参数是signColor,即指定指示器颜色。取值为css颜色代码,例如: 'red'    '#f00'  都可以。如果未指定这个参数,默认为红色

第五个参数是回调函数  
 

callback:function(res){//在res中,包含以下内容: id(本条数据的记录id)、field(要修改的字段名)、value(新值)取值分别为:res.idres.fieldres.value}        

系统判断内容是否有修改,如果有修改,会回调这个函数,您需要在这个函数中写提交后端的代码,如果没有修改,不会回调这个函数。

2、回调函数里的代码怎么写?

回调函数一般就是用ajax往后端提交,有一点需要注意,当后端返回到前端处理结果之后,还需要调用一下封装代码中的一个方法。伪代码如下:

const that = this;
ajax({url:'',data: {id:res.id,field:res.field,value:res.value},success: function(data){if(data.code > 0){alert('修改成功');that.success()               //这是新添加的} else {that.fail();                 //这是新添加的}},error:function(e){alert("服务器错误,请稍候再试");that.fail();                     //这是新添加的}
})

懂ajax的应该看明白了。就是加了  that.success()  和that.fail()。必须要调用,否则页面上的内容不会改变。这样,后端就收到了要修改哪条记录的哪个字段,新值是多少。

上面是对纯文本内容的修改,假如要修改的是性别呢,我们希望用户只选择男或女,并不希望输入其它内容。还有就是特殊的内容一般也只让选择不让手动修改。我们来看看select和radio的用法。

从html的修改开始。还是以table的td为例。

假设原始html内容为:

<td>sos</td>

如果想让用户双击后变成select,需要这样修改:

<td modify data-id="3" data-field="gender" data-datalist='[{"value":"0", "title":"不知道"},{"value":"1", "title":"男"},{"value":"2", "title":"女"}]' data-datatype="select">sos<td>

其中data-id和data-field和上面一样,一个指定数据表的id(唯一索引),一个指定字段名。后面datalist就是一个数据列表,用户可以选择里面的数据。data-datatype 指定元素类型,这里的值取的是select,意思是说当用户双击后,会变成select让用户选择data-datalist中的数据。如果想用radio,那只需要把data-datatype的值设置为radio即可。

因为radio没有blur事件,封装代码虽然处理了这种情况,但是还是不如select更方便。所以,2者选一的话建议使用select。

调用部分代码不变。

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

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

相关文章

涉密人员离职怎么做好安全管理?

在信息安全领域&#xff0c;涉密人员的离职安全管理具有极其重要的意义。一旦涉密人员离职&#xff0c;可能会对单位的信息安全造成威胁&#xff0c;因此必须采取有效的措施来确保涉密人员离职后的信息安全。 一、涉密人员离职安全管理的现状 目前&#xff0c;许多单位在涉密人…

QML22、常规组件Page

Page是一个容器控件,可以方便地向页面添加页眉和页脚项。 title : string 此属性保存页面标题。 header : Item 此属性保存页眉项。标题项被定位到顶部,并调整大小为页面的宽度。缺省值为空。 注意:指定一个ToolBar, TabBar,或DialogButtonBox作为页眉会自动将各自的ToolBar…

系列四、ThreadLocal的工作原理

一、内存结构图 二、工作原理 &#xff08;1&#xff09;Thread有一个类型为ThreadLocal.ThreadLocalMap threadLocals 的实例变量&#xff0c;即每个线程都有一个属于自己的ThreadLocalMap&#xff1b; &#xff08;2&#xff09;ThreadLocalMap内部维护着Entry数组&#xff0…

合理运用ChatGPT使用Python编写一个桌面便签应用

ChatGPT的编程能力也不差&#xff0c;本次我就一步一步提要求&#xff0c;让ChatGPT根据我的要求&#xff0c;编写出一个可用的&#xff0c;可打包运行的桌面便签。 代码 import sys from PyQt5.QtWidgets import QApplication, QMainWindow, QMenu, QAction, QSystemTrayIco…

EPSon打印机更换色带

1、打印机色带拆装视频 打印机色带更换 2、色带盒四周有多个卡扣&#xff0c;需从右到左依次轻微用力掰开&#xff0c;使盖板与盒体脱离&#xff0c;注意不要掰断卡扣。 3、如何将色带放入打印机色带盒&#xff1f; A、色带放入盒体时不可打乱打结&#xff0c;以免卡带&#x…

Js中clientX/Y、offsetX/Y和screenX/Y之间区别

Js中client、offset和screen的区别 前言图文解说实例代码解说 前言 本文主要讲解JavaScript中clientX、clientY、offsetX、offsetY、screenX、screenY之间的区别。 图文解说 在上图中&#xff0c;有三个框&#xff0c;第一个为屏幕&#xff0c;第二个为浏览器大小&#xff0c…

【2021集创赛】Diligent杯一等奖:基于Cortex-M3软核的智能识别称量平台

本作品参与极术社区组织的有奖征集|秀出你的集创赛作品风采,免费电子产品等你拿~活动。 杯赛题目&#xff1a;Diligent杯&#xff1a;基于FPGA开源软核的硬件加速智能平台 参赛组别&#xff1a;A组 设计任务&#xff1a; 利用业界主流软核处理器(仅限于Cortex-M系列及 RISC-V系…

使用hping3和wrk模拟泛洪

一、hping3 1、syn随机ip泛洪 hping3 --flood -S --rand-source -p 端口 目标ip hping3 -c 10000 -d 120 -S -p 80 --flood --rand-source 192.168.112.130​说明&#xff1a; -c 100000 packets 发送的数量 -d 120 packet的大小 -S 只发送syn packets -p 80 目标端口&am…

腾讯云服务器99元一年?假的,阿里云是99元

腾讯云服务器99元一年是真的吗&#xff1f;假的&#xff0c;不用99元&#xff0c;只要88元即可购买一台2核2G3M带宽的轻量应用服务器&#xff0c;99元太多了&#xff0c;88元就够了&#xff0c;腾讯云百科活动 txybk.com/go/txy 活动打开如下图&#xff1a; 腾讯云服务器价格 腾…

Oracle(2-5)Usage and Configuration of the Oracle Shared Server

文章目录 一、基础知识1、 Server Configurations服务器配置2、Dedicated server process专用服务器进程3、Oracle Shared ServerOracle共享服务器4、Benefits of Shared Server 共享服务器的优点5、Processing a Request 处理请求6、Configuring Shared Server 配置共享服务器…

BW4HANA 从头到脚 概念详解 ---- 持续更新中

1. 理解BW4HANA是干嘛的 好歹干了这么久的活了&#xff0c;从当初的啥也不懂到现在感觉啥都知道点&#xff0c;虽然知道的有限&#xff0c;但是也不是小白。渐渐的也知道了SAP开发的一些逻辑。本来咱是想当个BW的大牛的。但是现在感觉这条船要沉了是怎么回事。个人才稍微摸到点…

【SpringBoot】通过profiles设置环境

效果图&#xff0c;通过profiles设置环境 在父级pom.xml中添加配置 <profiles><profile><id>dev</id><properties><application.environment>dev</application.environment></properties><activation><activeByDefau…

jenkins springCloud项目优雅下线

文章目录 场景解决下线请求效果如图贴一个可用的部署脚本 场景 在 Spring Cloud 项目的微服务实例关闭时&#xff0c;需要首先从注册中心设置为下线&#xff0c;避免该服务的消费者继续请求该服务实例&#xff0c;导致请求失败如果我们在服务实例从注册中心取消注册后&#xff…

什么是强化学习

1.1 强化学习概述 强化学习&#xff08;reinforcement learning&#xff0c;RL&#xff09; 讨论的问题是智能体&#xff08;agent&#xff09;怎么在复杂、不确定的环境&#xff08;environment&#xff09;中最大化它能获得的奖励。如图 1.1 所示&#xff0c;强化学习由两部…

HashMap知识点总结

文章目录 HashMapConcurrentHashMap线程安全问题 HashMap 1、null作为key只能有一个&#xff0c;作为value可以有多个 2、容量&#xff1a; 1.7&#xff1a;默认161.8&#xff1a;初始化并未指定容量大小&#xff0c;第一次put才初始化容量 3、负载因子 默认0.75&#xff0…

NOSQL----redis的安装和基础命令

redis是什么 1.redis-------非关系型数据库 redis是非关系数据库的一种&#xff0c;也称为缓存型数据库。 非关系型数据库和关系型数据库 1.关系型数据库 关系型数据库是一个结构化的数据库&#xff0c;记录方式是行和列&#xff08;列&#xff1a;声明对象&#xff0c;行&am…

探索分销小程序商城开发和搭建

在数字化时代&#xff0c;小程序已经成为了企业营销的重要工具。其中&#xff0c;分销小程序商城以其独特的优势&#xff0c;吸引了众多企业的关注。下面给大家介绍如何开发分销小程序商城&#xff0c;以及小程序分销搭建的流程。 首先&#xff0c;我们需要明确什么是分销小程…

中国信息通信研究院发布《中国金融科技生态白皮书》(2023)

加gzh“大数据食铁兽”&#xff0c;回复“20231122”&#xff0c;获取材料完整版 导读 本白皮书是中国信息通信研究院连续第六年针对金融科技领域的跟踪研究成果&#xff0c;聚焦过去一年来国内外金融科技领域新的发展情况&#xff0c;重点分析了中国金融科技产业、技术、市…

七牛云产品使用介绍之CDN篇

上一篇介绍了七牛云的Kodo对象存储&#xff0c;并用Java SDK实现将本地文件上传到bucekt&#xff0c;接下来是对CDN产品的介绍 CDN&#xff08;内容分发网络&#xff09;&#xff1a;通过多级缓存实现对Kodo中的资源或者自己网站的资源的加速访问&#xff0c;让你的系统更快更强…

AMEYA360:村田首款1608M尺寸/100V静电容量1µF的MLCC实现商品化

株式会社村田制作所成功开发了用于基站、服务器和数据中心48V线路的多层陶瓷电容器“GRM188D72A105KE01”并已量产。该产品在1608M(1.60.8mm)尺寸、100V的额定电压下可实现1μF的超大静电容量(村田调查数据&#xff0c;截至2023年11月20日)。目前可向村田申请免费样品。 随着5G…