疫情可视化(visualization)

地图可视化


更多有趣示例 尽在 知屋安砖社区

示例

在这里插入图片描述

HTML

<div class="wrapper">
          <h1>THREEJS - A breast cancer visualization by joni.giuro</h1>
          <canvas width="750" height="500" id="source"></canvas>
          <div id="container">
          </div>
          <div class="dida">
            <h2 class="subtitle">Out of 100'000 women, how many were diagnosed with breast cancer?</h2>
            <ul class="inline-list">
                <!--<li class="item"><a class="button" href="#" id="zero">10+</a></li>-->
                <li class="item"><a class="button active" href="#" id="one">22-40</a></li>
                <li class="item"><a class="button" href="#" id="two">41-57</a></li>
                <li class="item"><a class="button" href="#" id="three">58-74</a></li>
                <li class="item"><a class="button" href="#" id="four">75-92</a></li>
                <li class="item"><a class="button" href="#" id="five">92+</a></li>
            </ul>
          </div>
          <p>
             Nothing too fancy. I took the original image from here:
            <a href="http://www.worldwidebreastcancer.com/wp-content/uploads/2011/08/breastcancerstatsworldwide.jpg">Breast cancer stats</a>, converted it to grayscale, placed it on a canvas and read pixel information from it. Once I have collected the color informations I need, I then proceed to create a lot of cubes with threejs. After a while the bars get resized. That's the ideal breast cancer graph
          </p>
          <div class="me">
            <ul class="block-list">
              <li class="item"><a href="http://zhereicome.com/experiments">Blog</a></li>
              <li class="item"><a href="https://twitter.com/JoniGiuro">Twittur</a></li>
            </ul>
          </div>
        </div>

CSS

@import url(https://fonts.googleapis.com/css?family=Alef);

body {
  background: url('https://subtlepatterns.com/patterns/ticks.png') center center repeat;
}
.wrapper {
  font-family: Alef, Helvetica, Arial, sans-serif;
  text-align: center;
  color: #444;
  text-shadow: 0 1px white;
}

a {
  color: #33b0d6;
}

#source-image {
    display: none;
}

#source {
    display: none;
}

#container {
    position: relative;
    width: 750px;
    margin: 0 auto;
}

canvas {
    margin: 0 auto;
    text-align: center;
    border: 1px solid black;
    background: rgba(255,255,255,.5);
}

.inline-list {
  list-style: none;
  width: 740px;
  margin: 0 auto;
  padding: 0;
}

.item {
    width: 15%;
    display: inline-block;
}

.block-list .item {
    display: block;
    text-align: center;
    width: 100%;
}

.button {
    display: block;
    padding: 10px;
    font-size: 20px;
    background: #444;
    color: #FFF;
    text-shadow: none;
    border-radius: 3px;
    text-decoration: none;
    font-weight: bold;
}

.button:hover {
    background: #777;
    box-shadow: 0 0 5px rgba(0,0,0,.5);
}

.button.active {
    background-color: #BADA55;
}

JS

var BreastCancer = {

            commonStuff : {
                tileSize: 15,
                heightGain: 1/30,
                width: 750,
                height: 420,
                sourceContext: null,
                dataArray: [],
                cubesArray: [],
                hReached : false,
                vReached : true,
                mouseDown : false,
                fighting : false,
                currentShow: 44,
                minShow: 22
            },

            doNotInit: function() {
                this.copySourceToCanvas();
            },

            copySourceToCanvas : function() {
                var self = this;
                var sourceCanvas = document.getElementById('source');
                var ctx = sourceCanvas.getContext('2d');
                this.commonStuff.sourceContext = ctx;

                var img = new Image();

                img.onload = function() {
                    ctx.drawImage(img, 0, 0);
                    self.collectData();
                };

                img.src = '';
            },

            collectData : function() {
                var width = this.commonStuff.width;
                var height = this.commonStuff.height;
                var tileSize = this.commonStuff.tileSize;
                var ctx = this.commonStuff.sourceContext;
                var dataArray = this.commonStuff.dataArray;

                for ( var i = 0 ; i < width ; i += tileSize ) {
                    for ( var q = 0 ; q < height ; q += tileSize ) {
                        var pixelInfo = ctx.getImageData(i, q, 1, 1);
                        dataArray.push({
                          alpha: 255 - pixelInfo.data[2],
                          column: i,
                          row: q  
                        });

                    }
                }

                //this.boxesFlat();
                this.drawBoxes();
            },

            //THIS WAS THE SAME THING ON A 2D CANVAS, JUST FOR TESTING PURPOSES
            boxesFlat : function() {
                var data = this.commonStuff.dataArray;
                var arrayLength = data.length;
                var tileSize = this.commonStuff.tileSize;
                var width = this.commonStuff.width;
                var height = this.commonStuff.height;
                var row = -1;
                var ctx = this.commonStuff.sourceContext;

                for ( var i = 0 ; i < arrayLength ; i++ ) {
                    var item = data[i];

                    ctx.fillStyle = "rgb("+ (255 - item.alpha) + "," +(255 - item.alpha)+ "," + (255 - item.alpha)+ ")" ;
                    ctx.fillRect( item.column, item.row, 4, 4 )

                }
            },
            //CREATE ALL THE BOXES
            drawBoxes : function(  ) {
                var data = this.commonStuff.dataArray;
                var arrayLength = data.length;
                var tileSize = this.commonStuff.tileSize;
                var width = this.commonStuff.width;
                var height = this.commonStuff.height;
                var row = -1;
                var self = this;


                //THREEJS STUFF

                // set the scene size
                var WIDTH = 750,
                    HEIGHT = 500

                // set some camera attributes
                var VIEW_ANGLE = 45,
                    ASPECT = WIDTH / HEIGHT,
                    NEAR = 0.1,
                    FAR = 10000;

                // get the DOM element to attach to
                // - assume we've got jQuery to hand
                var $container = $('#container');

                // create a WebGL renderer, camera
                // and a scene
                var renderer = new THREE.WebGLRenderer();
                var camera =
                  new THREE.PerspectiveCamera(
                    VIEW_ANGLE,
                    ASPECT,
                    NEAR,
                    FAR);

                var scene = new THREE.Scene();

                // add the camera to the scene
                scene.add(camera);

                // the camera starts at 0,0,0
                // so pull it back
                camera.position.z = 700;
                camera.position.y = 0;
                //camera.lookAt(0,0,0)

                // start the renderer
                renderer.setSize(WIDTH, HEIGHT);

                // attach the render-supplied DOM element
                $container.append(renderer.domElement);

                  // create a point light
                var ambientLight =
                  new THREE.AmbientLight(0x404040);

                  var pointLight = new THREE.DirectionalLight(0xFFFFFF);

                // set its position
                pointLight.position.x = 10;
                pointLight.position.y = 50;
                pointLight.position.z = 130;

                // add to the scene
                scene.add(pointLight);
                scene.add(ambientLight);

                var whiteMaterial =
                  new THREE.MeshLambertMaterial(
                    {
                      color: 0xFFFFFF
                    });

                var greyMaterial =
                  new THREE.MeshLambertMaterial(
                    {
                      color: 0x444444
                    });

                var cubeTopGeo = new THREE.CubeGeometry(800, 10, 10);
                var cubeTop = new THREE.Mesh( cubeTopGeo, greyMaterial);
                cubeTop.position.y = 250;
                cubeTop.position.z = -20;

                var cubeBottomGeo = new THREE.CubeGeometry(800, 10, 10);
                var cubeBottom = new THREE.Mesh( cubeTopGeo, greyMaterial);
                cubeBottom.position.y = -250;
                cubeBottom.position.z = -20;

                scene.add(cubeTop)
                scene.add(cubeBottom)


                //SCENE
                var planeGeo = new THREE.PlaneGeometry(800, 500, 1, 1);
                var planeMesh = new THREE.Mesh( planeGeo, whiteMaterial);
                planeMesh.position.z = -20;

                scene.add(planeMesh)

                

                var otherMaterial =
                  new THREE.MeshLambertMaterial(
                    {
                      color: 0x2583db
                    });

                  // set up the sphere vars
                var radius = 8,
                    segments = 8,
                    rings = 8;

                var geometry = new THREE.CubeGeometry( tileSize - 4, tileSize - 4, tileSize - 4 );
                var sphere = new THREE.SphereGeometry( tileSize - 10, 4, 4 );
                
                //LOOP TO CREATE ALL THE BLOCKS FROM THE DATA ARRAY PREVIOUSLY POPULATED
                for ( var i = 0 ; i < arrayLength ; i ++ ) {
                    var item = data[i];

                    // create the boxes material
                var material =
                  new THREE.MeshLambertMaterial(
                    {
                      //color: 0xEA80B0
                      color: 0x444444
                    });

                    if(item.alpha > 30) {
                        var mesh = new THREE.Mesh( geometry, material );
                    } else {
                        var mesh = new THREE.Mesh( sphere, otherMaterial );
                    }
                    //new mesh
                    

                    //position x
                    //mesh.position.x = item.column - 350;
                    mesh.position.x = 0;
                    //position y
                    //mesh.position.y = (20 - (item.row - 200));
                    mesh.position.y = 0;

                    //if it is not sea
                    /*if(item.alpha > 30) {
                        mesh.position.z = ((item.alpha ) * this.commonStuff.heightGain) * 10;
                        mesh.scale.set(1 ,1 , (item.alpha + .1) * this.commonStuff.heightGain);
                    }*/
                    
                    //had to add .1 because inverting a matrix3 thing with 0 was logging an error
                    

                    this.commonStuff.cubesArray.push({
                        mesh: mesh,
                        positionX : item.column - 370,
                        positionY :(20 - (item.row - 180)),
                        //animatable: item.alpha < 10,
                        alpha: 100/255 * item.alpha
                    })

                    // add the box to the scene
                    scene.add(mesh);

                }
                var m = 0;
                var loop = this.commonStuff.cubesArray.length;
                var scaleCancer = false;
                
                //Didn't use request animation, don't even know why..
                var t = window.setInterval( function() {
                    m++;
                    // animation loop!
                    for ( var i = 0 ; i < loop ; i++ ) {
                        var current = self.commonStuff.cubesArray[i];
                        if( m == 30 ) {
                            self.commonStuff.hReached = true;
                            self.commonStuff.vReached = false;
                        }

                        if( m == 80 ) {
                            self.commonStuff.vReached = true;
                        }

                        if (m == 150) {
                            scaleCancer = true;
                        }

                        if( !self.commonStuff.hReached ) {
                            current.mesh.position.x = current.mesh.position.x + ( current.positionX - current.mesh.position.x ) / 10;
                        }

                        if( !self.commonStuff.vReached ) {
                            current.mesh.position.y = current.mesh.position.y + ( current.positionY - current.mesh.position.y ) / 10;
                        }

                        if ( current.alpha < self.commonStuff.currentShow && current.alpha > self.commonStuff.minShow) {
                            //console.log('ye')
                            current.mesh.scale.z = current.mesh.scale.z + ( current.alpha / 10 - current.mesh.scale.z) / 20 ;
                            current.mesh.material.color.setHex(0xEA80B0);
                        } else {

                            if (current.alpha > 12 ) {
                                current.mesh.scale.z = current.mesh.scale.z + ( 1 - current.mesh.scale.z) / 20;
                                current.mesh.material.color.setHex(0x444444);
                            } else {
                                current.mesh.position.z = Math.sin( (m + current.mesh.position.x) / 20 ) * 8
                            }
                            
                        }


                        
                    }

                    if (self.commonStuff.mouseDown) {
                        camera.position.z -= 2;
                    }

                    camera.lookAt( scene.position );
                    renderer.render(scene, camera);
                }, 10);

                
                //LISTENERS
                var container = document.getElementById('container');

                //var zero = document.getElementById('zero');
                var one = document.getElementById('one');
                var two = document.getElementById('two');
                var three = document.getElementById('three');
                var four = document.getElementById('four');
                var five = document.getElementById('five');

                /*zero.onmousedown = function(e) {
                    self.commonStuff.minShow = 0;
                    self.commonStuff.currentShow = 10;
                    e.preventDefault();
                };*/

                one.onmousedown = function(e) {
                    self.commonStuff.minShow = 22;
                    self.commonStuff.currentShow = 44;
                    $('.button').removeClass('active');
                    $(this).addClass('active');
                    e.preventDefault();
                };

                two.onmousedown = function(e) {
                    self.commonStuff.minShow = 44;
                    self.commonStuff.currentShow = 57;
                    $('.button').removeClass('active');
                    $(this).addClass('active');
                    e.preventDefault();
                };

                three.onmousedown = function(e) {
                    self.commonStuff.minShow = 57;
                    self.commonStuff.currentShow = 74;
                    $('.button').removeClass('active');
                    $(this).addClass('active');
                    e.preventDefault();
                };

                four.onmousedown = function(e) {
                    self.commonStuff.minShow = 74;
                    self.commonStuff.currentShow = 92;
                    $('.button').removeClass('active');
                    $(this).addClass('active');
                    e.preventDefault();
                };

                five.onmousedown = function(e) {
                    self.commonStuff.minShow = 92;
                    self.commonStuff.currentShow = 150;
                    $('.button').removeClass('active');
                    $(this).addClass('active');
                    e.preventDefault();
                };

                container.onmousedown = function(e) {
                    //self.commonStuff.mouseDown = true;
                    self.commonStuff.currentShow = Math.random() * 100;
                }

                container.onmouseup = function(e) {
                    //self.commonStuff.mouseDown = false;
                }

                //Camera movement
                container.onmousemove = function(e) {
                    camera.position.x = (e.offsetX - 350) / 2;
                    camera.position.y = (e.offsetY - 250) / 2;
                }
                
            }

        }
        
        //DIDN'T USE INIT because I don't want to init a friggin' cancer :(
        BreastCancer.doNotInit();
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值