react編寫可編輯標題示例詳解
需求
因為自己換工作到了新公司,上周入職,以前沒有使用過react框架,雖然前面有學(xué)習(xí)過react,但是并沒有實踐經(jīng)驗
這個需求最終的效果是和石墨標題修改實現(xiàn)一樣的效果

初始需求
- 文案支持可編輯
- 用戶點擊位置即光標定位處
- 超過50字讀的時候,超出部分進行截斷
- 當(dāng)用戶把所有內(nèi)容刪除時,失去焦點時文案設(shè)置為 “無文案”三個字
- 編輯區(qū)域隨著編輯內(nèi)容的寬度而變化,最大寬度1000px 500px
- 失去焦點時保存文案內(nèi)容
方案設(shè)計
在看到第一眼需求的時候,想到的時候用span和input進行切換,但是這個肯定是滿足不了需求中第2點,所以首先這個需求肯定不會是兩個 標簽切換,只能一個標簽承擔(dān)展示和編輯的功能,第一反應(yīng)是用html屬性contentEditable,就有了我的第一個套方案,后因為需求的第三點實現(xiàn)上存在問題,所以被迫換了方案二(使用input標簽),下面我們詳細說說為啥棄用方案1選用方案二以及在這過程中遇到的問題。
方案一 span + contentEditable
思路
- 利用h5提供contentEditble,可實現(xiàn)需求點的1/2/5
- 監(jiān)聽focus事件和input時間,可以實現(xiàn)需求點4
- 監(jiān)聽blur事件,可以實現(xiàn)需求點3
但是 需求點中的3點,因為是用字數(shù)做的截斷,在這個方案中是實現(xiàn)不了的,所以我給出的建議方案是編輯的時候不做截斷,非編輯的時候做截斷段(是否失去焦點可用作判斷是否為編輯態(tài)的依據(jù))
代碼如下
演示demo:
import React, { useState, useRef, useEffect } from 'react';
import ReactDom from 'react-dom';
interface EditTextProps {
text: string;
// 告知父組件文案已被修改
changeText?: (text: string) => void;
}
const EditText = function (props: EditTextProps) {
useEffect(() => {
setShowText(props.text);
}, [props.text]);
const [showText, setShowText] = useState('');
const [isBlank, setIsBlank] = useState(false);
const [isFocus, setIsFocus] = useState(false);
const textRef = useRef<HTMLDivElement>(null);
const onFocus = () => {
setIsFocus(true)
}
const onInput = () => {
// 避免失去焦點的時候,標題區(qū)域明顯的閃動
setIsBlank(!textRef.current?.innerHTML);
}
const onBlur = () => {
const newTitle = textRef.current?.innerHTML || '無標題';
const oldTitle = props.text;
setIsFocus(false);
setIsBlank(false);
// 文案更新
if (newTitle !== oldTitle) {
props?.changeText(newTitle);
setShowText(getCharsByLength(newTitle, 50));
}
else {
// 文案不更新
setShowText(getCharsByLength(newTitle, 50));
if(textRef.current) {
textRef.current.innerHTML = getCharsByLength(newTitle, 50)
}
}
}
// 獲取前l(fā)ength個字符
const getCharsByLength = (title: string, length: number) => {
const titleLength = title.length;
// 假設(shè)都是非中文字符,一個中文字符的寬度可以顯示兩個非中文字符
let maxLength = length * 2;
const result = [];
for (let i = 0; i < titleLength; i++) {
const char = title[i];
// 中文字符寬度2,非中文字符寬度1
maxLength -= /[\u4e00-\u9fa5]/.test(char) ? 2 : 1;
result.push(char);
if (maxLength <= 0) {
break;
}
}
if (result.length < titleLength) {
result.push('...');
}
return result.join('');
};
return <div className="title">
{isFocus && isBlank ? <span className="title-blank">無標題</span> : ''}
<span
className="title-text"
contentEditable
suppressContentEditableWarning
ref={textRef}
onFocus={onFocus}
onInput={onInput}
onBlur={onBlur}
>{showText}</span>
</div>;
};
在這個方案中遇到的問題
如果在用戶修改之前的文案就是【無標題】,此時用戶刪除了文案所有的內(nèi)容【將文案置空】,此時失去焦點,根據(jù)需求我們應(yīng)該展示【無標題】,可是在代碼邏輯中 進行了setShowText(getCharsByLength(newTitle, 50));的處理,在不斷試探中,發(fā)現(xiàn)修改前后的showText一摸一樣,無法觸發(fā)dom的更新,針對這個問題我找到了兩個解決方式
- 方式一 在不需要更新標題,用戶觸發(fā)了失去焦點,但是并沒有修改標題時,先把showText設(shè)置為空,在setTimeout中設(shè)置會以前的標題。
嘗試了一下這個方案,從使用角度來說并不會特別明顯的閃動。不過個人覺得這個方案代碼看著很怪異
const onBlur = () => {
const newTitle = textRef.current?.innerHTML || '無標題';
const oldTitle = props.text;
setIsFocus(false);
setIsBlank(false);
// 文案更新
if (newTitle !== oldTitle) {
props?.changeText(newTitle);
setShowText(getCharsByLength(newTitle, 50));
}
else {
// 文案不更新
setShowText('');
setTimeout(() => {
setShowText(getCharsByLength(newTitle, 50));
}, 0)
}
}
- 方式二 利用ref
const onBlur = () => {
const newTitle = textRef.current?.innerHTML || '無標題';
const oldTitle = props.text;
setIsFocus(false);
setIsBlank(false);
// 文案更新
if (newTitle !== oldTitle) {
props?.changeText(newTitle);
setShowText(getCharsByLength(newTitle, 50));
}
else {
// 文案不更新
setShowText(getCharsByLength(newTitle, 50));
if(textRef.current) {
textRef.current.innerHTML = getCharsByLength(newTitle, 50)
}
}
}
存在的問題
- 無法用字數(shù)做限制
- 如果用寬度做限制,可以出現(xiàn)截斷的效果,但是內(nèi)容無法滑動
方案二 直接用input處理展示和編輯
采用修改input框樣式的方法,讓input展示和可編輯文案。整體的效果和文章開頭展示的效果一致。 canEdit這個參數(shù)時我后面加的,用來控制EditText組件是否可以編輯。遇到的問題見面后面。 演示demo:
import React, { useState, useEffect, useRef, useLayoutEffect } from 'react';
interface EditTextProps {
text: string;
canEdit?: boolean;
changeText?: (text: string) => void;
}
function EditText(props: EditTextProps) {
// 根據(jù)span獲取寬度
const witdthRef = useRef<HTMLDivElement>(null);
const [showText, setShowText] = useState('');
const [isFocus, setIsFocus] = useState(false);
const [inputWith, setInputWith] = useState(100);
const minTitleWidth = 70;
const maxTitleWidth = 500;
useEffect(() => {
setShowText(props.text);
}, [props.text]);
useLayoutEffect(() => {
dealInputWidth();
}, [showText]);
const dealInputWidth = () => {
const offsetWidth = witdthRef?.current?.offsetWidth || minTitleWidth;
// +5 防止出現(xiàn) 截斷
const width = offsetWidth < maxTitleWidth ? offsetWidth + 5 : maxTitleWidth;
setInputWith(width);
};
const titleFocus = () => {
setIsFocus(true);
};
const titleInput = (e: React.ChangeEvent<HTMLInputElement>) => {
const newTitle = e.target.value;
setShowText(newTitle);
};
const titleBlur = () => {
const newTitle = showText || '無標題';
const oldTitle = props.text;
setIsFocus(false);
if (showText !== oldTitle) {
setShowText(newTitle);
setIsFocus(false);
if (props?.changeText) {
props.changeText(newTitle);
}
} else {
setIsFocus(false);
setShowText(newTitle);
}
};
return (
<div className='wrap'>
{props.canEdit ? (
<input
value={showText}
style={{ width: inputWith }}
onFocus={titleFocus}
onChange={titleInput}
onBlur={titleBlur}
className='input'
placeholder="無標題"
/>
) : (
''
)}
{/* 為了計算文字的寬度 */}
<span ref={witdthRef} className={props.canEdit ? 'width' : 'text'}>
{showText}
</span>
</div>
);
}
踩到的坑
input自帶寬度,無法實現(xiàn)寬度隨著文案的改變而改變。
在方案一做出來后,就和UI進行了溝通在【編輯的時候用字數(shù)做截斷實現(xiàn)不了】,給出了一個建議的方案【編輯的時候不做截斷】,但是設(shè)計同學(xué)覺得不截斷的方案過丑,,,,,然后她就說能實現(xiàn) 【石墨標題編輯】時,類似的效果交互嗎???于是我就開啟了研究石墨的效果的征途中。
只發(fā)現(xiàn) 石墨用了一個input實現(xiàn)了不錯的效果,input后面放了一個span標簽,我體驗的時候,一直在想為什么會有一個span標簽?zāi)兀??(小朋友,是不是滿臉疑問)
直到我發(fā)現(xiàn)input自帶寬度,無法隨著內(nèi)容的寬度的改變而改變。此時才恍然大悟span標簽的作用。
我也采用了利用span標簽的寬度的方式來控input輸入內(nèi)容的寬度。
開玩笑,咋可能這么順利,我遇到了第二個問題
用useEffect 來監(jiān)控 witdthRef.current.offsetWidth時,拿到的是上次文案的寬度 經(jīng)過查閱資料,我發(fā)現(xiàn)了useLayoutEffect這個hook,真香
以上就是react編寫可編輯標題示例詳解的詳細內(nèi)容,更多關(guān)于react編寫可編輯標題的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
React Hooks 實現(xiàn)和由來以及解決的問題詳解
這篇文章主要介紹了React Hooks 實現(xiàn)和由來以及解決的問題詳解,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01
基于React Native 0.52實現(xiàn)輪播圖效果
這篇文章主要為大家詳細介紹了基于React Native 0.52實現(xiàn)輪播圖效果,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-11-11
React Native全面屏狀態(tài)欄和底部導(dǎo)航欄適配教程詳細講解
最近在寫 React Native 項目,調(diào)試應(yīng)用時發(fā)現(xiàn)頂部狀態(tài)欄和底部全面屏手勢指示條區(qū)域不是透明的,看起來很難受。研究了一下這個問題,現(xiàn)在總結(jié)一下解決方案,這篇文章主要介紹了React Native全面屏狀態(tài)欄和底部導(dǎo)航欄適配教程2023-01-01
使用 React 和 Threejs 創(chuàng)建一個VR全景項目的過程詳解
這篇文章主要介紹了使用 React 和 Threejs 創(chuàng)建一個VR全景項目的過程詳解,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-04-04
React Router 中實現(xiàn)嵌套路由和動態(tài)路由的示例
React Router 是一個非常強大和靈活的路由庫,它為 React 應(yīng)用程序提供了豐富的導(dǎo)航和 URL 管理功能,能夠幫助我們構(gòu)建復(fù)雜的單頁應(yīng)用和多頁應(yīng)用,這篇文章主要介紹了React Router 中如何實現(xiàn)嵌套路由和動態(tài)路由,需要的朋友可以參考下2023-05-05

