2023-05-23 12:05:00 +02:00
// SPDX-License-Identifier: MPL-2.0
// Copyright (c) 2023 Leon van Kammen/NLNET
2023-05-09 17:42:29 +02:00
var $hx _exports = typeof exports != "undefined" ? exports : typeof window != "undefined" ? window : typeof self != "undefined" ? self : this ;
( function ( $global ) { "use strict" ;
$hx _exports [ "xrfragment" ] = $hx _exports [ "xrfragment" ] || { } ;
var EReg = function ( r , opt ) {
this . r = new RegExp ( r , opt . split ( "u" ) . join ( "" ) ) ;
} ;
EReg . _ _name _ _ = true ;
EReg . prototype = {
match : function ( s ) {
if ( this . r . global ) {
this . r . lastIndex = 0 ;
}
this . r . m = this . r . exec ( s ) ;
this . r . s = s ;
return this . r . m != null ;
}
, split : function ( s ) {
var d = "#__delim__#" ;
return s . replace ( this . r , d ) . split ( d ) ;
}
} ;
var HxOverrides = function ( ) { } ;
HxOverrides . _ _name _ _ = true ;
HxOverrides . cca = function ( s , index ) {
var x = s . charCodeAt ( index ) ;
if ( x != x ) {
return undefined ;
}
return x ;
} ;
HxOverrides . substr = function ( s , pos , len ) {
if ( len == null ) {
len = s . length ;
} else if ( len < 0 ) {
if ( pos == 0 ) {
len = s . length + len ;
} else {
return "" ;
}
}
return s . substr ( pos , len ) ;
} ;
HxOverrides . now = function ( ) {
return Date . now ( ) ;
} ;
Math . _ _name _ _ = true ;
var Reflect = function ( ) { } ;
Reflect . _ _name _ _ = true ;
Reflect . field = function ( o , field ) {
try {
return o [ field ] ;
} catch ( _g ) {
return null ;
}
} ;
Reflect . fields = function ( o ) {
var a = [ ] ;
if ( o != null ) {
var hasOwnProperty = Object . prototype . hasOwnProperty ;
for ( var f in o ) {
if ( f != "__id__" && f != "hx__closures__" && hasOwnProperty . call ( o , f ) ) {
a . push ( f ) ;
}
}
}
return a ;
} ;
Reflect . deleteField = function ( o , field ) {
if ( ! Object . prototype . hasOwnProperty . call ( o , field ) ) {
return false ;
}
delete ( o [ field ] ) ;
return true ;
} ;
var Std = function ( ) { } ;
Std . _ _name _ _ = true ;
Std . string = function ( s ) {
return js _Boot . _ _string _rec ( s , "" ) ;
} ;
Std . parseInt = function ( x ) {
if ( x != null ) {
var _g = 0 ;
var _g1 = x . length ;
while ( _g < _g1 ) {
var i = _g ++ ;
var c = x . charCodeAt ( i ) ;
if ( c <= 8 || c >= 14 && c != 32 && c != 45 ) {
var nc = x . charCodeAt ( i + 1 ) ;
var v = parseInt ( x , nc == 120 || nc == 88 ? 16 : 10 ) ;
if ( isNaN ( v ) ) {
return null ;
} else {
return v ;
}
}
}
}
return null ;
} ;
var StringTools = function ( ) { } ;
StringTools . _ _name _ _ = true ;
StringTools . isSpace = function ( s , pos ) {
var c = HxOverrides . cca ( s , pos ) ;
if ( ! ( c > 8 && c < 14 ) ) {
return c == 32 ;
} else {
return true ;
}
} ;
StringTools . ltrim = function ( s ) {
var l = s . length ;
var r = 0 ;
while ( r < l && StringTools . isSpace ( s , r ) ) ++ r ;
if ( r > 0 ) {
return HxOverrides . substr ( s , r , l - r ) ;
} else {
return s ;
}
} ;
StringTools . rtrim = function ( s ) {
var l = s . length ;
var r = 0 ;
while ( r < l && StringTools . isSpace ( s , l - r - 1 ) ) ++ r ;
if ( r > 0 ) {
return HxOverrides . substr ( s , 0 , l - r ) ;
} else {
return s ;
}
} ;
StringTools . trim = function ( s ) {
return StringTools . ltrim ( StringTools . rtrim ( s ) ) ;
} ;
StringTools . replace = function ( s , sub , by ) {
return s . split ( sub ) . join ( by ) ;
} ;
var haxe _iterators _ArrayIterator = function ( array ) {
this . current = 0 ;
this . array = array ;
} ;
haxe _iterators _ArrayIterator . _ _name _ _ = true ;
haxe _iterators _ArrayIterator . prototype = {
hasNext : function ( ) {
return this . current < this . array . length ;
}
, next : function ( ) {
return this . array [ this . current ++ ] ;
}
} ;
var js _Boot = function ( ) { } ;
js _Boot . _ _name _ _ = true ;
js _Boot . _ _string _rec = function ( o , s ) {
if ( o == null ) {
return "null" ;
}
if ( s . length >= 5 ) {
return "<...>" ;
}
var t = typeof ( o ) ;
if ( t == "function" && ( o . _ _name _ _ || o . _ _ename _ _ ) ) {
t = "object" ;
}
switch ( t ) {
case "function" :
return "<function>" ;
case "object" :
if ( ( ( o ) instanceof Array ) ) {
var str = "[" ;
s += "\t" ;
var _g = 0 ;
var _g1 = o . length ;
while ( _g < _g1 ) {
var i = _g ++ ;
str += ( i > 0 ? "," : "" ) + js _Boot . _ _string _rec ( o [ i ] , s ) ;
}
str += "]" ;
return str ;
}
var tostr ;
try {
tostr = o . toString ;
} catch ( _g ) {
return "???" ;
}
if ( tostr != null && tostr != Object . toString && typeof ( tostr ) == "function" ) {
var s2 = o . toString ( ) ;
if ( s2 != "[object Object]" ) {
return s2 ;
}
}
var str = "{\n" ;
s += "\t" ;
var hasp = o . hasOwnProperty != null ;
var k = null ;
for ( k in o ) {
if ( hasp && ! o . hasOwnProperty ( k ) ) {
continue ;
}
if ( k == "prototype" || k == "__class__" || k == "__super__" || k == "__interfaces__" || k == "__properties__" ) {
continue ;
}
if ( str . length != 2 ) {
str += ", \n" ;
}
str += s + k + " : " + js _Boot . _ _string _rec ( o [ k ] , s ) ;
}
s = s . substring ( 1 ) ;
str += "\n" + s + "}" ;
return str ;
case "string" :
return o ;
default :
return String ( o ) ;
}
} ;
var xrfragment _Parser = $hx _exports [ "xrfragment" ] [ "Parser" ] = function ( ) { } ;
xrfragment _Parser . _ _name _ _ = true ;
xrfragment _Parser . parse = function ( key , value , resultMap ) {
var Frag _h = Object . create ( null ) ;
Frag _h [ "prio" ] = xrfragment _XRF . ASSET | xrfragment _XRF . T _INT ;
Frag _h [ "#" ] = xrfragment _XRF . ASSET | xrfragment _XRF . T _PREDEFINED _VIEW ;
Frag _h [ "class" ] = xrfragment _XRF . ASSET | xrfragment _XRF . T _STRING ;
Frag _h [ "src" ] = xrfragment _XRF . ASSET | xrfragment _XRF . T _URL ;
Frag _h [ "pos" ] = xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . ROUNDROBIN | xrfragment _XRF . T _VECTOR3 | xrfragment _XRF . T _STRING _OBJ | xrfragment _XRF . EMBEDDED | xrfragment _XRF . NAVIGATOR ;
Frag _h [ "href" ] = xrfragment _XRF . ASSET | xrfragment _XRF . T _URL | xrfragment _XRF . T _PREDEFINED _VIEW ;
Frag _h [ "q" ] = xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . T _STRING | xrfragment _XRF . EMBEDDED ;
Frag _h [ "scale" ] = xrfragment _XRF . QUERY _OPERATOR | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . ROUNDROBIN | xrfragment _XRF . T _VECTOR3 | xrfragment _XRF . EMBEDDED ;
Frag _h [ "rot" ] = xrfragment _XRF . QUERY _OPERATOR | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . ROUNDROBIN | xrfragment _XRF . T _VECTOR3 | xrfragment _XRF . EMBEDDED | xrfragment _XRF . NAVIGATOR ;
Frag _h [ "translate" ] = xrfragment _XRF . QUERY _OPERATOR | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . ROUNDROBIN | xrfragment _XRF . T _VECTOR3 | xrfragment _XRF . EMBEDDED ;
Frag _h [ "visible" ] = xrfragment _XRF . QUERY _OPERATOR | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . ROUNDROBIN | xrfragment _XRF . T _INT | xrfragment _XRF . EMBEDDED ;
Frag _h [ "env" ] = xrfragment _XRF . ASSET | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . T _STRING | xrfragment _XRF . EMBEDDED ;
Frag _h [ "t" ] = xrfragment _XRF . ASSET | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . ROUNDROBIN | xrfragment _XRF . T _VECTOR2 | xrfragment _XRF . NAVIGATOR | xrfragment _XRF . EMBEDDED ;
Frag _h [ "gravity" ] = xrfragment _XRF . ASSET | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . T _VECTOR3 | xrfragment _XRF . EMBEDDED ;
Frag _h [ "physics" ] = xrfragment _XRF . ASSET | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . T _VECTOR3 | xrfragment _XRF . EMBEDDED ;
Frag _h [ "fov" ] = xrfragment _XRF . ASSET | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . T _INT | xrfragment _XRF . NAVIGATOR | xrfragment _XRF . EMBEDDED ;
Frag _h [ "clip" ] = xrfragment _XRF . ASSET | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . T _VECTOR2 | xrfragment _XRF . NAVIGATOR | xrfragment _XRF . EMBEDDED ;
Frag _h [ "fog" ] = xrfragment _XRF . ASSET | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . T _STRING | xrfragment _XRF . NAVIGATOR | xrfragment _XRF . EMBEDDED ;
Frag _h [ "namespace" ] = xrfragment _XRF . ASSET | xrfragment _XRF . T _STRING ;
Frag _h [ "SPDX" ] = xrfragment _XRF . ASSET | xrfragment _XRF . T _STRING ;
Frag _h [ "unit" ] = xrfragment _XRF . ASSET | xrfragment _XRF . T _STRING ;
Frag _h [ "description" ] = xrfragment _XRF . ASSET | xrfragment _XRF . T _STRING ;
Frag _h [ "session" ] = xrfragment _XRF . ASSET | xrfragment _XRF . T _URL | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . NAVIGATOR | xrfragment _XRF . EMBEDDED | xrfragment _XRF . PROMPT ;
if ( value . length == 0 && ! Object . prototype . hasOwnProperty . call ( Frag _h , key ) ) {
resultMap [ key ] = new xrfragment _XRF ( key , xrfragment _XRF . PV _EXECUTE | xrfragment _XRF . NAVIGATOR ) ;
return true ;
}
if ( key . split ( "." ) . length > 1 && value . split ( "." ) . length > 1 ) {
resultMap [ key ] = new xrfragment _XRF ( key , xrfragment _XRF . ASSET | xrfragment _XRF . PV _OVERRIDE | xrfragment _XRF . T _STRING | xrfragment _XRF . PROP _BIND ) ;
return true ;
}
if ( Object . prototype . hasOwnProperty . call ( Frag _h , key ) ) {
var v = new xrfragment _XRF ( key , Frag _h [ key ] ) ;
if ( ! v . validate ( value ) ) {
console . log ( "src/xrfragment/Parser.hx:75:" , "⚠ fragment '" + key + "' has incompatible value (" + value + ")" ) ;
return false ;
}
if ( xrfragment _Parser . debug ) {
console . log ( "src/xrfragment/Parser.hx:78:" , "✔ " + key + ": " + v . string ) ;
}
resultMap [ key ] = v ;
}
return true ;
} ;
var xrfragment _Query = $hx _exports [ "xrfragment" ] [ "Query" ] = function ( str ) {
this . isNumber = new EReg ( "^[0-9\\.]+$" , "" ) ;
this . isClass = new EReg ( "^[-]?class$" , "" ) ;
this . isExclude = new EReg ( "^-" , "" ) ;
this . isProp = new EReg ( "^.*:[><=!]?" , "" ) ;
this . q = { } ;
this . str = "" ;
if ( str != null ) {
this . parse ( str ) ;
}
} ;
xrfragment _Query . _ _name _ _ = true ;
xrfragment _Query . prototype = {
toObject : function ( ) {
return this . q ;
}
, expandAliases : function ( token ) {
var classAlias = new EReg ( "^(-)?\\." , "" ) ;
if ( classAlias . match ( token ) ) {
return StringTools . replace ( token , "." , "class:" ) ;
} else {
return token ;
}
}
, get : function ( ) {
return this . q ;
}
, parse : function ( str , recurse ) {
if ( recurse == null ) {
recurse = false ;
}
var _gthis = this ;
var token = str . split ( " " ) ;
var q = { } ;
var process = function ( str , prefix ) {
if ( prefix == null ) {
prefix = "" ;
}
str = StringTools . trim ( str ) ;
var k = str . split ( ":" ) [ 0 ] ;
var v = str . split ( ":" ) [ 1 ] ;
var filter = { } ;
if ( q [ prefix + k ] ) {
filter = q [ prefix + k ] ;
}
filter [ "rules" ] = filter [ "rules" ] != null ? filter [ "rules" ] : [ ] ;
if ( _gthis . isProp . match ( str ) ) {
var oper = "" ;
if ( str . indexOf ( "*" ) != - 1 ) {
oper = "*" ;
}
if ( str . indexOf ( ">" ) != - 1 ) {
oper = ">" ;
}
if ( str . indexOf ( "<" ) != - 1 ) {
oper = "<" ;
}
if ( str . indexOf ( ">=" ) != - 1 ) {
oper = ">=" ;
}
if ( str . indexOf ( "<=" ) != - 1 ) {
oper = "<=" ;
}
if ( _gthis . isExclude . match ( k ) ) {
oper = "!=" ;
k = HxOverrides . substr ( k , 1 , null ) ;
} else {
v = HxOverrides . substr ( v , oper . length , null ) ;
}
if ( oper . length == 0 ) {
oper = "=" ;
}
if ( _gthis . isClass . match ( k ) ) {
filter [ prefix + k ] = oper != "!=" ;
q [ v ] = filter ;
} else {
var rule = { } ;
if ( _gthis . isNumber . match ( v ) ) {
rule [ oper ] = parseFloat ( v ) ;
} else {
rule [ oper ] = v ;
}
filter [ "rules" ] . push ( rule ) ;
q [ k ] = filter ;
}
return ;
} else {
filter [ "id" ] = _gthis . isExclude . match ( str ) ? false : true ;
var key = _gthis . isExclude . match ( str ) ? HxOverrides . substr ( str , 1 , null ) : str ;
q [ key ] = filter ;
}
} ;
var _g = 0 ;
var _g1 = token . length ;
while ( _g < _g1 ) {
var i = _g ++ ;
process ( this . expandAliases ( token [ i ] ) ) ;
}
return this . q = q ;
}
, test : function ( obj ) {
var qualify = false ;
var _g = 0 ;
var _g1 = Reflect . fields ( obj ) ;
while ( _g < _g1 . length ) {
var k = _g1 [ _g ] ;
++ _g ;
var v = Std . string ( Reflect . field ( obj , k ) ) ;
if ( this . testProperty ( k , v ) ) {
qualify = true ;
}
}
var _g = 0 ;
var _g1 = Reflect . fields ( obj ) ;
while ( _g < _g1 . length ) {
var k = _g1 [ _g ] ;
++ _g ;
var v = Std . string ( Reflect . field ( obj , k ) ) ;
if ( this . testProperty ( k , v , true ) ) {
qualify = false ;
}
}
return qualify ;
}
, testProperty : function ( property , value , exclude ) {
var conds = 0 ;
var fails = 0 ;
var qualify = 0 ;
var testprop = function ( expr ) {
conds += 1 ;
fails += expr ? 0 : 1 ;
return expr ;
} ;
if ( this . q [ value ] != null ) {
var v = this . q [ value ] ;
if ( v [ property ] != null ) {
return v [ property ] ;
}
}
var _g = 0 ;
var _g1 = Reflect . fields ( this . q ) ;
while ( _g < _g1 . length ) {
var k = _g1 [ _g ] ;
++ _g ;
var filter = Reflect . field ( this . q , k ) ;
if ( filter . rules == null ) {
continue ;
}
var rules = filter . rules ;
var _g2 = 0 ;
while ( _g2 < rules . length ) {
var rule = rules [ _g2 ] ;
++ _g2 ;
if ( exclude ) {
if ( Reflect . field ( rule , "!=" ) != null && testprop ( ( value == null ? "null" : "" + value ) == Std . string ( Reflect . field ( rule , "!=" ) ) ) && exclude ) {
++ qualify ;
}
} else {
if ( Reflect . field ( rule , "*" ) != null && testprop ( parseFloat ( value ) != null ) ) {
++ qualify ;
}
if ( Reflect . field ( rule , ">" ) != null && testprop ( parseFloat ( value ) > parseFloat ( Reflect . field ( rule , ">" ) ) ) ) {
++ qualify ;
}
if ( Reflect . field ( rule , "<" ) != null && testprop ( parseFloat ( value ) < parseFloat ( Reflect . field ( rule , "<" ) ) ) ) {
++ qualify ;
}
if ( Reflect . field ( rule , ">=" ) != null && testprop ( parseFloat ( value ) >= parseFloat ( Reflect . field ( rule , ">=" ) ) ) ) {
++ qualify ;
}
if ( Reflect . field ( rule , "<=" ) != null && testprop ( parseFloat ( value ) <= parseFloat ( Reflect . field ( rule , "<=" ) ) ) ) {
++ qualify ;
}
if ( Reflect . field ( rule , "=" ) != null && ( testprop ( value == Reflect . field ( rule , "=" ) ) || testprop ( parseFloat ( value ) == parseFloat ( Reflect . field ( rule , "=" ) ) ) ) ) {
++ qualify ;
}
}
}
}
return qualify > 0 ;
}
} ;
var xrfragment _URI = $hx _exports [ "xrfragment" ] [ "URI" ] = function ( ) { } ;
xrfragment _URI . _ _name _ _ = true ;
xrfragment _URI . parse = function ( url , filter ) {
var store = { } ;
if ( url . indexOf ( "#" ) == - 1 ) {
return store ;
}
var fragment = url . split ( "#" ) ;
var splitArray = fragment [ 1 ] . split ( "&" ) ;
var _g = 0 ;
var _g1 = splitArray . length ;
while ( _g < _g1 ) {
var i = _g ++ ;
var splitByEqual = splitArray [ i ] . split ( "=" ) ;
var regexPlus = new EReg ( "\\+" , "g" ) ;
var key = splitByEqual [ 0 ] ;
var value = "" ;
if ( splitByEqual . length > 1 ) {
var s = regexPlus . split ( splitByEqual [ 1 ] ) . join ( " " ) ;
value = decodeURIComponent ( s . split ( "+" ) . join ( " " ) ) ;
}
var ok = xrfragment _Parser . parse ( key , value , store ) ;
}
if ( filter != null && filter != 0 ) {
var _g = 0 ;
var _g1 = Reflect . fields ( store ) ;
while ( _g < _g1 . length ) {
var key = _g1 [ _g ] ;
++ _g ;
var xrf = store [ key ] ;
if ( ! xrf . is ( filter ) ) {
Reflect . deleteField ( store , key ) ;
}
}
}
return store ;
} ;
var xrfragment _XRF = $hx _exports [ "xrfragment" ] [ "XRF" ] = function ( _fragment , _flags ) {
this . fragment = _fragment ;
this . flags = _flags ;
} ;
xrfragment _XRF . _ _name _ _ = true ;
xrfragment _XRF . set = function ( flag , flags ) {
return flags | flag ;
} ;
xrfragment _XRF . unset = function ( flag , flags ) {
return flags & ~ flag ;
} ;
xrfragment _XRF . prototype = {
is : function ( flag ) {
return ( this . flags & flag ) != 0 ;
}
, validate : function ( value ) {
this . guessType ( this , value ) ;
if ( value . split ( "|" ) . length > 1 ) {
this . args = [ ] ;
var args = value . split ( "|" ) ;
var _g = 0 ;
var _g1 = args . length ;
while ( _g < _g1 ) {
var i = _g ++ ;
var x = new xrfragment _XRF ( this . fragment , this . flags ) ;
this . guessType ( x , args [ i ] ) ;
this . args . push ( x ) ;
}
}
if ( this . fragment == "q" ) {
this . query = new xrfragment _Query ( value ) . get ( ) ;
}
var ok = true ;
if ( ! ( ( this . args ) instanceof Array ) ) {
if ( this . is ( xrfragment _XRF . T _VECTOR3 ) && ! ( typeof ( this . x ) == "number" && typeof ( this . y ) == "number" && typeof ( this . z ) == "number" ) ) {
ok = false ;
}
if ( this . is ( xrfragment _XRF . T _VECTOR2 ) && ! ( typeof ( this . x ) == "number" && typeof ( this . y ) == "number" ) ) {
ok = false ;
}
var tmp ;
if ( this . is ( xrfragment _XRF . T _INT ) ) {
var v = this . int ;
tmp = ! ( typeof ( v ) == "number" && ( ( v | 0 ) === v ) ) ;
} else {
tmp = false ;
}
if ( tmp ) {
ok = false ;
}
}
return ok ;
}
, guessType : function ( v , str ) {
v . string = str ;
if ( str . split ( "," ) . length > 1 ) {
var xyz = str . split ( "," ) ;
if ( xyz . length > 0 ) {
v . x = parseFloat ( xyz [ 0 ] ) ;
}
if ( xyz . length > 1 ) {
v . y = parseFloat ( xyz [ 1 ] ) ;
}
if ( xyz . length > 2 ) {
v . z = parseFloat ( xyz [ 2 ] ) ;
}
}
if ( xrfragment _XRF . isColor . match ( str ) ) {
v . color = str ;
}
if ( xrfragment _XRF . isFloat . match ( str ) ) {
v . float = parseFloat ( str ) ;
}
if ( xrfragment _XRF . isInt . match ( str ) ) {
v . int = Std . parseInt ( str ) ;
}
}
} ;
if ( typeof ( performance ) != "undefined" ? typeof ( performance . now ) == "function" : false ) {
HxOverrides . now = performance . now . bind ( performance ) ;
}
String . _ _name _ _ = true ;
Array . _ _name _ _ = true ;
js _Boot . _ _toStr = ( { } ) . toString ;
xrfragment _Parser . error = "" ;
xrfragment _Parser . debug = false ;
xrfragment _XRF . ASSET = 1 ;
xrfragment _XRF . PROP _BIND = 2 ;
xrfragment _XRF . QUERY _OPERATOR = 4 ;
xrfragment _XRF . PROMPT = 8 ;
xrfragment _XRF . ROUNDROBIN = 16 ;
xrfragment _XRF . NAVIGATOR = 32 ;
xrfragment _XRF . EMBEDDED = 64 ;
xrfragment _XRF . PV _OVERRIDE = 128 ;
xrfragment _XRF . PV _EXECUTE = 256 ;
xrfragment _XRF . T _COLOR = 8192 ;
xrfragment _XRF . T _INT = 16384 ;
xrfragment _XRF . T _FLOAT = 32768 ;
xrfragment _XRF . T _VECTOR2 = 65536 ;
xrfragment _XRF . T _VECTOR3 = 131072 ;
xrfragment _XRF . T _URL = 262144 ;
xrfragment _XRF . T _PREDEFINED _VIEW = 524288 ;
xrfragment _XRF . T _STRING = 1048576 ;
xrfragment _XRF . T _STRING _OBJ = 2097152 ;
xrfragment _XRF . T _STRING _OBJ _PROP = 4194304 ;
xrfragment _XRF . isColor = new EReg ( "^#([A-Fa-f0-9]{6}|[A-Fa-f0-9]{3})$" , "" ) ;
xrfragment _XRF . isInt = new EReg ( "^[0-9]+$" , "" ) ;
xrfragment _XRF . isFloat = new EReg ( "^[0-9]+\\.[0-9]+$" , "" ) ;
xrfragment _XRF . isVector = new EReg ( "([,]+|\\w)" , "" ) ;
xrfragment _XRF . isUrl = new EReg ( "(://)?\\..*" , "" ) ;
xrfragment _XRF . isUrlOrPretypedView = new EReg ( "(^#|://)?\\..*" , "" ) ;
xrfragment _XRF . isString = new EReg ( ".*" , "" ) ;
} ) ( { } ) ;
var xrfragment = $hx _exports [ "xrfragment" ] ;
2023-05-12 22:06:21 +02:00
// wrapper to survive in/outside modules
2023-05-09 17:42:29 +02:00
2023-05-12 22:06:21 +02:00
xrfragment . InteractiveGroup = function ( THREE , renderer , camera ) {
let {
Group ,
Matrix4 ,
Raycaster ,
Vector2
} = THREE
const _pointer = new Vector2 ( ) ;
const _event = { type : '' , data : _pointer } ;
class InteractiveGroup extends Group {
constructor ( renderer , camera ) {
super ( ) ;
if ( ! renderer || ! camera ) return
2023-05-22 17:18:15 +02:00
// extract camera when camera-rig is passed
camera . traverse ( ( n ) => String ( n . type ) . match ( /Camera/ ) ? camera = n : null )
2023-05-12 22:06:21 +02:00
const scope = this ;
const raycaster = new Raycaster ( ) ;
const tempMatrix = new Matrix4 ( ) ;
2023-05-17 21:31:28 +02:00
function nocollide ( ) {
if ( nocollide . tid ) return // ratelimit
_event . type = "nocollide"
scope . children . map ( ( c ) => c . dispatchEvent ( _event ) )
nocollide . tid = setTimeout ( ( ) => nocollide . tid = null , 100 )
}
2023-05-12 22:06:21 +02:00
// Pointer Events
const element = renderer . domElement ;
function onPointerEvent ( event ) {
//event.stopPropagation();
const rect = renderer . domElement . getBoundingClientRect ( ) ;
_pointer . x = ( event . clientX - rect . left ) / rect . width * 2 - 1 ;
_pointer . y = - ( event . clientY - rect . top ) / rect . height * 2 + 1 ;
raycaster . setFromCamera ( _pointer , camera ) ;
const intersects = raycaster . intersectObjects ( scope . children , false ) ;
if ( intersects . length > 0 ) {
const intersection = intersects [ 0 ] ;
const object = intersection . object ;
const uv = intersection . uv ;
_event . type = event . type ;
_event . data . set ( uv . x , 1 - uv . y ) ;
object . dispatchEvent ( _event ) ;
2023-05-17 21:31:28 +02:00
} else nocollide ( )
2023-05-12 22:06:21 +02:00
}
element . addEventListener ( 'pointerdown' , onPointerEvent ) ;
element . addEventListener ( 'pointerup' , onPointerEvent ) ;
element . addEventListener ( 'pointermove' , onPointerEvent ) ;
element . addEventListener ( 'mousedown' , onPointerEvent ) ;
element . addEventListener ( 'mouseup' , onPointerEvent ) ;
element . addEventListener ( 'mousemove' , onPointerEvent ) ;
element . addEventListener ( 'click' , onPointerEvent ) ;
// WebXR Controller Events
// TODO: Dispatch pointerevents too
const events = {
'move' : 'mousemove' ,
'select' : 'click' ,
'selectstart' : 'mousedown' ,
'selectend' : 'mouseup'
} ;
function onXRControllerEvent ( event ) {
const controller = event . target ;
tempMatrix . identity ( ) . extractRotation ( controller . matrixWorld ) ;
raycaster . ray . origin . setFromMatrixPosition ( controller . matrixWorld ) ;
raycaster . ray . direction . set ( 0 , 0 , - 1 ) . applyMatrix4 ( tempMatrix ) ;
const intersections = raycaster . intersectObjects ( scope . children , false ) ;
if ( intersections . length > 0 ) {
const intersection = intersections [ 0 ] ;
const object = intersection . object ;
const uv = intersection . uv ;
_event . type = events [ event . type ] ;
_event . data . set ( uv . x , 1 - uv . y ) ;
if ( _event . type != "mousemove" ) {
console . log ( event . type + " => " + _event . type )
}
object . dispatchEvent ( _event ) ;
2023-05-17 21:31:28 +02:00
} else nocollide ( )
2023-05-12 22:06:21 +02:00
}
const controller1 = renderer . xr . getController ( 0 ) ;
controller1 . addEventListener ( 'move' , onXRControllerEvent ) ;
controller1 . addEventListener ( 'select' , onXRControllerEvent ) ;
controller1 . addEventListener ( 'selectstart' , onXRControllerEvent ) ;
controller1 . addEventListener ( 'selectend' , onXRControllerEvent ) ;
const controller2 = renderer . xr . getController ( 1 ) ;
controller2 . addEventListener ( 'move' , onXRControllerEvent ) ;
controller2 . addEventListener ( 'select' , onXRControllerEvent ) ;
controller2 . addEventListener ( 'selectstart' , onXRControllerEvent ) ;
controller2 . addEventListener ( 'selectend' , onXRControllerEvent ) ;
}
}
return new InteractiveGroup ( renderer , camera )
}
let xrf = xrfragment
xrf . frag = { }
2023-05-17 21:31:28 +02:00
xrf . model = { }
2023-05-12 22:06:21 +02:00
xrf . init = function ( opts ) {
2023-05-09 17:42:29 +02:00
opts = opts || { }
let XRF = function ( ) {
alert ( "queries are not implemented (yet)" )
}
2023-05-12 22:06:21 +02:00
for ( let i in opts ) xrf [ i ] = opts [ i ]
for ( let i in xrf . XRF ) xrf . XRF [ i ] // shortcuts to constants (NAVIGATOR e.g.)
xrf . Parser . debug = xrf . debug
if ( opts . loaders ) Object . values ( opts . loaders ) . map ( xrf . patchLoader )
2023-05-17 21:31:28 +02:00
2023-05-12 22:06:21 +02:00
xrf . patchRenderer ( opts . renderer )
2023-05-18 12:32:57 +02:00
xrf . navigator . init ( )
2023-05-12 22:06:21 +02:00
return xrf
2023-05-09 17:42:29 +02:00
}
2023-05-12 22:06:21 +02:00
xrf . patchRenderer = function ( renderer ) {
renderer . xr . addEventListener ( 'sessionstart' , ( ) => xrf . baseReferenceSpace = renderer . xr . getReferenceSpace ( ) ) ;
renderer . xr . enabled = true ;
2023-05-10 19:12:15 +02:00
renderer . render = ( ( render ) => function ( scene , camera ) {
2023-05-12 22:06:21 +02:00
if ( xrf . model && xrf . model . render )
xrf . model . render ( scene , camera )
2023-05-10 19:12:15 +02:00
render ( scene , camera )
} ) ( renderer . render . bind ( renderer ) )
}
2023-05-12 22:06:21 +02:00
xrf . patchLoader = function ( loader ) {
2023-05-09 17:42:29 +02:00
loader . prototype . load = ( ( load ) => function ( url , onLoad , onProgress , onError ) {
load . call ( this ,
url ,
2023-05-12 22:06:21 +02:00
( model ) => { onLoad ( model ) ; xrf . parseModel ( model , url ) } ,
2023-05-09 17:42:29 +02:00
onProgress ,
onError )
} ) ( loader . prototype . load )
}
2023-05-12 22:06:21 +02:00
xrf . getFile = ( url ) => url . split ( "/" ) . pop ( ) . replace ( /#.*/ , '' )
2023-05-09 17:42:29 +02:00
2023-05-12 22:06:21 +02:00
xrf . parseModel = function ( model , url ) {
let file = xrf . getFile ( url )
2023-05-09 17:42:29 +02:00
model . file = file
model . render = function ( ) { }
2023-05-18 17:11:11 +02:00
// eval embedded XR fragments
2023-05-17 21:31:28 +02:00
model . scene . traverse ( ( mesh ) => xrf . eval . mesh ( mesh , model ) )
2023-05-09 17:42:29 +02:00
}
2023-05-12 22:06:21 +02:00
xrf . getLastModel = ( ) => xrf . model . last
2023-05-09 17:42:29 +02:00
2023-05-12 22:06:21 +02:00
xrf . eval = function ( url , model ) {
2023-05-09 17:42:29 +02:00
let notice = false
2023-05-12 22:06:21 +02:00
model = model || xrf . model
let { THREE , camera } = xrf
let frag = xrf . URI . parse ( url , xrf . XRF . NAVIGATOR )
2023-05-09 17:42:29 +02:00
let meshes = frag . q ? [ ] : [ camera ]
for ( let i in meshes ) {
for ( let k in frag ) {
let mesh = meshes [ i ]
if ( ! String ( k ) . match ( /(pos|rot)/ ) ) notice = true
2023-05-12 22:06:21 +02:00
let opts = { frag , mesh , model , camera : xrf . camera , scene : xrf . scene , renderer : xrf . renderer , THREE : xrf . THREE }
xrf . eval . fragment ( k , opts )
2023-05-09 17:42:29 +02:00
}
}
if ( notice ) alert ( "only 'pos' and 'rot' XRF.NAVIGATOR-flagged XR fragments are supported (for now)" )
}
2023-05-12 22:06:21 +02:00
xrf . eval . mesh = ( mesh , model ) => {
if ( mesh . userData ) {
let frag = { }
for ( let k in mesh . userData ) xrf . Parser . parse ( k , mesh . userData [ k ] , frag )
for ( let k in frag ) {
let opts = { frag , mesh , model , camera : xrf . camera , scene : xrf . scene , renderer : xrf . renderer , THREE : xrf . THREE }
2023-05-17 21:31:28 +02:00
mesh . userData . XRF = frag // allow fragment impl to access XRF obj already
2023-05-12 22:06:21 +02:00
xrf . eval . fragment ( k , opts )
}
}
}
xrf . eval . fragment = ( k , opts ) => {
// call native function (xrf/env.js e.g.), or pass it to user decorator
let func = xrf . frag [ k ] || function ( ) { }
if ( xrf [ k ] ) xrf [ k ] ( func , opts . frag [ k ] , opts )
else func ( opts . frag [ k ] , opts )
}
xrf . reset = ( ) => {
2023-05-17 21:31:28 +02:00
const disposeObject = ( obj ) => {
if ( obj . children . length > 0 ) obj . children . forEach ( ( child ) => disposeObject ( child ) ) ;
if ( obj . geometry ) obj . geometry . dispose ( ) ;
if ( obj . material ) {
if ( obj . material . map ) obj . material . map . dispose ( ) ;
obj . material . dispose ( ) ;
2023-05-12 22:06:21 +02:00
}
2023-05-22 15:03:23 +02:00
obj . clear ( )
obj . removeFromParent ( )
2023-05-17 21:31:28 +02:00
return true
} ;
2023-05-22 15:03:23 +02:00
let nodes = [ ]
xrf . scene . traverse ( ( child ) => child . isXRF ? nodes . push ( child ) : false )
nodes . map ( disposeObject ) // leave non-XRF objects intact
2023-05-18 12:32:57 +02:00
xrf . interactive = xrf . InteractiveGroup ( xrf . THREE , xrf . renderer , xrf . camera )
xrf . add ( xrf . interactive )
2023-05-12 22:06:21 +02:00
}
2023-05-17 21:31:28 +02:00
xrf . parseUrl = ( url ) => {
const urlObj = new URL ( url . match ( /:\/\// ) ? url : String ( ` https://fake.com/ ${ url } ` ) . replace ( /\/\// , '/' ) )
let dir = url . substring ( 0 , url . lastIndexOf ( '/' ) + 1 )
const file = urlObj . pathname . substring ( urlObj . pathname . lastIndexOf ( '/' ) + 1 ) ;
const hash = url . match ( /#/ ) ? url . replace ( /.*#/ , '' ) : ''
const ext = file . split ( '.' ) . pop ( )
return { urlObj , dir , file , hash , ext }
}
2023-05-18 12:32:57 +02:00
xrf . add = ( object ) => {
object . isXRF = true // mark for easy deletion when replacing scene
xrf . scene . add ( object )
}
2023-05-22 14:10:44 +02:00
2023-05-17 21:31:28 +02:00
xrf . navigator = { }
2023-05-12 22:06:21 +02:00
2023-05-18 12:32:57 +02:00
xrf . navigator . to = ( url , event ) => {
2023-05-18 17:11:11 +02:00
if ( ! url ) throw 'xrf.navigator.to(..) no url given'
2023-05-12 22:06:21 +02:00
return new Promise ( ( resolve , reject ) => {
2023-05-17 21:31:28 +02:00
let { urlObj , dir , file , hash , ext } = xrf . parseUrl ( url )
2023-05-18 17:11:11 +02:00
console . log ( "xrfragment: navigating to " + url )
2023-05-22 17:18:15 +02:00
if ( ! file || xrf . model . file == file ) { // we're already loaded
2023-05-18 17:11:11 +02:00
document . location . hash = ` # ${ hash } ` // just update the hash
xrf . eval ( url , xrf . model ) // and eval URI XR fragments
return resolve ( xrf . model )
}
2023-05-12 22:06:21 +02:00
if ( xrf . model && xrf . model . scene ) xrf . model . scene . visible = false
const Loader = xrf . loaders [ ext ]
if ( ! Loader ) throw 'xrfragment: no loader passed to xrfragment for extension .' + ext
2023-05-18 12:32:57 +02:00
xrf . reset ( ) // clear xrf objects from scene
2023-05-12 22:06:21 +02:00
// force relative path
if ( dir ) dir = dir [ 0 ] == '.' ? dir : ` . ${ dir } `
const loader = new Loader ( ) . setPath ( dir )
loader . load ( file , function ( model ) {
2023-05-18 12:39:47 +02:00
model . file = file
2023-05-18 12:32:57 +02:00
xrf . add ( model . scene )
2023-05-12 22:06:21 +02:00
xrf . model = model
2023-05-18 17:11:11 +02:00
xrf . eval ( url , model ) // and eval URI XR fragments
xrf . navigator . pushState ( file , hash )
2023-05-12 22:06:21 +02:00
resolve ( model )
} )
} )
}
2023-05-17 21:31:28 +02:00
xrf . navigator . init = ( ) => {
if ( xrf . navigator . init . inited ) return
2023-05-12 22:06:21 +02:00
window . addEventListener ( 'popstate' , function ( event ) {
2023-05-18 12:32:57 +02:00
xrf . navigator . to ( document . location . search . substr ( 1 ) + document . location . hash , event )
2023-05-12 22:06:21 +02:00
} )
2023-05-17 21:31:28 +02:00
xrf . navigator . init . inited = true
2023-05-12 22:06:21 +02:00
}
2023-05-18 17:11:11 +02:00
xrf . navigator . pushState = ( file , hash ) => {
2023-05-18 12:32:57 +02:00
if ( file == document . location . search . substr ( 1 ) ) return // page is in its default state
window . history . pushState ( { } , ` ${ file } # ${ hash } ` , document . location . pathname + ` ? ${ file } # ${ hash } ` )
2023-05-12 22:06:21 +02:00
}
2023-06-02 12:00:21 +02:00
/ *
* ( promise - able ) EVENTS
*
* example :
*
* xrf . addEventListener ( 'foo' , ( e ) => {
* // let promise = e.promise()
* console . log ( "navigating to: " + e . detail . destination . url )
* // promise.resolve()
* // promise.reject("not going to happen")
* } )
*
* xrf . emit ( 'foo' , 123 )
* xrf . emit ( 'foo' , 123 ) . then ( ... ) . catch ( ... ) . finally ( ... )
* /
xrf . addEventListener = function ( eventName , callback ) {
if ( ! this . _listeners ) this . _listeners = [ ]
if ( ! this . _listeners [ eventName ] ) {
// create a new array for this event name if it doesn't exist yet
this . _listeners [ eventName ] = [ ] ;
}
// add the callback to the listeners array for this event name
this . _listeners [ eventName ] . push ( callback ) ;
} ;
xrf . emit = function ( eventName , data ) {
return xrf . emit . promise ( eventName , data )
}
xrf . emit . normal = function ( eventName , data ) {
if ( ! xrf . _listeners ) xrf . _listeners = [ ]
var callbacks = xrf . _listeners [ eventName ]
if ( callbacks ) {
for ( var i = 0 ; i < callbacks . length ; i ++ ) {
callbacks [ i ] ( data ) ;
}
}
} ;
xrf . emit . promise = function ( e , opts ) {
opts . XRF = xrf // always pass root XRF obj
return new Promise ( ( resolve , reject ) => {
opts . promise = ( ) => {
opts . promise . halted = true
return { resolve , reject }
}
xrf . emit . normal ( e , opts )
if ( ! opts . promise . halted ) resolve ( )
} )
}
2023-05-12 22:06:21 +02:00
xrf . frag . env = function ( v , opts ) {
2023-05-09 17:42:29 +02:00
let { mesh , model , camera , scene , renderer , THREE } = opts
let env = mesh . getObjectByName ( v . string )
env . material . map . mapping = THREE . EquirectangularReflectionMapping ;
scene . environment = env . material . map
2023-05-12 22:06:21 +02:00
//scene.texture = env.material.map
2023-05-09 17:42:29 +02:00
renderer . toneMapping = THREE . ACESFilmicToneMapping ;
2023-05-12 22:06:21 +02:00
renderer . toneMappingExposure = 2 ;
// apply to meshes *DISABLED* renderer.environment does this
const maxAnisotropy = renderer . capabilities . getMaxAnisotropy ( ) ;
setTimeout ( ( ) => {
scene . traverse ( ( mesh ) => {
//if (mesh.material && mesh.material.map && mesh.material.metalness == 1.0) {
// mesh.material = new THREE.MeshBasicMaterial({ map: mesh.material.map });
// mesh.material.dithering = true
// mesh.material.map.anisotropy = maxAnisotropy;
// mesh.material.needsUpdate = true;
//}
//if (mesh.material && mesh.material.metalness == 1.0 ){
// mesh.material = new THREE.MeshBasicMaterial({
// color:0xffffff,
// emissive: mesh.material.map,
// envMap: env.material.map,
// side: THREE.DoubleSide,
// flatShading: true
// })
// mesh.material.needsUpdate = true
// //mesh.material.envMap = env.material.map;
// //mesh.material.envMap.intensity = 5;
// //mesh.material.needsUpdate = true;
//}
} ) ;
} , 500 )
2023-05-09 17:42:29 +02:00
console . log ( ` └ applied image ' ${ v . string } ' as environment map ` )
}
2023-05-22 15:03:23 +02:00
/ * *
2023-05-23 14:41:24 +02:00
*
2023-05-22 15:03:23 +02:00
* navigation , portals & mutations
2023-05-23 14:41:24 +02:00
*
2023-05-22 15:03:23 +02:00
* | fragment | type | scope | example value |
2023-05-23 14:41:24 +02:00
* | ` href ` | string ( uri or predefined view ) | 🔒 | ` #pos=1,1,0 ` < br > ` #pos=1,1,0&rot=90,0,0 ` < br > ` #pos=pyramid ` < br > ` #pos=lastvisit|pyramid ` < br > ` ://somefile.gltf#pos=1,1,0 ` < br > |
*
* [ [ » example implementation | https : //github.com/coderofsalvation/xrfragment/blob/main/src/3rd/three/xrf/href.js]]<br>
* [ [ » example 3 D asset | https : //github.com/coderofsalvation/xrfragment/blob/main/example/assets/href.gltf#L192]]<br>
* [ [ » discussion | https : //github.com/coderofsalvation/xrfragment/issues/1]]<br>
2023-05-22 15:03:23 +02:00
*
* [ img [ xrfragment . jpg ] ]
2023-05-23 14:41:24 +02:00
*
*
2023-05-22 15:03:23 +02:00
* ! ! ! spec 1.0
2023-05-23 14:41:24 +02:00
*
* 1. an '' external '' - or '' file URI '' fully replaces the current scene and assumes ` pos=0,0,0&rot=0,0,0 ` by default ( unless specified )
*
2023-05-22 15:03:23 +02:00
* 2. navigation should not happen when queries ( ` q= ` ) are present in local url : queries will apply ( ` pos= ` , ` rot= ` e . g . ) to the targeted object ( s ) instead .
2023-05-23 14:41:24 +02:00
*
2023-05-22 15:03:23 +02:00
* 3. navigation should not happen '' immediately '' when user is more than 2 meter away from the portal / object containing the href ( to prevent accidental navigation e . g . )
2023-05-23 14:41:24 +02:00
*
* 4. URL navigation should always be reflected in the client ( in case of javascript : see [ [ here | https : //github.com/coderofsalvation/xrfragment/blob/dev/src/3rd/three/navigator.js]] for an example navigator).
*
* 5. In XR mode , the navigator back / forward - buttons should be always visible ( using a wearable e . g . , see [ [ here | https : //github.com/coderofsalvation/xrfragment/blob/dev/example/aframe/sandbox/index.html#L26-L29]] for an example wearable)
*
* [ img [ navigation . png ] ]
*
2023-05-22 15:03:23 +02:00
* /
2023-05-12 22:06:21 +02:00
xrf . frag . href = function ( v , opts ) {
2023-05-09 17:42:29 +02:00
let { mesh , model , camera , scene , renderer , THREE } = opts
2023-05-22 14:10:44 +02:00
const world = {
pos : new THREE . Vector3 ( ) ,
scale : new THREE . Vector3 ( ) ,
quat : new THREE . Quaternion ( )
}
2023-05-17 21:31:28 +02:00
mesh . getWorldPosition ( world . pos )
mesh . getWorldScale ( world . scale )
2023-05-22 14:10:44 +02:00
mesh . getWorldQuaternion ( world . quat ) ;
2023-05-17 21:31:28 +02:00
mesh . position . copy ( world . pos )
mesh . scale . copy ( world . scale )
2023-05-22 14:10:44 +02:00
mesh . setRotationFromQuaternion ( world . quat ) ;
2023-05-17 21:31:28 +02:00
2023-05-22 14:10:44 +02:00
// detect equirectangular image
2023-05-10 19:12:15 +02:00
let texture = mesh . material . map
2023-05-17 21:31:28 +02:00
if ( texture && texture . source . data . height == texture . source . data . width / 2 ) {
texture . mapping = THREE . ClampToEdgeWrapping
texture . needsUpdate = true
2023-05-09 17:42:29 +02:00
2023-05-22 14:10:44 +02:00
// poor man's equi-portal
mesh . material = new THREE . ShaderMaterial ( {
side : THREE . DoubleSide ,
uniforms : {
pano : { value : texture } ,
selected : { value : false } ,
} ,
vertexShader : `
vec3 portalPosition ;
varying vec3 vWorldPosition ;
varying float vDistanceToCenter ;
varying float vDistance ;
void main ( ) {
vDistanceToCenter = clamp ( length ( position - vec3 ( 0.0 , 0.0 , 0.0 ) ) , 0.0 , 1.0 ) ;
portalPosition = ( modelMatrix * vec4 ( 0.0 , 0.0 , 0.0 , 1.0 ) ) . xyz ;
vDistance = length ( portalPosition - cameraPosition ) ;
vWorldPosition = ( modelMatrix * vec4 ( position , 1.0 ) ) . xyz ;
gl _Position = projectionMatrix * modelViewMatrix * vec4 ( position , 1.0 ) ;
}
` ,
fragmentShader : `
# define RECIPROCAL _PI2 0.15915494
uniform sampler2D pano ;
uniform bool selected ;
varying float vDistanceToCenter ;
varying float vDistance ;
varying vec3 vWorldPosition ;
void main ( ) {
vec3 direction = normalize ( vWorldPosition - cameraPosition ) ;
vec2 sampleUV ;
sampleUV . y = - clamp ( direction . y * 0.5 + 0.5 , 0.0 , 1.0 ) ;
sampleUV . x = atan ( direction . z , - direction . x ) * - RECIPROCAL _PI2 ;
2023-05-22 15:03:23 +02:00
sampleUV . x += 0.33 ; // adjust focus to AFRAME's a-scene.components.screenshot.capture()
2023-05-22 14:10:44 +02:00
vec4 color = texture2D ( pano , sampleUV ) ;
// Convert color to grayscale (lazy lite approach to not having to match tonemapping/shaderstacking of THREE.js)
float luminance = 0.2126 * color . r + 0.7152 * color . g + 0.0722 * color . b ;
vec4 grayscale _color = selected ? color : vec4 ( vec3 ( luminance ) + vec3 ( 0.33 ) , color . a ) ;
gl _FragColor = grayscale _color ;
}
` ,
} ) ;
mesh . material . needsUpdate = true
}
2023-05-12 22:06:21 +02:00
2023-05-17 21:31:28 +02:00
let teleport = mesh . userData . XRF . href . exec = ( e ) => {
2023-06-02 12:00:21 +02:00
xrf
. emit ( 'href' , { click : true , mesh , xrf : v } ) // let all listeners agree
. then ( ( ) => xrf . navigator . to ( v . string ) ) // ok let's surf to HREF!
2023-05-22 14:10:44 +02:00
}
let selected = ( state ) => ( ) => {
if ( mesh . selected == state ) return // nothing changed
if ( mesh . material . uniforms ) mesh . material . uniforms . selected . value = state
else mesh . material . color . r = mesh . material . color . g = mesh . material . color . b = state ? 2.0 : 1.0
// update mouse cursor
if ( ! renderer . domElement . lastCursor )
renderer . domElement . lastCursor = renderer . domElement . style . cursor
renderer . domElement . style . cursor = state ? 'pointer' : renderer . domElement . lastCursor
2023-06-02 12:00:21 +02:00
xrf
. emit ( 'href' , { selected : state , mesh , xrf : v } ) // let all listeners agree
. then ( ( ) => mesh . selected = state )
2023-05-12 22:06:21 +02:00
}
2023-05-22 14:10:44 +02:00
if ( ! opts . frag . q ) { // query means an action
2023-05-17 21:31:28 +02:00
mesh . addEventListener ( 'click' , teleport )
2023-05-22 14:10:44 +02:00
mesh . addEventListener ( 'mousemove' , selected ( true ) )
mesh . addEventListener ( 'nocollide' , selected ( false ) )
2023-05-17 21:31:28 +02:00
}
2023-05-22 15:03:23 +02:00
// lazy add mesh (because we're inside a recursive traverse)
setTimeout ( ( mesh ) => {
xrf . interactive . add ( mesh )
} , 20 , mesh )
2023-05-09 17:42:29 +02:00
}
2023-05-22 15:03:23 +02:00
/ * *
2023-05-23 14:41:24 +02:00
* > above solutions were abducted from [ [ this | https : //i.imgur.com/E3En0gJ.png]] and [[this|https://i.imgur.com/lpnTz3A.png]] survey result
2023-05-22 15:03:23 +02:00
*
2023-05-23 14:41:24 +02:00
* ! ! ! Demo
*
* < $videojs controls = "controls" aspectratio = "16:9" preload = "auto" poster = "" fluid = "fluid" class = "vjs-big-play-centered" >
* < source src = "https://coderofsalvation.github.io/xrfragment.media/href.mp4" type = "video/mp4" / >
* < / $ v i d e o j s >
*
* > capture of < a href = "./example/aframe/sandbox" target = "_blank" > aframe / sandbox < / a >
2023-05-22 15:03:23 +02:00
* /
2023-05-12 22:06:21 +02:00
xrf . frag . pos = function ( v , opts ) {
2023-05-22 15:03:23 +02:00
//if( renderer.xr.isPresenting ) return // too far away
2023-05-09 17:42:29 +02:00
let { frag , mesh , model , camera , scene , renderer , THREE } = opts
console . log ( " └ setting camera position to " + v . string )
2023-05-22 17:18:15 +02:00
if ( ! frag . q ) {
if ( true ) { //!renderer.xr.isPresenting ){
console . dir ( camera )
camera . position . x = v . x
camera . position . y = v . y
camera . position . z = v . z
}
/ *
else { // XR
let cameraWorldPosition = new THREE . Vector3 ( )
camera . object3D . getWorldPosition ( this . cameraWorldPosition )
let newRigWorldPosition = new THREE . Vector3 ( v . x , v . y , x . z )
// Finally update the cameras position
let newRigLocalPosition . copy ( this . newRigWorldPosition )
if ( camera . object3D . parent ) {
camera . object3D . parent . worldToLocal ( newRigLocalPosition )
}
camera . setAttribute ( 'position' , newRigLocalPosition )
// Also take the headset/camera rotation itself into account
if ( this . data . rotateOnTeleport ) {
this . teleportOcamerainQuaternion
. setFromEuler ( new THREE . Euler ( 0 , this . teleportOcamerain . object3D . rotation . y , 0 ) )
this . teleportOcamerainQuaternion . invert ( )
this . teleportOcamerainQuaternion . multiply ( this . hitEntityQuaternion )
// Rotate the camera based on calculated teleport ocamerain rotation
this . cameraRig . object3D . setRotationFromQuaternion ( this . teleportOcamerainQuaternion )
}
console . log ( "XR" )
const offsetPosition = { x : - v . x , y : - v . y , z : - v . z , w : 1 } ;
const offsetRotation = new THREE . Quaternion ( ) ;
const transform = new XRRigidTransform ( offsetPosition , offsetRotation ) ;
const teleportSpaceOffset = xrf . baseReferenceSpace . getOffsetReferenceSpace ( transform ) ;
renderer . xr . setReferenceSpace ( teleportSpaceOffset ) ;
}
* /
}
2023-05-09 17:42:29 +02:00
}
2023-05-12 22:06:21 +02:00
xrf . frag . q = function ( v , opts ) {
let { frag , mesh , model , camera , scene , renderer , THREE } = opts
console . log ( " └ running query " )
for ( let i in v . query ) {
let target = v . query [ i ]
// remove objects if requested
if ( target . id != undefined && ( target . mesh = scene . getObjectByName ( i ) ) ) {
target . mesh . visible = target . id
target . mesh . parent . remove ( target . mesh )
console . log ( ` └ removing mesh: ${ i } ` )
} else console . log ( ` └ mesh not found: ${ i } ` )
}
}
xrf . frag . rot = function ( v , opts ) {
2023-05-09 17:42:29 +02:00
let { mesh , model , camera , scene , renderer , THREE } = opts
2023-05-22 14:10:44 +02:00
console . log ( " └ setting camera rotation to " + v . string )
camera . rotation . set (
v . x * Math . PI / 180 ,
v . y * Math . PI / 180 ,
v . z * Math . PI / 180
)
2023-05-23 14:41:24 +02:00
camera . updateMatrixWorld ( )
2023-05-09 17:42:29 +02:00
}
2023-05-12 22:06:21 +02:00
// *TODO* use webgl instancing
xrf . frag . src = function ( v , opts ) {
2023-05-09 17:42:29 +02:00
let { mesh , model , camera , scene , renderer , THREE } = opts
2023-05-17 21:31:28 +02:00
2023-05-09 17:42:29 +02:00
if ( v . string [ 0 ] == "#" ) { // local
console . log ( " └ instancing src" )
2023-05-12 22:06:21 +02:00
let frag = xrfragment . URI . parse ( v . string )
2023-05-09 17:42:29 +02:00
// Get an instance of the original model
2023-05-17 21:31:28 +02:00
let sceneInstance = new THREE . Group ( )
sceneInstance . isSrc = true
// prevent infinite recursion #1: skip src-instanced models
for ( let i in model . scene . children ) {
let child = model . scene . children [ i ]
if ( child . isSrc ) continue ;
sceneInstance . add ( model . scene . children [ i ] . clone ( ) )
}
sceneInstance . position . copy ( mesh . position )
sceneInstance . scale . copy ( mesh . scale )
sceneInstance . updateMatrixWorld ( true ) // needed because we're going to move portals to the interactive-group
// apply embedded XR fragments
setTimeout ( ( ) => {
sceneInstance . traverse ( ( m ) => {
if ( m . userData && m . userData . src ) return ; //delete m.userData.src // prevent infinite recursion
xrf . eval . mesh ( m , { scene , recursive : true } )
} )
// apply URI XR Fragments inside src-value
for ( var i in frag ) {
xrf . eval . fragment ( i , Object . assign ( opts , { frag , model : { scene : sceneInstance } , scene : sceneInstance } ) )
}
// Add the instance to the scene
model . scene . add ( sceneInstance ) ;
2023-05-18 12:32:57 +02:00
} , 10 )
2023-05-09 17:42:29 +02:00
}
}
window . AFRAME . registerComponent ( 'xrf' , {
schema : {
} ,
init : function ( ) {
if ( ! AFRAME . XRF ) this . initXRFragments ( )
2023-05-18 17:11:11 +02:00
if ( this . data ) {
2023-05-17 21:31:28 +02:00
AFRAME . XRF . navigator . to ( this . data )
2023-05-12 22:06:21 +02:00
. then ( ( model ) => {
let gets = [ ... document . querySelectorAll ( '[xrf-get]' ) ]
2023-05-12 22:40:09 +02:00
gets . map ( ( g ) => g . emit ( 'update' ) )
2023-05-12 22:06:21 +02:00
} )
}
2023-05-09 17:42:29 +02:00
} ,
2023-05-12 22:06:21 +02:00
2023-05-09 17:42:29 +02:00
initXRFragments : function ( ) {
2023-05-17 21:31:28 +02:00
2023-05-18 12:32:57 +02:00
//window.addEventListener('popstate', clear )
//window.addEventListener('pushstate', clear )
2023-05-17 21:31:28 +02:00
2023-05-09 17:42:29 +02:00
// enable XR fragments
2023-05-17 21:31:28 +02:00
let aScene = document . querySelector ( 'a-scene' )
2023-05-09 17:42:29 +02:00
let XRF = AFRAME . XRF = xrfragment . init ( {
THREE ,
2023-05-12 22:06:21 +02:00
camera : aScene . camera ,
2023-05-09 17:42:29 +02:00
scene : aScene . object3D ,
renderer : aScene . renderer ,
debug : true ,
2023-05-12 22:06:21 +02:00
loaders : { gltf : THREE . GLTFLoader } // which 3D assets (exts) to check for XR fragments?
2023-05-09 17:42:29 +02:00
} )
2023-05-12 22:06:21 +02:00
if ( ! XRF . camera ) throw 'xrfragment: no camera detected, please declare <a-entity camera..> ABOVE entities with xrf-attributes'
2023-05-09 17:42:29 +02:00
2023-05-12 22:06:21 +02:00
// override the camera-related XR Fragments so the camera-rig is affected
let camOverride = ( xrf , v , opts ) => {
2023-05-22 18:58:05 +02:00
opts . camera = document . querySelector ( '[camera]' ) . object3D . parent
2023-05-12 22:06:21 +02:00
xrf ( v , opts )
2023-05-09 17:42:29 +02:00
}
2023-05-12 22:06:21 +02:00
XRF . pos = camOverride
2023-05-12 22:40:09 +02:00
2023-05-23 14:41:24 +02:00
// in order to set the rotation programmatically
// we need to disable look-controls
XRF . rot = ( xrf , v , opts ) => {
let { renderer } = opts ;
let look = document . querySelector ( '[look-controls]' )
if ( look ) look . removeAttribute ( "look-controls" )
camOverride ( xrf , v , opts )
}
// convert portal to a-entity so AFRAME
// raycaster can find & execute it
XRF . href = ( xrf , v , opts ) => {
camOverride ( xrf , v , opts )
2023-05-12 22:40:09 +02:00
let { mesh , camera } = opts ;
let el = document . createElement ( "a-entity" )
el . setAttribute ( "xrf-get" , mesh . name )
2023-05-22 18:58:05 +02:00
el . setAttribute ( "class" , "ray" )
2023-05-17 21:31:28 +02:00
el . addEventListener ( "click" , mesh . userData . XRF . href . exec )
2023-05-12 22:40:09 +02:00
$ ( 'a-scene' ) . appendChild ( el )
}
2023-05-18 12:32:57 +02:00
// cleanup xrf-get objects when resetting scene
XRF . reset = ( ( reset ) => ( ) => {
2023-05-22 15:03:23 +02:00
reset ( )
2023-05-18 12:32:57 +02:00
console . log ( "aframe reset" )
let els = [ ... document . querySelectorAll ( '[xrf-get]' ) ]
els . map ( ( el ) => document . querySelector ( 'a-scene' ) . removeChild ( el ) )
} ) ( XRF . reset )
2023-05-22 17:18:15 +02:00
2023-05-22 18:58:05 +02:00
// undo lookup-control shenanigans (which blocks updating camerarig position in VR)
aScene . addEventListener ( 'enter-vr' , ( ) => document . querySelector ( '[camera]' ) . object3D . parent . matrixAutoUpdate = true )
2023-05-12 22:06:21 +02:00
} ,
} )
2023-05-22 18:58:05 +02:00
window . AFRAME . registerComponent ( 'xrf-button' , {
schema : {
label : {
default : 'label'
} ,
width : {
default : 0.11
} ,
toggable : {
default : false
} ,
textSize : {
default : 0.66
} ,
color : {
default : '#111'
} ,
textColor : {
default : '#fff'
} ,
hicolor : {
default : '#555555'
} ,
action : {
default : ''
}
} ,
init : function ( ) {
var el = this . el ;
var labelEl = this . labelEl = document . createElement ( 'a-entity' ) ;
this . color = this . data . color
el . setAttribute ( 'geometry' , {
primitive : 'box' ,
width : this . data . width ,
height : 0.05 ,
depth : 0.005
} ) ;
el . setAttribute ( 'material' , {
color : this . color ,
transparent : true ,
opacity : 0.3
} ) ;
el . setAttribute ( 'pressable' , '' ) ;
labelEl . setAttribute ( 'position' , '0 0 0.01' ) ;
labelEl . setAttribute ( 'text' , {
value : this . data . label ,
color : this . data . textColor ,
align : 'center'
} ) ;
labelEl . setAttribute ( 'scale' , ` ${ this . data . textSize } ${ this . data . textSize } ${ this . data . textSize } ` ) ;
this . el . appendChild ( labelEl ) ;
this . bindMethods ( ) ;
this . el . addEventListener ( 'stateadded' , this . stateChanged ) ;
this . el . addEventListener ( 'stateremoved' , this . stateChanged ) ;
this . el . addEventListener ( 'pressedstarted' , this . onPressedStarted ) ;
this . el . addEventListener ( 'pressedended' , this . onPressedEnded ) ;
this . el . addEventListener ( 'mouseenter' , ( e ) => this . onMouseEnter ( e ) ) ;
this . el . addEventListener ( 'mouseleave' , ( e ) => this . onMouseLeave ( e ) ) ;
if ( this . data . action ) {
this . el . addEventListener ( 'click' , new Function ( this . data . action ) )
}
} ,
bindMethods : function ( ) {
this . stateChanged = this . stateChanged . bind ( this ) ;
this . onPressedStarted = this . onPressedStarted . bind ( this ) ;
this . onPressedEnded = this . onPressedEnded . bind ( this ) ;
} ,
update : function ( oldData ) {
if ( oldData . label !== this . data . label ) {
this . labelEl . setAttribute ( 'text' , 'value' , this . data . label ) ;
}
} ,
stateChanged : function ( ) {
var color = this . el . is ( 'pressed' ) ? this . data . hicolor : this . color ;
this . el . setAttribute ( 'material' , {
color : color
} ) ;
} ,
onMouseEnter : function ( ) {
this . el . setAttribute ( 'material' , { color : this . data . hicolor } ) ;
} ,
onMouseLeave : function ( ) {
this . el . setAttribute ( 'material' , { color : this . color } ) ;
} ,
onPressedStarted : function ( ) {
var el = this . el ;
el . setAttribute ( 'material' , {
color : this . data . hicolor
} ) ;
el . emit ( 'click' ) ;
if ( this . data . togabble ) {
if ( el . is ( 'pressed' ) ) {
el . removeState ( 'pressed' ) ;
} else {
el . addState ( 'pressed' ) ;
}
}
} ,
onPressedEnded : function ( ) {
if ( this . el . is ( 'pressed' ) ) {
return ;
}
this . el . setAttribute ( 'material' , {
color : this . color
} ) ;
}
} ) ;
2023-05-12 22:06:21 +02:00
window . AFRAME . registerComponent ( 'xrf-get' , {
2023-05-10 19:12:15 +02:00
schema : {
2023-05-12 22:06:21 +02:00
name : { type : 'string' } ,
2023-05-17 21:31:28 +02:00
clone : { type : 'boolean' , default : false }
2023-05-10 19:12:15 +02:00
} ,
init : function ( ) {
2023-05-12 22:06:21 +02:00
2023-05-10 19:12:15 +02:00
var el = this . el ;
2023-05-12 22:06:21 +02:00
var meshname = this . data . name || this . data ;
this . el . addEventListener ( 'update' , ( evt ) => {
2023-05-12 22:40:09 +02:00
let scene = AFRAME . XRF . scene
2023-05-12 22:06:21 +02:00
let mesh = scene . getObjectByName ( meshname ) ;
if ( ! mesh ) {
console . error ( "mesh with name '" + meshname + "' not found in model" )
2023-05-10 19:12:15 +02:00
return ;
}
2023-05-17 21:31:28 +02:00
if ( ! this . data . clone ) mesh . parent . remove ( mesh )
2023-05-12 22:06:21 +02:00
////mesh.updateMatrixWorld();
this . el . object3D . position . setFromMatrixPosition ( scene . matrixWorld ) ;
this . el . object3D . quaternion . setFromRotationMatrix ( scene . matrixWorld ) ;
2023-05-18 12:32:57 +02:00
mesh . xrf = true // mark for deletion by xrf
2023-05-17 21:31:28 +02:00
this . el . setObject3D ( 'mesh' , mesh ) ;
if ( ! this . el . id ) this . el . setAttribute ( "id" , ` xrf- ${ mesh . name } ` )
2023-05-12 22:06:21 +02:00
} )
2023-05-10 19:12:15 +02:00
2023-05-22 18:58:05 +02:00
if ( this . el . className == "ray" ) this . el . emit ( "update" )
2023-05-12 22:40:09 +02:00
2023-05-10 19:12:15 +02:00
}
2023-05-12 22:06:21 +02:00
2023-05-10 19:12:15 +02:00
} ) ;
2023-05-12 22:06:21 +02:00
2023-05-22 18:58:05 +02:00
window . AFRAME . registerComponent ( 'xrf-wear' , {
schema : {
el : { type : "selector" } ,
position : { type : "vec3" } ,
rotation : { type : "vec3" }
} ,
init : function ( ) {
$ ( 'a-scene' ) . addEventListener ( 'enter-vr' , ( e ) => this . wear ( e ) )
$ ( 'a-scene' ) . addEventListener ( 'exit-vr' , ( e ) => this . unwear ( e ) )
} ,
wear : function ( ) {
if ( ! this . wearable ) {
let d = this . data
this . wearable = new THREE . Group ( )
this . el . object3D . children . map ( ( c ) => this . wearable . add ( c ) )
this . wearable . position . set ( d . position . x , d . position . y , d . position . z )
this . wearable . rotation . set ( d . rotation . x , d . rotation . y , d . rotation . z )
}
this . data . el . object3D . add ( this . wearable )
} ,
unwear : function ( ) {
this . data . el . remove ( this . wearable )
this . wearable . children . map ( ( c ) => this . el . object3D . add ( c ) )
delete this . wearable
}
} )