当前位置: 首页 > article >正文

【示例】Vue AntV G6 base64自定义img 动画效果,自适应宽高屏

需求:拓扑图中需要用动画的线条连接node,在此之前将HTML页面改成了vue页面。需要使用到G6的registerEdge 自定义边,小车的图片需要转成base64格式(并翻转),可以通过base64转image查看原来的样子。

另外,通过动态控制div的scale自适应拉伸尺寸的计算来达到自适应宽高屏幕的效果。

【完整代码】

<template>
    <div class="body-wrap" ref="contentWrapper">
        <div class="graph-wrap">
            <div ref="graphContainer"></div>
        </div>
    </div>
</template>

<script>
import G6 from '@antv/g6';
const { getLabelPosition, transform } = G6.Util;

export default {
    name: 'CustomEdgeGraph',
    data() {
        return {
            designWidth: 1920, // 设计稿宽度
            designHeight: 1080, // 设计稿高度
            currentScale: [1, 1],
            nodeData: {
                // 点集
                nodes: [
                    {
                        id: 'point1',
                        x: 110,
                        y: 200,
                    },
                    {
                        id: 'point2',
                        x: 110,
                        y: 200,
                    },
                    {
                        id: 'point3',
                        x: 130,
                        y: 630,
                    },
                    {
                        id: 'point4',
                        x: 130,
                        y: 630,
                    },
                    {
                        id: 'center',
                        x: 500,
                        y: 400,
                    },
                    {
                        id: 'oil_star',
                        x: 1600,
                        y: 1030,
                    },
                    {
                        id: 'oil_end',
                        x: 100,
                        y: 780,
                    },
                    {
                        id: 'car_star',
                        x: 1600,
                        y: 990,
                    },
                    {
                        id: 'car_end',
                        x: 100,
                        y: 750,
                    },

                ],
                // 边集
                edges: [
                    {
                        source: 'point1',
                        target: 'center',
                        curveOffset: -180,
                    },
                    {
                        source: 'point2',
                        target: 'center',
                        curveOffset: 160,
                    },
                    {
                        source: 'point3',
                        target: 'center',
                        curveOffset: -120,
                    },
                    {
                        source: 'point4',
                        target: 'center',
                        curveOffset: 90,
                    },

                    // 动画线
                    {
                        source: 'point1',
                        target: 'center',
                        curveOffset: -180,
                        type: 'line-growth',
                        style: {
                            lineAppendWidth: 5,
                            lineWidth: 5, // 线宽
                            stroke: 'l(0) 0:rgba(34,255,242,0.06) 0.25:rgba(34,255,242,0.5) 0.5:rgba(34,255,242,1) 0.75:rgba(34,255,242,0.5)  1:rgba(34,255,242,0.06)', // 线的颜色
                        }
                    },
                    {
                        source: 'point2',
                        target: 'center',
                        curveOffset: 160,
                        type: 'line-growth',
                        style: {
                            lineAppendWidth: 5,
                            lineWidth: 5, // 线宽
                            stroke: 'l(0) 0:rgba(34,255,242,0.06) 0.25:rgba(34,255,242,0.5) 0.5:rgba(34,255,242,1) 0.75:rgba(34,255,242,0.5)  1:rgba(34,255,242,0.06)', // 线的颜色
                        }
                    },
                    {
                        source: 'point3',
                        target: 'center',
                        curveOffset: -120,
                        type: 'line-growth',
                        style: {
                            lineAppendWidth: 5,
                            lineWidth: 5, // 线宽
                            stroke: 'l(0) 0:rgba(34,255,242,0.06) 0.25:rgba(34,255,242,0.5) 0.5:rgba(34,255,242,1) 0.75:rgba(34,255,242,0.5)  1:rgba(34,255,242,0.06)', // 线的颜色
                        }
                    },
                    {
                        source: 'point4',
                        target: 'center',
                        curveOffset: 90,
                        type: 'line-growth',
                        style: {
                            lineAppendWidth: 5,
                            lineWidth: 5, // 线宽
                            stroke: 'l(0) 0:rgba(34,255,242,0.06) 0.25:rgba(34,255,242,0.5) 0.5:rgba(34,255,242,1) 0.75:rgba(34,255,242,0.5)  1:rgba(34,255,242,0.06)', // 线的颜色
                        }
                    },
                    {
                        source: 'oil_star',
                        target: 'oil_end',
                        type: 'arrow-running',
                        style: {
                            lineAppendWidth: 10,
                            lineWidth: 10, // 线宽
                            stroke: 'l(0) 0:rgba(188,133,26,0.16) 0.95:rgba(188,133,26,0.16)  1:rgba(188,133,26,0)', // 线的颜色
                            // 
                        },
                        // assign the control points to control the bending positions
                        // 开车线路的中间拐点
                        controlPoints: [
                            {
                                x: 600,
                                y: 1030,
                            },
                        ],
                    },
                    {
                        source: 'car_star',
                        target: 'car_end',
                        type: 'car-running',
                        style: {
                            lineAppendWidth: 10,
                            lineWidth: 10, // 线宽
                            //stroke:'red'
                            stroke: 'rgba(255,255,255,0)', // 线的颜色
                            // 
                        },
                        // assign the control points to control the bending positions
                        // 开车的中间拐点
                        controlPoints: [
                            {
                                x: 600,
                                y: 990,
                            },
                        ],
                    },

                ]
            }
        }
    },
    beforeDestroy() {
        window.removeEventListener('resize', this.onWindowResize);
    },
    mounted() {
        this.initGraph();
        this.updateScale();
        window.addEventListener('resize', this.updateScale);

    },
    methods: {
        // 自适应拉伸尺寸计算
        updateScale() {
            const { designWidth, designHeight } = this;
            const windowWidth = window.innerWidth;
            const windowHeight = window.innerHeight;

            // 计算宽高比
            const widthRatio = (windowWidth / designWidth).toFixed(4) - 0.002;
            const heightRatio = (windowHeight / designHeight).toFixed(4) - 0.002;
            const scale = [widthRatio, heightRatio];

            // 更新缩放比例
            this.currentScale = scale;
            this.applyTransform();
        },
        applyTransform() {
            const contentWrapper = this.$refs.contentWrapper;
            if (contentWrapper) {
                contentWrapper.style.transform = `scale(${this.currentScale[0]},${this.currentScale[1]})`;
                contentWrapper.style.transformOrigin = 'top left';
            }
        },
        // 自适应拉伸尺寸计算 END

        // 初始化图表
        initGraph() {
            const car1 = ''
            const cube = ''
            
            // 自定义边类型
            G6.registerEdge(
                'line-growth',
                {
                    afterDraw(cfg, group) {
                        const shape = group.get('children')[0];
                        const length = shape.getTotalLength();
                        shape.animate(
                            (ratio) => {
                                // the operations in each frame. Ratio ranges from 0 to 1 indicating the prograss of the animation. Returns the modified configurations
                                const startLen = ratio * length;
                                // Calculate the lineDash
                                const cfg = {
                                    lineDash: [startLen, length - startLen],
                                };
                                return cfg;
                            },
                            {
                                repeat: true, // Whether executes the animation repeatly
                                duration: 2000, // the duration for executing once
                            },
                        );
                    },
                },
                'arc', // extend the built-in edge 'cubic' arc 
            );
            G6.registerEdge(
                "arrow-running",
                {
                    afterDraw(cfg, group) {
                        // get the first shape in the group, it is the edge's path here=
                        const shape = group.get("children")[0];
                        const length = shape.getTotalLength();
                        let circleCount = Math.ceil(length / 100);

                        circleCount = circleCount === 0 ? 1 : circleCount;

                        const _loop = function _loop(i) {
                            const delay = 0;
                            const start = shape.getPoint(0);
                            const image = group.addShape("image", {
                                attrs: {
                                    x: start.x,
                                    y: start.y - 9,
                                    width: 14,
                                    height: 14,
                                    img: cube,
                                }
                            });

                            // animation for the red circle
                            image.animate(
                                (ratio) => {
                                    ratio += i / circleCount;
                                    if (ratio > 1) {
                                        ratio %= 1;
                                    }
                                    const tmpPoint = shape.getPoint(ratio);
                                    const pos = getLabelPosition(shape, ratio);
                                    let matrix = [1, 0, 0, 0, 1, 0, 0, 0, 1];
                                    matrix = transform(matrix, [
                                        ["t", -tmpPoint.x - 9, -tmpPoint.y],
                                        ["r", pos.angle],
                                        ["t", tmpPoint.x, tmpPoint.y]
                                    ]);

                                    // returns the modified configurations here, x and y here
                                    return {
                                        x: tmpPoint.x,
                                        y: tmpPoint.y - 9,
                                        matrix
                                    };
                                },
                                {
                                    repeat: true, // Whether executes the animation repeatly
                                    // duration: 3000 // the duration for executing once
                                    duration: 8 * length,
                                    easing: 'easeLinear',
                                    // delay:5 * length,
                                }
                            );
                        }
                        for (let i = 0; i < circleCount; i++) {
                            _loop(i);
                        }
                    }
                },
                "polyline" // extend the built-in edge 'cubic'
            );
            G6.registerEdge(
                "car-running",
                {
                    afterDraw(cfg, group) {
                        // get the first shape in the group, it is the edge's path here=
                        const shape = group.get("children")[0];
                        const image = group.addShape("image", {
                            attrs: {
                                x: 0,
                                y: 0,
                                width: 73,
                                height: 41,
                                img: car1,
                            }
                        });
                        // animation for the red circle
                        image.animate(
                            (ratio) => {
                                // the operations in each frame. Ratio ranges from 0 to 1 indicating the prograss of the animation. Returns the modified configurations
                                // get the position on the edge according to the ratio
                                const tmpPoint = shape.getPoint(ratio);
                                const pos = getLabelPosition(shape, ratio);
                                let matrix = [1, 0, 0, 0, 1, 0, 0, 0, 1];
                                // var op =1
                                if (tmpPoint.x > 1653) {
                                    matrix = transform(matrix, [
                                        ["t", -tmpPoint.x - 50, -tmpPoint.y],
                                        ["r", pos.angle],
                                        ["t", tmpPoint.x, tmpPoint.y]
                                    ]);
                                } else {
                                    matrix = transform(matrix, [
                                        ["t", -tmpPoint.x, -tmpPoint.y + 10],
                                        ["r", pos.angle],
                                        ["t", tmpPoint.x, tmpPoint.y]
                                    ]);
                                }
                                var op = 1
                                // 当车跑到某点时变成不透明 0.5
                                // if (tmpPoint.x > 1483) {
                                //     op = 1
                                // } else {
                                //     op = 0.5
                                // }
                                // returns the modified configurations here, x and y here
                                return {
                                    x: tmpPoint.x,
                                    y: tmpPoint.y - 36,
                                    matrix,
                                    opacity: op
                                };
                            },
                            {
                                repeat: true, // Whether executes the animation repeatly
                                duration: 9000 // the duration for executing once
                            }
                        );
                    }
                },
                "polyline" // extend the built-in edge 'cubic'
            );
            this.$nextTick(() => {
                // 初始化图表
                const graph = new G6.Graph({
                    container: this.$refs.graphContainer,
                    width: 1920,
                    height: 1080,
                    modes: {
                        default: ['drag-node', 'zoom-canvas', 'click-select']
                    },
                    defaultNode: { // 节点样式修改
                        type: 'circle', // 设置节点为图片
                        size: [0, 0], // 节点大小
                        anchorPoints: [
                            [0.5, 0.5],
                            [0.5, 0.5],
                        ],
                    },
                    defaultEdge: { // 边通用配置
                        type: 'arc',
                        labelCfg: {
                            autoRotate: true,
                        },
                        style: {
                            lineAppendWidth: 3,
                            lineWidth: 3, // 线宽
                            stroke: 'rgba(9,237,224,0.3)', // 线的颜色
                        },
                    },
                });

                // 设置初始数据
                graph.data(this.nodeData);

                // 渲染图表
                graph.render();

                if (typeof window !== 'undefined')
                    window.onresize = () => {
                        if (!graph || graph.get('destroyed')) return;
                    };
            })
        }
    }
};
</script>

<style lang="less" scoped>
.body-wrap {
    overflow: hidden;
    width: 1920px;
    height: 1080px;
    background: #040d27;
    background-size: cover;
    transform-origin: 0 0;
}

.graph-wrap {
    width: 1920px;
    height: 1080px;
    background: none;
}
</style>

【小车原图示例】

【工具网站】

base64图片在线转换工具 - 站长工具

G6


http://www.kler.cn/a/449922.html

相关文章:

  • nmap扫描优化
  • 基于Spring Boot的九州美食城商户一体化系统
  • 写给Pythoner的前端进阶指南(五):事件驱动模型
  • 自动驾驶控制算法-横向误差微分方程LQR前馈控制
  • python中的字典数据和标准json格式区别
  • leetcode之hot100---240搜索二维矩阵II(C++)
  • nvm下载管理node版本
  • 在 Ubuntu 上安装 MySQL 的详细指南
  • 今日总结 2024-12-23
  • 利用 Jsoup 进行高效 Web 抓取与 HTML 处理
  • JSON 中的 $ref 问题及 DisableCircularReferenceDetect 解析
  • 突破续航瓶颈:数字样机技术引领新能源汽车复合制动新方向
  • SQL server学习09-数据库编程(上)
  • Java与容器化:如何使用Docker和Kubernetes优化Java应用的部署
  • SpringBoot中使用AOP切面编程实现登录拦截
  • 全面解析 Kubernetes 流量负载均衡:iptables 与 IPVS 模式
  • pyQt5实现目标检测可视化001
  • 基于Spring Boot的旅游推荐系统
  • Linux下mysql 8.0安装教程
  • [LeetCode-Python版] 定长滑动窗口8——2461. 长度为 K 子数组中的最大和
  • Windows和Linux安全配置和加固
  • [Effective C++]条款38-39 复合和private继承
  • IT运维的365天--021 服务器上的dns设置后不起作用
  • aws(学习笔记第十九课) 使用ECS和Fargate进行容器开发
  • 云计算中的容器技术(如Docker)是什么?
  • 一些elasticsearch重要概念与配置参数