2023-05-22 14:10:44 +02:00
// contentLoaders = {".gltf" : () => .....} and so on
export function loadFile ( contentLoaders , multiple ) {
return ( ) => {
2023-07-04 17:22:05 +02:00
window . notify ( "if you're on Meta browser, file-uploads might be disabled" )
2023-05-22 14:10:44 +02:00
let input = document . createElement ( 'input' ) ;
input . type = 'file' ;
input . multiple = multiple ;
input . accept = Object . keys ( contentLoaders ) . join ( "," ) ;
input . onchange = ( ) => {
let files = Array . from ( input . files ) ;
let file = files . slice ? files [ 0 ] : files
for ( var i in contentLoaders ) {
let r = new RegExp ( '\\' + i + '$' )
if ( file . name . match ( r ) ) return contentLoaders [ i ] ( file )
}
alert ( file . name + " is not supported" )
} ;
input . click ( ) ;
}
}
export function setupConsole ( el ) {
if ( ! el ) return setTimeout ( ( ) => setupConsole ( $ ( '.lil-gui' ) ) , 200 )
let $console = document . createElement ( 'textarea' )
$console . style . position = 'absolute'
$console . style . display = 'block'
$console . style . zIndex = 2000 ;
$console . style . background = "transparent !important"
$console . style . pointerEvents = 'none'
$console . style . top = '70px'
$console . style . padding = '10px'
$console . style . margin = '10px'
$console . style . background = '#000'
$console . style . left = $console . style . right = $console . style . bottom = 0 ;
$console . style . color = '#A6F' ;
$console . style . fontSize = '10px' ;
$console . style . fontFamily = 'Courier'
$console . style . border = '0'
$console . innerHTML = "XRFRAGMENT CONSOLE OUTPUT:\n"
el . appendChild ( $console )
console . log = ( ( log ) => function ( ) {
let str = ( [ ... arguments ] ) . join ( " " )
2023-06-22 08:49:24 +02:00
let s = str ;
2023-05-22 14:10:44 +02:00
log ( s )
let lines = String ( $console . innerHTML + "\n" + s ) . split ( "\n" )
while ( lines . length > 200 ) lines . shift ( )
$console . innerHTML = lines . join ( "\n" )
$console . scrollTop = $console . scrollHeight ;
} ) ( console . log . bind ( console ) )
}
2023-05-23 14:57:45 +02:00
export function setupUrlBar ( el , XRF ) {
2023-10-15 13:35:54 +02:00
let inIframe = window . location !== window . parent . location
2023-10-13 11:45:17 +02:00
let ids = [ '#overlay' , 'a#embed' , 'a#source' , 'a#model' , '#qrcode' ]
2023-10-12 17:04:46 +02:00
let showButtons = ( ) => {
ids . map ( ( i ) => $ ( i ) . style . display = 'block' )
$ ( 'a#more' ) . style . display = 'none'
2023-10-15 13:35:54 +02:00
if ( inIframe ) $ ( '#uri' ) . style . display = 'block'
2023-05-22 14:10:44 +02:00
}
2023-10-12 17:58:02 +02:00
$ ( 'a#more' ) . addEventListener ( 'click' , ( ) => showButtons ( ) )
2023-10-12 17:04:46 +02:00
2023-10-13 11:45:17 +02:00
XRF . addEventListener ( 'hash' , ( ) => reflectUrl ( ) )
const reflectUrl = window . reflectUrl = ( url ) => {
el . value = url || document . location . search . substr ( 1 ) + document . location . hash
let QR = window . QR
QR . canvas = document . getElementById ( 'qrcode' )
QR . draw ( document . location . href , QR . canvas )
}
2023-10-12 17:04:46 +02:00
reflectUrl ( )
2023-05-22 14:10:44 +02:00
}
function SnackBar ( userOptions ) {
var snackbar = this || ( window . snackbar = { } ) ;
var _Interval ;
var _Message ;
var _Element ;
var _Container ;
var _OptionDefaults = {
message : "Operation performed successfully." ,
dismissible : true ,
2023-06-22 08:49:24 +02:00
timeout : 7000 ,
2023-05-22 14:10:44 +02:00
status : ""
}
var _Options = _OptionDefaults ;
function _Create ( ) {
2023-06-22 13:59:17 +02:00
let _Containers = [ ... document . querySelectorAll ( ".js-snackbar-container" ) ]
_Containers . map ( ( c ) => c . remove ( ) )
_Container = null
2023-05-22 14:10:44 +02:00
if ( ! _Container ) {
// need to create a new container for notifications
_Container = document . createElement ( "div" ) ;
_Container . classList . add ( "js-snackbar-container" ) ;
document . body . appendChild ( _Container ) ;
}
2023-05-23 17:22:00 +02:00
_Container . innerHTML = ''
2023-05-22 14:10:44 +02:00
_Element = document . createElement ( "div" ) ;
_Element . classList . add ( "js-snackbar__wrapper" ) ;
let innerSnack = document . createElement ( "div" ) ;
innerSnack . classList . add ( "js-snackbar" , "js-snackbar--show" ) ;
if ( _Options . status ) {
_Options . status = _Options . status . toLowerCase ( ) . trim ( ) ;
let status = document . createElement ( "span" ) ;
status . classList . add ( "js-snackbar__status" ) ;
if ( _Options . status === "success" || _Options . status === "green" ) {
status . classList . add ( "js-snackbar--success" ) ;
}
else if ( _Options . status === "warning" || _Options . status === "alert" || _Options . status === "orange" ) {
status . classList . add ( "js-snackbar--warning" ) ;
}
else if ( _Options . status === "danger" || _Options . status === "error" || _Options . status === "red" ) {
status . classList . add ( "js-snackbar--danger" ) ;
}
else {
status . classList . add ( "js-snackbar--info" ) ;
}
innerSnack . appendChild ( status ) ;
}
_Message = document . createElement ( "span" ) ;
_Message . classList . add ( "js-snackbar__message" ) ;
2023-07-04 15:43:15 +02:00
_Message . innerHTML = _Options . message ;
2023-05-22 14:10:44 +02:00
innerSnack . appendChild ( _Message ) ;
if ( _Options . dismissible ) {
let closeBtn = document . createElement ( "span" ) ;
closeBtn . classList . add ( "js-snackbar__close" ) ;
closeBtn . innerText = "\u00D7" ;
closeBtn . onclick = snackbar . Close ;
innerSnack . appendChild ( closeBtn ) ;
}
_Element . style . height = "0px" ;
_Element . style . opacity = "0" ;
_Element . style . marginTop = "0px" ;
_Element . style . marginBottom = "0px" ;
_Element . appendChild ( innerSnack ) ;
_Container . appendChild ( _Element ) ;
if ( _Options . timeout !== false ) {
_Interval = setTimeout ( snackbar . Close , _Options . timeout ) ;
}
}
var _ConfigureDefaults = function ( ) {
// if no options given, revert to default
if ( userOptions === undefined ) {
return ;
}
if ( userOptions . message !== undefined ) {
_Options . message = userOptions . message ;
}
if ( userOptions . dismissible !== undefined ) {
if ( typeof ( userOptions . dismissible ) === "string" ) {
_Options . dismissible = ( userOptions . dismissible === "true" ) ;
}
else if ( typeof ( userOptions . dismissible ) === "boolean" ) {
_Options . dismissible = userOptions . dismissible ;
}
else {
console . debug ( "Invalid option provided for 'dismissable' [" + userOptions . dismissible + "] is of type " + ( typeof userOptions . dismissible ) ) ;
}
}
if ( userOptions . timeout !== undefined ) {
if ( typeof ( userOptions . timeout ) === "boolean" && userOptions . timeout === false ) {
_Options . timeout = false ;
}
else if ( typeof ( userOptions . timeout ) === "string" ) {
_Options . timeout = parseInt ( userOptions . timeout ) ;
}
if ( typeof ( userOptions . timeout ) === "number" ) {
if ( userOptions . timeout === Infinity ) {
_Options . timeout = false ;
}
else if ( userOptions . timeout >= 0 ) {
_Options . timeout = userOptions . timeout ;
}
else {
console . debug ( "Invalid timeout entered. Must be greater than or equal to 0." ) ;
}
_Options . timeout = userOptions . timeout ;
}
}
if ( userOptions . status !== undefined ) {
_Options . status = userOptions . status ;
}
}
snackbar . Open = function ( ) {
let contentHeight = _Element . firstElementChild . scrollHeight ; // get the height of the content
_Element . style . height = contentHeight + "px" ;
_Element . style . opacity = 1 ;
_Element . style . marginTop = "5px" ;
_Element . style . marginBottom = "5px" ;
_Element . addEventListener ( "transitioned" , function ( ) {
_Element . removeEventListener ( "transitioned" , arguments . callee ) ;
_Element . style . height = null ;
} )
}
snackbar . Close = function ( ) {
if ( _Interval )
clearInterval ( _Interval ) ;
let snackbarHeight = _Element . scrollHeight ; // get the auto height as a px value
let snackbarTransitions = _Element . style . transition ;
_Element . style . transition = "" ;
requestAnimationFrame ( function ( ) {
_Element . style . height = snackbarHeight + "px" ; // set the auto height to the px height
_Element . style . opacity = 1 ;
_Element . style . marginTop = "0px" ;
_Element . style . marginBottom = "0px" ;
_Element . style . transition = snackbarTransitions
requestAnimationFrame ( function ( ) {
_Element . style . height = "0px" ;
_Element . style . opacity = 0 ;
} )
} ) ;
setTimeout ( function ( ) {
2023-05-23 17:22:00 +02:00
try { _Container . removeChild ( _Element ) ; } catch ( e ) { }
2023-05-22 14:10:44 +02:00
} , 1000 ) ;
} ;
_ConfigureDefaults ( ) ;
_Create ( ) ;
snackbar . Open ( ) ;
}
export function notify ( scope ) {
return function notify ( str , opts ) {
str = String ( str )
opts = opts || { }
if ( ! opts . status ) {
opts . status = "info"
if ( str . match ( /error/g ) ) opts . status = "danger"
if ( str . match ( /warning/g ) ) opts . status = "warning"
}
2023-11-08 18:28:18 +01:00
opts = Object . assign ( { message : str , status , timeout : 4000 } , opts )
2023-05-22 14:10:44 +02:00
SnackBar ( opts )
}
}
2023-07-04 15:43:15 +02:00
2023-10-15 13:35:54 +02:00
export function download ( ) {
function fetchAndDownload ( dataurl , filename ) {
var a = document . createElement ( "a" ) ;
a . href = dataurl ;
a . setAttribute ( "download" , filename ) ;
a . click ( ) ;
return false ;
}
let file = document . location . search . replace ( /\?/ , '' )
fetchAndDownload ( file , file )
}
2023-07-04 15:43:15 +02:00
export function embed ( ) {
2023-11-29 16:45:21 +01:00
// *TODO* this should be part of the XRF Threejs framework
if ( typeof THREE == 'undefined' ) THREE = xrf . THREE
let radToDeg = THREE . MathUtils . radToDeg
let toDeg = ( x ) => x / ( Math . PI / 180 )
let camera = document . querySelector ( '[camera]' ) . object3D . parent // *TODO* fix for threejs
// *TODO* add camera direction
let direction = new xrf . THREE . Vector3 ( )
camera . getWorldDirection ( direction )
const pitch = Math . asin ( direction . y ) ;
const yaw = Math . atan2 ( direction . x , direction . z ) ;
const pitchInDegrees = pitch * 180 / Math . PI ;
const yawInDegrees = yaw * 180 / Math . PI ;
2023-10-13 11:45:17 +02:00
let lastPos = ` pos= ${ camera . position . x . toFixed ( 2 ) } , ${ camera . position . y . toFixed ( 2 ) } , ${ camera . position . z . toFixed ( 2 ) } `
2023-11-29 16:45:21 +01:00
let newHash = document . location . hash . replace ( /[&]?(pos|rot)=[0-9\.-]+,[0-9\.-]+,[0-9\.-]+/ , '' )
2023-10-13 11:45:17 +02:00
newHash += ` & ${ lastPos } `
document . location . hash = newHash . replace ( /&&/ , '&' )
2023-11-29 16:45:21 +01:00
. replace ( /#&/ , '' )
2023-10-15 13:38:49 +02:00
// copy url to clipboard
var dummy = document . createElement ( 'input' ) ,
text = window . location . href ;
document . body . appendChild ( dummy ) ;
dummy . value = text ;
dummy . select ( ) ;
document . execCommand ( 'copy' ) ;
document . body . removeChild ( dummy ) ;
2023-10-13 11:45:17 +02:00
// End of *TODO*
window . notify ( ` <b>Link copied to clipboard!</b> ❤️<br>ps. to embed this experience in your website,<br>copy/paste the following into your HTML:<br><input type="text" value="<iframe src=' ${ document . location . href } '><br></iframe>" id="share"/> ` , { timeout : 10000 } )
2023-07-04 15:43:15 +02:00
}