8.1 sec in total
989 ms
4.3 sec
2.9 sec
Visit blog.revolutionanalytics.com now to see the best up-to-date Blog Revolutionanalytics content for United States and also check out these interesting facts you probably never knew about blog.revolutionanalytics.com
Learn about using open source R for big data analysis, predictive modeling, data science
Visit blog.revolutionanalytics.comWe analyzed Blog.revolutionanalytics.com page load time and found that the first response time was 989 ms and then it took 7.2 sec to load all DOM resources and completely render a web page. This is a poor result, as 80% of websites can load faster.
blog.revolutionanalytics.com performance score
name
value
score
weighting
Value4.0 s
24/100
10%
Value4.0 s
49/100
25%
Value26.6 s
0/100
10%
Value1,800 ms
10/100
30%
Value0.003
100/100
15%
Value11.9 s
17/100
10%
989 ms
132 ms
23 ms
60 ms
62 ms
Our browser made a total of 63 requests to load all elements on the main page. We found that 5% of them (3 requests) were addressed to the original Blog.revolutionanalytics.com, 19% (12 requests) were made to Revolution-computing.typepad.com and 17% (11 requests) were made to Static.typepad.com. The less responsive or slowest element that took the longest time to load (1.4 sec) belongs to the original domain Blog.revolutionanalytics.com.
Page size can be reduced by 843.8 kB (32%)
2.6 MB
1.8 MB
In fact, the total size of Blog.revolutionanalytics.com main page is 2.6 MB. This result falls beyond the top 1M of websites and identifies a large and not optimized web page that may take ages to load. 65% of websites need less resources to load. Images take 1.9 MB which makes up the majority of the site volume.
Potential reduce by 99.6 kB
HTML content can be minified and compressed by a website’s server. The most efficient way is to compress content using GZIP which reduces data amount travelling through the network between server and browser. HTML code on this page is well minified. It is highly recommended that content of this web page should be compressed using GZIP, as it can save up to 99.6 kB or 82% of the original size.
Potential reduce by 318.1 kB
Image size optimization can help to speed up a website loading time. The chart above shows the difference between the size before and after optimization. Obviously, Blog Revolutionanalytics needs image optimization as it can save up to 318.1 kB or 17% of the original volume. The most popular and efficient tools for JPEG and PNG image optimization are Jpegoptim and PNG Crush.
Potential reduce by 158.7 kB
It’s better to minify JavaScript in order to improve website performance. The diagram shows the current total size of all JavaScript files against the prospective JavaScript size after its minification and compression. It is highly recommended that all JavaScript files should be compressed and minified as it can save up to 158.7 kB or 58% of the original size.
Potential reduce by 267.5 kB
CSS files minification is very important to reduce a web page rendering time. The faster CSS files can load, the earlier a page can be rendered. Blog.revolutionanalytics.com needs all CSS files to be minified and compressed as it can save up to 267.5 kB or 83% of the original size.
Number of requests can be reduced by 27 (52%)
52
25
The browser has sent 52 CSS, Javascripts, AJAX and image requests in order to completely render the main page of Blog Revolutionanalytics. We recommend that multiple CSS and JavaScript files should be merged into one by each type, as it can help reduce assets requests from 11 to 1 for JavaScripts and from 11 to 1 for CSS and as a result speed up the page load time.
blog.revolutionanalytics.com
989 ms
styles.css
132 ms
thumbnail-gallery-min.js
23 ms
recommend.js
60 ms
f9665e15b74b393bd2d6.js
62 ms
print.css
207 ms
munchkin.js
17 ms
quant.js
6 ms
base-weblog.css
235 ms
bxslider.css
159 ms
widgets.css
171 ms
recentpostsfancy.css
162 ms
tipjar.css
155 ms
theme-asterisk_white_wide.css
258 ms
theme-asterisk.css
166 ms
ga.js
58 ms
blog-header.jpg
23 ms
6a010534b1db25970b01b7c81c073f970b-800wi
198 ms
6a0105360ba1c6970c01b8d1a1f814970c-450wi
368 ms
t_small-b.png
205 ms
gprofile_button-16.png
22 ms
blogtrottr-button-91x17px.gif
551 ms
6a010534b1db25970b01bb08c0b6cd970d-800wi
621 ms
6a010534b1db25970b01b8d1a60b99970c-800wi
608 ms
6a010534b1db25970b01bb08c0c22b970d-800wi
553 ms
6a010534b1db25970b01b7c81c055b970b-800wi
550 ms
6a010534b1db25970b01b8d1a4d670970c-800wi
553 ms
6a010534b1db25970b01b8d1a3a567970c-800wi
234 ms
6a010534b1db25970b01bb08be59bf970d-800wi
551 ms
6a010534b1db25970b01b8d1a40341970c-800wi
1086 ms
6a010534b1db25970b01bb08bea7f5970d-800wi
701 ms
6a010534b1db25970b01b8d19eecfc970c-800wi
699 ms
6a010534b1db25970b01bb08bdb2c9970d-800wi
629 ms
body-bg.png
17 ms
body-bg-wt.png
18 ms
-8mzWkuOxz8
185 ms
GTrVjWj5y7Q
200 ms
embed-7346e9a735aaeeacede51d1efaad6208f95d9edbc222155878bfddc2da28fa87.css
109 ms
bivn_threejs_dense.html
1420 ms
VipqzK6_LBU
161 ms
__utm.gif
16 ms
munchkin.js
49 ms
__utm.gif
44 ms
stats
82 ms
__utm.gif
31 ms
pixel;r=434072528;a=p-fcYWUmj5YbYKM;tags=typepad.extended;fpan=1;fpa=P0-1594868788-1456855374630;ns=0;ce=1;cm=;je=0;sr=1024x768x32;enc=n;dst=0;et=1456855374629;tzo=-180;ref=;url=http%3A%2F%2Fblog.revolutionanalytics.com%2F;ogl=
34 ms
beacon.js
29 ms
www-embed-player-vflQrT_sR.css
83 ms
www-embed-player.js
111 ms
collect
73 ms
visitWebPage
175 ms
IHNpemVPYmouZ2V0V2lkdGgoKSA6IGVsLm9mZnNldFdpZHRoLAogICAgICAgICAgICAgIGg6IHNpemVPYmogPyBzaXplT2JqLmdldEhlaWdodCgpIDogZWwub2Zmc2V0SGVpZ2h0CiAgICAgICAgICAgIH07CiAgICAgICAgICAgIGlmIChzaXplLncgPT09IDAgJiYgc2l6ZS5oID09PSAwKQogICAgICAgICAgICAgIHJldHVybjsKICAgICAgICAgICAgaWYgKHNpemUudyA9PT0gbGFzdFNpemUudyAmJiBzaXplLmggPT09IGxhc3RTaXplLmgpCiAgICAgICAgICAgICAgcmV0dXJuOwogICAgICAgICAgICBsYXN0U2l6ZSA9IHNpemU7CiAgICAgICAgICAgIGJpbmRpbmcucmVzaXplKGVsLCBzaXplLncsIHNpemUuaCwgaW5pdFJlc3VsdCk7CiAgICAgICAgICB9KTsKICAgICAgICB9CgogICAgICAgIHZhciBzY3JpcHREYXRhID0gZG9jdW1lbnQucXVlcnlTZWxlY3Rvcigic2NyaXB0W2RhdGEtZm9yPSciICsgZWwuaWQgKyAiJ11bdHlwZT0nYXBwbGljYXRpb24vanNvbiddIik7CiAgICAgICAgaWYgKHNjcmlwdERhdGEpIHsKICAgICAgICAgIHZhciBkYXRhID0gSlNPTi5wYXJzZShzY3JpcHREYXRhLnRleHRDb250ZW50IHx8IHNjcmlwdERhdGEudGV4dCk7CiAgICAgICAgICAvLyBSZXNvbHZlIHN0cmluZ3MgbWFya2VkIGFzIGphdmFzY3JpcHQgbGl0ZXJhbHMgdG8gb2JqZWN0cwogICAgICAgICAgZm9yICh2YXIgaSA9IDA7IGRhdGEuZXZhbHMgJiYgaSA8IGRhdGEuZXZhbHMubGVuZ3RoOyBpKyspIHsKICAgICAgICAgICAgd2luZG93LkhUTUxXaWRnZXRzLmV2YWx1YXRlU3RyaW5nTWVtYmVyKGRhdGEueCwgZGF0YS5ldmFsc1tpXSk7CiAgICAgICAgICB9CiAgICAgICAgICBiaW5kaW5nLnJlbmRlclZhbHVlKGVsLCBkYXRhLngsIGluaXRSZXN1bHQpOwogICAgICAgIH0KICAgICAgfQogICAgfQogIH0KCiAgLy8gV2FpdCB1bnRpbCBhZnRlciB0aGUgZG9jdW1lbnQgaGFzIGxvYWRlZCB0byByZW5kZXIgdGhlIHdpZGdldHMuCiAgaWYgKGRvY3VtZW50LmFkZEV2ZW50TGlzdGVuZXIpIHsKICAgIGRvY3VtZW50LmFkZEV2ZW50TGlzdGVuZXIoIkRPTUNvbnRlbnRMb2FkZWQiLCBmdW5jdGlvbigpIHsKICAgICAgZG9jdW1lbnQucmVtb3ZlRXZlbnRMaXN0ZW5lcigiRE9NQ29udGVudExvYWRlZCIsIGFyZ3VtZW50cy5jYWxsZWUsIGZhbHNlKTsKICAgICAgc3RhdGljUmVuZGVyKCk7CiAgICB9LCBmYWxzZSk7CiAgfSBlbHNlIGlmIChkb2N1bWVudC5hdHRhY2hFdmVudCkgewogICAgZG9jdW1lbnQuYXR0YWNoRXZlbnQoIm9ucmVhZHlzdGF0ZWNoYW5nZSIsIGZ1bmN0aW9uKCkgewogICAgICBpZiAoZG9jdW1lbnQucmVhZHlTdGF0ZSA9PT0gImNvbXBsZXRlIikgewogICAgICAgIGRvY3VtZW50LmRldGFjaEV2ZW50KCJvbnJlYWR5c3RhdGVjaGFuZ2UiLCBhcmd1bWVudHMuY2FsbGVlKTsKICAgICAgICBzdGF0aWNSZW5kZXIoKTsKICAgICAgfQogICAgfSk7CiAgfQoKCiAgd2luZG93LkhUTUxXaWRnZXRzLmdldEF0dGFjaG1lbnRVcmwgPSBmdW5jdGlvbihkZXBuYW1lLCBrZXkpIHsKICAgIC8vIElmIG5vIGtleSwgZGVmYXVsdCB0byB0aGUgZmlyc3QgaXRlbQogICAgaWYgKHR5cGVvZihrZXkpID09PSAidW5kZWZpbmVkIikKICAgICAga2V5ID0gMTsKCiAgICB2YXIgbGluayA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKGRlcG5hbWUgKyAiLSIgKyBrZXkgKyAiLWF0dGFjaG1lbnQiKTsKICAgIGlmICghbGluaykgewogICAgICB0aHJvdyBuZXcgRXJyb3IoIkF0dGFjaG1lbnQgIiArIGRlcG5hbWUgKyAiLyIgKyBrZXkgKyAiIG5vdCBmb3VuZCBpbiBkb2N1bWVudCIpOwogICAgfQogICAgcmV0dXJuIGxpbmsuZ2V0QXR0cmlidXRlKCJocmVmIik7CiAgfTsKCiAgd2luZG93LkhUTUxXaWRnZXRzLmRhdGFmcmFtZVRvRDMgPSBmdW5jdGlvbihkZikgewogICAgdmFyIG5hbWVzID0gW107CiAgICB2YXIgbGVuZ3RoOwogICAgZm9yICh2YXIgbmFtZSBpbiBkZikgewogICAgICAgIGlmIChkZi5oYXNPd25Qcm9wZXJ0eShuYW1lKSkKICAgICAgICAgICAgbmFtZXMucHVzaChuYW1lKTsKICAgICAgICBpZiAodHlwZW9mKGRmW25hbWVdKSAhPT0gIm9iamVjdCIgfHwgdHlwZW9mKGRmW25hbWVdLmxlbmd0aCkgPT09ICJ1bmRlZmluZWQiKSB7CiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcigiQWxsIGZpZWxkcyBtdXN0IGJlIGFycmF5cyIpOwogICAgICAgIH0gZWxzZSBpZiAodHlwZW9mKGxlbmd0aCkgIT09ICJ1bmRlZmluZWQiICYmIGxlbmd0aCAhPT0gZGZbbmFtZV0ubGVuZ3RoKSB7CiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcigiQWxsIGZpZWxkcyBtdXN0IGJlIGFycmF5cyBvZiB0aGUgc2FtZSBsZW5ndGgiKTsKICAgICAgICB9CiAgICAgICAgbGVuZ3RoID0gZGZbbmFtZV0ubGVuZ3RoOwogICAgfQogICAgdmFyIHJlc3VsdHMgPSBbXTsKICAgIHZhciBpdGVtOwogICAgZm9yICh2YXIgcm93ID0gMDsgcm93IDwgbGVuZ3RoOyByb3crKykgewogICAgICAgIGl0ZW0gPSB7fTsKICAgICAgICBmb3IgKHZhciBjb2wgPSAwOyBjb2wgPCBuYW1lcy5sZW5ndGg7IGNvbCsrKSB7CiAgICAgICAgICAgIGl0ZW1bbmFtZXNbY29sXV0gPSBkZltuYW1lc1tjb2xdXVtyb3ddOwogICAgICAgIH0KICAgICAgICByZXN1bHRzLnB1c2goaXRlbSk7CiAgICB9CiAgICByZXR1cm4gcmVzdWx0czsKICB9OwoKICB3aW5kb3cuSFRNTFdpZGdldHMudHJhbnNwb3NlQXJyYXkyRCA9IGZ1bmN0aW9uKGFycmF5KSB7CiAgICAgIHZhciBuZXdBcnJheSA9IGFycmF5WzBdLm1hcChmdW5jdGlvbihjb2wsIGkpIHsKICAgICAgICAgIHJldHVybiBhcnJheS5tYXAoZnVuY3Rpb24ocm93KSB7CiAgICAgICAgICAgICAgcmV0dXJuIHJvd1tpXQogICAgICAgICAgfSkKICAgICAgfSk7CiAgICAgIHJldHVybiBuZXdBcnJheTsKICB9OwogIC8vIFNwbGl0IHZhbHVlIGF0IHNwbGl0Q2hhciwgYnV0IGFsbG93IHNwbGl0Q2hhciB0byBiZSBlc2NhcGVkCiAgLy8gdXNpbmcgZXNjYXBlQ2hhci4gQW55IG90aGVyIGNoYXJhY3RlcnMgZXNjYXBlZCBieSBlc2NhcGVDaGFyCiAgLy8gd2lsbCBiZSBpbmNsdWRlZCBhcyB1c3VhbCAoaW5jbHVkaW5nIGVzY2FwZUNoYXIgaXRzZWxmKS4KICBmdW5jdGlvbiBzcGxpdFdpdGhFc2NhcGUodmFsdWUsIHNwbGl0Q2hhciwgZXNjYXBlQ2hhcikgewogICAgdmFyIHJlc3VsdHMgPSBbXTsKICAgIHZhciBlc2NhcGVNb2RlID0gZmFsc2U7CiAgICB2YXIgY3VycmVudFJlc3VsdCA9ICIiOwogICAgZm9yICh2YXIgcG9zID0gMDsgcG9zIDwgdmFsdWUubGVuZ3RoOyBwb3MrKykgewogICAgICBpZiAoIWVzY2FwZU1vZGUpIHsKICAgICAgICBpZiAodmFsdWVbcG9zXSA9PT0gc3BsaXRDaGFyKSB7CiAgICAgICAgICByZXN1bHRzLnB1c2goY3VycmVudFJlc3VsdCk7CiAgICAgICAgICBjdXJyZW50UmVzdWx0ID0gIiI7CiAgICAgICAgfSBlbHNlIGlmICh2YWx1ZVtwb3NdID09PSBlc2NhcGVDaGFyKSB7CiAgICAgICAgICBlc2NhcGVNb2RlID0gdHJ1ZTsKICAgICAgICB9IGVsc2UgewogICAgICAgICAgY3VycmVudFJlc3VsdCArPSB2YWx1ZVtwb3NdOwogICAgICAgIH0KICAgICAgfSBlbHNlIHsKICAgICAgICBjdXJyZW50UmVzdWx0ICs9IHZhbHVlW3Bvc107CiAgICAgICAgZXNjYXBlTW9kZSA9IGZhbHNlOwogICAgICB9CiAgICB9CiAgICBpZiAoY3VycmVudFJlc3VsdCAhPT0gIiIpIHsKICAgICAgcmVzdWx0cy5wdXNoKGN1cnJlbnRSZXN1bHQpOwogICAgfQogICAgcmV0dXJuIHJlc3VsdHM7CiAgfQogIC8vIEZ1bmN0aW9uIGF1dGhvcmVkIGJ5IFlpaHVpL0pKIEFsbGFpcmUKICB3aW5kb3cuSFRNTFdpZGdldHMuZXZhbHVhdGVTdHJpbmdNZW1iZXIgPSBmdW5jdGlvbihvLCBtZW1iZXIpIHsKICAgIHZhciBwYXJ0cyA9IHNwbGl0V2l0aEVzY2FwZShtZW1iZXIsICcuJywgJ1xcJyk7CiAgICBmb3IgKHZhciBpID0gMCwgbCA9IHBhcnRzLmxlbmd0aDsgaSA8IGw7IGkrKykgewogICAgICB2YXIgcGFydCA9IHBhcnRzW2ldOwogICAgICAvLyBwYXJ0IG1heSBiZSBhIGNoYXJhY3RlciBvciAnbnVtZXJpYycgbWVtYmVyIG5hbWUKICAgICAgaWYgKG8gIT09IG51bGwgJiYgdHlwZW9mIG8gPT09ICJvYmplY3QiICYmIHBhcnQgaW4gbykgewogICAgICAgIGlmIChpID09IChsIC0gMSkpIHsgLy8gaWYgd2UgYXJlIGF0IHRoZSBlbmQgb2YgdGhlIGxpbmUgdGhlbiBldmFsdWxhdGUKICAgICAgICAgIGlmICh0eXBlb2Ygb1twYXJ0XSA9PT0gInN0cmluZyIpCiAgICAgICAgICAgIG9bcGFydF0gPSBldmFsKCIoIiArIG9bcGFydF0gKyAiKSIpOwogICAgICAgIH0gZWxzZSB7IC8vIG90aGVyd2lzZSBjb250aW51ZSB0byBuZXh0IGVtYmVkZGVkIG9iamVjdAogICAgICAgICAgbyA9IG9bcGFydF07CiAgICAgICAgfQogICAgICB9CiAgICB9CiAgfTsKfSkoKTsKCg==
9 ms
www.revolutionanalytics.com
464 ms
yrK4EqpSRdoujKQmsJoZ5WaLgeULvcWnibQMQSxOOnM.js
69 ms
ad_status.js
219 ms
zN7GBFwfMP4uA6AR0HCoLQ.ttf
204 ms
RxZJdnzeo3R5zSexge8UUaCWcynf_cDxXwCLxiixG1c.ttf
214 ms
ZGVmaW5lKHRhKToib2JqZWN0Ij09dHlwZW9mIG1vZHVsZSYmbW9kdWxlLmV4cG9ydHMmJihtb2R1bGUuZXhwb3J0cz10YSksdGhpcy5kMz10YX0oKTs=
64 ms
KGIudGltZT0wLGIuYWN0aXZlPSEwKTpjb25zb2xlLndhcm4oImFuaW1hdGlvblsiK2ErIl0gdW5kZWZpbmVkIil9O1RIUkVFLk1vcnBoQmxlbmRNZXNoLnByb3RvdHlwZS5zdG9wQW5pbWF0aW9uPWZ1bmN0aW9uKGEpe2lmKGE9dGhpcy5hbmltYXRpb25zTWFwW2FdKWEuYWN0aXZlPSExfTsKVEhSRUUuTW9ycGhCbGVuZE1lc2gucHJvdG90eXBlLnVwZGF0ZT1mdW5jdGlvbihhKXtmb3IodmFyIGI9MCxjPXRoaXMuYW5pbWF0aW9uc0xpc3QubGVuZ3RoO2I8YztiKyspe3ZhciBkPXRoaXMuYW5pbWF0aW9uc0xpc3RbYl07aWYoZC5hY3RpdmUpe3ZhciBlPWQuZHVyYXRpb24vZC5sZW5ndGg7ZC50aW1lKz1kLmRpcmVjdGlvbiphO2lmKGQubWlycm9yZWRMb29wKXtpZihkLnRpbWU+ZC5kdXJhdGlvbnx8MD5kLnRpbWUpZC5kaXJlY3Rpb24qPS0xLGQudGltZT5kLmR1cmF0aW9uJiYoZC50aW1lPWQuZHVyYXRpb24sZC5kaXJlY3Rpb25CYWNrd2FyZHM9ITApLDA+ZC50aW1lJiYoZC50aW1lPTAsZC5kaXJlY3Rpb25CYWNrd2FyZHM9ITEpfWVsc2UgZC50aW1lJT1kLmR1cmF0aW9uLDA+ZC50aW1lJiYoZC50aW1lKz1kLmR1cmF0aW9uKTt2YXIgZj1kLnN0YXJ0RnJhbWUrVEhSRUUuTWF0aC5jbGFtcChNYXRoLmZsb29yKGQudGltZS9lKSwwLGQubGVuZ3RoLTEpLGc9ZC53ZWlnaHQ7CmYhPT1kLmN1cnJlbnRGcmFtZSYmKHRoaXMubW9ycGhUYXJnZXRJbmZsdWVuY2VzW2QubGFzdEZyYW1lXT0wLHRoaXMubW9ycGhUYXJnZXRJbmZsdWVuY2VzW2QuY3VycmVudEZyYW1lXT0xKmcsdGhpcy5tb3JwaFRhcmdldEluZmx1ZW5jZXNbZl09MCxkLmxhc3RGcmFtZT1kLmN1cnJlbnRGcmFtZSxkLmN1cnJlbnRGcmFtZT1mKTtlPWQudGltZSVlL2U7ZC5kaXJlY3Rpb25CYWNrd2FyZHMmJihlPTEtZSk7dGhpcy5tb3JwaFRhcmdldEluZmx1ZW5jZXNbZC5jdXJyZW50RnJhbWVdPWUqZzt0aGlzLm1vcnBoVGFyZ2V0SW5mbHVlbmNlc1tkLmxhc3RGcmFtZV09KDEtZSkqZ319fTsK
26 ms
x-javascript;base64,LyoqCiAqIEBhdXRob3IgYWx0ZXJlZHEgLyBodHRwOi8vYWx0ZXJlZHF1YWxpYS5jb20vCiAqIEBhdXRob3IgbXIuZG9vYiAvIGh0dHA6Ly9tcmRvb2IuY29tLwogKi8KCnZhciBEZXRlY3RvciA9IHsKCgljYW52YXM6ICEhIHdpbmRvdy5DYW52YXNSZW5kZXJpbmdDb250ZXh0MkQsCgl3ZWJnbDogKCBmdW5jdGlvbiAoKSB7IHRyeSB7IHZhciBjYW52YXMgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCAnY2FudmFzJyApOyByZXR1cm4gISEgKCB3aW5kb3cuV2ViR0xSZW5kZXJpbmdDb250ZXh0ICYmICggY2FudmFzLmdldENvbnRleHQoICd3ZWJnbCcgKSB8fCBjYW52YXMuZ2V0Q29udGV4dCggJ2V4cGVyaW1lbnRhbC13ZWJnbCcgKSApICk7IH0gY2F0Y2goIGUgKSB7IHJldHVybiBmYWxzZTsgfSB9ICkoKSwKCXdvcmtlcnM6ICEhIHdpbmRvdy5Xb3JrZXIsCglmaWxlYXBpOiB3aW5kb3cuRmlsZSAmJiB3aW5kb3cuRmlsZVJlYWRlciAmJiB3aW5kb3cuRmlsZUxpc3QgJiYgd2luZG93LkJsb2IsCgoJZ2V0V2ViR0xFcnJvck1lc3NhZ2U6IGZ1bmN0aW9uICgpIHsKCgkJdmFyIGVsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCAnZGl2JyApOwoJCWVsZW1lbnQuaWQgPSAnd2ViZ2wtZXJyb3ItbWVzc2FnZSc7CgkJZWxlbWVudC5zdHlsZS5mb250RmFtaWx5ID0gJ21vbm9zcGFjZSc7CgkJZWxlbWVudC5zdHlsZS5mb250U2l6ZSA9ICcxM3B4JzsKCQllbGVtZW50LnN0eWxlLmZvbnRXZWlnaHQgPSAnbm9ybWFsJzsKCQllbGVtZW50LnN0eWxlLnRleHRBbGlnbiA9ICdjZW50ZXInOwoJCWVsZW1lbnQuc3R5bGUuYmFja2dyb3VuZCA9ICcjZmZmJzsKCQllbGVtZW50LnN0eWxlLmNvbG9yID0gJyMwMDAnOwoJCWVsZW1lbnQuc3R5bGUucGFkZGluZyA9ICcxLjVlbSc7CgkJZWxlbWVudC5zdHlsZS53aWR0aCA9ICc0MDBweCc7CgkJZWxlbWVudC5zdHlsZS5tYXJnaW4gPSAnNWVtIGF1dG8gMCc7CgoJCWlmICggISB0aGlzLndlYmdsICkgewoKCQkJZWxlbWVudC5pbm5lckhUTUwgPSB3aW5kb3cuV2ViR0xSZW5kZXJpbmdDb250ZXh0ID8gWwoJCQkJJ1lvdXIgZ3JhcGhpY3MgY2FyZCBkb2VzIG5vdCBzZWVtIHRvIHN1cHBvcnQgPGEgaHJlZj0iaHR0cDovL2tocm9ub3Mub3JnL3dlYmdsL3dpa2kvR2V0dGluZ19hX1dlYkdMX0ltcGxlbWVudGF0aW9uIiBzdHlsZT0iY29sb3I6IzAwMCI+V2ViR0w8L2E+LjxiciAvPicsCgkJCQknRmluZCBvdXQgaG93IHRvIGdldCBpdCA8YSBocmVmPSJodHRwOi8vZ2V0LndlYmdsLm9yZy8iIHN0eWxlPSJjb2xvcjojMDAwIj5oZXJlPC9hPi4nCgkJCV0uam9pbiggJ1xuJyApIDogWwoJCQkJJ1lvdXIgYnJvd3NlciBkb2VzIG5vdCBzZWVtIHRvIHN1cHBvcnQgPGEgaHJlZj0iaHR0cDovL2tocm9ub3Mub3JnL3dlYmdsL3dpa2kvR2V0dGluZ19hX1dlYkdMX0ltcGxlbWVudGF0aW9uIiBzdHlsZT0iY29sb3I6IzAwMCI+V2ViR0w8L2E+Ljxici8+JywKCQkJCSdGaW5kIG91dCBob3cgdG8gZ2V0IGl0IDxhIGhyZWY9Imh0dHA6Ly9nZXQud2ViZ2wub3JnLyIgc3R5bGU9ImNvbG9yOiMwMDAiPmhlcmU8L2E+LicKCQkJXS5qb2luKCAnXG4nICk7CgoJCX0KCgkJcmV0dXJuIGVsZW1lbnQ7CgoJfSwKCglhZGRHZXRXZWJHTE1lc3NhZ2U6IGZ1bmN0aW9uICggcGFyYW1ldGVycyApIHsKCgkJdmFyIHBhcmVudCwgaWQsIGVsZW1lbnQ7CgoJCXBhcmFtZXRlcnMgPSBwYXJhbWV0ZXJzIHx8IHt9OwoKCQlwYXJlbnQgPSBwYXJhbWV0ZXJzLnBhcmVudCAhPT0gdW5kZWZpbmVkID8gcGFyYW1ldGVycy5wYXJlbnQgOiBkb2N1bWVudC5ib2R5OwoJCWlkID0gcGFyYW1ldGVycy5pZCAhPT0gdW5kZWZpbmVkID8gcGFyYW1ldGVycy5pZCA6ICdvbGRpZSc7CgoJCWVsZW1lbnQgPSBEZXRlY3Rvci5nZXRXZWJHTEVycm9yTWVzc2FnZSgpOwoJCWVsZW1lbnQuaWQgPSBpZDsKCgkJcGFyZW50LmFwcGVuZENoaWxkKCBlbGVtZW50ICk7CgoJfQoKfTsKCi8vIGJyb3dzZXJpZnkgc3VwcG9ydAppZiAoIHR5cGVvZiBtb2R1bGUgPT09ICdvYmplY3QnICkgewoKCW1vZHVsZS5leHBvcnRzID0gRGV0ZWN0b3I7Cgp9Cg==
84 ms
x-javascript;base64,/**
 * @author mrdoob / http://mrdoob.com/
 * @author supereggbert / http://www.paulbrunt.co.uk/
 * @author julianwa / https://github.com/julianwa
 */

THREE.RenderableObject = function () {

	this.id = 0;

	this.object = null;
	this.z = 0;

};

//

THREE.RenderableFace = function () {

	this.id = 0;

	this.v1 = new THREE.RenderableVertex();
	this.v2 = new THREE.RenderableVertex();
	this.v3 = new THREE.RenderableVertex();

	this.normalModel = new THREE.Vector3();

	this.vertexNormalsModel = [ new THREE.Vector3(), new THREE.Vector3(), new THREE.Vector3() ];
	this.vertexNormalsLength = 0;

	this.color = new THREE.Color();
	this.material = null;
	this.uvs = [ new THREE.Vector2(), new THREE.Vector2(), new THREE.Vector2() ];

	this.z = 0;

};

//

THREE.RenderableVertex = function () {

	this.position = new THREE.Vector3();
	this.positionWorld = new THREE.Vector3();
	this.positionScreen = new THREE.Vector4();

	this.visible = true;

};

THREE.RenderableVertex.prototype.copy = function ( vertex ) {

	this.positionWorld.copy( vertex.positionWorld );
	this.positionScreen.copy( vertex.positionScreen );

};

//

THREE.RenderableLine = function () {

	this.id = 0;

	this.v1 = new THREE.RenderableVertex();
	this.v2 = new THREE.RenderableVertex();

	this.vertexColors = [ new THREE.Color(), new THREE.Color() ];
	this.material = null;

	this.z = 0;

};

//

THREE.RenderableSprite = function () {

	this.id = 0;

	this.object = null;

	this.x = 0;
	this.y = 0;
	this.z = 0;

	this.rotation = 0;
	this.scale = new THREE.Vector2();

	this.material = null;

};

//

THREE.Projector = function () {

	var _object, _objectCount, _objectPool = [], _objectPoolLength = 0,
	_vertex, _vertexCount, _vertexPool = [], _vertexPoolLength = 0,
	_face, _faceCount, _facePool = [], _facePoolLength = 0,
	_line, _lineCount, _linePool = [], _linePoolLength = 0,
	_sprite, _spriteCount, _spritePool = [], _spritePoolLength = 0,

	_renderData = { objects: [], lights: [], elements: [] },

	_vA = new THREE.Vector3(),
	_vB = new THREE.Vector3(),
	_vC = new THREE.Vector3(),

	_vector3 = new THREE.Vector3(),
	_vector4 = new THREE.Vector4(),

	_clipBox = new THREE.Box3( new THREE.Vector3( - 1, - 1, - 1 ), new THREE.Vector3( 1, 1, 1 ) ),
	_boundingBox = new THREE.Box3(),
	_points3 = new Array( 3 ),
	_points4 = new Array( 4 ),

	_viewMatrix = new THREE.Matrix4(),
	_viewProjectionMatrix = new THREE.Matrix4(),

	_modelMatrix,
	_modelViewProjectionMatrix = new THREE.Matrix4(),

	_normalMatrix = new THREE.Matrix3(),

	_frustum = new THREE.Frustum(),

	_clippedVertex1PositionScreen = new THREE.Vector4(),
	_clippedVertex2PositionScreen = new THREE.Vector4();

	//

	this.projectVector = function ( vector, camera ) {

		console.warn( 'THREE.Projector: .projectVector() is now vector.project().' );
		vector.project( camera );

	};

	this.unprojectVector = function ( vector, camera ) {

		console.warn( 'THREE.Projector: .unprojectVector() is now vector.unproject().' );
		vector.unproject( camera );

	};

	this.pickingRay = function ( vector, camera ) {

		console.error( 'THREE.Projector: .pickingRay() is now raycaster.setFromCamera().' );

	};

	//

	var RenderList = function () {

		var normals = [];
		var uvs = [];

		var object = null;
		var material = null;

		var normalMatrix = new THREE.Matrix3();

		var setObject = function ( value ) {

			object = value;
			material = object.material;

			normalMatrix.getNormalMatrix( object.matrixWorld );

			normals.length = 0;
			uvs.length = 0;

		};

		var projectVertex = function ( vertex ) {

			var position = vertex.position;
			var positionWorld = vertex.positionWorld;
			var positionScreen = vertex.positionScreen;

			positionWorld.copy( position ).applyMatrix4( _modelMatrix );
			positionScreen.copy( positionWorld ).applyMatrix4( _viewProjectionMatrix );

			var invW = 1 / positionScreen.w;

			positionScreen.x *= invW;
			positionScreen.y *= invW;
			positionScreen.z *= invW;

			vertex.visible = positionScreen.x >= - 1 && positionScreen.x <= 1 &&
					 positionScreen.y >= - 1 && positionScreen.y <= 1 &&
					 positionScreen.z >= - 1 && positionScreen.z <= 1;

		};

		var pushVertex = function ( x, y, z ) {

			_vertex = getNextVertexInPool();
			_vertex.position.set( x, y, z );

			projectVertex( _vertex );

		};

		var pushNormal = function ( x, y, z ) {

			normals.push( x, y, z );

		};

		var pushUv = function ( x, y ) {

			uvs.push( x, y );

		};

		var checkTriangleVisibility = function ( v1, v2, v3 ) {

			if ( v1.visible === true || v2.visible === true || v3.visible === true ) return true;

			_points3[ 0 ] = v1.positionScreen;
			_points3[ 1 ] = v2.positionScreen;
			_points3[ 2 ] = v3.positionScreen;

			return _clipBox.isIntersectionBox( _boundingBox.setFromPoints( _points3 ) );

		};

		var checkBackfaceCulling = function ( v1, v2, v3 ) {

			return ( ( v3.positionScreen.x - v1.positionScreen.x ) *
				    ( v2.positionScreen.y - v1.positionScreen.y ) -
				    ( v3.positionScreen.y - v1.positionScreen.y ) *
				    ( v2.positionScreen.x - v1.positionScreen.x ) ) < 0;

		};

		var pushLine = function ( a, b ) {

			var v1 = _vertexPool[ a ];
			var v2 = _vertexPool[ b ];

			_line = getNextLineInPool();

			_line.id = object.id;
			_line.v1.copy( v1 );
			_line.v2.copy( v2 );
			_line.z = ( v1.positionScreen.z + v2.positionScreen.z ) / 2;

			_line.material = object.material;

			_renderData.elements.push( _line );

		};

		var pushTriangle = function ( a, b, c ) {

			var v1 = _vertexPool[ a ];
			var v2 = _vertexPool[ b ];
			var v3 = _vertexPool[ c ];

			if ( checkTriangleVisibility( v1, v2, v3 ) === false ) return;

			if ( material.side === THREE.DoubleSide || checkBackfaceCulling( v1, v2, v3 ) === true ) {

				_face = getNextFaceInPool();

				_face.id = object.id;
				_face.v1.copy( v1 );
				_face.v2.copy( v2 );
				_face.v3.copy( v3 );
				_face.z = ( v1.positionScreen.z + v2.positionScreen.z + v3.positionScreen.z ) / 3;

				for ( var i = 0; i < 3; i ++ ) {

					var offset = arguments[ i ] * 3;
					var normal = _face.vertexNormalsModel[ i ];

					normal.set( normals[ offset ], normals[ offset + 1 ], normals[ offset + 2 ] );
					normal.applyMatrix3( normalMatrix ).normalize();

					var offset2 = arguments[ i ] * 2;

					var uv = _face.uvs[ i ];
					uv.set( uvs[ offset2 ], uvs[ offset2 + 1 ] );

				}

				_face.vertexNormalsLength = 3;

				_face.material = object.material;

				_renderData.elements.push( _face );

			}

		};

		return {
			setObject: setObject,
			projectVertex: projectVertex,
			checkTriangleVisibility: checkTriangleVisibility,
			checkBackfaceCulling: checkBackfaceCulling,
			pushVertex: pushVertex,
			pushNormal: pushNormal,
			pushUv: pushUv,
			pushLine: pushLine,
			pushTriangle: pushTriangle
		}

	};

	var renderList = new RenderList();

	this.projectScene = function ( scene, camera, sortObjects, sortElements ) {

		_faceCount = 0;
		_lineCount = 0;
		_spriteCount = 0;

		_renderData.elements.length = 0;

		if ( scene.autoUpdate === true ) scene.updateMatrixWorld();
		if ( camera.parent === undefined ) camera.updateMatrixWorld();

		_viewMatrix.copy( camera.matrixWorldInverse.getInverse( camera.matrixWorld ) );
		_viewProjectionMatrix.multiplyMatrices( camera.projectionMatrix, _viewMatrix );

		_frustum.setFromMatrix( _viewProjectionMatrix );

		//

		_objectCount = 0;

		_renderData.objects.length = 0;
		_renderData.lights.length = 0;

		scene.traverseVisible( function ( object ) {

			if ( object instanceof THREE.Light ) {

				_renderData.lights.push( object );

			} else if ( object instanceof THREE.Mesh || object instanceof THREE.Line || object instanceof THREE.Sprite ) {

				if ( object.material.visible === false ) return;

				if ( object.frustumCulled === false || _frustum.intersectsObject( object ) === true ) {

					_object = getNextObjectInPool();
					_object.id = object.id;
					_object.object = object;

					_vector3.setFromMatrixPosition( object.matrixWorld );
					_vector3.applyProjection( _viewProjectionMatrix );
					_object.z = _vector3.z;

					_renderData.objects.push( _object );

				}

			}

		} );

		if ( sortObjects === true ) {

			_renderData.objects.sort( painterSort );

		}

		//

		for ( var o = 0, ol = _renderData.objects.length; o < ol; o ++ ) {

			var object = _renderData.objects[ o ].object;
			var geometry = object.geometry;

			renderList.setObject( object );

			_modelMatrix = object.matrixWorld;

			_vertexCount = 0;

			if ( object instanceof THREE.Mesh ) {

				if ( geometry instanceof THREE.BufferGeometry ) {

					var attributes = geometry.attributes;
					var offsets = geometry.offsets;

					if ( attributes.position === undefined ) continue;

					var positions = attributes.position.array;

					for ( var i = 0, l = positions.length; i < l; i += 3 ) {

						renderList.pushVertex( positions[ i ], positions[ i + 1 ], positions[ i + 2 ] );

					}

					if ( attributes.normal !== undefined ) {

						var normals = attributes.normal.array;

						for ( var i = 0, l = normals.length; i < l; i += 3 ) {

							renderList.pushNormal( normals[ i ], normals[ i + 1 ], normals[ i + 2 ] );

						}

					}

					if ( attributes.uv !== undefined ) {

						var uvs = attributes.uv.array;

						for ( var i = 0, l = uvs.length; i < l; i += 2 ) {

							renderList.pushUv( uvs[ i ], uvs[ i + 1 ] );

						}

					}

					if ( attributes.index !== undefined ) {

						var indices = attributes.index.array;

						if ( offsets.length > 0 ) {

							for ( var o = 0; o < offsets.length; o ++ ) {

								var offset = offsets[ o ];
								var index = offset.index;

								for ( var i = offset.start, l = offset.start + offset.count; i < l; i += 3 ) {

									renderList.pushTriangle( indices[ i ] + index, indices[ i + 1 ] + index, indices[ i + 2 ] + index );

								}

							}

						} else {

							for ( var i = 0, l = indices.length; i < l; i += 3 ) {

								renderList.pushTriangle( indices[ i ], indices[ i + 1 ], indices[ i + 2 ] );

							}

						}

					} else {

						for ( var i = 0, l = positions.length / 3; i < l; i += 3 ) {

							renderList.pushTriangle( i, i + 1, i + 2 );

						}

					}

				} else if ( geometry instanceof THREE.Geometry ) {

					var vertices = geometry.vertices;
					var faces = geometry.faces;
					var faceVertexUvs = geometry.faceVertexUvs[ 0 ];

					_normalMatrix.getNormalMatrix( _modelMatrix );

					var isFaceMaterial = object.material instanceof THREE.MeshFaceMaterial;
					var objectMaterials = isFaceMaterial === true ? object.material : null;

					for ( var v = 0, vl = vertices.length; v < vl; v ++ ) {

						var vertex = vertices[ v ];
						renderList.pushVertex( vertex.x, vertex.y, vertex.z );

					}

					for ( var f = 0, fl = faces.length; f < fl; f ++ ) {

						var face = faces[ f ];

						var material = isFaceMaterial === true
							 ? objectMaterials.materials[ face.materialIndex ]
							 : object.material;

						if ( material === undefined ) continue;

						var side = material.side;

						var v1 = _vertexPool[ face.a ];
						var v2 = _vertexPool[ face.b ];
						var v3 = _vertexPool[ face.c ];

						if ( material.morphTargets === true ) {

							var morphTargets = geometry.morphTargets;
							var morphInfluences = object.morphTargetInfluences;

							var v1p = v1.position;
							var v2p = v2.position;
							var v3p = v3.position;

							_vA.set( 0, 0, 0 );
							_vB.set( 0, 0, 0 );
							_vC.set( 0, 0, 0 );

							for ( var t = 0, tl = morphTargets.length; t < tl; t ++ ) {

								var influence = morphInfluences[ t ];

								if ( influence === 0 ) continue;

								var targets = morphTargets[ t ].vertices;

								_vA.x += ( targets[ face.a ].x - v1p.x ) * influence;
								_vA.y += ( targets[ face.a ].y - v1p.y ) * influence;
								_vA.z += ( targets[ face.a ].z - v1p.z ) * influence;

								_vB.x += ( targets[ face.b ].x - v2p.x ) * influence;
								_vB.y += ( targets[ face.b ].y - v2p.y ) * influence;
								_vB.z += ( targets[ face.b ].z - v2p.z ) * influence;

								_vC.x += ( targets[ face.c ].x - v3p.x ) * influence;
								_vC.y += ( targets[ face.c ].y - v3p.y ) * influence;
								_vC.z += ( targets[ face.c ].z - v3p.z ) * influence;

							}

							v1.position.add( _vA );
							v2.position.add( _vB );
							v3.position.add( _vC );

							renderList.projectVertex( v1 );
							renderList.projectVertex( v2 );
							renderList.projectVertex( v3 );

						}

						if ( renderList.checkTriangleVisibility( v1, v2, v3 ) === false ) continue;

						var visible = renderList.checkBackfaceCulling( v1, v2, v3 );

						if ( side !== THREE.DoubleSide ) {
							if ( side === THREE.FrontSide && visible === false ) continue;
							if ( side === THREE.BackSide && visible === true ) continue;
						}

						_face = getNextFaceInPool();

						_face.id = object.id;
						_face.v1.copy( v1 );
						_face.v2.copy( v2 );
						_face.v3.copy( v3 );

						_face.normalModel.copy( face.normal );

						if ( visible === false && ( side === THREE.BackSide || side === THREE.DoubleSide ) ) {

							_face.normalModel.negate();

						}

						_face.normalModel.applyMatrix3( _normalMatrix ).normalize();

						var faceVertexNormals = face.vertexNormals;

						for ( var n = 0, nl = Math.min( faceVertexNormals.length, 3 ); n < nl; n ++ ) {

							var normalModel = _face.vertexNormalsModel[ n ];
							normalModel.copy( faceVertexNormals[ n ] );

							if ( visible === false && ( side === THREE.BackSide || side === THREE.DoubleSide ) ) {

								normalModel.negate();

							}

							normalModel.applyMatrix3( _normalMatrix ).normalize();

						}

						_face.vertexNormalsLength = faceVertexNormals.length;

						var vertexUvs = faceVertexUvs[ f ];

						if ( vertexUvs !== undefined ) {

							for ( var u = 0; u < 3; u ++ ) {

								_face.uvs[ u ].copy( vertexUvs[ u ] );

							}

						}

						_face.color = face.color;
						_face.material = material;

						_face.z = ( v1.positionScreen.z + v2.positionScreen.z + v3.positionScreen.z ) / 3;

						_renderData.elements.push( _face );

					}

				}

			} else if ( object instanceof THREE.Line ) {

				if ( geometry instanceof THREE.BufferGeometry ) {

					var attributes = geometry.attributes;

					if ( attributes.position !== undefined ) {

						var positions = attributes.position.array;

						for ( var i = 0, l = positions.length; i < l; i += 3 ) {

							renderList.pushVertex( positions[ i ], positions[ i + 1 ], positions[ i + 2 ] );

						}

						if ( attributes.index !== undefined ) {

							var indices = attributes.index.array;

							for ( var i = 0, l = indices.length; i < l; i += 2 ) {

								renderList.pushLine( indices[ i ], indices[ i + 1 ] );

							}

						} else {

							var step = object.mode === THREE.LinePieces ? 2 : 1;

							for ( var i = 0, l = ( positions.length / 3 ) - 1; i < l; i += step ) {

								renderList.pushLine( i, i + 1 );

							}

						}

					}

				} else if ( geometry instanceof THREE.Geometry ) {

					_modelViewProjectionMatrix.multiplyMatrices( _viewProjectionMatrix, _modelMatrix );

					var vertices = object.geometry.vertices;

					if ( vertices.length === 0 ) continue;

					v1 = getNextVertexInPool();
					v1.positionScreen.copy( vertices[ 0 ] ).applyMatrix4( _modelViewProjectionMatrix );

					// Handle LineStrip and LinePieces
					var step = object.mode === THREE.LinePieces ? 2 : 1;

					for ( var v = 1, vl = vertices.length; v < vl; v ++ ) {

						v1 = getNextVertexInPool();
						v1.positionScreen.copy( vertices[ v ] ).applyMatrix4( _modelViewProjectionMatrix );

						if ( ( v + 1 ) % step > 0 ) continue;

						v2 = _vertexPool[ _vertexCount - 2 ];

						_clippedVertex1PositionScreen.copy( v1.positionScreen );
						_clippedVertex2PositionScreen.copy( v2.positionScreen );

						if ( clipLine( _clippedVertex1PositionScreen, _clippedVertex2PositionScreen ) === true ) {

							// Perform the perspective divide
							_clippedVertex1PositionScreen.multiplyScalar( 1 / _clippedVertex1PositionScreen.w );
							_clippedVertex2PositionScreen.multiplyScalar( 1 / _clippedVertex2PositionScreen.w );

							_line = getNextLineInPool();

							_line.id = object.id;
							_line.v1.positionScreen.copy( _clippedVertex1PositionScreen );
							_line.v2.positionScreen.copy( _clippedVertex2PositionScreen );

							_line.z = Math.max( _clippedVertex1PositionScreen.z, _clippedVertex2PositionScreen.z );

							_line.material = object.material;

							if ( object.material.vertexColors === THREE.VertexColors ) {

								_line.vertexColors[ 0 ].copy( object.geometry.colors[ v ] );
								_line.vertexColors[ 1 ].copy( object.geometry.colors[ v - 1 ] );

							}

							_renderData.elements.push( _line );

						}

					}

				}

			} else if ( object instanceof THREE.Sprite ) {

				_vector4.set( _modelMatrix.elements[ 12 ], _modelMatrix.elements[ 13 ], _modelMatrix.elements[ 14 ], 1 );
				_vector4.applyMatrix4( _viewProjectionMatrix );

				var invW = 1 / _vector4.w;

				_vector4.z *= invW;

				if ( _vector4.z >= - 1 && _vector4.z <= 1 ) {

					_sprite = getNextSpriteInPool();
					_sprite.id = object.id;
					_sprite.x = _vector4.x * invW;
					_sprite.y = _vector4.y * invW;
					_sprite.z = _vector4.z;
					_sprite.object = object;

					_sprite.rotation = object.rotation;

					_sprite.scale.x = object.scale.x * Math.abs( _sprite.x - ( _vector4.x + camera.projectionMatrix.elements[ 0 ] ) / ( _vector4.w + camera.projectionMatrix.elements[ 12 ] ) );
					_sprite.scale.y = object.scale.y * Math.abs( _sprite.y - ( _vector4.y + camera.projectionMatrix.elements[ 5 ] ) / ( _vector4.w + camera.projectionMatrix.elements[ 13 ] ) );

					_sprite.material = object.material;

					_renderData.elements.push( _sprite );

				}

			}

		}

		if ( sortElements === true ) {

			_renderData.elements.sort( painterSort );

		}

		return _renderData;

	};

	// Pools

	function getNextObjectInPool() {

		if ( _objectCount === _objectPoolLength ) {

			var object = new THREE.RenderableObject();
			_objectPool.push( object );
			_objectPoolLength ++;
			_objectCount ++;
			return object;

		}

		return _objectPool[ _objectCount ++ ];

	}

	function getNextVertexInPool() {

		if ( _vertexCount === _vertexPoolLength ) {

			var vertex = new THREE.RenderableVertex();
			_vertexPool.push( vertex );
			_vertexPoolLength ++;
			_vertexCount ++;
			return vertex;

		}

		return _vertexPool[ _vertexCount ++ ];

	}

	function getNextFaceInPool() {

		if ( _faceCount === _facePoolLength ) {

			var face = new THREE.RenderableFace();
			_facePool.push( face );
			_facePoolLength ++;
			_faceCount ++;
			return face;

		}

		return _facePool[ _faceCount ++ ];


	}

	function getNextLineInPool() {

		if ( _lineCount === _linePoolLength ) {

			var line = new THREE.RenderableLine();
			_linePool.push( line );
			_linePoolLength ++;
			_lineCount ++
			return line;

		}

		return _linePool[ _lineCount ++ ];

	}

	function getNextSpriteInPool() {

		if ( _spriteCount === _spritePoolLength ) {

			var sprite = new THREE.RenderableSprite();
			_spritePool.push( sprite );
			_spritePoolLength ++;
			_spriteCount ++
			return sprite;

		}

		return _spritePool[ _spriteCount ++ ];

	}

	//

	function painterSort( a, b ) {

		if ( a.z !== b.z ) {

			return b.z - a.z;

		} else if ( a.id !== b.id ) {

			return a.id - b.id;

		} else {

			return 0;

		}

	}

	function clipLine( s1, s2 ) {

		var alpha1 = 0, alpha2 = 1,

		// Calculate the boundary coordinate of each vertex for the near and far clip planes,
		// Z = -1 and Z = +1, respectively.
		bc1near =  s1.z + s1.w,
		bc2near =  s2.z + s2.w,
		bc1far =  - s1.z + s1.w,
		bc2far =  - s2.z + s2.w;

		if ( bc1near >= 0 && bc2near >= 0 && bc1far >= 0 && bc2far >= 0 ) {

			// Both vertices lie entirely within all clip planes.
			return true;

		} else if ( ( bc1near < 0 && bc2near < 0 ) || ( bc1far < 0 && bc2far < 0 ) ) {

			// Both vertices lie entirely outside one of the clip planes.
			return false;

		} else {

			// The line segment spans at least one clip plane.

			if ( bc1near < 0 ) {

				// v1 lies outside the near plane, v2 inside
				alpha1 = Math.max( alpha1, bc1near / ( bc1near - bc2near ) );

			} else if ( bc2near < 0 ) {

				// v2 lies outside the near plane, v1 inside
				alpha2 = Math.min( alpha2, bc1near / ( bc1near - bc2near ) );

			}

			if ( bc1far < 0 ) {

				// v1 lies outside the far plane, v2 inside
				alpha1 = Math.max( alpha1, bc1far / ( bc1far - bc2far ) );

			} else if ( bc2far < 0 ) {

				// v2 lies outside the far plane, v2 inside
				alpha2 = Math.min( alpha2, bc1far / ( bc1far - bc2far ) );

			}

			if ( alpha2 < alpha1 ) {

				// The line segment spans two boundaries, but is outside both of them.
				// (This can't happen when we're only clipping against just near/far but good
				//  to leave the check here for future usage if other clip planes are added.)
				return false;

			} else {

				// Update the s1 and s2 vertices to match the clipped line segment.
				s1.lerp( s2, alpha1 );
				s2.lerp( s1, 1 - alpha2 );

				return true;

			}

		}

	}

};

84 ms
ICdyZXBlYXQteScKCQkJCQkJIDogJ25vLXJlcGVhdCcKCQkpOwoKCX0KCglmdW5jdGlvbiBwYXR0ZXJuUGF0aCggeDAsIHkwLCB4MSwgeTEsIHgyLCB5MiwgdTAsIHYwLCB1MSwgdjEsIHUyLCB2MiwgdGV4dHVyZSApIHsKCgkJaWYgKCB0ZXh0dXJlIGluc3RhbmNlb2YgVEhSRUUuRGF0YVRleHR1cmUgKSByZXR1cm47CgoJCWlmICggdGV4dHVyZS5oYXNFdmVudExpc3RlbmVyKCAndXBkYXRlJywgb25UZXh0dXJlVXBkYXRlICkgPT09IGZhbHNlICkgewoKCQkJaWYgKCB0ZXh0dXJlLmltYWdlICE9PSB1bmRlZmluZWQgJiYgdGV4dHVyZS5pbWFnZS53aWR0aCA+IDAgKSB7CgoJCQkJdGV4dHVyZVRvUGF0dGVybiggdGV4dHVyZSApOwoKCQkJfQoKCQkJdGV4dHVyZS5hZGRFdmVudExpc3RlbmVyKCAndXBkYXRlJywgb25UZXh0dXJlVXBkYXRlICk7CgoJCX0KCgkJdmFyIHBhdHRlcm4gPSBfcGF0dGVybnNbIHRleHR1cmUuaWQgXTsKCgkJaWYgKCBwYXR0ZXJuICE9PSB1bmRlZmluZWQgKSB7CgoJCQlzZXRGaWxsU3R5bGUoIHBhdHRlcm4gKTsKCgkJfSBlbHNlIHsKCgkJCXNldEZpbGxTdHlsZSggJ3JnYmEoMCwwLDAsMSknICk7CgkJCV9jb250ZXh0LmZpbGwoKTsKCgkJCXJldHVybjsKCgkJfQoKCQkvLyBodHRwOi8vZXh0cmVtZWx5c2F0aXNmYWN0b3J5dG90YWxpdGFyaWFuaXNtLmNvbS9ibG9nLz9wPTIxMjAKCgkJdmFyIGEsIGIsIGMsIGQsIGUsIGYsIGRldCwgaWRldCwKCQlvZmZzZXRYID0gdGV4dHVyZS5vZmZzZXQueCAvIHRleHR1cmUucmVwZWF0LngsCgkJb2Zmc2V0WSA9IHRleHR1cmUub2Zmc2V0LnkgLyB0ZXh0dXJlLnJlcGVhdC55LAoJCXdpZHRoID0gdGV4dHVyZS5pbWFnZS53aWR0aCAqIHRleHR1cmUucmVwZWF0LngsCgkJaGVpZ2h0ID0gdGV4dHVyZS5pbWFnZS5oZWlnaHQgKiB0ZXh0dXJlLnJlcGVhdC55OwoKCQl1MCA9ICggdTAgKyBvZmZzZXRYICkgKiB3aWR0aDsKCQl2MCA9ICggdjAgKyBvZmZzZXRZICkgKiBoZWlnaHQ7CgoJCXUxID0gKCB1MSArIG9mZnNldFggKSAqIHdpZHRoOwoJCXYxID0gKCB2MSArIG9mZnNldFkgKSAqIGhlaWdodDsKCgkJdTIgPSAoIHUyICsgb2Zmc2V0WCApICogd2lkdGg7CgkJdjIgPSAoIHYyICsgb2Zmc2V0WSApICogaGVpZ2h0OwoKCQl4MSAtPSB4MDsgeTEgLT0geTA7CgkJeDIgLT0geDA7IHkyIC09IHkwOwoKCQl1MSAtPSB1MDsgdjEgLT0gdjA7CgkJdTIgLT0gdTA7IHYyIC09IHYwOwoKCQlkZXQgPSB1MSAqIHYyIC0gdTIgKiB2MTsKCgkJaWYgKCBkZXQgPT09IDAgKSByZXR1cm47CgoJCWlkZXQgPSAxIC8gZGV0OwoKCQlhID0gKCB2MiAqIHgxIC0gdjEgKiB4MiApICogaWRldDsKCQliID0gKCB2MiAqIHkxIC0gdjEgKiB5MiApICogaWRldDsKCQljID0gKCB1MSAqIHgyIC0gdTIgKiB4MSApICogaWRldDsKCQlkID0gKCB1MSAqIHkyIC0gdTIgKiB5MSApICogaWRldDsKCgkJZSA9IHgwIC0gYSAqIHUwIC0gYyAqIHYwOwoJCWYgPSB5MCAtIGIgKiB1MCAtIGQgKiB2MDsKCgkJX2NvbnRleHQuc2F2ZSgpOwoJCV9jb250ZXh0LnRyYW5zZm9ybSggYSwgYiwgYywgZCwgZSwgZiApOwoJCV9jb250ZXh0LmZpbGwoKTsKCQlfY29udGV4dC5yZXN0b3JlKCk7CgoJfQoKCWZ1bmN0aW9uIGNsaXBJbWFnZSggeDAsIHkwLCB4MSwgeTEsIHgyLCB5MiwgdTAsIHYwLCB1MSwgdjEsIHUyLCB2MiwgaW1hZ2UgKSB7CgoJCS8vIGh0dHA6Ly9leHRyZW1lbHlzYXRpc2ZhY3Rvcnl0b3RhbGl0YXJpYW5pc20uY29tL2Jsb2cvP3A9MjEyMAoKCQl2YXIgYSwgYiwgYywgZCwgZSwgZiwgZGV0LCBpZGV0LAoJCXdpZHRoID0gaW1hZ2Uud2lkdGggLSAxLAoJCWhlaWdodCA9IGltYWdlLmhlaWdodCAtIDE7CgoJCXUwICo9IHdpZHRoOyB2MCAqPSBoZWlnaHQ7CgkJdTEgKj0gd2lkdGg7IHYxICo9IGhlaWdodDsKCQl1MiAqPSB3aWR0aDsgdjIgKj0gaGVpZ2h0OwoKCQl4MSAtPSB4MDsgeTEgLT0geTA7CgkJeDIgLT0geDA7IHkyIC09IHkwOwoKCQl1MSAtPSB1MDsgdjEgLT0gdjA7CgkJdTIgLT0gdTA7IHYyIC09IHYwOwoKCQlkZXQgPSB1MSAqIHYyIC0gdTIgKiB2MTsKCgkJaWRldCA9IDEgLyBkZXQ7CgoJCWEgPSAoIHYyICogeDEgLSB2MSAqIHgyICkgKiBpZGV0OwoJCWIgPSAoIHYyICogeTEgLSB2MSAqIHkyICkgKiBpZGV0OwoJCWMgPSAoIHUxICogeDIgLSB1MiAqIHgxICkgKiBpZGV0OwoJCWQgPSAoIHUxICogeTIgLSB1MiAqIHkxICkgKiBpZGV0OwoKCQllID0geDAgLSBhICogdTAgLSBjICogdjA7CgkJZiA9IHkwIC0gYiAqIHUwIC0gZCAqIHYwOwoKCQlfY29udGV4dC5zYXZlKCk7CgkJX2NvbnRleHQudHJhbnNmb3JtKCBhLCBiLCBjLCBkLCBlLCBmICk7CgkJX2NvbnRleHQuY2xpcCgpOwoJCV9jb250ZXh0LmRyYXdJbWFnZSggaW1hZ2UsIDAsIDAgKTsKCQlfY29udGV4dC5yZXN0b3JlKCk7CgoJfQoKCS8vIEhpZGUgYW50aS1hbGlhcyBnYXBzCgoJZnVuY3Rpb24gZXhwYW5kKCB2MSwgdjIsIHBpeGVscyApIHsKCgkJdmFyIHggPSB2Mi54IC0gdjEueCwgeSA9IHYyLnkgLSB2MS55LAoJCWRldCA9IHggKiB4ICsgeSAqIHksIGlkZXQ7CgoJCWlmICggZGV0ID09PSAwICkgcmV0dXJuOwoKCQlpZGV0ID0gcGl4ZWxzIC8gTWF0aC5zcXJ0KCBkZXQgKTsKCgkJeCAqPSBpZGV0OyB5ICo9IGlkZXQ7CgoJCXYyLnggKz0geDsgdjIueSArPSB5OwoJCXYxLnggLT0geDsgdjEueSAtPSB5OwoKCX0KCgkvLyBDb250ZXh0IGNhY2hlZCBtZXRob2RzLgoKCWZ1bmN0aW9uIHNldE9wYWNpdHkoIHZhbHVlICkgewoKCQlpZiAoIF9jb250ZXh0R2xvYmFsQWxwaGEgIT09IHZhbHVlICkgewoKCQkJX2NvbnRleHQuZ2xvYmFsQWxwaGEgPSB2YWx1ZTsKCQkJX2NvbnRleHRHbG9iYWxBbHBoYSA9IHZhbHVlOwoKCQl9CgoJfQoKCWZ1bmN0aW9uIHNldEJsZW5kaW5nKCB2YWx1ZSApIHsKCgkJaWYgKCBfY29udGV4dEdsb2JhbENvbXBvc2l0ZU9wZXJhdGlvbiAhPT0gdmFsdWUgKSB7CgoJCQlpZiAoIHZhbHVlID09PSBUSFJFRS5Ob3JtYWxCbGVuZGluZyApIHsKCgkJCQlfY29udGV4dC5nbG9iYWxDb21wb3NpdGVPcGVyYXRpb24gPSAnc291cmNlLW92ZXInOwoKCQkJfSBlbHNlIGlmICggdmFsdWUgPT09IFRIUkVFLkFkZGl0aXZlQmxlbmRpbmcgKSB7CgoJCQkJX2NvbnRleHQuZ2xvYmFsQ29tcG9zaXRlT3BlcmF0aW9uID0gJ2xpZ2h0ZXInOwoKCQkJfSBlbHNlIGlmICggdmFsdWUgPT09IFRIUkVFLlN1YnRyYWN0aXZlQmxlbmRpbmcgKSB7CgoJCQkJX2NvbnRleHQuZ2xvYmFsQ29tcG9zaXRlT3BlcmF0aW9uID0gJ2Rhcmtlcic7CgoJCQl9CgoJCQlfY29udGV4dEdsb2JhbENvbXBvc2l0ZU9wZXJhdGlvbiA9IHZhbHVlOwoKCQl9CgoJfQoKCWZ1bmN0aW9uIHNldExpbmVXaWR0aCggdmFsdWUgKSB7CgoJCWlmICggX2NvbnRleHRMaW5lV2lkdGggIT09IHZhbHVlICkgewoKCQkJX2NvbnRleHQubGluZVdpZHRoID0gdmFsdWU7CgkJCV9jb250ZXh0TGluZVdpZHRoID0gdmFsdWU7CgoJCX0KCgl9CgoJZnVuY3Rpb24gc2V0TGluZUNhcCggdmFsdWUgKSB7CgoJCS8vICJidXR0IiwgInJvdW5kIiwgInNxdWFyZSIKCgkJaWYgKCBfY29udGV4dExpbmVDYXAgIT09IHZhbHVlICkgewoKCQkJX2NvbnRleHQubGluZUNhcCA9IHZhbHVlOwoJCQlfY29udGV4dExpbmVDYXAgPSB2YWx1ZTsKCgkJfQoKCX0KCglmdW5jdGlvbiBzZXRMaW5lSm9pbiggdmFsdWUgKSB7CgoJCS8vICJyb3VuZCIsICJiZXZlbCIsICJtaXRlciIKCgkJaWYgKCBfY29udGV4dExpbmVKb2luICE9PSB2YWx1ZSApIHsKCgkJCV9jb250ZXh0LmxpbmVKb2luID0gdmFsdWU7CgkJCV9jb250ZXh0TGluZUpvaW4gPSB2YWx1ZTsKCgkJfQoKCX0KCglmdW5jdGlvbiBzZXRTdHJva2VTdHlsZSggdmFsdWUgKSB7CgoJCWlmICggX2NvbnRleHRTdHJva2VTdHlsZSAhPT0gdmFsdWUgKSB7CgoJCQlfY29udGV4dC5zdHJva2VTdHlsZSA9IHZhbHVlOwoJCQlfY29udGV4dFN0cm9rZVN0eWxlID0gdmFsdWU7CgoJCX0KCgl9CgoJZnVuY3Rpb24gc2V0RmlsbFN0eWxlKCB2YWx1ZSApIHsKCgkJaWYgKCBfY29udGV4dEZpbGxTdHlsZSAhPT0gdmFsdWUgKSB7CgoJCQlfY29udGV4dC5maWxsU3R5bGUgPSB2YWx1ZTsKCQkJX2NvbnRleHRGaWxsU3R5bGUgPSB2YWx1ZTsKCgkJfQoKCX0KCglmdW5jdGlvbiBzZXRMaW5lRGFzaCggdmFsdWUgKSB7CgoJCWlmICggX2NvbnRleHRMaW5lRGFzaC5sZW5ndGggIT09IHZhbHVlLmxlbmd0aCApIHsKCgkJCV9jb250ZXh0LnNldExpbmVEYXNoKCB2YWx1ZSApOwoJCQlfY29udGV4dExpbmVEYXNoID0gdmFsdWU7CgoJCX0KCgl9Cgp9Owo=
5 ms
x-javascript;base64,/* scatterplotThree.js
 * A set of example Javascript functions that support a threejs-based 3d
 * scatterplot in R, geared for use with the htmlwidgets and shiny packages.
 */
HTMLWidgets.widget(
{

  name: "scatterplotThree",
  type: "output",

  initialize: function(el, width, height)
  {
    var r = render_init(el, width, height, false);
    var c = new THREE.PerspectiveCamera(39, r.domElement.width/r.domElement.height, 1E-5, 10);
    var s = new THREE.Scene();
    return {renderer:r, camera:c, scene: s, width: parseInt(width), height: parseInt(height)};
  },

  resize: function(el, width, height, stuff)
  {
    stuff.renderer.clear();
    stuff.renderer.setSize(parseInt(width), parseInt(height));
    stuff.camera.projectionMatrix = new THREE.Matrix4().makePerspective(stuff.camera.fov,  stuff.renderer.domElement.width/stuff.renderer.domElement.height, stuff.camera.near, stuff.camera.far);
    stuff.camera.lookAt(stuff.scene.position);
    stuff.renderer.render(stuff.scene, stuff.camera);
  },

  renderValue: function(el, x, stuff)
  {
    stuff.renderer = render_init(el, stuff.width, stuff.height, x.options.renderer, x.options.labelmargin);
    if(x.bg) stuff.renderer.setClearColor(new THREE.Color(x.bg));
    scatter(el, x, stuff);
  }
})


function render_init(el, width, height, choice, labelmargin)
{
  var r;
  if(choice=="webgl-buffered") choice = "webgl";  // deprecated. All WebGL is buffered now
  if(Detector.webgl && (choice=="auto" || choice=="webgl"))
  {
    r = new THREE.WebGLRenderer({antialias: true});
    GL=true;
  } else
  {
    r = new THREE.CanvasRenderer();
    GL=false;
  }
  r.setSize(parseInt(width), parseInt(height));
  r.setClearColor("white");
  d3.select(el).node().innerHTML="";
  d3.select(el).node().appendChild(r.domElement);
  d3.select(el).append("div").text(" ").attr("id","coordinate_label");
  document.getElementById("coordinate_label").style.zIndex = "100";
  document.getElementById("coordinate_label").style.position = "absolute";
  document.getElementById("coordinate_label").style.top = "0";
  document.getElementById("coordinate_label").style.margin = labelmargin;
  return r;
}

// x.options list of options including:
// x.options.axisLabels  3 element list of axis labels
// x.options.grid true/false draw xz grid (requires xtick.length==ztick.length)
// x.options.stroke (optional) stroke color (canvas renderer only)
// x.options.color (optional) either a single color or a vector of colors
// x.options.size (optional) either a single size or a vector of sizes
// x.options.renderer, one of "auto" "canvas" "webgl" or "webgl-buffered"
// x.options.labels (optional) vector of point labels
// x.options 
//   xtick:[0,0.5,1]
//   xticklab:["1","2","3"]
//   ytick:[0,0.5,1]
//   yticklab:["10","20","30"]
//   ztick:[0,0.5,1]
//   zticklab:["-1","0","1"]
//   NOTE: ticks must be in [0,1].
// x.data JSON 3-column data matrix. Data are assumed to be already
//   scaled in a unit box (that is, all coordinates are assumed to lie in the
//   interval [0,1]).

function scatter(el, x, obj)
{
  obj.camera = new THREE.PerspectiveCamera(39, obj.renderer.domElement.width/obj.renderer.domElement.height, 1E-5, 10);
  obj.camera.position.z = 2;
  obj.camera.position.x = 2.55;
  obj.camera.position.y = 1.25;

  obj.scene = new THREE.Scene();
  var group = new THREE.Object3D();      // contains non-point plot elements
  var pointgroup = new THREE.Object3D(); // contains point elements
  group.name = "group";
  pointgroup.name = "pointgroup";
  obj.scene.add( group );
  obj.scene.add( pointgroup );
  obj.raycaster = new THREE.Raycaster();
  obj.raycaster.params.PointCloud.threshold = 0.05; // XXX Investigate these units...


// program for drawing a Canvas point
  var program = function ( context )
  {
    context.beginPath();
    context.arc( 0, 0, 0.5, 0, Math.PI*2, true );
    if(x.options.stroke)
    {
      context.strokeStyle = x.options.stroke;
      context.lineWidth = 0.15;
      context.stroke();
    }
    context.fill();
  };
// add the points
  var j;
  if(GL)
  {
    var geometry = new THREE.BufferGeometry();
    var positions = new Float32Array( x.data.length );
    var colors = new Float32Array( x.data.length );
    var col = new THREE.Color("steelblue");
    var scale = 0.07;
    if(x.options.size && !Array.isArray(x.options.size)) scale = 0.07 * x.options.size;
    for ( var i = 0; i < x.data.length; i++ )
    {
      positions[i] = x.data[i];
    }
    for(var i=0;i<x.data.length/3;i++)
    {
      j = i*3;
      if(x.options.color)
      {
        if(Array.isArray(x.options.color)) col = new THREE.Color(x.options.color[i]);
        else col = new THREE.Color(x.options.color);
      }
      colors[j] = col.r;
      colors[j+1] = col.g;
      colors[j+2] = col.b;
    }
    geometry.addAttribute( 'position', new THREE.BufferAttribute( positions, 3 ) );
    geometry.addAttribute( 'color', new THREE.BufferAttribute( colors, 3 ) );
    geometry.computeBoundingSphere();
    var pcmaterial = new THREE.PointCloudMaterial( { size: scale, vertexColors: THREE.VertexColors } );
    var particleSystem = new THREE.PointCloud( geometry, pcmaterial );
    pointgroup.add( particleSystem );
  }
  else {
    var col = new THREE.Color("steelblue");
    var scale = 0.03;
    for ( var i = 0; i < x.data.length/3; i++ )
    {
      j = i*3;
      if(x.options.color)
      {
        if(Array.isArray(x.options.color)) col = new THREE.Color(x.options.color[i]);
        else col = new THREE.Color(x.options.color);
      }
      if(x.options.size)
      {
        if(Array.isArray(x.options.size)) scale = 0.03*x.options.size[i];
        else scale = 0.03*x.options.size;
      }
      var material = new THREE.SpriteCanvasMaterial( {
          color: col, program: program , opacity:0.9} );
      var particle = new THREE.Sprite( material );
      particle.position.x = x.data[j];
      particle.position.y = x.data[j+1];
      particle.position.z = x.data[j+2];
      particle.scale.x = particle.scale.y = scale;
// Label points.
      if(x.options.labels)
      {
        if(Array.isArray(x.options.labels)) particle.name = x.options.labels[i];
        else particle.name = x.options.labels;
      }
      pointgroup.add( particle );
    }
  }

// helper function to add text to object
  function addText(object, string, scale, x, y, z, color)
  {
    var canvas = document.createElement('canvas');
    var size = 256;
    canvas.width = size;
    canvas.height = size;
    var context = canvas.getContext('2d');
    context.fillStyle = "#" + color.getHexString();
    context.textAlign = 'center';
    context.font = '24px Arial';
    context.fillText(string, size / 2, size / 2);
    var amap = new THREE.Texture(canvas);
    amap.needsUpdate = true;
    var mat = new THREE.SpriteMaterial({
      map: amap,
      transparent: true,
      useScreenCoordinates: false,
      color: 0xffffff });
    sp = new THREE.Sprite(mat);
    sp.scale.set( scale, scale, scale );
    sp.position.x = x;
    sp.position.y = y;
    sp.position.z = z;
    object.add(sp);
  }

// Set up the axes
  var axisColor = new THREE.Color("#000000");
  if(x.bg)
  {
    var bgcolor = new THREE.Color(x.bg);
    axisColor.r = 1 - bgcolor.r;
    axisColor.g = 1 - bgcolor.g;
    axisColor.b = 1 - bgcolor.b;
  }

  var fontSize = Math.max(Math.round(1/4), 8);
  var fontOffset = Math.min(Math.round(fontSize/4), 8);
  var xAxisGeo = new THREE.Geometry();
  var yAxisGeo = new THREE.Geometry();
  var zAxisGeo = new THREE.Geometry();
  function v(x,y,z){ return new THREE.Vector3(x,y,z); }
  xAxisGeo.vertices.push(v(0, 0, 0), v(1, 0, 0));
  yAxisGeo.vertices.push(v(0, 0, 0), v(0, 1, 0));
  zAxisGeo.vertices.push(v(0, 0, 0), v(0, 0, 1));
  var xAxis = new THREE.Line(xAxisGeo, new THREE.LineBasicMaterial({color: axisColor, linewidth: 1}));
  var yAxis = new THREE.Line(yAxisGeo, new THREE.LineBasicMaterial({color: axisColor, linewidth: 1}));
  var zAxis = new THREE.Line(zAxisGeo, new THREE.LineBasicMaterial({color: axisColor, linewidth: 1}));
  xAxis.type = THREE.Lines;
  yAxis.type = THREE.Lines;
  zAxis.type = THREE.Lines;
  group.add(xAxis);
  group.add(yAxis);
  group.add(zAxis);
  if(x.options.axisLabels)
  {
    addText(group, x.options.axisLabels[0], 0.8, 1.1, 0, 0, axisColor)
    addText(group, x.options.axisLabels[1], 0.8, 0, 1.1, 0, axisColor)
    addText(group, x.options.axisLabels[2], 0.8, 0, 0, 1.1, axisColor)
  }


// Ticks and tick labels
  var tickColor = axisColor;
  tickColor.r = Math.min(tickColor.r + 0.2, 1);
  tickColor.g = Math.min(tickColor.g + 0.2, 1);
  tickColor.b = Math.min(tickColor.b + 0.2, 1);
  function tick(length, thickness, axis, ticks, ticklabels)
  {
    for(var j=0; j<ticks.length; j++)
    {
      var tick = new THREE.Geometry();
      var a1 = ticks[j]; var a2 = ticks[j]; var a3=ticks[j];
      var b1 = length; var b2 = -length; var b3=-0.05;
      var c1 = 0; var c2 = 0; var c3=-0.08;
      if(axis==1){a1=length; b1=ticks[j]; c1=0; a2=-length; b2=ticks[j]; c2=0; a3=0.08; b3=ticks[j]; c3=-0.05;}
      else if(axis==2){a1=0; b1=length; c1=ticks[j];a2=0;b2=-length;c2=ticks[j]; a3=-0.08; b3=-0.05; c3=ticks[j];}
      tick.vertices.push(v(a1,b1,c1),v(a2,b2,c2));
      if(ticklabels)
        addText(group, ticklabels[j], 0.5, a3, b3, c3, tickColor);
      var tl = new THREE.Line(tick, new THREE.LineBasicMaterial({color: tickColor, linewidth: thickness}));
      tl.type=THREE.Lines;
      group.add(tl);
    }
  }
  if(x.options.xtick) tick(0.005,3,0,x.options.xtick,x.options.xticklab);
  if(x.options.ytick) tick(0.005,3,1,x.options.ytick,x.options.yticklab);
  if(x.options.ztick) tick(0.005,3,2,x.options.ztick,x.options.zticklab);

// Grid
  if(x.options.grid && x.options.xtick && x.options.ztick && x.options.xtick.length==x.options.ztick.length)
  {
    for(var j=1; j<x.options.xtick.length; j++)
    {
      var gridline = new THREE.Geometry();
      gridline.vertices.push(v(x.options.xtick[j],0,0),v(x.options.xtick[j],0,1));
      var gl = new THREE.Line(gridline, new THREE.LineBasicMaterial({color: tickColor, linewidth: 1}));
      gl.type=THREE.Lines;
      group.add(gl);
      gridline = new THREE.Geometry();
      gridline.vertices.push(v(0,0,x.options.ztick[j]),v(1,0,x.options.ztick[j]));
      gl = new THREE.Line(gridline, new THREE.LineBasicMaterial({color: tickColor, linewidth: 1}));
      gl.type=THREE.Lines;
      group.add(gl);
    }
  }


  var mouse = new THREE.Vector2();
  var down = false;
  var sx = 0, sy = 0;
  el.onmousedown = function (ev)
  {
    down = true; sx = ev.clientX; sy = ev.clientY;
  };
  el.onmouseup = function(){ down = false; };
  function mousewheel(event)
  {
    var fovMAX = 100;
    var fovMIN = 2;
    event.wheelDeltaY = event.wheelDeltaY || -10*event.detail || event.wheelDelta;
    if(GL) obj.camera.fov -= event.wheelDeltaY * 0.02;
    else obj.camera.fov -= event.wheelDeltaY * 0.0075;
    obj.camera.fov = Math.max( Math.min( obj.camera.fov, fovMAX ), fovMIN );
    obj.camera.projectionMatrix = new THREE.Matrix4().makePerspective(obj.camera.fov,  obj.renderer.domElement.width/obj.renderer.domElement.height, obj.camera.near, obj.camera.far);
    render();
  }
  el.onmousewheel = function(ev) {ev.preventDefault();};
  el.addEventListener('DOMMouseScroll', mousewheel, true);
  el.addEventListener('mousewheel', mousewheel, true);

  el.onmousemove = function(ev)
  { 
    ev.preventDefault();

    var canvasRect = this.getBoundingClientRect();
    mouse.x = 2 * ( ev.pageX - canvasRect.left ) / canvasRect.width - 1;
    mouse.y = -2 * ( ev.pageY - canvasRect.top ) / canvasRect.height + 1;

    if (down) {
      var dx = ev.clientX - sx;
      var dy = ev.clientY - sy;
      group.rotation.y += dx*0.01;
      pointgroup.rotation.y += dx*0.01;
      obj.camera.position.y += 0.05*dy;
      if(obj.camera.position.y < -8) obj.camera.position.y = -8;
      if(obj.camera.position.y > 8) obj.camera.position.y = 8;
      sx += dx;
      sy += dy;
      render();
    } else
    {
      onMouseHover();
    }
  };

  function onMouseHover()
  {
    var label = "";
    // update the picking ray with the camera and mouse position
    obj.raycaster.setFromCamera( mouse, obj.camera );
    // calculate objects intersecting the picking ray
    var particles = obj.scene.getObjectByName('pointgroup');
    if(GL)
    {
      var intersects = obj.raycaster.intersectObject( particles, true );
      if(intersects.length > 0) {
        if(x.options.labels)
        {
          if(Array.isArray(x.options.labels)) label = x.options.labels[intersects[0].index];
          else label = x.options.labels;
        }
      }
    } else
    {
      var intersects = obj.raycaster.intersectObjects( particles.children );
      // add new labels to the points that are being hovered over now
      if ( intersects.length > 0 ) {
        label = intersects[0].object.name;
      }
    }
    document.getElementById("coordinate_label").innerHTML = label;
  }

  function render()
  { 
    obj.renderer.clear();
    obj.camera.lookAt(obj.scene.position);
    obj.renderer.render(obj.scene, obj.camera);
  }

  render();
// See the note about rendering in the globe.js widget.
}

5 ms
blog.revolutionanalytics.com accessibility score
Names and labels
These are opportunities to improve the semantics of the controls in your application. This may enhance the experience for users of assistive technology, like a screen reader.
Impact
Issue
<frame> or <iframe> elements do not have a title
Form elements do not have associated labels
Internationalization and localization
These are opportunities to improve the interpretation of your content by users in different locales.
Impact
Issue
<html> element does not have a [lang] attribute
blog.revolutionanalytics.com best practices score
Trust and Safety
Impact
Issue
Does not use HTTPS
Ensure CSP is effective against XSS attacks
User Experience
Impact
Issue
Serves images with low resolution
General
Impact
Issue
Detected JavaScript libraries
Browser errors were logged to the console
blog.revolutionanalytics.com SEO score
Content Best Practices
Format your HTML in a way that enables crawlers to better understand your app’s content.
Impact
Issue
Links do not have descriptive text
Crawling and Indexing
To appear in search results, crawlers need access to your app.
Impact
Issue
Links are not crawlable
EN
N/A
UTF-8
Language claimed in HTML meta tag should match the language actually used on the web page. Otherwise Blog.revolutionanalytics.com can be misinterpreted by Google and other search engines. Our service has detected that English is used on the page, and neither this language nor any other was claimed in <html> or <meta> tags. Our system also found out that Blog.revolutionanalytics.com main page’s claimed encoding is utf-8. Use of this encoding format is the best practice as the main page visitors from all over the world won’t have any issues with symbol transcription.
blog.revolutionanalytics.com
Open Graph description is not detected on the main page of Blog Revolutionanalytics. Lack of Open Graph description can be counter-productive for their social media presence, as such a description allows converting a website homepage (or other pages) into good-looking, rich and well-structured posts, when it is being shared on Facebook and other social media. For example, adding the following code snippet into HTML <head> tag will help to represent this web page correctly in social networks: