HTML5数据列分组,HTML5数据可视化第三弹:萌萌哒拓扑图分组

2023-11-23 00:10

本文主要是介绍HTML5数据列分组,HTML5数据可视化第三弹:萌萌哒拓扑图分组,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

最近忙着给客户折腾一个复杂的多层嵌套关系。客户一句话“要好看!”,哥就忙白了头啊,还好最终搞定了。

需求描述

先简单描述下这次客户的需求。

现实应用中,网络拓扑图结构可能很简单,也可能非常复杂。

比如这种节点较多的单层拓扑:

a4c26d1e5885305701be709a3d33442f.png

稍复杂一些的:

a4c26d1e5885305701be709a3d33442f.png

再复杂一些的:

a4c26d1e5885305701be709a3d33442f.png

在这些拓扑图中常见的场景是,很多网络节点需要组成一组,这常被称为“网元组”。一般来说,网元组会有一个形状,双击可以展开/闭合。如下图:

a4c26d1e5885305701be709a3d33442f.png

a4c26d1e5885305701be709a3d33442f.png

这次客户的需求中,最大的难点就是需要有五层网元组的嵌套,五层同时展开时,要求清晰美观。常规的分组形状有圆形、矩形、平行四边形等,无论哪种形状,分组多了后,就会产生审美疲劳。比如,我让设计师mm简单地把五层嵌套画个图,它看起来会是这样:

a4c26d1e5885305701be709a3d33442f.png

我把这个图给客户看了以后,客户表示希望“结构能够更加清晰”。那天,魔都大雨倾盆,我抓耳挠腮一个下午,终于有了一些灵感。

颜色

怎样才能使得结构效果更加清晰?我想到的是用颜色。颜色永远是图形设计里面的第一要素。如果分组颜色千篇一律,自然就看不太清楚包含关系。但是颜色太多五颜六色,显然也不符合电信UI系统的风格。那颜色要如何设置呢?层层嵌套的分组,层层……叮!你有没有想到一种蔬菜?

a4c26d1e5885305701be709a3d33442f.png

(此处哼唱“如果你可以一层一层一层地剥开我的心……”一百遍……)

这颗大洋葱看上去层次分明,是因为它的颜色从内到外有一定规律的变化:渐变。说到渐变,又想起最近GF推荐的一枚游戏,大概内容是按照颜色的渐变规律来排列一些方块。(很无聊有没有?)

a4c26d1e5885305701be709a3d33442f.png

不过,前端设计中,配色倒是很重要的一个环节。

总之,想好了用渐变的配色来使嵌套组更清晰后,就大胆地尝试一下:

var group=newtwaver.Group();

group.setStyle('group.fill.color', style[3]);

group.setStyle('group.deep',0);

group.setStyle('group.outline.width', style[1]);

group.setStyle('group.outline.color', style[0]);

group.setStyle('group.shape','roundrect'); group.setStyle('select.style','none');

group.setStyle('vector.outline.pattern', style[2]);

group.setStyle('label.font','14px "Microsoft Yahei"');

group.setStyle('whole.alpha',0.8);

group.setStyle('group.padding',-10);

group.setStyle('label.position','topright.topleft');

group.setName(name);

group.setLocation(100+150*level,300);

box.add(group);

数据量更大的时候,看看分组是不是会更加清晰?

a4c26d1e5885305701be709a3d33442f.png

更多颜色

这个图做出来之后,拿给周围几个同事看,大家纷纷表示不错,但是似乎有一些死板,不够生动。生动。。那就是要活灵活现的,于是我继续抓耳挠腮,又想到了一些瓜果蔬菜:

a4c26d1e5885305701be709a3d33442f.png

a4c26d1e5885305701be709a3d33442f.png

果然还是配色不够明艳啊。我又让设计师mm给找了几个色值调整了一下:

a4c26d1e5885305701be709a3d33442f.png

折角

有了写花瓣层叠的感觉,是不是舒服多了?不过,方方正正的组的形状,还是太死板,缺乏立体感。看到桌子上的一张折了角的白纸,突然有了灵感。

给方形的组做一个折角效果,不知效果如何。要做这个效果,就要重写group的绘制,自己接管2d绘制了。Group的形状将不再是一个矩形,而是一个切角的矩形。

//draw round rect body.var roundRadius=10;

ctx.save();

ctx.beginPath();

ctx.moveTo(rect.x+roundRadius, rect.y);

ctx.lineTo(rect.x+rect.width-60, rect.y);

ctx.lineTo(rect.x+rect.width, rect.y+28);

ctx.lineTo(rect.x+rect.width, rect.y+rect.height-roundRadius);

ctx.quadraticCurveTo(rect.x+rect.width, rect.y+rect.height, rect.x+rect.width-roundRadius, rect.y+rect.height);

ctx.lineTo(rect.x+roundRadius, rect.y+rect.height);

ctx.quadraticCurveTo(rect.x, rect.y+rect.height, rect.x, rect.y+rect.height-roundRadius);

ctx.lineTo(rect.x, rect.y+roundRadius);

ctx.quadraticCurveTo(rect.x, rect.y, rect.x+roundRadius, rect.y);

ctx.save();

ctx.shadowOffsetX=4;

ctx.shadowOffsetY=4;

ctx.shadowBlur=4;

ctx.shadowColor="#555555";

ctx.fill();

ctx.restore();

ctx.lineWidth=node.getStyle('group.outline.width');

ctx.strokeStyle=node.getStyle('group.outline.color');

ctx.stroke(); ctx.restore();

通过绘制一个带圆角的矩形并切掉一个角,stroke到画布上。看看效果:

a4c26d1e5885305701be709a3d33442f.png

再通过增加圆角、切角、增加阴影、设置不同的边框线宽,让分组进一步产生“层层递进”的感觉。现在就剩下画折角的细节了。

折角这里,需要画一个被折下来的直角三角形。三角形的颜色,应该是“纸”的背面颜色。这里小心定义三角形的形状,并进行填充:

//draw corlor.ctx.fillStyle=node.getStyle('group.outline.color'); ctx.lineWidth=node.getStyle('group.outline.width');

ctx.lineJoin='bevel';

ctx.beginPath();

ctx.moveTo(rect.x+rect.width-60, rect.y);

ctx.lineTo(rect.x+rect.width-23-10, rect.y+47-10);

ctx.quadraticCurveTo(rect.x+rect.width-23, rect.y+46, rect.x+rect.width-23+10, rect.y+47-10);

ctx.lineTo(rect.x+rect.width, rect.y+28);

ctx.closePath();

ctx.save();

ctx.shadowOffsetX=4;

ctx.shadowOffsetY=4;

ctx.shadowBlur=4;

ctx.shadowColor="#777777";

ctx.fill(); ctx.restore();

ctx.strokeStyle=node.getStyle('group.outline.color');

ctx.stroke();

效果如下,立体感出来以后,是不是生动了很多?

a4c26d1e5885305701be709a3d33442f.png

这里要注意的是,折角的阴影也要设置,并且填充要使用和边框相同的颜色,增加“折纸”的立体感。

小细节

折纸效果有了,不过左侧上方略显空旷,于是利用canvas的2d来练练手,画个path看看:

ctx.save();

ctx.strokeStyle='#27A3DA';

ctx.lineWidth=2;

ctx.beginPath();

ctx.moveTo(rect.x+31, rect.y+5);

ctx.lineTo(rect.x+25, rect.y+20);

ctx.bezierCurveTo(rect.x+25, rect.y+26, rect.x+28, rect.y+28, rect.x+32, rect.y+23);

ctx.lineTo(rect.x+42, rect.y-2);

ctx.bezierCurveTo(rect.x+42, rect.y-12, rect.x+32, rect.y-10, rect.x+32, rect.y-5);

ctx.lineTo(rect.x+29, rect.y-1);

ctx.shadowOffsetX=1;

ctx.shadowOffsetY=1;

ctx.shadowBlur=1;

ctx.shadowColor="#aaaaaa";

ctx.stroke();

ctx.restore();

运行一下,你猜是神马?

a4c26d1e5885305701be709a3d33442f.png

哈哈,一个小回形针瞬间跃然纸上了,感觉萌萌哒!为了增加立体感,回形针也是要设置阴影的,不过偏移不要太大、颜色要淡一些,像这样:

a4c26d1e5885305701be709a3d33442f.png

适当明艳的色彩,加上折角、阴影和小回形针,这回这个层层嵌套总算是清晰又好看了吧?

运行一下,拖拖拽拽,因为之前已经做了不少图标、线条的样式,所以总体效果还是很不错的!

a4c26d1e5885305701be709a3d33442f.png

a4c26d1e5885305701be709a3d33442f.png

后记

之前也说,HTML5的canvas,虽然已经不是什么新鲜技术了,但当技术本身不再有壁垒,我们更应该注重的是实际业务中的应用,比如在画这种组织结构关系非常复杂的拓扑图时,如何让图形更加清晰、易懂,让技术真正落到实处。如果你有更加好的这类拓扑图的解决方案,也可以和我联系交流:tw-service@servasoft.com

这篇关于HTML5数据列分组,HTML5数据可视化第三弹:萌萌哒拓扑图分组的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/413909

相关文章

vite搭建vue3项目的搭建步骤

《vite搭建vue3项目的搭建步骤》本文主要介绍了vite搭建vue3项目的搭建步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录1.确保Nodejs环境2.使用vite-cli工具3.进入项目安装依赖1.确保Nodejs环境

Nginx搭建前端本地预览环境的完整步骤教学

《Nginx搭建前端本地预览环境的完整步骤教学》这篇文章主要为大家详细介绍了Nginx搭建前端本地预览环境的完整步骤教学,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录项目目录结构核心配置文件:nginx.conf脚本化操作:nginx.shnpm 脚本集成总结:对前端的意义很多

Linux下利用select实现串口数据读取过程

《Linux下利用select实现串口数据读取过程》文章介绍Linux中使用select、poll或epoll实现串口数据读取,通过I/O多路复用机制在数据到达时触发读取,避免持续轮询,示例代码展示设... 目录示例代码(使用select实现)代码解释总结在 linux 系统里,我们可以借助 select、

前端缓存策略的自解方案全解析

《前端缓存策略的自解方案全解析》缓存从来都是前端的一个痛点,很多前端搞不清楚缓存到底是何物,:本文主要介绍前端缓存的自解方案,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、为什么“清缓存”成了技术圈的梗二、先给缓存“把个脉”:浏览器到底缓存了谁?三、设计思路:把“发版”做成“自愈”四、代码

通过React实现页面的无限滚动效果

《通过React实现页面的无限滚动效果》今天我们来聊聊无限滚动这个现代Web开发中不可或缺的技术,无论你是刷微博、逛知乎还是看脚本,无限滚动都已经渗透到我们日常的浏览体验中,那么,如何优雅地实现它呢?... 目录1. 早期的解决方案2. 交叉观察者:IntersectionObserver2.1 Inter

Vue3视频播放组件 vue3-video-play使用方式

《Vue3视频播放组件vue3-video-play使用方式》vue3-video-play是Vue3的视频播放组件,基于原生video标签开发,支持MP4和HLS流,提供全局/局部引入方式,可监听... 目录一、安装二、全局引入三、局部引入四、基本使用五、事件监听六、播放 HLS 流七、更多功能总结在 v

JS纯前端实现浏览器语音播报、朗读功能的完整代码

《JS纯前端实现浏览器语音播报、朗读功能的完整代码》在现代互联网的发展中,语音技术正逐渐成为改变用户体验的重要一环,下面:本文主要介绍JS纯前端实现浏览器语音播报、朗读功能的相关资料,文中通过代码... 目录一、朗读单条文本:① 语音自选参数,按钮控制语音:② 效果图:二、朗读多条文本:① 语音有默认值:②

vue监听属性watch的用法及使用场景详解

《vue监听属性watch的用法及使用场景详解》watch是vue中常用的监听器,它主要用于侦听数据的变化,在数据发生变化的时候执行一些操作,:本文主要介绍vue监听属性watch的用法及使用场景... 目录1. 监听属性 watch2. 常规用法3. 监听对象和route变化4. 使用场景附Watch 的

前端导出Excel文件出现乱码或文件损坏问题的解决办法

《前端导出Excel文件出现乱码或文件损坏问题的解决办法》在现代网页应用程序中,前端有时需要与后端进行数据交互,包括下载文件,:本文主要介绍前端导出Excel文件出现乱码或文件损坏问题的解决办法,... 目录1. 检查后端返回的数据格式2. 前端正确处理二进制数据方案 1:直接下载(推荐)方案 2:手动构造

Vue实现路由守卫的示例代码

《Vue实现路由守卫的示例代码》Vue路由守卫是控制页面导航的钩子函数,主要用于鉴权、数据预加载等场景,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着... 目录一、概念二、类型三、实战一、概念路由守卫(Navigation Guards)本质上就是 在路