分类 JavaScript 下的文章

用温度监测模块记录了咖啡烘焙过程中的咖啡豆温度数据,保存为CSV文件。根据该数据生成“烘焙曲线”和“升温曲线”,需要用Excel之类的表格处理软件生成对应的“折线图”,比较麻烦。然后写了个简单的网页,可以选择CSV文件,自动生成两个曲线图。生成图表的JavaScript库,采用了Frappe Charts。记录一下Frappe Charts的使用。

优点:

  1. 简单。根据官方示例代码,即可生成图表。
  2. 支持混合图表。通过多种图表同时显示,更好表达出数据的含义。
  3. 折线图(line chart)支持序列数据,并优化显示。包括:平滑曲线、优化X轴标签显示(避免显示全部而导致太密集)等。

缺点:

  1. 项目太久没更新(起码3年了),很多issue都没有处理。
  2. 扩展性差。几乎没有插入特殊处理代码的地方。
  3. 可生成Y轴标签(以虚线显示),但不能生成X轴标签。目前使用条形图(bar chart)实现X轴标签的效果(指定X轴标签上显示一条竖线)。
  4. 折线图(line chart)数据点的提示框(tooltip),不能根据不同的数据做显示定制。由于提示框是生成图表时,预先生成的,并且按数据顺序执行每个数据点的各组数据的处理。例如有3组数据,那么每次执行数据处理都记录其自增ID,再用该ID除以3求余数,就能确定当次所处理的是第几组数据,并做对应的处理。
  5. 折线图(line chart)不能设置缺失数据,也不能隐藏指定数据点。暂时无解决方案,只能用零值表示缺失数据。

本文记录了从jQurey转到原生JavaScript开发的相关处理。

一 历史

二十一世纪初,IE 6还在统治浏览器的时代,出现了一批JavaScript框架。除了提高前端开发效率,还屏蔽了各个浏览器的JavaScript接口差异。那时有3个产品印象比较深刻:

  1. prototype,http://prototypejs.org/
    其特点是在原生JavaScript基础上做扩展,定义通用的方法或接口,屏蔽各个浏览器的差异。很轻量,个人比较喜欢。
  2. Ext JS,https://www.sencha.com/products/extjs/
    数据与界面分离,提供丰富的UI组建,便于页面开发。当时浏览器JavaScript性能不高,用起来不够流畅,不适合简单排版布局的页面。但是对于开发一些管理系统,确实很方便。
  3. jQuery,https://jquery.com/
    最大特别是查找HTML元素很方便(前提是熟悉其搜索语法),有点函数式编程的味道。在那个需要手工修改HTML界面的年代,确实很方便。

二 当前

看看当前的浏览器,已经是Webkit内核的天下,加上IE已亡,ECMAScript 6普及……各个浏览器的JavaScript兼容性大大提高。所以,我们可以直接采用浏览器原生JavaScript,替代jQuery这类用于遍历或搜索DOM的框架。当然,复杂的界面,主要是响应式前端框架(AngularJS、React、VUE)的世界。

三 实现方法

主要参考这个文章,从jQuery转到原生JavaScript。

另外,对于页面上的异步请求(ajax),该文章没有提出timeout的处理。以下整理一个示例:

// 请求错误的类,用于传递错误信息
let RespError = class {
  constructor(code, msg, respJson) {
    this.code = code;
    this.msg = msg;
    this.respJson = respJson;
  }
};

// POST提交Json数据。调用ajaxJson方法前加上async就是同步调用,直接调用就是异步调用
// 默认超时10秒
let ajaxJson = async (url, formParam={}, onSuccess=(respJson)=>{}, 
    onFailed=(respError)=>{}, timeoutSec=10) => {
  let controller = new AbortController();
  let timeoutId = setTimeout(() => {
    // 超时后停止请求
    controller.abort();
    // 抛出超时的错误
    onFailed(new RespError(-1, 'TIMEOUT', null));
  }, timeoutSec * 1000);
  try {
    // 发起请求
    let response = await fetch(url, {
        signal: controller.signal, // 用于接收中断请求信号
        method: 'POST',
        cache: 'no-cache',
        headers: {
          // 声明请求的参数是JSON
          'Content-Type': 'application/json; charset=UTF-8'
        },
        body: JSON.stringify(formParam)
    });
    // 注意,响应的数据只能获取一次,包括response.json()和response.text()
    let respJson = await response.json();
    if(!response.ok) {
      // 请求失败,抛出自定义的错误对象
      throw new RespError(response.status, response.statusText, respJson);
    }
    onSuccess(respJson);
  } catch(err) {
    onFailed(err instanceof RespError ? err : new RespError(-1, err.message, null));
  } finally {
    // 请求结束,停止执行定时函数。避免相应成功后,抛出超时的错误。
    clearTimeout(timeoutId);
  }
};

关于Fetch的使用,参考:

  1. Fetch API 教程
    https://www.ruanyifeng.com/blog/2020/12/fetch-tutorial.html
  2. mdn web docs - 使用 Fetch
    https://developer.mozilla.org/zh-CN/docs/Web/API/Fetch_API/Using_Fetch

四 后记

推荐搭配 petite-vue ,实现数据与页面元素的绑定。