使用 D3.js 实现下拉菜单联动更新可视化图表

使用 d3.js 实现下拉菜单联动更新可视化图表

本文档将指导你如何使用 D3.js 创建一个动态图表,该图表能够根据 HTML 下拉菜单的选择进行数据更新。我们将重点讲解如何监听下拉菜单的 `change` 事件,并利用 D3.js 的 `join`, `enter`, `update`, `exit` 模式高效地更新图表元素,实现数据驱动视图的动态变化。

教程内容

1. 数据准备

首先,我们需要准备用于生成图表的数据。以下代码模拟生成了一组包含年份、名称、排名和数值的数据集 src。

// desired permutation lengthconst length = 4;// build array from the above lengthconst perm = Array.from(Array(length).keys()).map((d) => d + 1);// generate corresponding alphabets for nameconst name = perm.map((x) => String.fromCharCode(x - 1 + 65));// permutation functionfunction permute(permutation) {    var length = permutation.length,        result = [permutation.slice()],        c = new Array(length).fill(0),        i = 1,        k, p;    while (i < length) {        if (c[i]  i + 2000);// generate a yearly constant based on year to generate final value as per the rank {year-name}const constant = year.map(d => Math.round(d * Math.random()));const src =    year.map((y, i) => {        return name.map((d, j) => {            return {                Name: d,                Year: y,                Rank: permut[i][j],                Const: constant[i],                Value: Math.round(constant[i] / permut[i][j])            };        });    }).flat();

2. 创建 HTML 下拉菜单

接下来,我们使用 D3.js 在 HTML 页面中创建一个下拉菜单。该下拉菜单将包含数据集 src 中的所有年份选项。

const select = d3.select('body')    .append('div', 'dropdown')    .style('position', 'absolute')    .style('top', '400px')    .append('select')    .attr('name', 'input')    .classed('Year', true);select.selectAll('option')    .data(year)    .enter()    .append('option')    .text((d) => d)    .attr("value", (d) => d)

3. 监听下拉菜单的 change 事件

为了使图表能够根据下拉菜单的选择进行更新,我们需要监听下拉菜单的 change 事件。当用户选择不同的年份时,我们将获取选中的年份值,并调用 draw() 函数来更新图表。

select.on("change", event => {    const filterYr = +event.currentTarget.value;    draw(filterYr);});

注意:

使用 event.currentTarget.value 获取选中的年份值。使用 + 运算符将年份值转换为数字类型。将选中的年份值作为参数传递给 draw() 函数。

4. 创建 SVG 画布

绘制图表之前,我们需要创建一个 SVG 画布。

// namespace// define dimensionconst width = 1536;const height = 720;const svgns = "http://www.w3.org/2000/svg";const svg = d3.select("svg");svg.attr("xmlns", svgns).attr("viewBox", `0 0 ${width} ${height}`);svg    .append("rect")    .attr("class", "vBoxRect")    .style("overflow", "visible")    .attr("width", `${width}`)    .attr("height", `${height}`)    .attr("stroke", "black")    .attr("fill", "white");const padding = {    top: 70,    bottom: 100,    left: 120,    right: 120};const multiplierH = 1; //controls the height of the visual containerconst multiplierW = 1; //controls the width of the visual containerconst boundHeight = height * multiplierH - padding.top - padding.bottom;const boundWidth = width * multiplierW - padding.right - padding.left;//create BOUND rect -- to be deleted latersvg    .append("rect")    .attr("class", "boundRect")    .attr("x", `${padding.left}`)    .attr("y", `${padding.top}`)    .attr("width", `${boundWidth}`)    .attr("height", `${boundHeight}`)    .attr("fill", "white");//create bound elementconst bound = svg    .append("g")    .attr("class", "bound")    .style("transform", `translate(${padding.left}px,${padding.top}px)`);const g = bound.append('g')    .classed('textContainer', true);

5. 绘制图表

draw() 函数负责根据选中的年份值过滤数据,并使用 D3.js 的 join, enter, update, exit 模式更新图表元素。

function draw(filterYr) {    // filter data as per dropdown    const data = src.filter(a => a.Year == filterYr);    const xAccessor = (d) => d.Year;    const yAccessor = (d) => d.Value;    const scaleX = d3        .scaleLinear()        .range([0, boundWidth])        .domain(d3.extent(data, xAccessor));    const scaleY = d3        .scaleLinear()        .range([boundHeight, 0])        .domain(d3.extent(data, yAccessor));    g.selectAll('text')        .data(data)        .join(            enter => enter.append('text')            .attr('x', (d, i) => scaleX(d.Year))            .attr('y', (d, i) => i)            .attr('dy', (d, i) => i * 30)            .text((d) => d.Year + '-------' + d.Value.toLocaleString())            .style("fill", "blue"),            update =>            update            .transition()            .duration(500)            .attr('x', (d, i) => scaleX(d.Year))            .attr('y', (d, i) => i)            .attr('dy', (d, i) => i * 30)            .text((d) => d.Year + '-------' + d.Value.toLocaleString())            .style("fill", "red")            /*,                        (exit) =>                        exit                        .style("fill", "black")                        .transition()                        .duration(1000)                        .attr("transform", (d, i) => `translate(${300},${30 + i * 30})`)                        .remove()*/        )}draw(filterYr);

代码解释:

data(data): 将过滤后的数据绑定到 SVG 元素。join(enter, update, exit): D3.js 的核心方法,用于处理数据的 enter, update 和 exit 状态。enter: 处理新增的数据,创建新的 SVG 元素。update: 处理已存在的数据,更新 SVG 元素的属性。exit: 处理被移除的数据,移除对应的 SVG 元素。.transition().duration(500): 为更新操作添加过渡效果,使图表变化更加平滑。

完整代码

                D3.js Dropdown Update                    // desired permutation length        const length = 4;        // build array from the above length        const perm = Array.from(Array(length).keys()).map((d) => d + 1);        // generate corresponding alphabets for name        const name = perm.map((x) => String.fromCharCode(x - 1 + 65));        // permutation function        function permute(permutation) {            var length = permutation.length,                result = [permutation.slice()],                c = new Array(length).fill(0),                i = 1,                k, p;            while (i < length) {                if (c[i]  i + 2000);        // generate a yearly constant based on year to generate final value as per the rank {year-name}        const constant = year.map(d => Math.round(d * Math.random()));        const src =            year.map((y, i) => {                return name.map((d, j) => {                    return {                        Name: d,                        Year: y,                        Rank: permut[i][j],                        Const: constant[i],                        Value: Math.round(constant[i] / permut[i][j])                    };                });            }).flat();        const select = d3.select('body')            .append('div', 'dropdown')            .style('position', 'absolute')            .style('top', '400px')            .append('select')            .attr('name', 'input')            .classed('Year', true);        select.selectAll('option')            .data(year)            .enter()            .append('option')            .text((d) => d)            .attr("value", (d) => d)        //get the dropdown value        const filterYr = parseFloat(d3.select('.Year').node().value);        select.on("change", event => {            const filterYr = +event.currentTarget.value;            draw(filterYr);        });        const width = 1536;        const height = 720;        const svgns = "http://www.w3.org/2000/svg";        const svg = d3.select("svg");        svg.attr("xmlns", svgns).attr("viewBox", `0 0 ${width} ${height}`);        svg            .append("rect")            .attr("class", "vBoxRect")            .style("overflow", "visible")            .attr("width", `${width}`)            .attr("height", `${height}`)            .attr("stroke", "black")            .attr("fill", "white");        const padding = {            top: 70,            bottom: 100,            left: 120,            right: 120        };        const multiplierH = 1; //controls the height of the visual container        const multiplierW = 1; //controls the width of the visual container        const boundHeight = height * multiplierH - padding.top - padding.bottom;        const boundWidth = width * multiplierW - padding.right - padding.left;        //create BOUND rect -- to be deleted later        svg            .append("rect")            .attr("class", "boundRect")            .attr("x", `${padding.left}`)            .attr("y", `${padding.top}`)            .attr("width", `${boundWidth}`)            .attr("height", `${boundHeight}`)            .attr("fill", "white");        //create bound element        const bound = svg            .append("g")            .attr("class", "bound")            .style("transform", `translate(${padding.left}px,${padding.top}px)`);        const g = bound.append('g')            .classed('textContainer', true);        function draw(filterYr) {            // filter data as per dropdown            const data = src.filter(a => a.Year == filterYr);            const xAccessor = (d) => d.Year;            const yAccessor = (d) => d.Value;            const scaleX = d3                .scaleLinear()                .range([0, boundWidth])                .domain(d3.extent(data, xAccessor));            const scaleY = d3                .scaleLinear()                .range([boundHeight, 0])                .domain(d3.extent(data, yAccessor));            g.selectAll('text')                .data(data)                .join(                    enter => enter.append('text')                    .attr('x', (d, i) => scaleX(d.Year))                    .attr('y', (d, i) => i)                    .attr('dy', (d, i) => i * 30)                    .text((d) => d.Year + '-------' + d.Value.toLocaleString())                    .style("fill", "blue"),                    update =>                    update                    .transition()                    .duration(500)                    .attr('x', (d, i) => scaleX(d.Year))                    .attr('y', (d, i) => i)                    .attr('dy', (d, i) => i * 30)                    .text((d) => d.Year + '-------' + d.Value.toLocaleString())                    .style("fill", "red")                    /*,                                (exit) =>                                exit                                .style("fill", "black")                                .transition()                                .duration(1000)                                .attr("transform", (d, i) => `translate(${300},${30 + i * 30})`)                                .remove()*/                )        }        draw(filterYr);    

总结

通过本教程,你学习了如何使用 D3.js 创建一个能够根据 HTML 下拉菜单的选择进行动态更新的图表。 关键步骤包括:

创建下拉菜单: 使用 D3.js 创建 HTML 下拉菜单,并绑定年份数据。监听 change 事件: 监听下拉菜单的 change 事件,获取选中的年份值。数据过滤: 根据选中的年份值过滤数据集。绘制图表: 使用 D3.js 的 join, enter, update, exit 模式更新图表元素。

希望本教程能够帮助你更好地理解 D3.js 的数据绑定和动态更新机制,并将其应用到你的可视化项目中。

以上就是使用 D3.js 实现下拉菜单联动更新可视化图表的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1584458.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月23日 00:54:00
下一篇 2025年12月23日 00:54:14

相关推荐

  • CSS文件加载失败时动态替换方案:利用onerror事件实现优雅降级

    本文探讨了在网页加载过程中,当主css文件无法找到或加载失败时,如何优雅地动态切换到备用css文件的解决方案。通过利用“标签的`onerror`事件,开发者可以实现一个简单而有效的故障转移机制,确保页面样式能够正常呈现,同时避免因同时加载多个样式表而产生的冲突。 在前端开发中,我们常常需…

    2025年12月23日 好文分享
    000
  • 动态加载CSS:主文件失败时如何优雅地切换备用样式表

    本文介绍一种在网页中实现css文件动态加载的策略。当主css文件因任何原因无法加载时,可以自动切换并应用一个备用css文件,从而避免样式冲突并确保页面始终拥有合适的视觉呈现。 在现代Web开发中,为页面引入样式表是常见操作。然而,有时我们希望引入一个主CSS文件,但如果该文件因网络问题、路径错误或服…

    2025年12月23日
    000
  • 整合JavaScript表单验证与jQuery AJAX提交:确保数据有效性

    本教程将指导您如何有效地将javascript客户端表单验证与jquery ajax表单提交机制相结合,确保在数据通过所有验证规则后才发起ajax请求。我们将通过一个实际案例,解决ajax提交绕过原生验证的问题,提升表单交互的健壮性和用户体验。 引言:表单验证与AJAX提交的挑战 在现代Web应用中…

    2025年12月23日
    000
  • AJAX发送数组到ASP.NET控制器:解决415错误及数据绑定指南

    本文深入探讨在使用jquery ajax向asp.net控制器发送数组或复杂对象时常见的415错误。我们将详细介绍客户端如何通过`json.stringify()`方法正确序列化数据,并设置正确的`content-type`。同时,也会讲解服务器端控制器如何使用`[httppost]`和`[from…

    2025年12月23日
    000
  • 解决Bootstrap按钮间距异常:深入理解与多种布局方案

    当bootstrap按钮并排显示时,有时会出现难以检查的额外间距。这通常是由于html源代码中内联块元素之间的空白字符(如换行符、空格)被浏览器渲染为视觉空间所致。本文将深入探讨这一现象的根源,并提供包括直接移除html空白、使用html注释、父元素`font-size: 0`以及更现代的flexb…

    2025年12月23日
    000
  • 如何在网页中实现全屏视频背景效果

    本教程详细讲解如何在网页中创建响应式全屏视频背景。通过运用css的position: fixed、min-width/height: 100%等属性,配合html5 标签,确保视频始终覆盖整个视口,并能在此之上叠加内容,提供沉浸式用户体验。 在现代网页设计中,全屏视频背景已成为一种流行的视觉元素,它…

    2025年12月23日
    000
  • 解决CSS伪元素遮挡页面点击事件的问题

    在使用CSS伪元素(如`::before`或`::after`)创建覆盖层时,可能会遇到伪元素遮挡下方元素,导致无法点击的问题。本文将详细介绍如何解决这个问题,通过调整`z-index`属性,确保伪元素在视觉上呈现为背景,同时不影响用户与页面元素的交互。 问题分析 当使用position: fixe…

    2025年12月23日
    000
  • html5怎么开发游戏_HTML5游戏开发引擎与Canvas绘图技术

    HTML5游戏开发依赖Canvas和JavaScript,通过绘图、动画循环与引擎工具实现跨平台游戏。使用Canvas绘制图形,结合requestAnimationFrame实现动画,推荐Phaser、PixiJS等引擎提升效率,需注意资源预加载、重绘优化、帧率控制及多设备适配等关键实践。 HTML…

    2025年12月23日
    000
  • 如何在网页中居中显示带链接的图片

    本教程详细介绍了如何在网页中实现带链接图片的水平居中显示。通过将图片元素转换为块级元素并利用CSS的自动外边距属性,可以轻松解决图片无法居中对齐的问题,并提供完整的HTML和CSS代码示例,确保图片在各种布局中都能正确居中。 在网页开发中,我们经常需要将图片与链接结合,并使其在页面中居中显示。然而,…

    2025年12月23日
    000
  • HTML5网页如何实现复制功能 HTML5网页剪贴板API的使用教程

    答案:HTML5的Clipboard API可通过JavaScript实现安全复制。需用户点击触发,在HTTPS或localhost环境下,调用navigator.clipboard.writeText()写入文本,配合try-catch处理错误,并为不支持的浏览器降级使用execCommand。 …

    2025年12月23日
    000
  • html官方站点入口_html网站免费制作链接

    html官方站点入口是https://www.w3.org/TR/html/,该网站提供HTML技术规范、元素使用示例、浏览器兼容性说明及开发工具推荐,支持多语言手册下载、代码实践指南、社区交流与标准测试,助力开发者学习与验证。 html官方站点入口在哪里?这是不少刚开始接触网页设计的朋友都关注的问…

    2025年12月23日
    000
  • HTML数据怎样进行安全防护 HTML数据采集的安全注意事项

    答案:HTML数据采集需兼顾技术安全与法律合规。1. 防止恶意内容注入,使用白名单过滤危险标签,转义特殊字符,并借助DOMPurify等工具净化HTML;2. 合理控制采集行为,遵守robots.txt、设置请求间隔、使用合法User-Agent以降低被封禁风险;3. 保障数据存储与传输安全,加密敏…

    2025年12月23日
    000
  • HTML5网页如何实现文件预览 HTML5网页文档查看的解决方案

    答案:HTML5文件预览依赖浏览器原生标签与前端技术结合,图片、PDF、文本可通过、、FileReader直接显示;Office文件需用OneDrive嵌入或前端库解析;音视频用、标签支持;安全上需防XSS、大文件分片处理,部分格式依赖服务端转换。 在HTML5网页中实现文件预览,主要依赖浏览器原生…

    2025年12月23日
    000
  • 如何在HTML中插入多语言切换_HTML语言包加载与动态内容替换

    答案:通过JSON语言包和data-lang-key属性实现多语言切换。准备lang-en.json和lang-zh.json文件,标记需翻译元素如Hello,用JavaScript加载对应语言包并替换文本内容,提供按钮切换语言,结合localStorage保存用户偏好,页面加载时读取首选语言自动切…

    2025年12月23日
    000
  • HTML5在线如何制作仪表盘 HTML5在线数据监控的开发指南

    答案:开发HTML5在线仪表盘需明确监控指标、搭建语义化页面结构、集成图表库如Chart.js实现可视化,并通过WebSocket或定时轮询接入实时数据,结合响应式设计与HTTPS安全部署确保稳定可用。 制作一个基于HTML5的在线仪表盘用于数据监控,关键在于结合HTML、CSS和JavaScrip…

    2025年12月23日
    000
  • HTML5怎么进行代码调试_HTML5开发调试技巧大全

    掌握HTML5调试技巧需先使用浏览器开发者工具,通过Elements面板检查DOM结构,Console查看JS错误与日志,Sources设置断点调试,Network监控资源加载;再验证HTML语义化标签正确性,避免嵌套与闭合错误,并用W3C验证器检测语法;针对Canvas和多媒体元素,确认上下文获取…

    2025年12月23日
    000
  • Formik中实现onChange事件触发与组件重渲染的教程

    本文旨在解决在formik表单中,`onchange`事件未能如预期触发以及如何确保其他组件能根据表单输入实时更新的问题。通过结合formik的内置机制与react的`usestate`,教程将详细阐述如何正确处理表单输入、同步数据流,并确保ui的及时响应,从而构建高效、可维护的react表单应用。…

    2025年12月23日
    000
  • JavaScript 实现数组随机显示且不重复

    本文将介绍如何使用 JavaScript 实现一个数组的随机显示功能,点击按钮后,数组中的元素会依次随机显示在页面上,且不会重复显示同一个元素,直到所有元素都显示完毕。同时,提供重置按钮,以便重新开始随机显示。 核心思路 实现该功能的关键在于以下几点: 数组乱序(Shuffle): 在显示之前,需要…

    2025年12月23日
    000
  • html5使用geolocation和地图API显示当前位置 html5使用LBS服务的集成

    首先通过HTML5 Geolocation获取用户坐标,再结合高德地图API渲染地图并标记位置。1. 在HTTPS环境下调用navigator.geolocation.getCurrentPosition()获取经纬度;2. 引入高德地图SDK,使用AMap.Map初始化地图,AMap.Marker…

    2025年12月23日
    000
  • HTML数据如何用于数据挖掘 HTML数据挖掘的完整工作流程

    明确目标后,从公开网站采集HTML数据,用Python等工具抓取并解析内容,提取价格、评论等信息,经清洗去噪后结构化存储,再进行统计分析或机器学习挖掘,最终可视化展示并建立自动化更新机制,实现全流程数据价值提取。 HTML数据本身是网页的结构化标记语言,主要用于展示内容。但对数据挖掘而言,HTML中…

    2025年12月23日
    000

发表回复

登录后才能评论
关注微信