3d viewer working onf f
This commit is contained in:
parent
027200f9ca
commit
68037c1b3d
20 changed files with 2862 additions and 78 deletions
|
|
@ -6,3 +6,7 @@ My STL
|
|||
|
||||
$ jekyll serve --watch
|
||||
|
||||
## Sources for the web-stuff
|
||||
|
||||
https://tonybox.net/posts/simple-stl-viewer/
|
||||
https://github.com/jhvanderschee/jekyllcodex/
|
||||
|
|
|
|||
|
|
@ -1,2 +1,2 @@
|
|||
exclude: [README.md,]
|
||||
exclude: [README.md,Makefile,licenses,rules.py,rules.mk,readfile.scad,.*,resources/gen/*.d]
|
||||
pygments: True
|
||||
|
|
|
|||
|
|
@ -11,6 +11,12 @@
|
|||
<link href='http://fonts.googleapis.com/css?family=Arvo:400,700|PT+Sans:400,700,400italic' rel='stylesheet' type='text/css'>
|
||||
<link href="{{ relative }}resources/css/bootstrap.min.css" rel="stylesheet">
|
||||
<link href="{{ relative }}resources/css/style.css" rel="stylesheet">
|
||||
<link href="{{ relative }}resources/css/lightbox.css" rel="stylesheet">
|
||||
<script src="{{ relative }}resources/js/three.min.js"></script>
|
||||
<script src="{{ relative }}resources/js/STLLoader.js"></script>
|
||||
<script src="{{ relative }}resources/js/OrbitControls.js"></script>
|
||||
<script src="{{ relative }}resources/js/stlviewer.js"></script>
|
||||
<script src="{{ relative }}resources/js/lightbox.js"></script>
|
||||
<title>{{ page.title }}</title>
|
||||
|
||||
</head>
|
||||
|
|
|
|||
|
|
@ -1 +0,0 @@
|
|||
main: main.scad
|
||||
6
index.md
6
index.md
|
|
@ -7,7 +7,11 @@ title: MySTL
|
|||
|
||||
{% for asset in site.data.assets %}
|
||||
{% for image in asset.images %}
|
||||

|
||||
{% if asset.stl %}
|
||||
<img src="{{ relative }}resources/{{ image }}" title="{{ asset.name }}" data-stl="{{ relative }}resources/{{ asset.stl }}">
|
||||
{% else %}
|
||||
<img src="{{ relative }}resources/{{ image }}" title="{{ asset.name }}">
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
{% endfor %}
|
||||
|
||||
|
|
|
|||
94
resources/css/lightbox.css
Normal file
94
resources/css/lightbox.css
Normal file
|
|
@ -0,0 +1,94 @@
|
|||
#lightbox {width: 100%; height: 100%; position: fixed; top: 0; left: 0; background: rgba(0,0,0,0.85); z-index: 9999999; line-height: 0; cursor: pointer; display: none;}
|
||||
#lightbox .img {
|
||||
position: relative;
|
||||
top: 50%;
|
||||
left: 50%;
|
||||
-ms-transform: translateX(-50%) translateY(-50%);
|
||||
-webkit-transform: translate(-50%,-50%);
|
||||
transform: translate(-50%,-50%);
|
||||
max-width: 100%;
|
||||
max-height: 100%;
|
||||
}
|
||||
#lightbox .img img {opacity: 0; pointer-events: none; width: auto;}
|
||||
@media screen and (min-width: 1200px) {
|
||||
#lightbox .img {
|
||||
max-width: 1200px;
|
||||
}
|
||||
}
|
||||
@media screen and (min-height: 1200px) {
|
||||
#lightbox .img {
|
||||
max-height: 1200px;
|
||||
}
|
||||
}
|
||||
#lightbox span {display: block; position: fixed; bottom: 13px; height: 1.5em; line-height: 1.4em; width: 100%; text-align: center; color: white; text-shadow:
|
||||
-1px -1px 0 #000,
|
||||
1px -1px 0 #000,
|
||||
-1px 1px 0 #000,
|
||||
1px 1px 0 #000;
|
||||
}
|
||||
|
||||
#lightbox span {display: none;}
|
||||
|
||||
#lightbox .videoWrapperContainer {
|
||||
position: relative;
|
||||
top: 50%;
|
||||
left: 50%;
|
||||
-ms-transform: translateX(-50%) translateY(-50%);
|
||||
-webkit-transform: translate(-50%,-50%);
|
||||
transform: translate(-50%,-50%);
|
||||
max-width: 900px;
|
||||
max-height: 100%;
|
||||
}
|
||||
#lightbox .videoWrapperContainer .videoWrapper {
|
||||
height: 0;
|
||||
line-height: 0;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
position: relative;
|
||||
padding-bottom: 56.333%; /* custom */
|
||||
background: black;
|
||||
}
|
||||
#lightbox .videoWrapper iframe {
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
border: 0;
|
||||
display: block;
|
||||
}
|
||||
#lightbox #prev, #lightbox #next {height: 50px; line-height: 36px; display: none; margin-top: -25px; position: fixed; top: 50%; padding: 0 15px; cursor: pointer; text-decoration: none; z-index: 99; color: white; font-size: 60px;}
|
||||
#lightbox.gallery #prev, #lightbox.gallery #next {display: block;}
|
||||
#lightbox #prev {left: 0;}
|
||||
#lightbox #next {right: 0;}
|
||||
#lightbox #close {height: 50px; width: 50px; position: fixed; cursor: pointer; text-decoration: none; z-index: 99; right: 0; top: 0;}
|
||||
#lightbox #close:after, #lightbox #close:before {position: absolute; margin-top: 22px; margin-left: 14px; content: ""; height: 3px; background: white; width: 23px;
|
||||
-webkit-transform-origin: 50% 50%;
|
||||
-moz-transform-origin: 50% 50%;
|
||||
-o-transform-origin: 50% 50%;
|
||||
transform-origin: 50% 50%;
|
||||
/* Safari */
|
||||
-webkit-transform: rotate(-45deg);
|
||||
/* Firefox */
|
||||
-moz-transform: rotate(-45deg);
|
||||
/* IE */
|
||||
-ms-transform: rotate(-45deg);
|
||||
/* Opera */
|
||||
-o-transform: rotate(-45deg);
|
||||
}
|
||||
#lightbox #close:after {
|
||||
/* Safari */
|
||||
-webkit-transform: rotate(45deg);
|
||||
/* Firefox */
|
||||
-moz-transform: rotate(45deg);
|
||||
/* IE */
|
||||
-ms-transform: rotate(45deg);
|
||||
/* Opera */
|
||||
-o-transform: rotate(45deg);
|
||||
}
|
||||
#lightbox, #lightbox * {
|
||||
-webkit-user-select: none;
|
||||
-moz-user-select: none;
|
||||
-ms-user-select: none;
|
||||
user-select: none;
|
||||
}
|
||||
1183
resources/js/OrbitControls.js
Normal file
1183
resources/js/OrbitControls.js
Normal file
File diff suppressed because it is too large
Load diff
393
resources/js/STLLoader.js
Normal file
393
resources/js/STLLoader.js
Normal file
|
|
@ -0,0 +1,393 @@
|
|||
/**
|
||||
* @author aleeper / http://adamleeper.com/
|
||||
* @author mrdoob / http://mrdoob.com/
|
||||
* @author gero3 / https://github.com/gero3
|
||||
* @author Mugen87 / https://github.com/Mugen87
|
||||
* @author neverhood311 / https://github.com/neverhood311
|
||||
*
|
||||
* Description: A THREE loader for STL ASCII files, as created by Solidworks and other CAD programs.
|
||||
*
|
||||
* Supports both binary and ASCII encoded files, with automatic detection of type.
|
||||
*
|
||||
* The loader returns a non-indexed buffer geometry.
|
||||
*
|
||||
* Limitations:
|
||||
* Binary decoding supports "Magics" color format (http://en.wikipedia.org/wiki/STL_(file_format)#Color_in_binary_STL).
|
||||
* There is perhaps some question as to how valid it is to always assume little-endian-ness.
|
||||
* ASCII decoding assumes file is UTF-8.
|
||||
*
|
||||
* Usage:
|
||||
* var loader = new THREE.STLLoader();
|
||||
* loader.load( './models/stl/slotted_disk.stl', function ( geometry ) {
|
||||
* scene.add( new THREE.Mesh( geometry ) );
|
||||
* });
|
||||
*
|
||||
* For binary STLs geometry might contain colors for vertices. To use it:
|
||||
* // use the same code to load STL as above
|
||||
* if (geometry.hasColors) {
|
||||
* material = new THREE.MeshPhongMaterial({ opacity: geometry.alpha, vertexColors: true });
|
||||
* } else { .... }
|
||||
* var mesh = new THREE.Mesh( geometry, material );
|
||||
*
|
||||
* For ASCII STLs containing multiple solids, each solid is assigned to a different group.
|
||||
* Groups can be used to assign a different color by defining an array of materials with the same length of
|
||||
* geometry.groups and passing it to the Mesh constructor:
|
||||
*
|
||||
* var mesh = new THREE.Mesh( geometry, material );
|
||||
*
|
||||
* For example:
|
||||
*
|
||||
* var materials = [];
|
||||
* var nGeometryGroups = geometry.groups.length;
|
||||
*
|
||||
* var colorMap = ...; // Some logic to index colors.
|
||||
*
|
||||
* for (var i = 0; i < nGeometryGroups; i++) {
|
||||
*
|
||||
* var material = new THREE.MeshPhongMaterial({
|
||||
* color: colorMap[i],
|
||||
* wireframe: false
|
||||
* });
|
||||
*
|
||||
* }
|
||||
*
|
||||
* materials.push(material);
|
||||
* var mesh = new THREE.Mesh(geometry, materials);
|
||||
*/
|
||||
|
||||
|
||||
THREE.STLLoader = function ( manager ) {
|
||||
|
||||
THREE.Loader.call( this, manager );
|
||||
|
||||
};
|
||||
|
||||
THREE.STLLoader.prototype = Object.assign( Object.create( THREE.Loader.prototype ), {
|
||||
|
||||
constructor: THREE.STLLoader,
|
||||
|
||||
load: function ( url, onLoad, onProgress, onError ) {
|
||||
|
||||
var scope = this;
|
||||
|
||||
var loader = new THREE.FileLoader( scope.manager );
|
||||
loader.setPath( scope.path );
|
||||
loader.setResponseType( 'arraybuffer' );
|
||||
loader.load( url, function ( text ) {
|
||||
|
||||
try {
|
||||
|
||||
onLoad( scope.parse( text ) );
|
||||
|
||||
} catch ( e ) {
|
||||
|
||||
if ( onError ) {
|
||||
|
||||
onError( e );
|
||||
|
||||
} else {
|
||||
|
||||
console.error( e );
|
||||
|
||||
}
|
||||
|
||||
scope.manager.itemError( url );
|
||||
|
||||
}
|
||||
|
||||
}, onProgress, onError );
|
||||
|
||||
},
|
||||
|
||||
parse: function ( data ) {
|
||||
|
||||
function isBinary( data ) {
|
||||
|
||||
var expect, face_size, n_faces, reader;
|
||||
reader = new DataView( data );
|
||||
face_size = ( 32 / 8 * 3 ) + ( ( 32 / 8 * 3 ) * 3 ) + ( 16 / 8 );
|
||||
n_faces = reader.getUint32( 80, true );
|
||||
expect = 80 + ( 32 / 8 ) + ( n_faces * face_size );
|
||||
|
||||
if ( expect === reader.byteLength ) {
|
||||
|
||||
return true;
|
||||
|
||||
}
|
||||
|
||||
// An ASCII STL data must begin with 'solid ' as the first six bytes.
|
||||
// However, ASCII STLs lacking the SPACE after the 'd' are known to be
|
||||
// plentiful. So, check the first 5 bytes for 'solid'.
|
||||
|
||||
// Several encodings, such as UTF-8, precede the text with up to 5 bytes:
|
||||
// https://en.wikipedia.org/wiki/Byte_order_mark#Byte_order_marks_by_encoding
|
||||
// Search for "solid" to start anywhere after those prefixes.
|
||||
|
||||
// US-ASCII ordinal values for 's', 'o', 'l', 'i', 'd'
|
||||
|
||||
var solid = [ 115, 111, 108, 105, 100 ];
|
||||
|
||||
for ( var off = 0; off < 5; off ++ ) {
|
||||
|
||||
// If "solid" text is matched to the current offset, declare it to be an ASCII STL.
|
||||
|
||||
if ( matchDataViewAt( solid, reader, off ) ) return false;
|
||||
|
||||
}
|
||||
|
||||
// Couldn't find "solid" text at the beginning; it is binary STL.
|
||||
|
||||
return true;
|
||||
|
||||
}
|
||||
|
||||
function matchDataViewAt( query, reader, offset ) {
|
||||
|
||||
// Check if each byte in query matches the corresponding byte from the current offset
|
||||
|
||||
for ( var i = 0, il = query.length; i < il; i ++ ) {
|
||||
|
||||
if ( query[ i ] !== reader.getUint8( offset + i, false ) ) return false;
|
||||
|
||||
}
|
||||
|
||||
return true;
|
||||
|
||||
}
|
||||
|
||||
function parseBinary( data ) {
|
||||
|
||||
var reader = new DataView( data );
|
||||
var faces = reader.getUint32( 80, true );
|
||||
|
||||
var r, g, b, hasColors = false, colors;
|
||||
var defaultR, defaultG, defaultB, alpha;
|
||||
|
||||
// process STL header
|
||||
// check for default color in header ("COLOR=rgba" sequence).
|
||||
|
||||
for ( var index = 0; index < 80 - 10; index ++ ) {
|
||||
|
||||
if ( ( reader.getUint32( index, false ) == 0x434F4C4F /*COLO*/ ) &&
|
||||
( reader.getUint8( index + 4 ) == 0x52 /*'R'*/ ) &&
|
||||
( reader.getUint8( index + 5 ) == 0x3D /*'='*/ ) ) {
|
||||
|
||||
hasColors = true;
|
||||
colors = new Float32Array( faces * 3 * 3 );
|
||||
|
||||
defaultR = reader.getUint8( index + 6 ) / 255;
|
||||
defaultG = reader.getUint8( index + 7 ) / 255;
|
||||
defaultB = reader.getUint8( index + 8 ) / 255;
|
||||
alpha = reader.getUint8( index + 9 ) / 255;
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
var dataOffset = 84;
|
||||
var faceLength = 12 * 4 + 2;
|
||||
|
||||
var geometry = new THREE.BufferGeometry();
|
||||
|
||||
var vertices = new Float32Array( faces * 3 * 3 );
|
||||
var normals = new Float32Array( faces * 3 * 3 );
|
||||
|
||||
for ( var face = 0; face < faces; face ++ ) {
|
||||
|
||||
var start = dataOffset + face * faceLength;
|
||||
var normalX = reader.getFloat32( start, true );
|
||||
var normalY = reader.getFloat32( start + 4, true );
|
||||
var normalZ = reader.getFloat32( start + 8, true );
|
||||
|
||||
if ( hasColors ) {
|
||||
|
||||
var packedColor = reader.getUint16( start + 48, true );
|
||||
|
||||
if ( ( packedColor & 0x8000 ) === 0 ) {
|
||||
|
||||
// facet has its own unique color
|
||||
|
||||
r = ( packedColor & 0x1F ) / 31;
|
||||
g = ( ( packedColor >> 5 ) & 0x1F ) / 31;
|
||||
b = ( ( packedColor >> 10 ) & 0x1F ) / 31;
|
||||
|
||||
} else {
|
||||
|
||||
r = defaultR;
|
||||
g = defaultG;
|
||||
b = defaultB;
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
for ( var i = 1; i <= 3; i ++ ) {
|
||||
|
||||
var vertexstart = start + i * 12;
|
||||
var componentIdx = ( face * 3 * 3 ) + ( ( i - 1 ) * 3 );
|
||||
|
||||
vertices[ componentIdx ] = reader.getFloat32( vertexstart, true );
|
||||
vertices[ componentIdx + 1 ] = reader.getFloat32( vertexstart + 4, true );
|
||||
vertices[ componentIdx + 2 ] = reader.getFloat32( vertexstart + 8, true );
|
||||
|
||||
normals[ componentIdx ] = normalX;
|
||||
normals[ componentIdx + 1 ] = normalY;
|
||||
normals[ componentIdx + 2 ] = normalZ;
|
||||
|
||||
if ( hasColors ) {
|
||||
|
||||
colors[ componentIdx ] = r;
|
||||
colors[ componentIdx + 1 ] = g;
|
||||
colors[ componentIdx + 2 ] = b;
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
geometry.setAttribute( 'position', new THREE.BufferAttribute( vertices, 3 ) );
|
||||
geometry.setAttribute( 'normal', new THREE.BufferAttribute( normals, 3 ) );
|
||||
|
||||
if ( hasColors ) {
|
||||
|
||||
geometry.setAttribute( 'color', new THREE.BufferAttribute( colors, 3 ) );
|
||||
geometry.hasColors = true;
|
||||
geometry.alpha = alpha;
|
||||
|
||||
}
|
||||
|
||||
return geometry;
|
||||
|
||||
}
|
||||
|
||||
function parseASCII( data ) {
|
||||
|
||||
var geometry = new THREE.BufferGeometry();
|
||||
var patternSolid = /solid([\s\S]*?)endsolid/g;
|
||||
var patternFace = /facet([\s\S]*?)endfacet/g;
|
||||
var faceCounter = 0;
|
||||
|
||||
var patternFloat = /[\s]+([+-]?(?:\d*)(?:\.\d*)?(?:[eE][+-]?\d+)?)/.source;
|
||||
var patternVertex = new RegExp( 'vertex' + patternFloat + patternFloat + patternFloat, 'g' );
|
||||
var patternNormal = new RegExp( 'normal' + patternFloat + patternFloat + patternFloat, 'g' );
|
||||
|
||||
var vertices = [];
|
||||
var normals = [];
|
||||
|
||||
var normal = new THREE.Vector3();
|
||||
|
||||
var result;
|
||||
|
||||
var groupCount = 0;
|
||||
var startVertex = 0;
|
||||
var endVertex = 0;
|
||||
|
||||
while ( ( result = patternSolid.exec( data ) ) !== null ) {
|
||||
|
||||
startVertex = endVertex;
|
||||
|
||||
var solid = result[ 0 ];
|
||||
|
||||
while ( ( result = patternFace.exec( solid ) ) !== null ) {
|
||||
|
||||
var vertexCountPerFace = 0;
|
||||
var normalCountPerFace = 0;
|
||||
|
||||
var text = result[ 0 ];
|
||||
|
||||
while ( ( result = patternNormal.exec( text ) ) !== null ) {
|
||||
|
||||
normal.x = parseFloat( result[ 1 ] );
|
||||
normal.y = parseFloat( result[ 2 ] );
|
||||
normal.z = parseFloat( result[ 3 ] );
|
||||
normalCountPerFace ++;
|
||||
|
||||
}
|
||||
|
||||
while ( ( result = patternVertex.exec( text ) ) !== null ) {
|
||||
|
||||
vertices.push( parseFloat( result[ 1 ] ), parseFloat( result[ 2 ] ), parseFloat( result[ 3 ] ) );
|
||||
normals.push( normal.x, normal.y, normal.z );
|
||||
vertexCountPerFace ++;
|
||||
endVertex ++;
|
||||
|
||||
}
|
||||
|
||||
// every face have to own ONE valid normal
|
||||
|
||||
if ( normalCountPerFace !== 1 ) {
|
||||
|
||||
console.error( 'THREE.STLLoader: Something isn\'t right with the normal of face number ' + faceCounter );
|
||||
|
||||
}
|
||||
|
||||
// each face have to own THREE valid vertices
|
||||
|
||||
if ( vertexCountPerFace !== 3 ) {
|
||||
|
||||
console.error( 'THREE.STLLoader: Something isn\'t right with the vertices of face number ' + faceCounter );
|
||||
|
||||
}
|
||||
|
||||
faceCounter ++;
|
||||
|
||||
}
|
||||
|
||||
var start = startVertex;
|
||||
var count = endVertex - startVertex;
|
||||
|
||||
geometry.addGroup( start, count, groupCount );
|
||||
groupCount ++;
|
||||
|
||||
}
|
||||
|
||||
geometry.setAttribute( 'position', new THREE.Float32BufferAttribute( vertices, 3 ) );
|
||||
geometry.setAttribute( 'normal', new THREE.Float32BufferAttribute( normals, 3 ) );
|
||||
|
||||
return geometry;
|
||||
|
||||
}
|
||||
|
||||
function ensureString( buffer ) {
|
||||
|
||||
if ( typeof buffer !== 'string' ) {
|
||||
|
||||
return THREE.LoaderUtils.decodeText( new Uint8Array( buffer ) );
|
||||
|
||||
}
|
||||
|
||||
return buffer;
|
||||
|
||||
}
|
||||
|
||||
function ensureBinary( buffer ) {
|
||||
|
||||
if ( typeof buffer === 'string' ) {
|
||||
|
||||
var array_buffer = new Uint8Array( buffer.length );
|
||||
for ( var i = 0; i < buffer.length; i ++ ) {
|
||||
|
||||
array_buffer[ i ] = buffer.charCodeAt( i ) & 0xff; // implicitly assumes little-endian
|
||||
|
||||
}
|
||||
|
||||
return array_buffer.buffer || array_buffer;
|
||||
|
||||
} else {
|
||||
|
||||
return buffer;
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// start
|
||||
|
||||
var binData = ensureBinary( data );
|
||||
|
||||
return isBinary( binData ) ? parseBinary( binData ) : parseASCII( ensureString( data ) );
|
||||
|
||||
}
|
||||
|
||||
} );
|
||||
|
|
@ -1,42 +1,3 @@
|
|||
function is_youtubelink(url) {
|
||||
var p = /^(?:https?:\/\/)?(?:www\.)?(?:youtu\.be\/|youtube\.com\/(?:embed\/|v\/|watch\?v=|watch\?.+&v=))((\w|-){11})(?:\S+)?$/;
|
||||
return (url.match(p)) ? RegExp.$1 : false;
|
||||
}
|
||||
function is_imagelink(url) {
|
||||
var p = /([a-z\-_0-9\/\:\.]*\.(jpg|jpeg|png|gif))/i;
|
||||
return (url.match(p)) ? true : false;
|
||||
}
|
||||
function is_vimeolink(url,el) {
|
||||
var id = false;
|
||||
var xmlhttp = new XMLHttpRequest();
|
||||
xmlhttp.onreadystatechange = function() {
|
||||
if (xmlhttp.readyState == XMLHttpRequest.DONE) { // XMLHttpRequest.DONE == 4
|
||||
if (xmlhttp.status == 200) {
|
||||
var response = JSON.parse(xmlhttp.responseText);
|
||||
id = response.video_id;
|
||||
console.log(id);
|
||||
el.classList.add('lightbox-vimeo');
|
||||
el.setAttribute('data-id',id);
|
||||
|
||||
el.addEventListener("click", function(event) {
|
||||
event.preventDefault();
|
||||
document.getElementById('lightbox').innerHTML = '<a id="close"></a><a id="next">›</a><a id="prev">‹</a><div class="videoWrapperContainer"><div class="videoWrapper"><iframe src="https://player.vimeo.com/video/'+el.getAttribute('data-id')+'/?autoplay=1&byline=0&title=0&portrait=0" webkitallowfullscreen mozallowfullscreen allowfullscreen></iframe></div></div>';
|
||||
document.getElementById('lightbox').style.display = 'block';
|
||||
|
||||
setGallery(this);
|
||||
});
|
||||
}
|
||||
else if (xmlhttp.status == 400) {
|
||||
alert('There was an error 400');
|
||||
}
|
||||
else {
|
||||
alert('something else other than 200 was returned');
|
||||
}
|
||||
}
|
||||
};
|
||||
xmlhttp.open("GET", 'https://vimeo.com/api/oembed.json?url='+url, true);
|
||||
xmlhttp.send();
|
||||
}
|
||||
function setGallery(el) {
|
||||
var elements = document.body.querySelectorAll(".gallery");
|
||||
elements.forEach(element => {
|
||||
|
|
@ -80,62 +41,48 @@ document.addEventListener("DOMContentLoaded", function() {
|
|||
|
||||
//create lightbox div in the footer
|
||||
var newdiv = document.createElement("div");
|
||||
newdiv.tabIndex = 0;
|
||||
newdiv.setAttribute('id',"lightbox");
|
||||
document.body.appendChild(newdiv);
|
||||
|
||||
//add classes to links to be able to initiate lightboxes
|
||||
var elements = document.querySelectorAll('a');
|
||||
var elements = document.querySelectorAll('img');
|
||||
elements.forEach(element => {
|
||||
var url = element.getAttribute('href');
|
||||
var url = element.getAttribute('data-stl');
|
||||
if(url) {
|
||||
if(url.indexOf('vimeo') !== -1 && !element.classList.contains('no-lightbox')) {
|
||||
is_vimeolink(url,element);
|
||||
}
|
||||
if(is_youtubelink(url) && !element.classList.contains('no-lightbox')) {
|
||||
element.classList.add('lightbox-youtube');
|
||||
element.setAttribute('data-id',is_youtubelink(url));
|
||||
}
|
||||
if(is_imagelink(url) && !element.classList.contains('no-lightbox')) {
|
||||
element.classList.add('lightbox-image');
|
||||
var href = element.getAttribute('href');
|
||||
var filename = href.split('/').pop();
|
||||
var split = filename.split(".");
|
||||
var name = split[0];
|
||||
element.setAttribute('title',name);
|
||||
}
|
||||
var stl = element.getAttribute('data-stl');
|
||||
element.classList.add('lightbox-image-stl');
|
||||
}
|
||||
});
|
||||
|
||||
//remove the clicked lightbox
|
||||
document.getElementById('lightbox').addEventListener("click", function(event) {
|
||||
if(event.target.id != 'next' && event.target.id != 'prev'){
|
||||
if(event.target.id == 'close'){
|
||||
this.innerHTML = '';
|
||||
document.getElementById('lightbox').style.display = 'none';
|
||||
}
|
||||
});
|
||||
|
||||
//add the youtube lightbox on click
|
||||
var elements = document.querySelectorAll('a.lightbox-youtube');
|
||||
// hit esc to dismiss lightbox
|
||||
document.getElementById('lightbox').addEventListener("keydown", function(event) {
|
||||
if(event.key == 'Escape'){
|
||||
this.innerHTML = '';
|
||||
document.getElementById('lightbox').style.display = 'none';
|
||||
}
|
||||
});
|
||||
|
||||
//add the stl lightbox on click
|
||||
var elements = document.querySelectorAll('img.lightbox-image-stl');
|
||||
elements.forEach(element => {
|
||||
element.addEventListener("click", function(event) {
|
||||
event.preventDefault();
|
||||
document.getElementById('lightbox').innerHTML = '<a id="close"></a><a id="next">›</a><a id="prev">‹</a><div class="videoWrapperContainer"><div class="videoWrapper"><iframe src="https://www.youtube.com/embed/'+this.getAttribute('data-id')+'?autoplay=1&showinfo=0&rel=0"></iframe></div>';
|
||||
document.getElementById('lightbox').innerHTML = '<a id="close"></a><a id="next">›</a><a id="prev">‹</a><div id="model" class="img" style="width: 100%; height:100%;" title="'+this.getAttribute('title')+'" ><</div><span>'+this.getAttribute('title')+'</span>';
|
||||
document.getElementById('lightbox').style.display = 'block';
|
||||
|
||||
document.getElementById('model').setAttribute('data-rotate', 'x');
|
||||
document.getElementById('model').setAttribute('data-zdistance', '4');
|
||||
STLViewer(document.getElementById("model"), this.getAttribute('data-stl'));
|
||||
setGallery(this);
|
||||
document.getElementById('lightbox').focus();
|
||||
});
|
||||
});
|
||||
|
||||
//add the image lightbox on click
|
||||
var elements = document.querySelectorAll('a.lightbox-image');
|
||||
elements.forEach(element => {
|
||||
element.addEventListener("click", function(event) {
|
||||
event.preventDefault();
|
||||
document.getElementById('lightbox').innerHTML = '<a id="close"></a><a id="next">›</a><a id="prev">‹</a><div class="img" style="background: url(\''+this.getAttribute('href')+'\') center center / contain no-repeat;" title="'+this.getAttribute('title')+'" ><img src="'+this.getAttribute('href')+'" alt="'+this.getAttribute('title')+'" /></div><span>'+this.getAttribute('title')+'</span>';
|
||||
document.getElementById('lightbox').style.display = 'block';
|
||||
|
||||
setGallery(this);
|
||||
});
|
||||
});
|
||||
|
||||
});
|
||||
});
|
||||
|
|
|
|||
86
resources/js/stlviewer.js
Normal file
86
resources/js/stlviewer.js
Normal file
|
|
@ -0,0 +1,86 @@
|
|||
|
||||
function STLViewerEnable(classname) {
|
||||
var models = document.getElementsByClassName(classname);
|
||||
for (var i = 0; i < models.length; i++) {
|
||||
STLViewer(models[i], models[i].getAttribute("data-src"));
|
||||
}
|
||||
}
|
||||
|
||||
function STLViewer(elem, model) {
|
||||
|
||||
var renderer = new THREE.WebGLRenderer({ antialias: true, alpha: true });
|
||||
var camera = new THREE.PerspectiveCamera(50, elem.clientWidth / elem.clientHeight, 1, 1000);
|
||||
|
||||
renderer.setSize(elem.clientWidth, elem.clientHeight);
|
||||
elem.appendChild(renderer.domElement);
|
||||
|
||||
var controls = new THREE.OrbitControls(camera, renderer.domElement);
|
||||
controls.enableDamping = true;
|
||||
controls.rotateSpeed = 0.05;
|
||||
controls.dampingFactor = 0.1;
|
||||
controls.enableZoom = false;
|
||||
controls.enablePan = false;
|
||||
controls.autoRotate = true;
|
||||
controls.autoRotateSpeed = 0.75;
|
||||
|
||||
var scene = new THREE.Scene();
|
||||
|
||||
scene.add(new THREE.HemisphereLight(0xffffff, 0x080820, 1.5));
|
||||
|
||||
(new THREE.STLLoader()).load(model, function (geometry) {
|
||||
|
||||
// Determine the color
|
||||
var colorString = elem.getAttribute("data-color")
|
||||
if(colorString != null) { var color = new THREE.Color(colorString); }
|
||||
else { var color = 0xff5533 }
|
||||
|
||||
// Set up the material
|
||||
var material = new THREE.MeshPhongMaterial({ color: color, specular: 100, shininess: 100 });
|
||||
var mesh = new THREE.Mesh(geometry, material);
|
||||
scene.add(mesh);
|
||||
|
||||
// Compute the middle
|
||||
var middle = new THREE.Vector3();
|
||||
geometry.computeBoundingBox();
|
||||
geometry.boundingBox.getCenter(middle);
|
||||
|
||||
// Center it
|
||||
mesh.geometry.applyMatrix(new THREE.Matrix4().makeTranslation( -middle.x, -middle.y, -middle.z ) );
|
||||
|
||||
// Rotate, if desired
|
||||
if(elem.getAttribute("data-rotate") == "x")
|
||||
mesh.rotation.x = -Math.PI/3
|
||||
|
||||
// Pull the camera away as needed
|
||||
var largestDimension = Math.max(geometry.boundingBox.max.x - geometry.boundingBox.min.x,
|
||||
geometry.boundingBox.max.y - geometry.boundingBox.min.y,
|
||||
geometry.boundingBox.max.z - geometry.boundingBox.min.z)
|
||||
camera.position.z = largestDimension * elem.getAttribute("data-zdistance");
|
||||
|
||||
|
||||
var animate = function () {
|
||||
requestAnimationFrame(animate);
|
||||
controls.update();
|
||||
renderer.render(scene, camera);
|
||||
}; animate();
|
||||
|
||||
|
||||
const observer = new MutationObserver(function(mutationList, observer) {
|
||||
mutationList.forEach(element => {
|
||||
if(element.removedNodes.length > 0) {
|
||||
renderer.dispose();
|
||||
geometry.dispose();
|
||||
material.dispose();
|
||||
observer.disconnect();
|
||||
}
|
||||
})
|
||||
});
|
||||
observer.observe(elem.parentElement, {childList: true});
|
||||
|
||||
window.addEventListener('resize', function () {
|
||||
renderer.setSize(elem.clientWidth, elem.clientHeight);
|
||||
camera.aspect = elem.clientWidth / elem.clientHeight;
|
||||
camera.updateProjectionMatrix();
|
||||
}, false);
|
||||
});
|
||||
}
|
||||
1068
resources/js/three.min.js
vendored
Normal file
1068
resources/js/three.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
Loading…
Reference in a new issue