用HTML DOM实现有条件地渲染网页元素(下)

简介: 用HTML DOM实现有条件地渲染网页元素(下)

0 前言

在  用HTML DOM实现有条件地渲染网页元素(上)_PurpleEndurer@5lcto的技术博客_51CTO博客 中,我使用HTML DOM完成了对原有代码的改造。现在我们要赋予用户更多的选择,比如允许用户输入自己喜欢的水果名称,并添加到水果名称列表中。

1 改进代码

我们将在 用HTML DOM实现有条件地渲染网页元素(上)_PurpleEndurer@5lcto的技术博客_51CTO博客的最终代码:

<script>
  function showFruit(e)
  {
  //var oUL =  document.getElementById('ulFruit');
  //alert(oUL.childElementCount);
  var coLi = document.getElementById('ulFruit').children;
  //alert(coLi.length);
  for (var i = 0; i < coLi.length; i++)
  {
    //alert(coLi[i].title);
    coLi[i].style.display = (e.target.value==coLi[i].innerText ? "list-item" : "none");
  } // for
  } // showFruit(v)
var aFruits = new Array(['苹果', 'Apple', 'red'],
    ['桔子', 'Orange', 'orange'],
    ['葡萄', 'Grape', 'purple'],
    ['芒果','Mango','yellow'],
    ['其它', 'Other', 'blue']);
function genFruitOption()
{
  var oDivFruit = document.getElementById('divFruit');
  var oP;//<p>对象
  var oIn;// <input>对象
  var oLa;// <label>对象
  for (var i=0; i<aFruits.length; i++)
  {
  //创建<p>
  oP = document.createElement("P");
   
    //创建<input type="raido">
  oIn = document.createElement("input");
  oIn.type = 'radio';   //类型
  oIn.name = "fruit";   //
  oIn.id = 'ra' + aFruits[i][1];  //id
  oIn.value = aFruits[i][0];  //值
    oIn.addEventListener("click", showFruit); //增加click事件监听器
  //创建<label>
  oLa = document.createElement('label')
  oLa.htmlFor = 'ra' + aFruits[i][1];
  oLa.appendChild(document.createTextNode(aFruits[i][0]));
    //向<div>容器追加网页元素<P><input><label>对象
  oDivFruit.appendChild(oP);
  oDivFruit.appendChild(oIn);
  oDivFruit.appendChild(oLa);
  } //for
} //genFruitOption()
function genFruitLi()
{
  var oUL =  document.getElementById('ulFruit');
  var oLi;
  for (var i=0; i<aFruits.length; i++)
  {
  oLi = document.createElement("li");
  oLi.appendChild(document.createTextNode(aFruits[i][0]));
  oLi.style.color = aFruits[i][2]; //设置颜色
  oUL.appendChild(oLi);
  } //for  
} //genFruitLi()
</script>
<p>用JavaScript响应click事件有条件地渲染网页元素</p>
<p>(引入HTML COM技术改造)</p>
<p style="margin-left:20%; color:purple; font-weight:bold;">
  by PurpleEndurer
</p>
<p>你喜欢哪种水果?</p>
<div id="divFruit">
<script>
  genFruitOption();
</script>
</div>  
<p>你喜欢的是:</p>
<ul id="ulFruit">
<script>
  genFruitLi();
</script>
</ul>

基础上进行改进。

1.1 修改网页元素描述代码

我们只需要修改2个地方。

1.1.1 修改技术改进说明

<p>(引入HTML COM技术改造)</p>

改为

<p>(允许用户添加自己喜欢的水果)</p>

1.1.2 增加代码提示用户输入和添加水果名称

我们在可选水果列表代码:

<p>你喜欢哪种水果?</p>
<div id="divFruit">
<script>
  genFruitOption();
</script>
</div>


的下方增加一段<p></p>代码,来输出3个页面元素:

一段提示用户输入水果名称的信息

一个文本框:用来给用户输入具体的水果名称

一个按钮:用来把输入的水果名称添加到水果列表中

即:

<p style="color:#0033ff">
  温馨提示:如果以上没有您喜欢的水果,您可以输入和添加喜欢的水果名称到列表中:
  <input type="text" id="itFruit" value=" 水果名称" />  
  <input type="button" value="添加" onclick="addFruit()" />
</p>

当用户点击“添加”按钮后,我们将执行addFruit函数,实现把用户输入的水果名称添加到水果列表的目标。

修改后的代码如下:

<p>用JavaScript响应click事件有条件地渲染网页元素</p>
<p>(允许用户添加自己喜欢的水果)</p>
<p style="margin-left:20%; color:purple; font-weight:bold;">
  by PurpleEndurer
</p>
<p>你喜欢哪种水果?</p>
<div id="divFruit">
<script>
  genFruitOption();
</script>
</div>  
<p style="color:#0033ff">
  温馨提示:如果以上没有您喜欢的水果,您可以输入和添加喜欢的水果名称到列表中:
  <input type="text" id="itFruit" value=" 水果名称" />  
  <input type="button" value="添加" onclick="addFruit()" />
</p>
<p>你喜欢的是:</p>
<ul id="ulFruit">
<script>
  genFruitLi();
</script>
</ul>

2.2 修改JavaScript脚本

改造主要集中在JavaScript脚本这块。

2.2.1 增加addFruit函数

addFruit函数的要提供的功能是把用户输入的水果名称添加到水果列表。

由于水果列表信息是存放在数组aFruits中的,所以我们要实现的第一步是把用户输入的水果信息添加到数组aFruits中。

2.2.1.1 确定新水果放在水果列表的位置

那么用户输入的水果信息添加到水果列表中的什么位置好呢?

从技术上来说,不管是放在列表开头,还是追加到列表的末尾,亦或插入到列表中的某个位置,这些都可以实现。

从视觉上来说,如果把用户输入的水果信息放在列表开头或末尾这两头,这样会让用户更容易找到自己添加的水果。

从情理上来说,用户输入的水果信息是追加进来的,放在列表的末尾更合情理。

所以我们的选择是把用户输入的水果信息放在列表的末尾,也就是作为数组aFruits的最后一行元素。

要把数组元素追加到数组的末尾,我们可以使用数组对象的push方法。

2.2.1.2 确定新水果的id

由于数组aFruits是二维数组,每一行存储一种水果的信息,共有3个成员,分别存储了水果名称,水果的id和水果颜色。

其中:

第1个成员水果名称由用户输入。

第2个成员水果id原则上不能跟原有水果的id重复,不应由用户输入,因为用户并知道原有的水果的id是什么。

我们的解法办法是这样来定义id:以fruit开头,再接上aFruits数组的长度。

至于第3个成员水果颜色,我可以先默认为'black'或''。

2.2.1.3 更新水果列表

在把用户输入的水果信息放在列表的末尾后,我们需要更新用户可选水果列表,把用户新添加的水果显示出来,提供给用户选择。

相应的,我们也要更新展示用户选择的水果结果的列表。

实现方法就是依次调用 genFruitOption函数和 genFruitLi函数。

具体的代码如下:

function addFruit()
{
  //获取用户添加的水果名称
  var sFruit = document.getElementById('itFruit').value;
  //aFruits.unshift([sFruit, ('fruit'+aFruits.length) , 'black']);//把用户添加的水果名称加入数组头部,成为第1个元素
  aFruits.push([sFruit, ('fruit'+aFruits.length) , 'black']);//把用户添加的水果名称加入数组尾部,成为最后一个元素
 
  genFruitOption();//更新用户可选水果的列表
  genFruitLi(); //更新展示用户选择水果的列表
}

2.2.2 修改 genFruitOption函数

genFruitOption函数的功能是生成水果列表,提供给用户进选择。

如果不做修改就反复调用,将会重复生成列表。

因此,我们需要做1个改进,就是输出新的水果列表前,要把原有列表项清除掉。

实现的方法有两种:

第一种方法简单了当,就是直接将列表容器的innerHTML置为''。

第二种方法文雅一点,就是使用HTML COM提供的parent.removeChild方法来逐一清除。

由于我们是要把原有列表项全部清除掉,而不只是清除其中一些,所以我们使用第一种方法,在genFruitOption函数中增加一行代码:

oDivFruit.innerHTML = '';//清空原有水果选项


即:

function genFruitOption()
{
  var oDivFruit = document.getElementById('divFruit');
  oDivFruit.innerHTML = '';//清空原有水果选项
  var oP;//<p>对象
  var oIn;// <input>对象
  var oLa;// <label>对象
  for (var i=0; i<aFruits.length; i++)
  {
  //创建<p>
  oP = document.createElement("P");
   
    //创建<input type="raido">
  oIn = document.createElement("input");
  oIn.type = 'radio';   //类型
  oIn.name = "fruit";   //
  oIn.id = 'ra' + aFruits[i][1];  //id
  oIn.value = aFruits[i][0];  //值
    oIn.addEventListener("click", showFruit); //增加click事件监听器
  //创建<label>
  oLa = document.createElement('label')
  oLa.htmlFor = 'ra' + aFruits[i][1];
  oLa.appendChild(document.createTextNode(aFruits[i][0]));
    //向<div>容器追加网页元素<P><input><label>对象
  oDivFruit.appendChild(oP);
  oDivFruit.appendChild(oIn);
  oDivFruit.appendChild(oLa);
  } //for
} //genFruitOption()

2.2.3 修改genFruitLi函数

genFruitLi函数的功能是显示用户选择的水果。

当用户添加自己的水果后,也需要进行相应的更新。

跟修改genFruitOption函数一样,为了避免重复生成列表,我们同样需要在输出新的水果列表前,要把原有列表项清除掉。

我们在genFruitOption函数中增加一行代码:

oUL.innerHTML = ''; //清空原有水果列表项

即:

function genFruitLi()
{
  var oUL =  document.getElementById('ulFruit');
  oUL.innerHTML = ''; //清空原有水果列表项  
  var oLi;
  //添加水果列表项
  for (var i=0; i<aFruits.length; i++)
  {
  oLi = document.createElement("li");
  oLi.appendChild(document.createTextNode(aFruits[i][0]));
  oLi.style.color = aFruits[i][2]; //设置颜色
  oUL.appendChild(oLi);
  } //for  
} //genFruitLi()

3 改造后的最终代码

综合以上修改后的最终代码如下:

<script>
  function showFruit(e)
  {
  //var oUL =  document.getElementById('ulFruit');
  //alert(oUL.childElementCount);
  var coLi = document.getElementById('ulFruit').children;
  //alert(coLi.length);
  for (var i = 0; i < coLi.length; i++)
  {
    //alert(coLi[i].title);
    coLi[i].style.display = (e.target.value==coLi[i].innerText ? "list-item" : "none");
  } // for
  } // showFruit(e)
var aFruits = new Array(['苹果', 'Apple', 'red'],
    ['桔子', 'Orange', 'orange'],
    ['葡萄', 'Grape', 'purple'],
    ['芒果','Mango','yellow'],
    ['其它', 'Other', 'blue']);
function genFruitOption()
{
  var oDivFruit = document.getElementById('divFruit');
  oDivFruit.innerHTML = '';//清空原有水果选项
  var oP;//<p>对象
  var oIn;// <input>对象
  var oLa;// <label>对象
  for (var i=0; i<aFruits.length; i++)
  {
  //创建<p>
  oP = document.createElement("P");
   
    //创建<input type="raido">
  oIn = document.createElement("input");
  oIn.type = 'radio';   //类型
  oIn.name = "fruit";   //
  oIn.id = 'ra' + aFruits[i][1];  //id
  oIn.value = aFruits[i][0];  //值
    oIn.addEventListener("click", showFruit); //增加click事件监听器
  //创建<label>
  oLa = document.createElement('label')
  oLa.htmlFor = 'ra' + aFruits[i][1];
  oLa.appendChild(document.createTextNode(aFruits[i][0]));
    //向<div>容器追加网页元素<P><input><label>对象
  oDivFruit.appendChild(oP);
  oDivFruit.appendChild(oIn);
  oDivFruit.appendChild(oLa);
  } //for
} //genFruitOption()
function genFruitLi()
{
  var oUL =  document.getElementById('ulFruit');
  oUL.innerHTML = ''; //清空原有水果列表项  
  var oLi;
  //添加水果列表项
  for (var i=0; i<aFruits.length; i++)
  {
  oLi = document.createElement("li");
  oLi.appendChild(document.createTextNode(aFruits[i][0]));
  oLi.style.color = aFruits[i][2]; //设置颜色
  oUL.appendChild(oLi);
  } //for  
} //genFruitLi()
function addFruit()
{
  //获取用户添加的水果名称
  var sFruit = document.getElementById('itFruit').value;
  //aFruits.unshift([sFruit, ('fruit'+aFruits.length) , 'black']);//把用户添加的水果名称加入数组头部,成为第1个元素
  aFruits.push([sFruit, ('fruit'+aFruits.length) , 'black']);//把用户添加的水果名称加入数组尾部,成为最后一个元素
  //alert(aFruits);
  genFruitOption();//更新用户可选水果的列表
        genFruitLi(); //更新展示用户选择水果的列表
}
</script>
<p>用JavaScript响应click事件有条件地渲染网页元素</p>
<p>(允许用户添加自己喜欢的水果)</p>
<p style="margin-left:20%; color:purple; font-weight:bold;">
  by PurpleEndurer
</p>
<p>你喜欢哪种水果?</p>
<div id="divFruit">
<script>
  genFruitOption();
</script>
</div>  
<p style="color:#0033ff">
  温馨提示:如果以上没有您喜欢的水果,您可以输入和添加喜欢的水果名称到列表中:
  <input type="text" id="itFruit" value="水果名称" />  
  <input type="button" value="添加" onclick="addFruit()" />
</p>
<p>你喜欢的是:</p>
<ul id="ulFruit">
<script>
  genFruitLi();
</script>
</ul>

4 代码运行效果

image.png

5 小结

我们通过改进代码为用户提供了增加自己水果的功能,为了突出重点,我们没有对用户输入的水果名称进行检查,比如用户是否输入了有效的水果名称,以及输入的水果名称是否与水果选项列表中原有的重复,等等。

对于水果id这一项,为了便于识记,我们的命令原则上是用水果的英母名做为id,而不是fruit+数组aFruits的长度,因为从这样的id上我们无法看出它对应的是什么水果。

一个可以考虑的改进方向是建立一个水果信息字典,把水果的中文或英文名称,水果对应英文名称(作为id),水果颜色存储起来,当用户输入水果的名称时,我们就在字典中查询,取得id和颜色信息。


相关文章
|
12天前
|
弹性计算 人工智能 架构师
阿里云携手Altair共拓云上工业仿真新机遇
2024年9月12日,「2024 Altair 技术大会杭州站」成功召开,阿里云弹性计算产品运营与生态负责人何川,与Altair中国技术总监赵阳在会上联合发布了最新的“云上CAE一体机”。
阿里云携手Altair共拓云上工业仿真新机遇
|
8天前
|
机器学习/深度学习 算法 大数据
【BetterBench博士】2024 “华为杯”第二十一届中国研究生数学建模竞赛 选题分析
2024“华为杯”数学建模竞赛,对ABCDEF每个题进行详细的分析,涵盖风电场功率优化、WLAN网络吞吐量、磁性元件损耗建模、地理环境问题、高速公路应急车道启用和X射线脉冲星建模等多领域问题,解析了问题类型、专业和技能的需要。
2523 18
【BetterBench博士】2024 “华为杯”第二十一届中国研究生数学建模竞赛 选题分析
|
8天前
|
机器学习/深度学习 算法 数据可视化
【BetterBench博士】2024年中国研究生数学建模竞赛 C题:数据驱动下磁性元件的磁芯损耗建模 问题分析、数学模型、python 代码
2024年中国研究生数学建模竞赛C题聚焦磁性元件磁芯损耗建模。题目背景介绍了电能变换技术的发展与应用,强调磁性元件在功率变换器中的重要性。磁芯损耗受多种因素影响,现有模型难以精确预测。题目要求通过数据分析建立高精度磁芯损耗模型。具体任务包括励磁波形分类、修正斯坦麦茨方程、分析影响因素、构建预测模型及优化设计条件。涉及数据预处理、特征提取、机器学习及优化算法等技术。适合电气、材料、计算机等多个专业学生参与。
1525 15
【BetterBench博士】2024年中国研究生数学建模竞赛 C题:数据驱动下磁性元件的磁芯损耗建模 问题分析、数学模型、python 代码
|
4天前
|
存储 关系型数据库 分布式数据库
GraphRAG:基于PolarDB+通义千问+LangChain的知识图谱+大模型最佳实践
本文介绍了如何使用PolarDB、通义千问和LangChain搭建GraphRAG系统,结合知识图谱和向量检索提升问答质量。通过实例展示了单独使用向量检索和图检索的局限性,并通过图+向量联合搜索增强了问答准确性。PolarDB支持AGE图引擎和pgvector插件,实现图数据和向量数据的统一存储与检索,提升了RAG系统的性能和效果。
|
10天前
|
编解码 JSON 自然语言处理
通义千问重磅开源Qwen2.5,性能超越Llama
击败Meta,阿里Qwen2.5再登全球开源大模型王座
596 14
|
1月前
|
运维 Cloud Native Devops
一线实战:运维人少,我们从 0 到 1 实践 DevOps 和云原生
上海经证科技有限公司为有效推进软件项目管理和开发工作,选择了阿里云云效作为 DevOps 解决方案。通过云效,实现了从 0 开始,到现在近百个微服务、数百条流水线与应用交付的全面覆盖,有效支撑了敏捷开发流程。
19283 30
|
10天前
|
人工智能 自动驾驶 机器人
吴泳铭:AI最大的想象力不在手机屏幕,而是改变物理世界
过去22个月,AI发展速度超过任何历史时期,但我们依然还处于AGI变革的早期。生成式AI最大的想象力,绝不是在手机屏幕上做一两个新的超级app,而是接管数字世界,改变物理世界。
498 49
吴泳铭:AI最大的想象力不在手机屏幕,而是改变物理世界
|
1月前
|
人工智能 自然语言处理 搜索推荐
阿里云Elasticsearch AI搜索实践
本文介绍了阿里云 Elasticsearch 在AI 搜索方面的技术实践与探索。
18845 20
|
1月前
|
Rust Apache 对象存储
Apache Paimon V0.9最新进展
Apache Paimon V0.9 版本即将发布,此版本带来了多项新特性并解决了关键挑战。Paimon自2022年从Flink社区诞生以来迅速成长,已成为Apache顶级项目,并广泛应用于阿里集团内外的多家企业。
17530 13
Apache Paimon V0.9最新进展
|
3天前
|
云安全 存储 运维
叮咚!您有一份六大必做安全操作清单,请查收
云安全态势管理(CSPM)开启免费试用
368 4
叮咚!您有一份六大必做安全操作清单,请查收