textareaCenter 未完结 其实就是iview的textarea的从写一遍 需求是光标上下居中
重点1: 一但赋值内容,光标会失去,导致光标到第一位
解决方法 设置一个状态位isChange,编辑的时候不进行watch更新,因为emit会自动改变外层的值,触发watch
解决方法2 找回上一次的值进行替换
重点2:div可编辑 contenteditable="true"
重点3:div内只能是文本 不要html
-webkit-user-modify: read-write-plaintext-only;
user-modify: read-write-plaintext-only;
重点4: 上下高度居中(多行)
display: table-cell; vertical-align: middle;
重点5:移动光标
// 光标移动到起始偏移位置(具体看上下文代码)
document.getSelection().collapse(_this.firstChild, txtAnchorOffset)
重点5:输入中文的长度计算
重点6:复制黏贴事件的处理(最大长度)光标如何恢复原始位置,超出最大长度如何处理
最后 fail 的地方是
1,没找到中文拼音输入多字,按空格等按键确认的文字是什么,就是没找到怎么获取
2. 没找到中文输入前的文字是什么
3. 多行的文字进行处理,我记得是个数组node
4. 没能恢复到没有输入的前的状态
小计:
发现iview的textarea,在这块也存在光标位置计算不准确,它是把文字删了,恢复之前的状态,但是光标位置没有恢复。
参考
-webkit-user-modify
DIV可编辑后,与限制输入及光标偏移的纠葛
https://www.cnblogs.com/echon-cw/p/9809524.html
限制可编辑div只能输入纯文本
https://www.cnblogs.com/libo0125ok/p/8143642.html
<!--
* @description text
!-->
<template>
<div ref="textareaCenter" id="textareaCenter"
@focus="isChange = false"
@blur="isChange = true"
class="textareaCenter" contenteditable="true" v-text="innerValue">
</div>
</template> <script>
// import privateZenMixins from '@/view/biz/input/components/privateZenMixins.js'
export default {
name: 'textareaCenter',
// mixins: [privateZenMixins],
components: {},
props: {
maxlength: {
type: Number,
default:
},
value: {
type: String,
default: ''
}
},
data () {
return {
isChange: true,
innerValue: this.value
}
},
watch: {
value (val) {
if (this.isChange) {
this.innerValue = val
}
},
innerValue (val) {
console.info('innerValue', val)
}
// value (val) {
// console.info('this.$refs.textareaCenterRef.target.innerText', this.$refs.textareaCenterRef.innerText)
// // console.info('watch value', val)
// // this.$nextTick(() => {
// // this.valueInner = this.getMaxText(val)
// // })
//
// // this.$emit('input', this.valueInner)
// // console.info('watch this.valueInner', this.valueInner)
// if (!this.lock) {
// console.info('watch lock val', val)
// this.valueInner = this.getMaxText(val)
// this.$refs.textareaCenterRef.innerText = this.getMaxText(val)
// }
// }
},
computed: {},
methods: {
// oEmit (event, oldtxt, firstChild, txtAnchorOffset)
oEmit (event) {
if (this.$refs.textareaCenter.innerText.replace(/\n/g, 'n').length >= this.maxlength) {
event.preventDefault()
}
console.info('oEmit', this.$refs.textareaCenter.innerText.replace(/\n/g, 'n').length)
this.$emit('input', this.$refs.textareaCenter.innerText)
},
init () {
let _this
let oldDate
let noteMax = this.maxlength
let oThis = this
// let that = oThis.$refs.textareaCenter.innerText
let content = document.getElementById('textareaCenter')
// 注册中文的输入事件,
let isCN = false
content.addEventListener('compositionstart', function (event) {
oThis.oEmit(event)
console.info('compositionstart')
isCN = true
console.info('kkk3', this.textContent.replace(/\n/g, 'n').length)
// 撤销预输入内容,必须否则会替代末尾字符 if (this.textContent.replace(/\n/g, 'n').length >= noteMax) {
event.preventDefault()
}
})
content.addEventListener('compositionend', function (event) {
oThis.oEmit(event)
console.info('compositionend')
isCN = false
console.info('oldDate', oldDate, 'event.clipboardData.getData(\'Text\').length')
oThis.$refs.textareaCenter.innerText = oThis.$refs.textareaCenter.innerText.slice(, noteMax)
document.getSelection().collapse(_this.firstChild, txtAnchorOffset)
})
// 注册文本输入事件,获取光标的起止偏移数据,如果是非中文以及超出长度的输入,则撤销本次操作
let txtAnchorOffset, txtFocusOffset
content.addEventListener('textInput', function (event) {
oThis.oEmit(event)
console.info('textInput', oThis.innerValue, '11jj', content.textContent)
let _sel = document.getSelection()
txtAnchorOffset = _sel.anchorOffset
txtFocusOffset = _sel.focusOffset console.info('kkk2', this.textContent.replace(/\n/g, 'n').length)
// 必须加上isCN的判断,否则获取不到正确的光标数据
if (!isCN && this.textContent.replace(/\n/g, 'n').length >= noteMax) {
event.preventDefault()
}
})
// 注册粘贴事件,获取粘贴数据的长度
let pastedLength
content.addEventListener('paste', function (event) {
oThis.oEmit(event)
console.info('paste')
if (!event.clipboardData) return
pastedLength = event.clipboardData.getData('Text').length
let txtLength = oThis.$refs.textareaCenter.innerText.replace(/\n/g, 'n').length
if ((pastedLength + txtLength) >= noteMax) {
event.preventDefault()
}
}) // 注册输入事件,对输入的数据进行
content.addEventListener('input', function (event) {
oThis.oEmit(event)
console.info('input')
setTimeout(function () {
oThis.oEmit(event)
console.info('input setTimeout')
// oldDate
if (!isCN) {
_this = content
console.info('kkk1', _this.textContent.replace('\n', '3333333n').length)
if (_this.textContent.replace(/\n/g, 'n').length > noteMax) {
let data = _this.textContent
if (pastedLength > ) {
oldDate = data.slice(, txtAnchorOffset) + data.slice(txtFocusOffset + pastedLength, data.length)
// 粘贴字符串长度置为0,以免影响到下一次判断。
pastedLength =
} else {
oldDate = data.slice(, txtAnchorOffset) + data.slice(txtFocusOffset, data.length)
}
// 再次截取最大长度字符串,防止溢出
_this.textContent = oldDate.slice(, noteMax)
// mThis.$emit('input', _this.textContent)
// oThis.oEmit(event, oldDate.slice(0, noteMax), _this.firstChild, txtAnchorOffset)
oThis.oEmit(event)
console.info('_this.textContent', _this.textContent)
// console.info('mThis', mThis)
// 光标移动到起始偏移位置
document.getSelection().collapse(_this.firstChild, txtAnchorOffset)
}
}
}, )
})
}
// getMaxText (txt) {
// let ret
// if (this.maxlength === 0) {
// ret = txt
// } else {
// if (txt.length > this.maxlength) {
// ret = txt.substr(0, this.maxlength)
// } else {
// ret = txt
// }
// }
// console.info('ret', ret)
// return ret
// },
// textareaCenterInput (item) {
// // this[attr] = api.getStrByteLen(this[attr], length)
// // console.info('textareaCenterInput', item)
// // innerText innerHTML
// // item.target.innerText = this.getMaxText(item.target.innerText)
// this.$emit('input', item.target.innerText)
// } },
created () {},
mounted () {
this.init()
console.info('value', this.value)
}
}
</script>
<style lang="less">
/*.textareaCenterWrap {*/
/*display: table;*/
/*width: 100%;*/
/*}*/
.textareaCenter {
background-color: #CCCCCC;
-webkit-user-modify: read-write-plaintext-only;
user-modify: read-write-plaintext-only;
width: %;
height: %;
margin: ;
/*!*display: inline-block;*!*/
display: table-cell;
text-align: left;
padding: 10px;
/*!*padding-left: 10px;*!*/
vertical-align: middle;
white-space: pre-wrap;
font-size: .14rem;
font-family: "微软雅黑 regular";
&:focus {
outline: #d5e8fc auto 0px !important;
}
} </style>
textareaCenter 未完结 其实就是iview的textarea的从写一遍 需求是光标上下居中的更多相关文章
- DIV当textarea使用,在聚焦的时候将光标移动到内容的末尾
#### DIV当textarea使用,在聚焦的时候将光标移动到内容的末尾 #### <style type="text/css"> .test_box { width ...
- 未完结第八节 JBPM流程节点
1.12个节点介绍 2.Node节点
- NOIP刷题建议(未完结)
1NOIP提高组真题 2NOI部分题目 为什么要写这个? 主要是一个人在硕大的机房里打(wan)代(you)码(xi),没多少时间了,所以打算来总结一下. 这个也是为我接下来冲刺做一个准备. 这个会不 ...
- 【Java线程安全】 — 常用数据结构及原理(未完结)
本文主要记录自己对于多线程安全的学习,先来记几个线程安全模型. 首先最重要的当然是volatile和AQS了: 我们知道,整个java.cuncurrent包的核心就是volatile,CAS加自旋悲 ...
- (未完结)“文远知行杯”GDET第十四届竞赛(网络赛共10题,仅整理出6题)
刚开学没多久就打了一个网络赛,通过这次网络赛我是发现我是真的菜... 放假前校赛的排名让我有些自满,寒假丝毫没有接触ACM,一直沉迷于Steam,这个真的值得好好反省. 虽然现在大一课有点多,在学校也 ...
- MySQL查询(未完结)
MySql查询 单表查询: 查询所有字段 SELECT * FROM 表名; '*' 代表所有字段 查询指定字段 SELECT 字段名1, 字段名2 FROM 表名; 按照指定条件查询记录 1. 查询 ...
- java 集合类复习(未完结)
JAVA常用数据结构及原理分析(面试总结) https://blog.csdn.net/qq_29631809/article/details/72599708 java 中几种常用数据结构 ht ...
- #linux 命令使用 cp -未完结版
下载了sublime 解压之后,想把文件夹放到opt目录,这里用命令cp将其复制过来 johnny@johnny-pc:~$ sudo cp -r ~/下载/Sublime_2.0.2 /opt/ [ ...
- [未完结]数字微分分析法的直线绘制(DDA)
注意! 本文被第1次更新,可能存在后续更新 直线画法 直线的斜截式方程 在二维空间下,一条直线的方程可以被描述为若干种形式,其中比较常见的一种是斜截式方程: \[y=kx+b\] 其中\(k\)称为直 ...
随机推荐
- Ho|H1|p-value|p值与U值|单侧检验
生物统计学 统计推断的过程: Ho:XXXX会发生 H1:XXXX不会发生 p:XXXX会发生的概率(概率计算过程),如果是小概率,则H0不可能发生,所以拒绝H0接受H1. 概率计算过程:先设定小概率 ...
- tomcat启动不了的问题
tomcat启动的几个问题 1.端口冲突 2.非端口冲突,需要加入配置host文件 日志文件: 解决办法:https://blog.csdn.net/u012949658/article/detail ...
- jmeter后置处理器之Json提取器详解
此提取器用于提取请求返回结果中的某个值或者某一组值,用法比正则表达式要简单,标准写法为$.key,其中key为返回结果map中的一个键,如果是多层则继续用.key进行即可,如果遇到key的value值 ...
- 实现一个简易的HashMap
实现一个键的类型为int,值的类型为int的HashMap 输入一个T,表示操作次数: 之后每行接一个操作,可以包括插入.删除.修改.查询.清空.判断是否有这个键: 因为是刚学完随手敲的,所以功能粗糙 ...
- tomcat——nginx负载均衡
Tomcat一般应用在这种小型系统中应用非常广泛,是开发调试jsp的首先应用.Tomcat和其他web软甲一样具有解析HTML语言的功能,但是处理效率远不及Apacge和Nginx,所以Tomcat一 ...
- Luogu_1280_尼克的任务
题目描述 尼克每天上班之前都连接上英特网,接收他的上司发来的邮件,这些邮件包含了尼克主管的部门当天要完成的全部任务,每个任务由一个开始时刻与一个持续时间构成. 尼克的一个工作日为N分钟,从第一分钟开始 ...
- JavaScript值类型和引用类型有哪些
JavaScript值类型和引用类型有哪些 (1)值类型:数值.布尔值.null.undefined. (2)引用类型:对象.数组.函数.
- Binder机制简析(三)
注册Service Service组件运行在Server进程中,首先要将Service注册到Service Manager中,再启动一个Binder线程池来等待和处理Client的通信请求. 注册过程 ...
- 算法拾遗[4]——STL用法
主要bb一下优先队列和字符串吧. 哦还有 bitset. 优先队列 定义很容易: priority_queue<int> pq; 内部是一个堆. 基本操作 pq.top() 取堆顶元素; ...
- markdoen语法
# 标题1 ## 标题2 ### 标题3 #### 标题4 ##### 标题5 ###### 标题6 1. 有序列表1 2. 有序列表2 <!--more--> + 无序列表 * 无序列表 ...