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 = this
			var 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 = this
			var 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.id

    res.field

    res.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。

调用部分代码不变。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

SpringCloud - 新版淘汰 Ribbon,在 OpenFeign 中整合 LoadBalancer 负载均衡

目录 一、LoadBalancer 负载均衡 1.1、前言 1.2、LoadBalancer 负载均衡底层实现原理 二、整合 OpenFeign LoadBalancer 2.1、所需依赖 2.2、具体实现 2.3、自定义负载均衡策略 一、LoadBalancer 负载均衡 1.1、前言 在 2020 年以前的 SpringCloud 采用 Ribbon 作为负载…

C语言矩阵乘积(ZZULIOJ1127:矩阵乘积)

题目描述 计算两个矩阵A和B的乘积。 输入第一行三个正整数m、p和n&#xff0c;0<m,n,p<10&#xff0c;表示矩阵A是m行p列&#xff0c;矩阵B是p行n列&#xff1b;接下来的m行是矩阵A的内容&#xff0c;每行p个整数&#xff0c;用空格隔开&#xff1b;最后的p行是矩阵B的内…

Java架构师软件架构开发

目录 1 基于架构的软件开发导论2 ABSD架构方法论3 ABSD方法论具体实现4 ABSD金融业案例5 基于特定领域的软件架构开发导论6 DSSA领域分析7 DSSA领域设计和实现8 DSSA国际电商平台架构案例9 架构思维方法论概述10 AT方法论和案例想学习架构师构建流程请跳转:Java架构师系统架构…

人工智能和AR/VR:AI在AR和VR中扮演什么角色?行业应用有哪些?

增强现实技术和虚拟现实技术&#xff08;AR/VR&#xff09;发展前景广阔&#xff0c;备受各大企业关注。事实上&#xff0c;近四分之三的行业领导者表示&#xff0c;他们预计这些沉浸式技术将于未来五年内成为主流。高盛公司报告称&#xff0c;到2025年&#xff0c;AR/VR行业值…

485 实验

485(一般称作 RS485/EIA-485)隶属于 OSI 模型物理层&#xff0c;是串行通讯的一种。电气特性规定 为 2 线&#xff0c;半双工&#xff0c;多点通信的类型。它的电气特性和 RS-232 大不一样。用缆线两端的电压差值 来表示传递信号。RS485 仅仅规定了接受端和发送端的电气特性。它…

【网络奇缘】- 计算机网络|性能指标|体系结构

&#x1f308;个人主页: Aileen_0v0&#x1f525;系列专栏: 一见倾心,再见倾城 --- 计算机网络~&#x1f4ab;个人格言:"没有罗马,那就自己创造罗马~" 目录 温故而知新 计算机网络性能指标 时延 时延带宽积 往返时延RTT 访问百度​编辑 访问b站 访问谷歌 …

Centos 7 安装yum(针对python卸载yum出错)

提前下载所需安装包&#xff0c;按照下面顺序安装即可完成&#xff0c;每个依赖包必须正确安装 下载地址&#xff1a;http://mirrors.163.com/centos/7/os/x86_64/Packages/ rpm -qa|grep python|xargs rpm -ev --allmatches --nodeps ##强制删除已安装程序及其关联 whereis …

VsCode学习

一、在VsCode 上编写第一个C语言 在VsCode上写代码都是先打开文件夹&#xff0c;这样也方便管理代码和编译器产生的可执行程序&#xff0c;VsCode生成的配置文件等。 1.1打开文件夹 写代码前&#xff0c;首先创立一个文件夹存储以后我们写的VsCode代码&#xff0c;便于管理。…

Scala如何写一个通用的游戏数据爬虫程序

以前想要获取一些网站数据的时候&#xff0c;都是通过人工手动复制粘贴&#xff0c;这样的效率及其低下。数据少无所谓&#xff0c;如果需要采集大量数据&#xff0c;手动就显得乏力了。半夜睡不着&#xff0c;爬起来写一段有关游戏商品数据的爬虫通用模板&#xff0c;希望能帮…

Modbus TCP

Modbus &#xff08;&#x1f446; 百度百科&#xff0c;放心跳转&#xff09; 起源 Modbus 由 Modicon 公司于 1979 年开发&#xff0c;是一种工业现场总线协议标准。 Modbus 通信协议具有多个变种&#xff0c;支持串口&#xff0c;以太网多个版本&#xff0c;其中最著名的…

基于STM32的色彩识别与分类算法优化

基于STM32的色彩识别与分类算法优化是一项与图像处理和机器学习相关的研究任务&#xff0c;旨在实现高效的色彩识别和分类算法在STM32微控制器上的运行。本文将介绍基于STM32的色彩识别与分类算法优化的原理和实现步骤&#xff0c;并提供相应的代码示例。 1. 色彩识别与分类概…

SpringBoot:kaptcha生成验证码

GitHub项目地址&#xff1a;GitHub - penggle/kaptcha: kaptcha - A kaptcha generation engine. kaptcha介绍 kaptcha官网&#xff08;Google Code Archive - Long-term storage for Google Code Project Hosting.&#xff09;对其介绍如下&#xff0c; kaptcha十分易于安装…

Android Spannable 使用​注意事项

1、当前示例中间的 "评论"&#xff0c;使用SpannableStringBuilder实现&#xff0c;点击评论会有高亮效果加粗&#xff0c;但再点击其它Bar时无法恢复默认样式。 2、因为SpannableString或SpannableStringBuilder中的效果是叠加的&#xff0c;恢复默认样式需要先移除…

Redisson分布式锁源码解析

一、使用Redisson步骤 Redisson各个锁基本所用Redisson各个锁基本所用Redisson各个锁基本所用 二、源码解析 lock锁 1&#xff09; 基本思想&#xff1a; lock有两种方法 一种是空参 另一种是带参 * 空参方法&#xff1a;会默认调用看门狗的过期时间30*1000&…

供应链和物流的自动化新时代

今天&#xff0c;当大多数人想到物流自动化时&#xff0c;他们会想到设备。机器人、无人机和自主卡车运输在大家的谈话中占主导地位。全自动化仓库的视频在网上流传&#xff0c;新闻主播们为就业问题绞尽脑汁。这种炒作是不完整的&#xff0c;它错过了供应链和物流公司的机会。…

黑马点评笔记 redis实现缓存

文章目录 什么是缓存?为什么要使用缓存 如何使用缓存功能实现缓存模型和思路代码实现 缓存更新策略数据库缓存不一致解决方案代码实现 什么是缓存? 缓存(Cache),就是数据交换的缓冲区,俗称的缓存就是缓冲区内的数据,一般从数据库中获取,存储于本地代码(例如: 例1:Static fi…

【VRTK】【VR开发】【Unity】8-可交互对象

【概述】 之前我们只是用了一个简单方块作为可交互对象。其实可交互对象可以有许多细节设置,包括具体抓握物体的哪个点,指定抓握的方向,指定Secondary Action允许两手互换抓握,双手抓握,用两手改变物体大小等。 【拾取物体】 要让一个物体能够被拾取,必须设置它为可互…

电脑磁盘怎么设置密码?磁盘加密软件哪个好?

电脑磁盘经常被用于存放各种重要数据&#xff0c;而为了避免数据泄露&#xff0c;我们需要为磁盘设置密码&#xff0c;以防止未授权人员使用磁盘。那么&#xff0c;电脑磁盘怎么设置密码呢&#xff1f;下面我们就一起来了解一下。 如何设置磁盘密码&#xff1f; 想要为磁盘设置…

uniapp 打包后各静态资源加载失败的问题(背景图,字体等)

原因: 1.部署地址不在域名根目录下 解决办法(推荐办法2): 办法1.如果部署在域名的文件夹下(例如h5), 则运行的基础路径修改为/h5/ 且注意路由模式 办法2.不修改运行的基础路径(还是./), 将代码中涉及背景图(background-image)和字体资源的路径前统一加,如图:

ubuntu操作系统中docker下Hadoop分布式前置环境配置实验

版本&#xff1a; centos7 hadoop 3.1.3 java JDK:1.8 集群规划&#xff1a; masterslave1slave2HDFS NameNode DataNode DataNode SecondryNameNode DataNode YARNNodeManager ResourceManage NodeManager NodeManager 1.docker容器&#xff1a; 把普通用户加入到docker组&am…