react项目配合diff实现文件对比差异功能

简介: 在React项目中,可以使用`diff`库实现文件内容对比差异功能。首先安装`diff`库,然后在组件中引入并使用`Diff.diffChars`或`Diff.diffLines`方法比较文本差异。通过循环遍历`diff`结果,可以生成不同样式的HTML元素来高亮显示文本差异。

效果图:

在这里插入图片描述
这是点击一个按钮随后显示一个Modal弹框,分为上中下三部分。
首先我们需要下载一个依赖包diff

npm install diff --save

这个引入的方式只能使用commonJS的形式引入,而不能使用ES6的形式引入

const Diff = require('diff');

里面涉及到的方法很多:

参考链接 :https://maokun.blog.csdn.net/article/details/104092182

我这里只用到了其中一个对比方法:

Diff.diffChars(oldStr, newStr[, options])

options:ignoreCase:true忽略大小写差异。默认为false。

返回值 :
在这里插入图片描述
分析:
在这里插入图片描述
返回数组对象,每个对象都有count和value属性
count:字符个数;
value:具体字符;
新增或者删除的对象属性:
added:新增标识,新增为true,否为undefined;
renoved:删除标识,删除为true,否为undefined;

基本使用版本:
拿来的例子
就是将读出来的数组进行循环,随后创建标签,最后整体追加到页面中进行展示。
在html中:

<pre id="display"></pre>
<script src="diff.js"></script>
<script>
var one = 'beep boop',
    other = 'beep boob blah',
    color = '',
    span = null;

var diff = Diff.diffChars(one, other),
    display = document.getElementById('display'),
    fragment = document.createDocumentFragment();

diff.forEach(function(part){
   
  // green for additions, red for deletions
  // grey for common parts
  color = part.added ? 'green' :
    part.removed ? 'red' : 'grey';
  span = document.createElement('span');
  span.style.color = color;
  span.appendChild(document
    .createTextNode(part.value));
  fragment.appendChild(span);
});

display.appendChild(fragment);
</script>

react第一版

此外我们还可以看到读文件后读出的换行符,所有我们还能根据这个换行符对整理好的数据进行展示:

import React, {
    Fragment } from 'react'
import {
    Modal, Divider, Spin } from 'antd';
import Button from '@com/Button'
const Diff = require('diff');
export default class DiffString extends React.Component {
   
    static defaultProps = {
   
        isModalVisible: false,
        title: "预览文件差异",
        changeIsModalVisibleState: () => {
    },
        oldStr: "",
        newStr: "",
        contextBoxLoading: false,
    }
    /**
     * oldStr  为上次文件
     * 
     * newStr 为最新文件 
     * 
     * contextBoxLoading  加载文件的loading
     * 
     * isModalVisible modal显示与否
     * 
     */

    handleCancel() {
   
        this.props.changeIsModalVisibleState()
    }
    render() {
   
        let {
    title, width, isModalVisible, contextBoxLoading, oldStr ,newStr, ...rest } = this.props
        console.log(Diff.diffChars(oldStr, newStr))
        return (
           <Modal
                title={
   <div>{
   title}<h4 style={
   {
    float: "right" }}>灰色:<span style={
   {
    color: "grey" }}>无变化</span>;绿色:<span style={
   {
    color: "green" }}>新增</span>;红色:<span style={
   {
    color: "red" }}>删除</span></h4></div>}
                visible={
   isModalVisible}
                maskClosable={
   true}
                style={
   {
    top: "20%", minWidth: "1000px" }}
                closable={
   false}
                width={
   width}
                {
   ...rest}
                footer={
   [
                    <Button rsType="noIcon" key={
   1} title="关闭" onClick={
   () => this.handleCancel()}></Button>
                ]}
            >
                <Spin tip="配置文件获取中..." spinning={
   contextBoxLoading} size="large" style={
   {
    width: "100%", height: '100%' }}>
                    <div style={
   {
    height: "600px", overflowY: "scroll" }}>
                        <div style={
   {
    wordBreak: "break-all", display: "flex" }}>
                            <div style={
   {
    width: "50%", paddingLeft: "30px" }}>
                                <h2 style={
   {
    fontWeight: "700" }}>已生效配置</h2>
                                {
   Diff.diffWordsWithSpace(newStr, oldStr).map((item, index) => {
   
                                    return (
                                        <Fragment key={
   index} >
                                            <span style={
   {
   
                                                fontSize: "0",
                                                marginBottom: "0",
                                                padding: "0",
                                                lineHeight: "40px",
                                                color: item.added ? 'red' :
                                                    item.removed ? 'green' : 'grey',
                                                fontWeight: item.added ? '600' :
                                                    item.removed ? '600' : 'normal',
                                            }}>{
   !item ? null : item.removed ? null : item.value.split('\n').map((i, indexx) => {
   
                                                return <Fragment key={
   indexx} >
                                                    {
   i && <span style={
   {
    marginBottom: "0", lineHeight: "22px", fontSize: "16px", whiteSpace: "pre" }} key={
   indexx}>{
   i}</span>}
                                                    {
   indexx !== item.value.split('\n').length - 1 ? <br /> : null}
                                                </Fragment>
                                            })} </span>
                                        </Fragment>
                                    )
                                })}
                            </div>
                            <Divider type="vertical" style={
   {
    width: "10px", height: "auto" }} />
                            <div style={
   {
    flex: "1", paddingLeft: "20px" }}>
                                <h2 style={
   {
    fontWeight: "700" }}>待下发</h2>
                                {
   Diff.diffWordsWithSpace(oldStr, newStr).map((item, index) => {
   
                                    return (
                                        <Fragment key={
   index} >
                                            <span style={
   {
   
                                                fontSize: "0",
                                                marginBottom: "0",
                                                padding: "0",
                                                lineHeight: "40px",
                                                color: item.added ? 'green' :
                                                    item.removed ? 'red' : 'grey',
                                                fontWeight: item.added ? '600' :
                                                    item.removed ? '600' : 'normal',
                                            }}>{
   !item ? null : item.removed ? null : item.value.split('\n').map((i, indexx) => {
   
                                                return <Fragment key={
   indexx} >
                                                    {
   console.log(i, item)}
                                                    {
   i && <span style={
   {
    marginBottom: "0", lineHeight: "22px", fontSize: "16px", whiteSpace: "pre" }} key={
   indexx}>{
   i}</span>}
                                                    {
   indexx !== item.value.split('\n').length - 1 ? <br /> : null}
                                                </Fragment>
                                            })} </span>
                                        </Fragment>
                                    )
                                })}
                            </div>

                        </div>
                    </div>
                </Spin>
            </Modal>
        )
    }
}

这里其实已经封装成了一个组件,使用了antd的Modal组件。
里面也是对数据进行循环,对样式进行控制,对内容进行切割,换行展示。

react第二版

最终我选择了使用行比较,最终代码如下,本人最终目的是为了留笔记,所以直接粘贴代码了:

/*
 * @Descripttion: 
 * @version: 
 * @Author: ZhangJunQing
 * @Date: 2021-11-12 17:35:44
 * @LastEditors: ZhangJunQing
 * @LastEditTime: 2022-01-12 15:08:08
 */
import React, {
    Fragment } from 'react'
import {
    Modal, Divider, Spin } from 'antd';
import Button from '@com/Button'
const Diff = require('diff');
export default class DiffString extends React.Component {
   
    static defaultProps = {
   
        isModalVisible: false,
        title: "预览文件差异",
        changeIsModalVisibleState: () => {
    },
        oldStr: "",
        newStr: "",
        contextBoxLoading: false,
    }
    /**
     * oldStr  为上次文件  已生效配置
     * 
     * newStr 为最新文件  待下发
     * 
     * contextBoxLoading  加载文件的loading
     * 
     * isModalVisible modal显示与否
     * 
     */
    state = {
   
        isSplitBRFlag: true,//是否为 对齐查看   这个变量目前不做动态改变  和changeisSplitBRFlagFun方法配合
        isOriginFlag: false
    }
    // 源文件切换变量
    changeOriginFileFun = () => {
   
        this.setState({
   
            isOriginFlag: !this.state.isOriginFlag
        })
    }
    // 是否为 对齐查看   
    changeisSplitBRFlagFun = () => {
   
        this.setState({
   
            isSplitBRFlag: !this.state.isSplitBRFlag
        })
    }
    handleCancel() {
   
        this.props.changeIsModalVisibleState()
    }
    render() {
   
        let {
    isSplitBRFlag, isOriginFlag } = this.state
        let {
    title, width, isModalVisible, contextBoxLoading, oldStr, newStr, ...rest } = this.props
        // console.log(Diff.diffLines(oldStr.replace(/\r/ig, ''), newStr.replace(/\r/ig, '')), "zjqqq")
        return (
            <Modal
                title={
   <div>{
   title}<h4 style={
   {
    float: "right" }}>灰色:<span style={
   {
    color: "grey" }}>无变化</span>;绿色:<span style={
   {
    color: "green" }}>新增</span>;红色:<span style={
   {
    color: "red" }}>删除</span></h4></div>}
                visible={
   isModalVisible}
                maskClosable={
   true}
                style={
   {
    top: "20%", minWidth: "1000px" }}
                closable={
   false}
                width={
   width}
                {
   ...rest}
                footer={
   [
                    <Button rsType="noIcon" key={
   1} title={
   !isOriginFlag ? "切换源文件" : "切换对比文件"} onClick={
   () => this.changeOriginFileFun()}></Button>,
                    <Button rsType="noIcon" key={
   2} title="关闭" onClick={
   () => this.handleCancel()}></Button>
                ]}
            >
                <Spin tip="配置文件获取中..." spinning={
   contextBoxLoading} size="large" style={
   {
    width: "100%", height: '100%' }}>
                    <div style={
   {
    height: "600px", overflowY: "scroll" }}>
                        <div style={
   {
    wordBreak: "break-all", display: "flex" }}>
                            <div style={
   {
    width: "50%", paddingLeft: "30px" }}>
                                <h2 style={
   {
    fontWeight: "700" }}>已生效配置</h2>
                                {
   Diff.diffLines(oldStr.replace(/\r/ig, ''), isOriginFlag ? oldStr.replace(/\r/ig, '') : newStr.replace(/\r/ig, '')).map((item, index) => {
   
                                    return (
                                        <Fragment key={
   index} >
                                            <span style={
   {
   
                                                fontSize: "0",
                                                marginBottom: "0",
                                                padding: "0",
                                                lineHeight: "40px",
                                                color: item.added ? 'green' :
                                                    item.removed ? 'red' : 'grey',
                                                fontWeight: item.added ? '600' :
                                                    item.removed ? '600' : '600',
                                            }}>{
   !item ? null : item.added ? (isSplitBRFlag ?
                                                item.value.split('\n').map((child, idx) => {
   
                                                    return <Fragment key={
   idx} >
                                                        {
   child && <span style={
   {
    marginBottom: "0", lineHeight: "22px", fontSize: "16px", whiteSpace: "pre", color: "transparent", userSelect: "none" }} key={
   idx}>{
   child}</span>}
                                                        {
   idx !== item.value.split('\n').length - 1 ? <br /> : null}
                                                    </Fragment>
                                                }) : null)
                                                : item.value.split('\n').map((i, indexx) => {
   
                                                    return <Fragment key={
   indexx} >
                                                        {
   i && <span style={
   {
    marginBottom: "0", lineHeight: "22px", fontSize: "16px", whiteSpace: "pre" }} key={
   indexx}>{
   i}</span>}
                                                        {
   indexx !== item.value.split('\n').length - 1 ? <br /> : null}
                                                    </Fragment>
                                                })} </span>
                                        </Fragment>
                                    )
                                })}
                            </div>
                            <Divider type="vertical" style={
   {
    width: "10px", height: "auto", minHeight: '600px' }} />
                            <div style={
   {
    flex: "1", paddingLeft: "20px" }}>
                                <h2 style={
   {
    fontWeight: "700" }}>待下发</h2>
                                {
   Diff.diffLines(isOriginFlag ? newStr.replace(/\r/ig, '') : oldStr.replace(/\r/ig, ''), newStr.replace(/\r/ig, '')).map((item, index) => {
   
                                    return (
                                        <Fragment key={
   index} >
                                            <span style={
   {
   
                                                fontSize: "0",
                                                marginBottom: "0",
                                                padding: "0",
                                                lineHeight: "40px",
                                                color: item.added ? 'green' :
                                                    item.removed ? 'red' : 'grey',
                                                fontWeight: item.added ? '600' :
                                                    item.removed ? '600' : '600',
                                            }}>{
   !item ? null : item.removed ? (isSplitBRFlag ?
                                                item.value.split('\n').map((child, idx) => {
   
                                                    return <Fragment key={
   idx} >
                                                        {
   child && <span style={
   {
    marginBottom: "0", lineHeight: "22px", fontSize: "16px", whiteSpace: "pre", color: "transparent", userSelect: "none" }} key={
   idx}>{
   child}</span>}
                                                        {
   idx !== item.value.split('\n').length - 1 ? <br /> : null}
                                                    </Fragment>
                                                }) : null)
                                                : item.value.split('\n').map((i, indexx) => {
   
                                                    return <Fragment key={
   indexx} >
                                                        {
   i && <span style={
   {
    marginBottom: "0", lineHeight: "22px", fontSize: "16px", whiteSpace: "pre" }} key={
   indexx}>{
   i}</span>}
                                                        {
   indexx !== item.value.split('\n').length - 1 ? <br /> : null}
                                                    </Fragment>
                                                })} </span>
                                        </Fragment>
                                    )
                                })}
                            </div>

                        </div>
                    </div>
                </Spin>
            </Modal>
        )
    }
}

在这里插入图片描述

上面做了两个操作 一个是切换源文件,就是只看两个文件,这个也进行了对比,目的是为了要整体格式,还有一种就是隐藏掉的那个按钮,出来的效果没有多余空行,上面多余的空行实现用了透明色和css不可选中的样式。

react第三版(改进版):

在这里插入图片描述

代码:

/*
/*
 * @Descripttion: 
 * @version: 
 * @Author: ZhangJunQing
 * @Date: 2021-11-12 17:35:44
 * @LastEditors: ZhangJunQing
 * @LastEditTime: 2022-01-18 14:31:20
 */
import React, {
    Fragment } from 'react'
import {
    Modal, Divider, Spin } from 'antd';
import Button from '@com/Button'
import {
    string } from 'prop-types';
const Diff = require('diff');
export default class DiffString extends React.Component {
   
    static defaultProps = {
   
        isModalVisible: false,
        title: "预览文件差异",
        changeIsModalVisibleState: () => {
    },
        oldStr: "",
        newStr: "",
        contextBoxLoading: false,
    }
    /**
     * oldStr  为上次文件  已生效配置
     * 
     * newStr 为最新文件  待下发
     * 
     * contextBoxLoading  加载文件的loading
     * 
     * isModalVisible modal显示与否
     * 
     */
    state = {
   
        isSplitBRFlag: true,//是否为 对齐查看   这个变量目前不做动态改变  和changeisSplitBRFlagFun方法配合
        isOriginFlag: false,
        defaultAddColor: "green",
        defaultDelColor: "red",
        defaultColor: "grey"
    }
    // 源文件切换变量
    changeOriginFileFun = () => {
   
        this.setState({
   
            isOriginFlag: !this.state.isOriginFlag,

        })

    }
    // 是否为 对齐查看   
    changeisSplitBRFlagFun = () => {
   
        this.setState({
   
            isSplitBRFlag: !this.state.isSplitBRFlag
        })
    }
    /**
     * 每次进来都显示 对比文件页面
     */
    handleCancel() {
   
        this.setState({
   
            isOriginFlag: false,//默认还是显示 对比文件页面
        })
        this.props.changeIsModalVisibleState()
    }
    /**
     * 这个方法 事件委托
     * 
     * 点击p标签 改变当前p标签的样式
     * 以及更改对面等行的p标签的样式
     * 
     * 点击左边 有背景的为defaultDelColor
     * 点击右边 有背景的为defaultAddColor
     * 不是增加不是删除的的为defaultColor
     * 
     * 2px solid #fff  默认边框 
     * 如果只给当前点击的加边框 对引起整个页面抖动
     * 
     * 所有每一个p标签都添加一个边框  颜色为底色
     * 
     * 这样每次点击的时候把所有的颜色都重置
     * 
     * 然后再将点击的p换颜色 即可
     */
    changeRightLineFun = (flag, ev) => {
   
        let {
    defaultAddColor, defaultDelColor } = this.state
        let RightFileId = flag === 'right' ? document.getElementById('RightFileId') : document.getElementById('LeftFileId')
        let LeftFileId = flag === 'right' ? document.getElementById('LeftFileId') : document.getElementById('RightFileId')
        let LeftFileIdP = LeftFileId.getElementsByTagName('p')
        let RightFileIdP = RightFileId.getElementsByTagName('p')
        ev = ev || window.event;
        let target = ev.target || ev.srcElement;
        let index;
        // 点击空P标签
        if (target.style.userSelect === 'none') {
   
            return false;
        }
        if (target.nodeName.toLowerCase() == "p") {
   
            for (let i = 0; i < RightFileIdP.length; i++) {
   
                if (RightFileIdP[i] === target)
                    index = i;
                RightFileIdP[i].style.border = "2px solid #fff"
            }
            for (let i = 0; i < LeftFileIdP.length; i++) {
   
                if (LeftFileIdP[i] === target)
                    index = i;
                LeftFileIdP[i].style.border = "2px solid #fff"
            }
            if (flag === 'right') {
   
                if (target.parentNode.style.color === defaultAddColor) {
   
                    target.style.border = `2px solid ${
     defaultAddColor}`;
                } else {
   
                    target.style.border = "2px solid #000";
                }
            } else {
   
                if (target.parentNode.style.color === defaultDelColor) {
   
                    target.style.border = `2px solid ${
     defaultDelColor}`;
                } else {
   
                    target.style.border = "2px solid #000";
                }
            }

            LeftFileIdP[index].style.border = "2px solid #000";
        }

    }
    render() {
   
        let {
    isSplitBRFlag, isOriginFlag, defaultAddColor, defaultDelColor, defaultColor } = this.state
        let {
    title, width, isModalVisible, contextBoxLoading, oldStr, newStr, fileName, ...rest } = this.props
        // console.log(Diff.diffLines(oldStr.replace(/\r/ig, ''), newStr.replace(/\r/ig, '')), "zjqqq")
        const styleObj = {
   
            lineHeight: "22px",
            border: "2px solid #fff",
            paddingTop: "2px",
            paddingBottom: "2px",
            borderRadius: "4px",
            fontSize: "16px",
            paddingLeft: "10px",
            marginBottom: "0",

        }
        return (
            <Modal
                title={
   <div style={
   {
    display: "flex", justifyContent: "space-between" }}><span style={
   {
    fontWeight: "500" }}>{
   title}</span><span style={
   {
    paddingLeft: "50px" }}>{
   fileName ? `文件名称:${
     fileName}` : ""}</span><h4 style={
   {
    float: "right" }}>灰色:<span style={
   {
    color: defaultColor }}>无变化</span>;绿色:<span style={
   {
    color: defaultAddColor }}>新增</span>;红色:<span style={
   {
    color: defaultDelColor }}>删除</span></h4></div>}
                visible={
   isModalVisible}
                maskClosable={
   true}
                style={
   {
    top: "20%", minWidth: "1000px" }}
                closable={
   false}
                width={
   width}
                {
   ...rest}
                footer={
   [
                    <Button rsType="noIcon" key="1" title={
   !isOriginFlag ? "查看源文件" : "查看对比文件"} onClick={
   () => this.changeOriginFileFun()}></Button>,
                    // <Button rsType="noIcon" key={1} title="切换对比方式" onClick={() => this.changeisSplitBRFlagFun()}></Button>,
                    <Button rsType="noIcon" key="2" title="关闭" onClick={
   () => this.handleCancel()}></Button>
                ]}
            >
                <Spin tip="配置文件获取中..." spinning={
   contextBoxLoading} size="large" style={
   {
    width: "100%", height: '100%' }}>
                    {
   /* 绑定 key   目前这个key有两种形式 false true 字符的形式 正好对应 查看源文件和对比文件两个key  */}
                    <div style={
   {
    height: "600px", overflowY: "scroll" }} key={
   String(isOriginFlag)}>
                        {
   /* // 为了每次都处于文件顶端  */}
                        <div style={
   {
    wordBreak: "break-all", display: "flex" }} >
                            <div style={
   {
    width: "50%", paddingLeft: "20px" }} id="LeftFileId">
                                <h2 style={
   {
    fontWeight: "700" }}>已生效配置</h2>
                                {
   Diff.diffLines(oldStr.replace(/\r/ig, ''), isOriginFlag ? oldStr.replace(/\r/ig, '') : newStr.replace(/\r/ig, '')).map((item, index) => {
   
                                    return (
                                        <Fragment key={
   index} >
                                            <div
                                                onClick={
   (e) => {
    this.changeRightLineFun("left", e) }}
                                                style={
   {
   
                                                    fontSize: "0",
                                                    marginBottom: "0",
                                                    padding: "0",
                                                    lineHeight: "10px",
                                                    color: item.added ? defaultAddColor :
                                                        item.removed ? defaultDelColor : defaultColor,
                                                    fontWeight: item.added ? '600' :
                                                        item.removed ? '600' : '600',
                                                }}>{
   !item ? null : item.added ? (isSplitBRFlag ?
                                                    item.value.split('\n').map((child, idx) => {
   
                                                        return <Fragment key={
   idx} >
                                                            {
   child && <p style={
   {
    whiteSpace: "pre", color: "transparent", userSelect: "none", ...styleObj }} >{
   child}</p>}
                                                            {
   idx !== item.value.split('\n').length - 1 ? <br /> : null}
                                                        </Fragment>
                                                    }) : null)
                                                    : item.value.split('\n').map((i, idx) => {
   
                                                        return <Fragment key={
   idx} >
                                                            {
   i && <p style={
   {
    backgroundColor: `${
     item.removed ? "rgb(255,227,227)" : "#fff"}`, whiteSpace: "pre", cursor: 'pointer', ...styleObj }} >{
   i}</p>}
                                                            {
   idx !== item.value.split('\n').length - 1 ? <br /> : null}
                                                        </Fragment>
                                                    })} </div>
                                        </Fragment>
                                    )
                                })}
                            </div>
                            <Divider type="vertical" style={
   {
    width: "10px", height: "auto", minHeight: '600px' }} />
                            <div style={
   {
    flex: "1", paddingLeft: "10px", paddingRight: "10px" }} id="RightFileId" >
                                <h2 style={
   {
    fontWeight: "700" }}>待下发</h2>
                                {
   Diff.diffLines(isOriginFlag ? newStr.replace(/\r/ig, '') : oldStr.replace(/\r/ig, ''), newStr.replace(/\r/ig, '')).map((item, index) => {
   
                                    return (
                                        <Fragment key={
   index} >
                                            <div
                                                onClick={
   (e) => {
    this.changeRightLineFun("right", e) }}
                                                style={
   {
   
                                                    fontSize: "0",
                                                    marginBottom: "0",
                                                    padding: "0",
                                                    lineHeight: "10px",
                                                    color: item.added ? defaultAddColor :
                                                        item.removed ? defaultDelColor : defaultColor,
                                                    fontWeight: item.added ? '600' :
                                                        item.removed ? '600' : '600',
                                                }}>{
   !item ? null : item.removed ? (isSplitBRFlag ?
                                                    item.value.split('\n').map((child, idx) => {
   
                                                        return <Fragment key={
   idx} >
                                                            {
   child && <p style={
   {
    whiteSpace: "pre", color: "transparent", userSelect: "none", ...styleObj }} >{
   child}</p>}
                                                            {
   idx !== item.value.split('\n').length - 1 ? <br /> : null}
                                                        </Fragment>
                                                    }) : null)
                                                    : item.value.split('\n').map((i, idx) => {
   
                                                        return <Fragment key={
   idx} >
                                                            {
   i && <p style={
   {
    backgroundColor: `${
     item.added ? "rgb(221,255,221)" : "#fff"}`, whiteSpace: "pre", cursor: 'pointer', ...styleObj }} key={
   idx}>{
   i}</p>}
                                                            {
   idx !== item.value.split('\n').length - 1 ? <br /> : null}
                                                        </Fragment>
                                                    })} </div>
                                        </Fragment>
                                    )
                                })}
                            </div>

                        </div>
                    </div>
                </Spin>
            </Modal>
        )
    }
}

加了一个点击事件,利用了事件委托,当然这里没有考虑效率问题,直接使用了事件委托写了。

总结来说diff这个npm还是很好用的,里面有很多的方法,可以按照字符比较、按照单词比较、按照行比较,但是传入的两个字符串,自己首先要先看看,不然由于后端传来的字符串的\r\n问题让我苦恼了很久,所以上面我统一过滤了\r,如果具体指导怎么用的,还是找官方文档吧。

最终版本(代码整理):

/*
 * @Descripttion: 
 * @version: 
 * @Author: ZhangJunQing
 * @Date: 2021-11-12 17:35:44
 * @LastEditors: ZhangJunQing
 * @LastEditTime: 2022-07-28 17:49:22
 */
import React, {
    Fragment } from 'react'
import {
    Modal, Divider, Spin } from 'antd';
import Button from '@com/Button'
import Worker from './workerJS/differ.worker'
// const Diff = require('diff');
const pStyleObj = {
   
    fontSize: "0",
    marginBottom: "0",
    padding: "0",
    lineHeight: "2px",
    wordBreak: "break-word",
    width: "100%",
    // minHeight: `${560}px`,
    overflowX: "auto"
}
const styleObj = {
   
    lineHeight: "22px",
    border: "2px solid #fff",
    paddingTop: "2px",
    paddingBottom: "2px",
    borderRadius: "4px",
    fontSize: "16px",
    paddingLeft: "10px",
    marginBottom: "0",
}
export default class DiffString extends React.PureComponent {
   
    /**
     * oldStr  为上次文件  已生效配置
     * 
     * newStr 为最新文件  待下发
     * 

     * 
     * isModalVisible modal显示与否
     * 
     */
    static defaultProps = {
   
        isModalVisible: false,
        title: "预览文件差异",
        leftTitle: "已生效配置",
        rightTitle: "待下发",
        changeIsModalVisibleState: () => {
    },
        oldStr: "",
        newStr: "",

        isDiffWords: false, //是否开启单词对比   但是这个也是在行对比的基础上进行改造的
        // 这个变量是为了 空行  使两边相同的内容行对齐
        isSplitBRFlag: true,//是否为 对齐查看   这个变量目前不做动态改变  和changeisSplitBRFlagFun方法配合
        headerHeight: 1,
    }
    /**
     * 
     * 注意:
     * 
     *  当开启 单词对比后   isDiffWords  :true   需要 设置  isSplitBRFlag false
     * 
     *     * contextBoxLoading  加载文件的loading
     */

    state = {
   
        isOriginFlag: false, //切换比较模式   默认为文件差异对比   true为源文件对比
        defaultAddColor: "green",
        defaultDelColor: "red",
        defaultColor: "grey",
        DiffLeft: [],
        DiffRight: [],
        leftSourceList: [],
        rightSourceList: [],
        contextBoxLoading: false,
    }
    myWorker = null
    componentDidMount() {
   
        this.myWorker = new Worker();
    }
    componentWillReceiveProps(props) {
   
        const {
    newStr, oldStr, isModalVisible } = props
        if (!isModalVisible) {
   
            this.myWorker.terminate()
            return false
        } else {
   
            this.myWorker = new Worker();
        }
        if (newStr === '' && oldStr === '') return
        console.time("收到数据直至结束花费时间")
        this.setState({
   
            contextBoxLoading: true
        })
        this.myWorker.postMessage({
    newStr: newStr.replace(/\r/ig, ''), oldStr: oldStr.replace(/\r/ig, '') })
        const that = this
        this.myWorker.onmessage = function (e) {
   
            const {
    DiffLeft, leftSourceList, rightSourceList } = e.data
            that.setState({
   
                DiffLeft,
                DiffRight: DiffLeft,
                leftSourceList,
                rightSourceList,
                contextBoxLoading: false
            }, () => {
   
                console.timeEnd("收到数据直至渲染结束花费时间")
            })
        }
    }
    componentWillUnmount() {
   
        this.myWorker.terminate()
    }
    // 源文件切换变量
    changeOriginFileFun = () => {
   
        this.setState({
   
            isOriginFlag: !this.state.isOriginFlag,
        })
    }


    // 是否为 对齐查看   
    // changeisSplitBRFlagFun = () => {
   
    //     this.setState({
   
    //         isSplitBRFlag: !this.state.isSplitBRFlag
    //     })
    // }


    /**
     * 每次进来都显示 对比文件页面
     */
    handleCancel() {
   
        this.setState({
   
            isOriginFlag: false,//默认还是显示 对比文件页面
        })
        this.props.changeIsModalVisibleState()
    }


    /**
     * 这个方法 事件委托
     * 
     * 点击p标签 改变当前p标签的样式
     * 以及更改对面等行的p标签的样式
     * 
     * 点击左边 有背景的为defaultDelColor
     * 点击右边 有背景的为defaultAddColor
     * 不是增加不是删除的的为defaultColor
     * 
     * 2px solid #fff  默认边框 
     * 如果只给当前点击的加边框 对引起整个页面抖动
     * 
     * 所有每一个p标签都添加一个边框  颜色为底色
     * 
     * 这样每次点击的时候把所有的颜色都重置
     * 
     * 然后再将点击的p换颜色 即可
     */
    changeRightLineFun = (flag, ev) => {
   
        ev = ev || window.event;
        let target = ev.target || ev.srcElement;
        let index;
        // 点击空P标签
        if (target.style.userSelect === 'none') {
   
            return false;
        }
        let {
    defaultAddColor, defaultDelColor } = this.state
        let RightFileId = flag === 'right' ? document.getElementById('RightFileId') : document.getElementById('LeftFileId')
        let LeftFileId = flag === 'right' ? document.getElementById('LeftFileId') : document.getElementById('RightFileId')
        let LeftFileIdP = LeftFileId.getElementsByTagName('p')
        let RightFileIdP = RightFileId.getElementsByTagName('p')
        if (target.nodeName.toLowerCase() == "p") {
   
            for (let i = 0; i < RightFileIdP.length; i++) {
   
                if (RightFileIdP[i] === target)
                    index = i;
                RightFileIdP[i].style.border = "2px solid #fff"
            }
            for (let i = 0; i < LeftFileIdP.length; i++) {
   
                if (LeftFileIdP[i] === target)
                    index = i;
                LeftFileIdP[i].style.border = "2px solid #fff"
            }
            if (flag === 'right') {
   
                if (target.parentNode.style.color === defaultAddColor) {
   
                    target.style.border = `2px solid ${
     defaultAddColor}`;
                } else {
   
                    target.style.border = "2px solid #000";
                }
            } else {
   
                if (target.parentNode.style.color === defaultDelColor) {
   
                    target.style.border = `2px solid ${
     defaultDelColor}`;
                } else {
   
                    target.style.border = "2px solid #000";
                }
            }
            LeftFileIdP[index].style.border = "2px solid #000";
        }
    }
    // 左右盒子样式
    returnDivStyle = (item) => {
   
        let {
    defaultAddColor, defaultDelColor, defaultColor, } = this.state;
        return {
   
            color: item?.added ? defaultAddColor :
                item?.removed ? defaultDelColor : defaultColor,
            fontWeight: item?.added ? '600' :
                item?.removed ? '600' : '600',
            ...pStyleObj
        }
    }
    // 判断是否换行
    returBrOrNULL = (idx, item) => {
   
        return idx !== item.value.split('\n').length - 1 ? <br /> : null
    }
    // 显示P标签
    returnPStyle = () => {
   
        let {
    defaultColor, } = this.state;
        return {
    backgroundColor: "#fff", whiteSpace: "pre", cursor: 'pointer', color: defaultColor, ...styleObj }
    }
    // 隐藏P标签
    returnTransparentP = (item) => {
   
        return (
            item?.value && item.value.split('\n').map((child, idx) => {
   
                //隐藏的元素 为了空格
                return <Fragment key={
   idx} >
                    {
   child && <p style={
   {
    whiteSpace: "pre", color: "transparent", userSelect: "none", ...styleObj }} >占位置</p>}
                    {
   this.returBrOrNULL(idx, item)}
                </Fragment>
            })
        )
    }
    // 左侧和右侧标题
    returnH2Fun = (title) => {
   
        const {
    headerHeight } = this.props
        return <h2 style={
   {
    fontWeight: "700", fontSize: "18px", lineHeight: "25px", height: `${
     25 * headerHeight}px` }}>{
   title}</h2>
    }
    // 返回右侧domlist
    returnRightDomFun = () => {
   
        let {
    isOriginFlag, defaultAddColor, defaultDelColor, defaultColor, DiffRight, rightSourceList } = this.state;
        const {
    isSplitBRFlag, isDiffWords } = this.props
        let domList = isOriginFlag ? rightSourceList : DiffRight
        return domList.map((item, index) => {
   
            return (
                <Fragment key={
   index} >
                    <div
                        onClick={
   (e) => {
    this.changeRightLineFun("right", e) }}
                        style={
   this.returnDivStyle()}>{
   !item ? null : item.removed ? (isSplitBRFlag ?
                            //隐藏的元素 为了空格
                            this.returnTransparentP(item) : null)
                            : item.value.split('\n').map((i, idx) => {
   
                                return <Fragment key={
   idx} >
                                    {
   /* 默认    */}
                                    {
   i && !isDiffWords ? <p style={
   {
    backgroundColor: `${
     item.added ? "rgb(221,255,221)" : "#fff"}`, whiteSpace: "pre", cursor: 'pointer', ...styleObj }} key={
   idx}>{
   i}</p> : null}
                                    {
   /* 开启单词对比 */}
                                    {
   i && isDiffWords ? <p style={
   this.returnPStyle()} >{
   
                                        i.split(':')[0]}:<span style={
   {
   
                                            backgroundColor: `${
     item.added ? "rgb(221,255,221)" : "#fff"}`, color: item.added ? defaultAddColor :
                                                item.removed ? defaultDelColor : defaultColor,
                                        }}>{
   i.split(':')[1]}</span></p> : null}
                                    {
   this.returBrOrNULL(idx, item)}
                                </Fragment>
                            })} </div>
                </Fragment>
            )
        })
    }
    // 返回左侧domlist
    returnLeftDomFun = () => {
   
        let {
    isOriginFlag, defaultAddColor, defaultDelColor, defaultColor, DiffLeft, leftSourceList } = this.state;
        const {
    isSplitBRFlag, isDiffWords } = this.props
        let domList = isOriginFlag ? leftSourceList : DiffLeft;
        return domList.map((item, index) => {
   
            return (
                <Fragment key={
   index} >
                    <div
                        onClick={
   (e) => {
    this.changeRightLineFun("left", e) }}
                        style={
   this.returnDivStyle(item)}>{
   !item ? null : item.added ? (isSplitBRFlag ?
                            this.returnTransparentP(item) : null)
                            : item.value.split('\n').map((i, idx) => {
   
                                return <Fragment key={
   idx} >
                                    {
   /* 默认    */}
                                    {
   i && !isDiffWords ? <p style={
   {
    backgroundColor: `${
     item.removed ? "rgb(255,227,227)" : "#fff"}`, whiteSpace: "pre", cursor: 'pointer', ...styleObj }} >{
   i}</p> : null}
                                    {
   /* 开启单词对比 */}
                                    {
   i && isDiffWords ? <p style={
   this.returnPStyle()} >{
   
                                        i.split(':')[0]}:<span style={
   {
   
                                            backgroundColor: `${
     item.removed ? "rgb(255,227,227)" : "#fff"}`, color: item.added ? defaultAddColor :
                                                item.removed ? defaultDelColor : defaultColor,
                                        }}>{
   i.split(':')[1]}</span></p> : null}
                                    {
   this.returBrOrNULL(idx, item)}
                                </Fragment>
                            })} </div>
                </Fragment>
            )
        })
    }
    render() {
   
        let {
    isOriginFlag, defaultAddColor, defaultDelColor, defaultColor, contextBoxLoading, } = this.state
        let {
    title, width, isModalVisible, oldStr, newStr, fileName, isDiffWords, isSplitBRFlag, rightTitle, leftTitle, headerHeight, ...rest } = this.props
        return !isModalVisible ? null : (
            <Modal
                title={
   <div style={
   {
    display: "flex", justifyContent: "space-between" }}><span style={
   {
    fontWeight: "500" }}>{
   title}</span><span style={
   {
    paddingLeft: "50px" }}>{
   fileName ? `文件名称:${
     fileName}` : ""}</span><h4 style={
   {
    float: "right" }}>灰色:<span style={
   {
    color: defaultColor }}>无变化</span>;绿色:<span style={
   {
    color: defaultAddColor }}>新增</span>;红色:<span style={
   {
    color: defaultDelColor }}>删除</span></h4></div>}
                visible={
   isModalVisible}
                maskClosable={
   true}
                style={
   {
    top: "20%", minWidth: "1000px" }}
                closable={
   false}
                width={
   width}
                {
   ...rest}
                footer={
   [
                    <Button rsType="noIcon" key="1" title={
   !isOriginFlag ? "查看源文件" : "查看对比文件"} onClick={
   () => this.changeOriginFileFun()}></Button>,
                    <Button rsType="noIcon" key="2" title="关闭" onClick={
   () => this.handleCancel()}></Button>
                ]}
            >
                <Spin
                    // tip="请稍等,可能需要一些时间..."
                    indicator={
   <img src={
   require("@/assets/loading.gif")} style={
   {
    width: "54px", height: "90px", position: "absolute", top: "50%", left: "50%", transform: "translate(-50%,-50%)" }} />}
                     spinning={
   contextBoxLoading} size="large" style={
   {
    width: "100%", height: '100%' }}>
                    {
   /* 绑定 key   目前这个key有两种形式 false true 字符的形式 正好对应 查看源文件和对比文件两个key  */}
                    <div style={
   {
    height: "600px", overflowY: "scroll", width: "100%" }} key={
   String(isOriginFlag)}>
                        {
   /* // 为了每次都处于文件顶端  */}
                        <div style={
   {
    wordBreak: "break-all", display: "flex", width: "100%" }} >
                            <div style={
   {
    width: "49%", paddingLeft: "20px" }} id="LeftFileId">
                                {
   this.returnH2Fun(leftTitle)}
                                {
   /* {Diff.diffLines(oldStr.replace(/\r/ig, ''), isOriginFlag ? oldStr.replace(/\r/ig, '') : newStr.replace(/\r/ig, '')) */}
                                {
   this.returnLeftDomFun()}
                            </div>
                            <Divider type="vertical" style={
   {
    width: "1%", height: "auto", minHeight: '600px' }} />
                            <div style={
   {
    width: "49%", paddingLeft: "10px", paddingRight: "10px" }} id="RightFileId" >
                                {
   this.returnH2Fun(rightTitle)}
                                {
   /* {Diff.diffLines(isOriginFlag ? newStr.replace(/\r/ig, '') : oldStr.replace(/\r/ig, ''), newStr.replace(/\r/ig, '')) */}
                                {
   this.returnRightDomFun()}
                            </div>
                        </div>
                    </div>
                </Spin>
            </Modal>
        )
    }
}

在线测试链接:
http://tangshisanbaishou.xyz/diff/index.html

目录
相关文章
|
2天前
|
前端开发 JavaScript
React项目路由懒加载lazy、Suspense,使第一次打开项目页面变快
本文介绍了在React项目中实现路由懒加载的方法,使用React提供的`lazy`和`Suspense`来优化项目首次加载的速度。通过将路由组件改为懒加载的方式,可以显著减少初始包的大小,从而加快首次加载速度。文章还展示了如何使用`Suspense`组件包裹`Switch`来实现懒加载过程中的fallback效果,并提供了使用前后的加载时间对比,说明了懒加载对性能的提升作用。
10 2
React项目路由懒加载lazy、Suspense,使第一次打开项目页面变快
|
2天前
|
前端开发 算法 JavaScript
React项目input输入框输入自动失去焦点
本文讨论了在React项目中如何处理input输入框自动失去焦点的问题,特别是在移动端开发中。文章提供了一个使用React Native的TouchableWithoutFeedback组件来监听点击事件,并在事件处理函数中通过调用Keyboard.dismiss()方法使输入框失去焦点的示例代码。这种方法可以确保在用户点击页面其他区域时,键盘能够收起,输入框失去焦点。
12 1
React项目input输入框输入自动失去焦点
|
2天前
|
前端开发 JavaScript Java
SpringBoot项目部署打包好的React、Vue项目刷新报错404
本文讨论了在SpringBoot项目中部署React或Vue打包好的前端项目时,刷新页面导致404错误的问题,并提供了两种解决方案:一是在SpringBoot启动类中配置错误页面重定向到index.html,二是将前端路由改为hash模式以避免刷新问题。
21 1
|
1天前
|
缓存 前端开发 JavaScript
在react项目中实现按钮权限createContext && useContext
文章介绍了在React项目中如何使用`createContext`和`useContext`来实现按钮级别的权限控制。
5 0
|
1月前
|
前端开发 JavaScript UED
React 基础与实践 | 青训营笔记
React 基础与实践 | 青训营笔记
41 0
|
2月前
|
前端开发 JavaScript Java
React 速通笔记
【7月更文挑战第17天】
36 1
|
前端开发
前端学习笔记202305学习笔记第二十九天-React keep alive原理之2
前端学习笔记202305学习笔记第二十九天-React keep alive原理之2
70 0
|
前端开发
前端学习笔记202306学习笔记第四十八天-react-admin marmelab之8
前端学习笔记202306学习笔记第四十八天-react-admin marmelab之7
47 0
|
4月前
|
前端开发 JavaScript
前端知识笔记(二十六)———React如何像Vue一样将css和js写在同一文件
前端知识笔记(二十六)———React如何像Vue一样将css和js写在同一文件
57 1
|
10月前
|
前端开发
前端笔记:React的form表单全部置空或者某个操作框置空的做法
在React框架前端开发中,经常会有弹出框的开发,涉及到弹出框,难免就会有表单。一般在关闭弹出框或者对表单联动时,往往都需要考虑对表单进行置空操作了。
86 0