基于 HTML5 WebGL 实现的医疗物流系统


Posted in HTML / CSS onOctober 08, 2019

前言

物联网( IoT ),简单的理解就是物体之间通过互联网进行链接。世界上的万事万物,都可以通过数据的改变进行智能化管理。IoT 的兴起在医疗行业中具有拯救生命的潜在作用。
不断的收集用户信息并且实时的进行诊断,所以未来  IoT 肯定在医疗行业的应用会呈覆盖性。下面是我最近做的一个医疗物流系统,用来观察医疗物流过程。

基于 HTML5 WebGL 实现的医疗物流系统

ht官网链接:http://www.hightopo.com/cn-index.html

demo链接: https://www.hightopo.com/demo/pivas/
 

实现过程

增加光源

 整个原场景其实是非常暗的,所以需要使用灯光的效果照亮整个场景,使其接近真实世界的场景。

 我们看下对比。

基于 HTML5 WebGL 实现的医疗物流系统

light 的一些属性:

type 代表灯光的类型

color 代表灯光的颜色

intensity 代表灯光的强度(1是最大值)

range 代表范围

addLight() {
        const skyBox = this.dm.getDataByTag('skyBox')

        // 限制视野在天空球之内
        this.gv.setSkyBox(skyBox)
        const light = new ht.Light()
        const lightSource = this.dm.getDataByTag('sunlight').p3()

        const config = {
            'light.type': 'point',
            'light.color': 'white',
            'light.intensity': 0.3,
            'light.range': 10000
        }

        light.s(config)
        light.p3(lightSource)this.dm.add(light)
 }

 

看向物体

 看到左下角的一个小窗口,其实是另一个3d场景,把它定位到左下角的,两个场景都使用了反序列化( deserialize )。

基于 HTML5 WebGL 实现的医疗物流系统

 因为要定位医疗箱移动,所以这里使用到了 flyTo 方法 。

var renderCanvas = function (medical, duration) {
    ht.Default.startAnim({
       duration,
       easing(v, t) {
         return t
       },
       action(v, t) {
         outScreenG3d.flyTo(medical, { direction: [-5, 3, 5], distance: 300 })
       }
    })
 }

 

封装动画

如果要实现这么多的动画,首先想到的是一个个物体进行移动的过程。医疗箱的行走、电梯的升降、传送带运送医疗箱等我们都可以对它们的动作进行封装。

如图可以看到医疗箱总是在动,所以定义了一个行走的动画,每次医疗箱行走的距离、行走方向、动画的配置都进行传参。

这里要说明的参数:

1.node(对应的元素)

2.fn(动画执行完进行回调的函数)

3.config(动画配置)

4.coord(方向轴)

// 行走动画
walkAnim(node, fn, config, coord) {
const { duration, space } = config

const positionArray = node.p3()

let isShadow = false
   let ShadowNode = null
    // 如果移动的元素是icu车或者供应车的话 获取它的阴影跟随元素移动
   if (node.getTag() === 'supply' || node.getTag() === 'icuCar') {
   
isShadow = true
   
ShadowNode = this.dm.getDataByTag(`${node.getTag()}Shadow`)
   }

   ht.Default.startAnim({
   
duration,
   
easing: function (t) {
      
return t
      },
      action(v, t) {
      
if (coord === 'x') {
          
node.p3(positionArray[0] + t * space, positionArray[1], positionArray[2])
          
isShadow && ShadowNode.p3(positionArray[0] + t * space, positionArray[1], positionArray[2])
      
} else if (coord === 'y') {
          
node.p3(positionArray[0], positionArray[1] + t * space, positionArray[2])
          
isShadow && ShadowNode.p3(positionArray[0], positionArray[1] + t * space, positionArray[2])
         } else {
         
node.p3(positionArray[0], positionArray[1], positionArray[2] + t * space)
         
isShadow && ShadowNode.p3(positionArray[0], positionArray[1], positionArray[2] + t * space)
         }
      },
      finishFunc() {
         typeof fn === 'function' && fn(node)
      }
   })
}

物体之间的影响

电梯的升降会影响很多东西,比如频台的移动会带着传送带和医疗箱,这里我用到了 sethost 吸附方法(吸附:节点指定宿主,宿主进行改变会影响节点)。

很多场景下非常合适,我需要电梯升降的过程中带用医疗箱和频台一起上升,还有医疗箱放到传送带的时候,医疗箱要动起来,感觉是这真的传送带在带动医疗箱进行运动。

这里要说明的参数:

1.node(操作的电梯元素)

2.medicalKit(医疗箱)

3.fn(动画执行完进行回调的函数)

4.status (电梯上升和下降的状态)

5.config(动画配置)

 

// 电梯升降动画
    elevatorAnim(node, medicalKit, fn, status, config) {
        const self = this
        // 获取电梯的index 让对应的频台也跟着动
        const elevatorIndex = node.getTag().replace(/[^0-9]/ig, '') - 0
        // 获取医疗箱的index 控制电梯升降的距离
        const medicalKitIndex = medicalKit.getTag().replace(/[^0-9]/ig, '') - 0
        const positionArray = node.p3()
        const station = self.dm.getDataByTag(`station${elevatorIndex}`)
        //吸附宿主
        station.setHost(node)
        medicalKit.setHost(node)
        // 设置升降状态
        if (elevatorIndex === 3) self.elevatorRunning = true
        // 升降距离 status 为 0 的时候是下降 最低部位的距离是固定的 所以只需要控制上升的距离
        const medicalKitLevel = self.returnMedicalKitLevel(medicalKitIndex)
        // 电梯的属性
        // 最低点的位置 Lowest
        // 如果有轨道的话 就去轨道的位置  否则就按照层数 orbitalP
        // 第一层的位置  distance
        let space
        const addSpace = medicalKitIndex === 7 ? 100 : 0
        if (status == 1) {
            space = config.orbitalP ? config.orbitalP : config.distance + addSpace + (400 * medicalKitLevel)
        } else {
            space = config.Lowest
        }
        // 下降状态时 医疗箱不会做动作
        if (status === 0) {
            medicalKit.setHost()
        }
        return ht.Default.startAnim({
            duration: config.orbitalP ? 2000 : (medicalKitLevel === 0 && elevatorIndex == 3 ? 700 : 2500 + (medicalKitLevel * 1000)),
            action(v, t) {
                node.p3(
                    positionArray[0],
                    positionArray[1] + ((space - positionArray[1]) * t),
                    positionArray[2]
                )
            },
            finishFunc() {
                station.setHost()
                typeof fn === 'function' && fn(node)
            }
        })
    }

 

动画方法

 动画的过程中有个问题需要处理就是等待电梯的动画,医疗箱在动画过程中,需要判断电梯是否在上升,如果不在地面的话,需要等待。

基于 HTML5 WebGL 实现的医疗物流系统

 我的思路是,当医疗箱走到离电梯一点距离的时候,需要判断电梯是否在上升状态,如果是的话,需要调用动画暂停的方法。

 当 elevatorRunning 为 false 的时候代表电梯没有运动,否则在运动中。

 电梯动画开始的时候设置为 true,结束后设置变量为 false,  就可以监控它的状态了。

 ht.Default.startAnim 方法返回一个实例,利用 action 方法, 实现轮询监听动画状态,然后进行操作。

 当 elevatorRunning 为 true 的话, 使用 anim.pause() 暂停当前动画。

 当 elevatorRunning 为 false 的话, 使用 anim.resume() 继续当前动画。

const anim = ht.Default.startAnim({
duration,

action(v, t) {


node.p3(
   

positionArray[0],
   

positionArray[1],
   

positionArray[2] - (tpMax - positionArray[2]) * t
   
);
   
if (index > 1 && self.elevatorRunning === true) {
       
if (node.p3()[2] <= stopMax) {
       

anim.pause();
       

const t = setInterval(() => {
       


if (self.elevatorRunning === false) {
          


anim.resume();
          


clearInterval(t);
           

}
       

}, 100);
   

}
 
 }
   },
   finishFunc() {
   
typeof fn === "function" && fn();
   }
});

 

事件监听(发布、订阅)

 因为需要监听某个当前动画的结束,然后进行相机位移。

 如图,我需要监听第一个 3d 场景中显示提示文字动画结束,然后执行第二个 3d 场景的显示。因为2个是不同的场景,是不能用回调的方法监听到的,所以这里就用到了 eventBus 事件总线。

 下面是 eventBus 的使用,第一个参数代表要监听的注册函数名,第二个是回调函数。

// 事件总线 监听事件
eventbus.on('animation1', _ => {
  const medical = dm.getDataByTag('medicalKit1')
  renderView(medical, dm, gv)
})

 

下面是 eventBus 发射的使用,第一个参数代表要触发的函数名,第二个是发射给函数的参数。

// 触发事件
eventbus.emit("animation1", null);

总结

 做完这个 demo 之后,除了对 HT for Web 更加熟练之外,对物联网也有了更深刻的概念。

 我身为一名前端工作者,在这个时代感觉非常的自豪,因为我能通过自己的技能创造出许多能造福和改善人们生活的东西。

 希望大家看到我的 demo 能够得到一些启发,同时也要相信自己能够创造不可能,为社会做出贡献。

HTML / CSS 相关文章推荐
css3 给背景设置渐变色的方法
Sep 12 HTML / CSS
button在IE6/7下的黑边去除方案
Dec 24 HTML / CSS
纯css3实现图片翻牌特效
Mar 10 HTML / CSS
纯CSS3实现8组超炫酷鼠标滑过图片动画
Mar 16 HTML / CSS
CSS3制作轮播图的一种方法
Nov 11 HTML / CSS
详解Canvas 实现炫丽的粒子运动效果(粒子生成文字)
Feb 01 HTML / CSS
HTML5拖放API实现拖放排序的实例代码
May 11 HTML / CSS
html5 datalist 选中option选项后的触发事件
Mar 05 HTML / CSS
AmazeUI中模态框的实现
Aug 19 HTML / CSS
关于HTML5+ API plusready的兼容问题
Nov 20 HTML / CSS
CSS元素定位之通过元素的标签或者元素的id、class属性定位详解
Sep 23 HTML / CSS
CSS 鼠标选中文字后改变背景色的实现代码
May 21 HTML / CSS
html2canvas生成清晰的图片实现打印的示例代码
Sep 30 #HTML / CSS
html5视频媒体标签video的使用方法及完整参数说明详解
Sep 27 #HTML / CSS
Html5实现首页动态视频背景的示例代码
Sep 25 #HTML / CSS
HTML5自定义元素播放焦点图动画的实现
Sep 25 #HTML / CSS
使用Html5中的cavas画一面国旗
Sep 25 #HTML / CSS
关于canvas绘制模糊问题的解决方法
Sep 24 #HTML / CSS
html svg生成环形进度条的实现方法
Sep 23 #HTML / CSS
You might like
php 文件上传类代码
2011/08/06 PHP
php验证session无效的解决方法
2014/11/04 PHP
PHP获取服务器端信息的方法
2014/11/28 PHP
使用PHP生成PDF方法详解
2015/01/23 PHP
PHP中strnatcmp()函数“自然排序算法”进行字符串比较用法分析(对比strcmp函数)
2016/01/07 PHP
IOS 开发之NSDictionary转换成JSON字符串
2017/08/14 PHP
php获取微信基础接口凭证Access_token
2018/08/23 PHP
php 的多进程操作实践案例分析
2020/02/28 PHP
File文件控件,选中文件(图片,flash,视频)即立即预览显示
2009/04/09 Javascript
基于jquery的下拉框改变动态添加和删除表格实现代码
2020/09/12 Javascript
html中的input标签的checked属性jquery判断代码
2012/09/19 Javascript
javascript遍历控件实例详细解析
2014/01/10 Javascript
JavaScript对Json的增删改属性详解
2016/06/02 Javascript
Nodejs全局安装和本地安装的不同之处
2016/07/04 NodeJs
整理一下常见的IE错误
2016/11/18 Javascript
纯原生js实现table表格的增删
2017/01/05 Javascript
ui-router中使用ocLazyLoad和resolve的具体方法
2017/10/18 Javascript
微信小程序实现点击按钮修改文字大小功能【附demo源码下载】
2017/12/06 Javascript
JS实现的tab页切换效果完整示例
2018/12/18 Javascript
关于JS模块化的知识点分享
2019/10/16 Javascript
js+css3实现简单时钟特效
2020/09/13 Javascript
JS实现放大镜效果
2020/09/21 Javascript
vue-resource 拦截器interceptors使用详解
2021/01/18 Vue.js
浅谈numpy库的常用基本操作方法
2018/01/09 Python
浅谈Python对内存的使用(深浅拷贝)
2018/01/17 Python
tensorflow实现图像的裁剪和填充方法
2018/07/27 Python
python关于矩阵重复赋值覆盖问题的解决方法
2019/07/19 Python
使用批处理脚本自动生成并上传NuGet包(操作方法)
2019/11/19 Python
西班牙用户之间买卖视频游戏的平台:Wakkap
2020/03/21 全球购物
教学改革实施方案
2014/03/31 职场文书
党员对十八届四中全会的期盼思想汇报范文
2014/10/17 职场文书
教师培训学习心得体会
2016/01/21 职场文书
Redis如何一键部署脚本
2021/04/12 Redis
MySQL派生表联表查询实战过程
2022/03/20 MySQL
Win11任务栏无法正常显示 资源管理器不停重启的解决方法
2022/07/07 数码科技
Python可视化神器pyecharts之绘制箱形图
2022/07/07 Python