From bc25f7fdf187d3dcbaf8d01b5f9a8348270823d4 Mon Sep 17 00:00:00 2001 From: Leon van Kammen Date: Fri, 16 Feb 2024 16:34:32 +0000 Subject: [PATCH] bugfix: better nav --- doc/RF6_XR_Fragments.png | Bin 27686 -> 27692 bytes doc/RFC_XR_Fragments.html | 25 +- doc/RFC_XR_Fragments.md | 2 + doc/RFC_XR_Fragments.txt | 752 +++++++++---------- doc/RFC_XR_Fragments.xml | 167 ++--- doc/RFC_XR_Macros.txt | 24 +- doc/RFC_XR_Macros.xml | 6 +- src/3rd/js/aframe/index.js | 1 + src/3rd/js/pubsub.js | 10 +- src/3rd/js/three/util/optimize.js | 4 +- src/3rd/js/three/xrf/#.js | 7 +- src/3rd/js/three/xrf/dynamic/filter.js | 2 +- src/3rd/js/three/xrf/src/fbx.js | 27 + test/generated/test.js | 861 +++++++++++++++++++-- test/generated/test.py | 998 ++++++++++++++++++++++--- 15 files changed, 2239 insertions(+), 647 deletions(-) create mode 100644 src/3rd/js/three/xrf/src/fbx.js diff --git a/doc/RF6_XR_Fragments.png b/doc/RF6_XR_Fragments.png index 128e19e526f43496a374e1937b2d84ca28306a7e..d8b8f8919ec153e6a44a50b67210c507e23567c7 100644 GIT binary patch literal 27692 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 LO{PZ54E%opn*JYt delta 27349 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 W3C Media Fragments media fragment -#t=0,2 #xywh -play/loop 3D animation from 0 seconds till 2 seconds +#t=0,2&loop +play (and loop) 3D animation from 0 seconds till 2 seconds @@ -448,6 +448,14 @@ That way, if the link gets shared, the XR Fragments implementation at http #foo=bar sets URI Template variable foo to the value #t=0 from existing object metadata (bar:#t=0 e.g.), This allows for reactive URI Template defined in object metadata elsewhere (src:://m.com/cat.mp4#{foo} e.g., to play media using media fragment URI). NOTE: metadata-key should not start with # + + +ANIMATION +#<tag_or_objectname>=<animationname> +string=string +#people=walk #people=noanim +assign a different animation to object(s) + @@ -546,13 +554,14 @@ That way, if the link gets shared, the XR Fragments implementation at http -media parameter (shader uniform) -u:= -float -vec2 + + ++0.5,+0.5 +scroll instantly by adding 0.5 to the current uv coordinates +

| media parameter (shader uniform) | u:= | u:color=1,0,0 | set shader uniform value |

* = this is extending the W3C media fragments with (missing) playback/viewport-control. Normally #t=0,2 implies setting start/stop-values AND starting playback, whereas #s=0&loop allows pausing a video, speeding up/slowing down media, as well as enabling/disabling looping.

@@ -666,7 +675,7 @@ For example, to render a portal with a preview-version of the scene, create an 3
  • the Y-coordinate of pos identifies the floorposition. This means that desktop-projections usually need to add 1.5m (average person height) on top (which is done automatically by VR/AR headsets).
  • set the position of the camera accordingly to the vector3 values of #pos
  • rot sets the rotation of the camera (only for non-VR/AR headsets)
  • -
  • t sets the playbackspeed and animation-range of the current scene animation(s) or src-mediacontent (video/audioframes e.g., use t=0,7,7 to ‘STOP’ at frame 7 e.g.)
  • +
  • t in the top-URL sets the playbackspeed and animation-range of the global scene animation
  • after scene load: in case an href does not mention any pos-coordinate, pos=0,0,0 will be assumed
  • @@ -825,6 +834,8 @@ Resizing will be happen accordingly to its placeholder object aquariumcube
  • ignore previous rule in special cases, like clicking an href using camera-portal collision (the back-button would cause a teleport-loop)

  • href-events should bubble upward the node-tree

  • + +
  • the end-user navigator back/forward buttons should repeat a back/forward action until a pos=... primitive is found (the inbetween interaction URI’s are only for UX research purposes)

  • » example implementation
    diff --git a/doc/RFC_XR_Fragments.md b/doc/RFC_XR_Fragments.md index cf15481..c435bf8 100644 --- a/doc/RFC_XR_Fragments.md +++ b/doc/RFC_XR_Fragments.md @@ -278,6 +278,8 @@ These are automatic fragment-to-metadata mappings, which only trigger if the 3D | | | +0.5,+0.5 | scroll instantly by adding 0.5 to the current uv coordinates | | | | 0.2,1,0.1,0.1 | scroll (lerp) to uv coordinate `0,2,1` with `0.1` units per second | | | | 0,0,0,+0.1 | scroll v coordinates with `0.1` units per second (infinitely) | +| | | +0.5,+0.5 | scroll instantly by adding 0.5 to the current uv coordinates | + | media parameter (shader uniform) | u:= | u:color=1,0,0 | set shader uniform value | > \* = this is extending the [W3C media fragments](https://www.w3.org/TR/media-frags/#mf-advanced) with (missing) playback/viewport-control. Normally `#t=0,2` implies setting start/stop-values AND starting playback, whereas `#s=0&loop` allows pausing a video, speeding up/slowing down media, as well as enabling/disabling looping. diff --git a/doc/RFC_XR_Fragments.txt b/doc/RFC_XR_Fragments.txt index 3c08de3..0f894af 100644 --- a/doc/RFC_XR_Fragments.txt +++ b/doc/RFC_XR_Fragments.txt @@ -3,7 +3,7 @@ Jens & Leon Internet Engineering Task Force L.R. van Kammen -Internet-Draft 9 February 2024 +Internet-Draft 16 February 2024 Intended status: Informational @@ -43,7 +43,7 @@ Status of This Memo time. It is inappropriate to use Internet-Drafts as reference material or to cite them other than as "work in progress." - This Internet-Draft will expire on 12 August 2024. + This Internet-Draft will expire on 19 August 2024. Copyright Notice @@ -53,7 +53,7 @@ Copyright Notice -van Kammen Expires 12 August 2024 [Page 1] +van Kammen Expires 19 August 2024 [Page 1] Internet-Draft XR Fragments February 2024 @@ -77,22 +77,22 @@ Table of Contents 4.1. List of metadata for 3D nodes . . . . . . . . . . . . . . 8 4.2. Fragment-to-metadata mapping . . . . . . . . . . . . . . 8 4.3. media fragments and datatypes . . . . . . . . . . . . . . 10 - 5. Spatial Referencing 3D . . . . . . . . . . . . . . . . . . . 12 + 5. Spatial Referencing 3D . . . . . . . . . . . . . . . . . . . 13 6. Navigating 3D . . . . . . . . . . . . . . . . . . . . . . . . 13 7. Top-level URL processing . . . . . . . . . . . . . . . . . . 14 8. Embedding XR content using src . . . . . . . . . . . . . . . 15 9. Navigating content href portals . . . . . . . . . . . . . . . 17 - 9.1. Walking surfaces . . . . . . . . . . . . . . . . . . . . 18 - 9.2. UX spec . . . . . . . . . . . . . . . . . . . . . . . . . 18 + 9.1. Walking surfaces . . . . . . . . . . . . . . . . . . . . 19 + 9.2. UX spec . . . . . . . . . . . . . . . . . . . . . . . . . 19 9.3. Scaling instanced content . . . . . . . . . . . . . . . . 19 - 10. XR Fragment: pos . . . . . . . . . . . . . . . . . . . . . . 19 - 11. XR Fragment: rot . . . . . . . . . . . . . . . . . . . . . . 19 - 12. XR Fragment: t . . . . . . . . . . . . . . . . . . . . . . . 19 - 13. XR audio/video integration . . . . . . . . . . . . . . . . . 20 - 14. XR Fragment filters . . . . . . . . . . . . . . . . . . . . . 20 - 14.1. including/excluding . . . . . . . . . . . . . . . . . . 21 + 10. XR Fragment: pos . . . . . . . . . . . . . . . . . . . . . . 20 + 11. XR Fragment: rot . . . . . . . . . . . . . . . . . . . . . . 20 + 12. XR Fragment: t . . . . . . . . . . . . . . . . . . . . . . . 20 + 13. XR audio/video integration . . . . . . . . . . . . . . . . . 21 + 14. XR Fragment filters . . . . . . . . . . . . . . . . . . . . . 21 + 14.1. including/excluding . . . . . . . . . . . . . . . . . . 22 14.2. Filter Parser . . . . . . . . . . . . . . . . . . . . . 22 - 15. Visible links . . . . . . . . . . . . . . . . . . . . . . . . 22 + 15. Visible links . . . . . . . . . . . . . . . . . . . . . . . . 23 16. Text in XR (tagging,linking to spatial objects) . . . . . . . 23 16.1. Default Data URI mimetype . . . . . . . . . . . . . . . 27 16.2. URL and Data URI . . . . . . . . . . . . . . . . . . . . 28 @@ -101,7 +101,7 @@ Table of Contents 18. Topic-based index-less Webrings . . . . . . . . . . . . . . . 32 19. URI Templates (RFC6570) . . . . . . . . . . . . . . . . . . . 33 20. Security Considerations . . . . . . . . . . . . . . . . . . . 33 - 21. FAQ . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 33 + 21. FAQ . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 34 22. authors . . . . . . . . . . . . . . . . . . . . . . . . . . . 34 23. IANA Considerations . . . . . . . . . . . . . . . . . . . . . 34 24. Acknowledgments . . . . . . . . . . . . . . . . . . . . . . . 34 @@ -109,7 +109,7 @@ Table of Contents -van Kammen Expires 12 August 2024 [Page 2] +van Kammen Expires 19 August 2024 [Page 2] Internet-Draft XR Fragments February 2024 @@ -165,7 +165,7 @@ Internet-Draft XR Fragments February 2024 -van Kammen Expires 12 August 2024 [Page 3] +van Kammen Expires 19 August 2024 [Page 3] Internet-Draft XR Fragments February 2024 @@ -221,7 +221,7 @@ Internet-Draft XR Fragments February 2024 -van Kammen Expires 12 August 2024 [Page 4] +van Kammen Expires 19 August 2024 [Page 4] Internet-Draft XR Fragments February 2024 @@ -277,7 +277,7 @@ Internet-Draft XR Fragments February 2024 -van Kammen Expires 12 August 2024 [Page 5] +van Kammen Expires 19 August 2024 [Page 5] Internet-Draft XR Fragments February 2024 @@ -333,7 +333,7 @@ Internet-Draft XR Fragments February 2024 -van Kammen Expires 12 August 2024 [Page 6] +van Kammen Expires 19 August 2024 [Page 6] Internet-Draft XR Fragments February 2024 @@ -376,24 +376,25 @@ Internet-Draft XR Fragments February 2024 | | | |xyz-coord | | | | |0.5,0,0 | +-----------------------+--------------------------------------+------------+------------+ - |W3C Media Fragments |media fragment |#t=0,2 #xywh|play/loop 3D| - |(https://www.w3.org/TR/|(#media%20fragments%20and%20datatypes)| |animation | - |media-frags/) | | |from 0 | + |W3C Media Fragments |media fragment |#t=0,2&loop |play (and | + |(https://www.w3.org/TR/|(#media%20fragments%20and%20datatypes)| |loop) 3D | + |media-frags/) | | |animation | + | | | |from 0 | | | | |seconds till| | | | |2 seconds | +-----------------------+--------------------------------------+------------+------------+ | | | |but can also| | | | |crop, | | | | |animate & | - | | | |configure | -van Kammen Expires 12 August 2024 [Page 7] +van Kammen Expires 19 August 2024 [Page 7] Internet-Draft XR Fragments February 2024 + | | | |configure | | | | |uv- | | | | |coordinates/| | | | |shader | @@ -441,15 +442,15 @@ Internet-Draft XR Fragments February 2024 | | | | |works when metadata-key| | | | | |starts with # | +----------------+--------------------------------------+-------------+---------------------+-----------------------+ - |*FOCUS* |# |string |#person |(and show) object(s) | -van Kammen Expires 12 August 2024 [Page 8] +van Kammen Expires 19 August 2024 [Page 8] Internet-Draft XR Fragments February 2024 + |*FOCUS* |# |string |#person |(and show) object(s) | | | | | |with tag: person or | | | | | |name person (XRWG | | | | | |lookup) | @@ -500,12 +501,15 @@ Internet-Draft XR Fragments February 2024 - -van Kammen Expires 12 August 2024 [Page 9] +van Kammen Expires 19 August 2024 [Page 9] Internet-Draft XR Fragments February 2024 + |*ANIMATION* |#= |string=string|#people=walk |assign a different | + | | | |#people=noanim |animation to object(s) | + +----------------+--------------------------------------+-------------+---------------------+-----------------------+ + Table 5 4.3. media fragments and datatypes @@ -514,79 +518,105 @@ Internet-Draft XR Fragments February 2024 | the 3D scene(file) *but also* media defined in src-metadata like | audio/video-files (mp3/mp4 e.g.) - +=========+=======================+===============+================+ - |type | syntax | example | info | - +=========+=======================+===============+================+ - |vector2 | x,y | 2,3.0 | 2-dimensional | - | | | | vector | - +---------+-----------------------+---------------+----------------+ - |vector3 | x,y,z | 2,3.0,4 | 3-dimensional | - | | | | vector | - +---------+-----------------------+---------------+----------------+ - |temporal | t=x | 0 | play from 0 | - |W3C media| | | seconds to end | - |fragment | | | (and stop) | - +---------+-----------------------+---------------+----------------+ - |temporal | t=x,y | 0,2 | play from 0 | - |W3C media| | | seconds till 2 | - |fragment | | | seconds (and | - | | | | stop) | - +---------+-----------------------+---------------+----------------+ - |temporal | s=x | 1 | set playback | - |W3C media| | | speed of | - |fragment | | | audio/video/3D | - |* | | | anim | - +---------+-----------------------+---------------+----------------+ - |temporal | [-]loop | loop | enable looped | - |W3C media| | | playback of | - |fragment | | | audio/video/3D | - |* | | | anim | - +---------+-----------------------+---------------+----------------+ - | | | -loop | disable looped | - | | | | playback (does | - | | | | not affect | - | | | | playbackstate | - | | | | of media) | - +---------+-----------------------+---------------+----------------+ - |vector2 | uv=uv,v,uspeed,vspeed | 0,0 | set uv offset | - | | | | instantly | - | | | | (default speed | - | | | | = 1,1) | - +---------+-----------------------+---------------+----------------+ - | | | +0.5,+0.5 | scroll | + +==========+=======================+===============+================+ + | type | syntax | example | info | + +==========+=======================+===============+================+ + | vector2 | x,y | 2,3.0 | 2-dimensional | + | | | | vector | + +----------+-----------------------+---------------+----------------+ + | vector3 | x,y,z | 2,3.0,4 | 3-dimensional | + | | | | vector | + +----------+-----------------------+---------------+----------------+ + | temporal | t=x | 0 | play from 0 | + | W3C | | | seconds to | + | media | | | end (and | + | fragment | | | stop) | + +----------+-----------------------+---------------+----------------+ + | temporal | t=x,y | 0,2 | play from 0 | + | W3C | | | seconds till | + | media | | | 2 seconds | + | fragment | | | (and stop) | + +----------+-----------------------+---------------+----------------+ + | temporal | s=x | 1 | set playback | + | W3C | | | speed of | + | media | | | audio/ | + | fragment | | | video/3D anim | + | * | | | | + +----------+-----------------------+---------------+----------------+ + | temporal | [-]loop | loop | enable looped | + | W3C | | | playback of | + | media | | | audio/ | + | fragment | | | video/3D anim | + | * | | | | + +----------+-----------------------+---------------+----------------+ + | | | -loop | disable | + | | | | looped | + | | | | playback | + | | | | (does not | + | | | | affect | -van Kammen Expires 12 August 2024 [Page 10] +van Kammen Expires 19 August 2024 [Page 10] Internet-Draft XR Fragments February 2024 - | | | | instantly by | - | | | | adding 0.5 to | - | | | | the current uv | - | | | | coordinates | - +---------+-----------------------+---------------+----------------+ - | | | 0.2,1,0.1,0.1 | scroll (lerp) | - | | | | to uv | - | | | | coordinate | - | | | | 0,2,1 with 0.1 | - | | | | units per | - | | | | second | - +---------+-----------------------+---------------+----------------+ - | | | 0,0,0,+0.1 | scroll v | - | | | | coordinates | - | | | | with 0.1 units | - | | | | per second | - | | | | (infinitely) | - +---------+-----------------------+---------------+----------------+ - |media | u:== | u:color=1,0,0 | set + shader uniform value | + + + + + + + + + +van Kammen Expires 19 August 2024 [Page 11] + +Internet-Draft XR Fragments February 2024 - Table 6 | * = this is extending the W3C media fragments | (https://www.w3.org/TR/media-frags/#mf-advanced) with (missing) @@ -609,15 +639,6 @@ Internet-Draft XR Fragments February 2024 * https://shaders.org/plasma.glsl#t=0&u:col1=1,0,0&u:col2=0,1,0 (red-green shader plasma starts playing from time-offset 0) - - - - -van Kammen Expires 12 August 2024 [Page 11] - -Internet-Draft XR Fragments February 2024 - - +──────────────────────────────────────────────────────────+ │ │ │ index.gltf#playall │ @@ -644,6 +665,15 @@ Internet-Draft XR Fragments February 2024 > NOTE: URI Template variables are immutable and respect scope: in other words, the end-user cannot modify `blue` by entering an URL like `#blue=.....` in the browser URL, and `blue` is not accessible by the plane/media-object (however `{play}` would work). + + + + +van Kammen Expires 19 August 2024 [Page 12] + +Internet-Draft XR Fragments February 2024 + + 5. Spatial Referencing 3D XR Fragments assume the following objectname-to-URIFragment mapping: @@ -666,14 +696,6 @@ Internet-Draft XR Fragments February 2024 | Every 3D fileformat supports named 3D object, and this name allows | URLs (fragments) to reference them (and their children objects). - - - -van Kammen Expires 12 August 2024 [Page 12] - -Internet-Draft XR Fragments February 2024 - - Clever nested design of 3D scenes allow great ways for re-using content, and/or previewing scenes. For example, to render a portal with a preview-version of the scene, @@ -701,12 +723,19 @@ Internet-Draft XR Fragments February 2024 | #rot=0,90,0 | vector3 | rotate camera | +--------------------+---------+-----------------------------+ + + +van Kammen Expires 19 August 2024 [Page 13] + +Internet-Draft XR Fragments February 2024 + + Table 7 - » example implementation + » example implementation (https://github.com/coderofsalvation/xrfragment/blob/main/src/3rd/js/ three/xrf/pos.js) - » discussion (https://github.com/coderofsalvation/xrfragment/ + » discussion (https://github.com/coderofsalvation/xrfragment/ issues/5) 1. the Y-coordinate of pos identifies the floorposition. This means @@ -715,23 +744,13 @@ Internet-Draft XR Fragments February 2024 2. set the position of the camera accordingly to the vector3 values of #pos 3. rot sets the rotation of the camera (only for non-VR/AR headsets) - 4. t sets the playbackspeed and animation-range of the current scene - animation(s) or src-mediacontent (video/audioframes e.g., use - t=0,7,7 to 'STOP' at frame 7 e.g.) + 4. t in the top-URL sets the playbackspeed and animation-range of + the global scene animation 5. after scene load: in case an href does not mention any pos- coordinate, pos=0,0,0 will be assumed - - - - -van Kammen Expires 12 August 2024 [Page 13] - -Internet-Draft XR Fragments February 2024 - - Here's an ascii representation of a 3D scene-graph which contains 3D - objects ◻ and their metadata: + objects ◻ and their metadata: +────────────────────────────────────────────────────────+ │ │ @@ -758,6 +777,15 @@ Internet-Draft XR Fragments February 2024 The URL-processing-flow for hypermedia browsers goes like this: + + + + +van Kammen Expires 19 August 2024 [Page 14] + +Internet-Draft XR Fragments February 2024 + + 1. IF a #cube matches a custom property-key (of an object) in the 3D file/scene (#cube: #......) THEN execute that predefined_view. @@ -774,18 +802,6 @@ Internet-Draft XR Fragments February 2024 5. IF a #cube matches anything else in the XR Word Graph (XRWG) draw wires to them (text or related objects). - - - - - - - -van Kammen Expires 12 August 2024 [Page 14] - -Internet-Draft XR Fragments February 2024 - - 8. Embedding XR content using src src is the 3D version of the based on the scale-property (of a geometry-less placeholder, an 'empty'-object in blender e.g.). For more info see Chapter Scaling. + + + + +van Kammen Expires 19 August 2024 [Page 16] + +Internet-Draft XR Fragments February 2024 + + 6. external src values should be served with appropriate mimetype (so the XR Fragment-compatible browser will now how to render it). The bare minimum supported mimetypes are: @@ -888,16 +919,6 @@ Internet-Draft XR Fragments February 2024 position to 0,0,0 13. when the enduser clicks an href with #t=1,0,0 (play) will be applied to all src mediacontent with a timeline (mp4/mp3 e.g.) - - - - - -van Kammen Expires 12 August 2024 [Page 16] - -Internet-Draft XR Fragments February 2024 - - 14. a non-euclidian portal can be rendered for flat 3D objects (using stencil buffer e.g.) in case ofspatial src-values (an object #world3 or URL world3.fbx e.g.). @@ -908,19 +929,31 @@ Internet-Draft XR Fragments February 2024 * image/jpg * text/plain;charset=utf-8 - » example implementation + » example implementation (https://github.com/coderofsalvation/xrfragment/blob/main/src/3rd/js/ three/xrf/src.js) - » example 3D asset + » example 3D asset (https://github.com/coderofsalvation/xrfragment/blob/main/example/ assets/src.gltf#L192) - » discussion (https://github.com/coderofsalvation/xrfragment/ + » discussion (https://github.com/coderofsalvation/xrfragment/ issues/4) 9. Navigating content href portals navigation, portals & mutations + + + + + + + +van Kammen Expires 19 August 2024 [Page 17] + +Internet-Draft XR Fragments February 2024 + + +==========+==================+============================+ | fragment | type | example value | +==========+==================+============================+ @@ -947,13 +980,6 @@ Internet-Draft XR Fragments February 2024 (https://github.com/coderofsalvation/xrfragment/blob/dev/src/3rd/ js/three/navigator.js) for an example navigator). - - -van Kammen Expires 12 August 2024 [Page 17] - -Internet-Draft XR Fragments February 2024 - - 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/ @@ -971,13 +997,26 @@ Internet-Draft XR Fragments February 2024 8. href-events should bubble upward the node-tree - » example implementation + 9. the end-user navigator back/forward buttons should repeat a back/ + forward action until a pos=... primitive is found (the inbetween + interaction URI's are only for UX research purposes) + + + + + +van Kammen Expires 19 August 2024 [Page 18] + +Internet-Draft XR Fragments February 2024 + + + » example implementation (https://github.com/coderofsalvation/xrfragment/blob/main/src/3rd/js/ three/xrf/href.js) - » example 3D asset + » example 3D asset (https://github.com/coderofsalvation/xrfragment/blob/main/example/ assets/href.gltf#L192) - » discussion (https://github.com/coderofsalvation/xrfragment/ + » discussion (https://github.com/coderofsalvation/xrfragment/ issues/1) 9.1. Walking surfaces @@ -1001,15 +1040,6 @@ Internet-Draft XR Fragments February 2024 another scene/file (and coordinate e.g. in case the URL contains XR Fragments). - - - - -van Kammen Expires 12 August 2024 [Page 18] - -Internet-Draft XR Fragments February 2024 - - 9.3. Scaling instanced content Sometimes embedded properties (like src) instance new objects. @@ -1028,6 +1058,14 @@ Internet-Draft XR Fragments February 2024 (maxsize=1.4 e.g.) * hide the ''placeholder'' object (material e.g.) * instance the src scene as a child of the existing object + + + +van Kammen Expires 19 August 2024 [Page 19] + +Internet-Draft XR Fragments February 2024 + + * calculate the bounding box of the instanced scene, and scale it accordingly (to 1.4 e.g.) @@ -1056,16 +1094,6 @@ Internet-Draft XR Fragments February 2024 * playposition is reset to framestart, when framestart or framestop is greater than 0 | - - - - - -van Kammen Expires 12 August 2024 [Page 19] - -Internet-Draft XR Fragments February 2024 - - | Example Value | Explanation | |-|-| | 1,1,100 | play loop between frame 1 and 100 | | 1,1,0 | play once from frame 1 (oneshot) | | 1,0,0 | play (previously set looprange if any) | | 0,0,0 | pause | | @@ -1083,6 +1111,17 @@ Internet-Draft XR Fragments February 2024 issues/10] (https://github.com/coderofsalvation/xrfragment/ issues/10])] + + + + + + +van Kammen Expires 19 August 2024 [Page 20] + +Internet-Draft XR Fragments February 2024 + + 13. XR audio/video integration To play global audio/video items: @@ -1105,23 +1144,6 @@ Internet-Draft XR Fragments February 2024 Include, exclude, hide/shows objects using space-separated strings: - - - - - - - - - - - - -van Kammen Expires 12 August 2024 [Page 20] - -Internet-Draft XR Fragments February 2024 - - +====================+===========================================+ | example | outcome | +====================+===========================================+ @@ -1147,6 +1169,15 @@ Internet-Draft XR Fragments February 2024 which used a dedicated q= variable (now deprecated and usable directly) + + + + +van Kammen Expires 19 August 2024 [Page 21] + +Internet-Draft XR Fragments February 2024 + + 14.1. including/excluding By default, selectors work like photoshop-layers: they scan for @@ -1171,13 +1202,6 @@ Internet-Draft XR Fragments February 2024 Table 11 - - -van Kammen Expires 12 August 2024 [Page 21] - -Internet-Draft XR Fragments February 2024 - - | NOTE 1: after an external embedded object has been instanced (src: | https://y.com/bar.fbx#room e.g.), filters do not affect them | anymore (reason: local tag/name collisions can be mitigated @@ -1188,11 +1212,11 @@ Internet-Draft XR Fragments February 2024 | colorbuffer (to allow children being still visible while their | parents are invisible). - » example implementation + » example implementation (https://github.com/coderofsalvation/xrfragment/blob/main/src/3rd/js/ - three/xrf/q.js) » example 3D asset + three/xrf/q.js) » example 3D asset (https://github.com/coderofsalvation/xrfragment/blob/main/example/ - assets/filter.gltf#L192) » discussion + assets/filter.gltf#L192) » discussion (https://github.com/coderofsalvation/xrfragment/issues/3) 14.2. Filter Parser @@ -1201,6 +1225,15 @@ Internet-Draft XR Fragments February 2024 1. create an associative array/object to store filter-arguments as objects + + + + +van Kammen Expires 19 August 2024 [Page 22] + +Internet-Draft XR Fragments February 2024 + + 2. detect object id's & properties foo=1 and foo (reference regex= ~/^.*=[><=]?/ ) 3. detect excluders like -foo,-foo=1,-.foo,-/foo (reference regex= @@ -1226,14 +1259,6 @@ Internet-Draft XR Fragments February 2024 #mytag e.g.) are triggered by the enduser (via toplevel URL or clicking href): - - - -van Kammen Expires 12 August 2024 [Page 22] - -Internet-Draft XR Fragments February 2024 - - 1. draw a wire from the enduser (preferabbly a bit below the camera, heartposition) to object(s) matching that ID (objectname) 2. draw a wire from the enduser (preferabbly a bit below the camera, @@ -1257,6 +1282,14 @@ Internet-Draft XR Fragments February 2024 How does XR Fragments interlink text with objects? + + + +van Kammen Expires 19 August 2024 [Page 23] + +Internet-Draft XR Fragments February 2024 + + | The XR Fragments does this by collapsing space into a *Word Graph* | (the *XRWG* example | (https://github.com/coderofsalvation/xrfragment/blob/feat/macros/ @@ -1282,14 +1315,6 @@ Internet-Draft XR Fragments February 2024 1. XR Fragments promotes (de)serializing a scene to the XRWG (example (https://github.com/coderofsalvation/xrfragment/blob/fe at/macros/src/3rd/js/XRWG.js)) - - - -van Kammen Expires 12 August 2024 [Page 23] - -Internet-Draft XR Fragments February 2024 - - 2. XR Fragments primes the XRWG, by collecting words from the tag and name-property of 3D objects. 3. XR Fragments primes the XRWG, by collecting words from @@ -1314,6 +1339,13 @@ Internet-Draft XR Fragments February 2024 scroll-to-text-fragment (https://github.com/WICG/scroll-to-text- fragment) will be considered) + + +van Kammen Expires 19 August 2024 [Page 24] + +Internet-Draft XR Fragments February 2024 + + Example: http://y.io/z.fbx | Derived XRWG (expressed as BibTex) @@ -1338,14 +1370,6 @@ Internet-Draft XR Fragments February 2024 | the #john@baroque-bib associates both text John and objectname | john, with tag baroque - - - -van Kammen Expires 12 August 2024 [Page 24] - -Internet-Draft XR Fragments February 2024 - - Another example: http://y.io/z.fbx | Derived XRWG (expressed as BibTex) @@ -1371,6 +1395,13 @@ Internet-Draft XR Fragments February 2024 | same XRWG, however on top of that 2 tages (house and todo) are now | associated with text/objectname/tag 'baroque'. + + +van Kammen Expires 19 August 2024 [Page 25] + +Internet-Draft XR Fragments February 2024 + + As seen above, the XRWG can expand bibs (https://github.com/coderofsalvation/hashtagbibs) (and the whole scene) to BibTeX. @@ -1392,16 +1423,6 @@ Internet-Draft XR Fragments February 2024 Table 12 - - - - - -van Kammen Expires 12 August 2024 [Page 25] - -Internet-Draft XR Fragments February 2024 - - | hashtagbibs (https://github.com/coderofsalvation/hashtagbibs) | potentially allow the enduser to annotate text/objects by | *speaking/typing/scanning associations*, which the XR Browser @@ -1429,6 +1450,14 @@ Internet-Draft XR Fragments February 2024 (contextmenu e.g.) of any XR text, anywhere anytime. 11. respect multi-line BiBTeX metadata in text because of the core principle (#core-principle) + + + +van Kammen Expires 19 August 2024 [Page 26] + +Internet-Draft XR Fragments February 2024 + + 12. Default font (unless specified otherwise) is a modern monospace font, for maximized tabular expressiveness (see the core principle (#core-principle)). @@ -1445,19 +1474,6 @@ Internet-Draft XR Fragments February 2024 Fictional chat: - - - - - - - - -van Kammen Expires 12 August 2024 [Page 26] - -Internet-Draft XR Fragments February 2024 - - Hey what about this: https://my.com/station.gltf#pos=0,0,1&rot=90,2,0&t=500,1000 I'm checking it right now I don't see everything..where's our text from yesterday? @@ -1490,6 +1506,14 @@ Internet-Draft XR Fragments February 2024 This indicates that: * utf-8 is supported by default + + + +van Kammen Expires 19 August 2024 [Page 27] + +Internet-Draft XR Fragments February 2024 + + * lines beginning with @ will not be rendered verbatim by default (read more (https://github.com/coderofsalvation/ hashtagbibs#hashtagbib-mimetypes)) @@ -1506,14 +1530,6 @@ Internet-Draft XR Fragments February 2024 Advantages: - - - -van Kammen Expires 12 August 2024 [Page 27] - -Internet-Draft XR Fragments February 2024 - - * auto-expanding of hashtagbibs (https://github.com/coderofsalvation/hashtagbibs) associations * out-of-the-box (de)multiplex human text and metadata in one go @@ -1537,6 +1553,23 @@ Internet-Draft XR Fragments February 2024 16.2. URL and Data URI + + + + + + + + + + + + +van Kammen Expires 19 August 2024 [Page 28] + +Internet-Draft XR Fragments February 2024 + + +--------------------------------------------------------------+ +------------------------+ | | | author.com/article.txt | | index.gltf | +------------------------+ @@ -1563,13 +1596,6 @@ Internet-Draft XR Fragments February 2024 | object note_canvas with 'todo', the enduser can type or speak | #note_canvas@todo - - -van Kammen Expires 12 August 2024 [Page 28] - -Internet-Draft XR Fragments February 2024 - - 16.3. XR Text example parser To prime the XRWG with text from plain text src-values, here's an @@ -1592,6 +1618,14 @@ xrtext = { }, decode: (str) => { + + + +van Kammen Expires 19 August 2024 [Page 29] + +Internet-Draft XR Fragments February 2024 + + // bibtex: ↓@ ↓ ↓property ↓end let pat = [ /@/, /^\S+[,{}]/, /},/, /}/ ] let tags = [], text='', i=0, prop='' @@ -1619,13 +1653,6 @@ xrtext = { return {text, tags} }, - - -van Kammen Expires 12 August 2024 [Page 29] - -Internet-Draft XR Fragments February 2024 - - encode: (text,tags) => { let str = text+"\n" for( let i in tags ){ @@ -1648,6 +1675,13 @@ Internet-Draft XR Fragments February 2024 | above can be used as a startingpoint for LLVM's to translate/ | steelman to a more formal form/language. + + +van Kammen Expires 19 August 2024 [Page 30] + +Internet-Draft XR Fragments February 2024 + + str = ` hello world here are some hashtagbibs followed by bibtex: @@ -1668,20 +1702,6 @@ console.log( xrtext.encode(text,tags) ) // multiplex text & bibtex back to This expands to the following (hidden by default) BibTex appendix: - - - - - - - - - -van Kammen Expires 12 August 2024 [Page 30] - -Internet-Draft XR Fragments February 2024 - - hello world here are some hashtagbibs followed by bibtex: @@ -1709,6 +1729,15 @@ Internet-Draft XR Fragments February 2024 1. defining a different transport protocol (https vs ipfs or DAT) in src or href values can make a difference + + + + +van Kammen Expires 19 August 2024 [Page 31] + +Internet-Draft XR Fragments February 2024 + + 2. mirroring files on another protocol using (HTTP) errorcode tags in src or href properties 3. in case of src: nesting a copy of the embedded object in the @@ -1721,23 +1750,6 @@ Internet-Draft XR Fragments February 2024 For example: - - - - - - - - - - - - -van Kammen Expires 12 August 2024 [Page 31] - -Internet-Draft XR Fragments February 2024 - - +────────────────────────────────────────────────────────+ │ │ │ index.gltf │ @@ -1774,6 +1786,14 @@ Internet-Draft XR Fragments February 2024 * href: schoolA.edu/projects.gltf#math&-topics math * href: schoolB.edu/projects.gltf#math&-courses math + + + +van Kammen Expires 19 August 2024 [Page 32] + +Internet-Draft XR Fragments February 2024 + + * href: university.edu/projects.gltf#math&-theme math | This would hide all object tagged with topic, courses or theme @@ -1784,16 +1804,6 @@ Internet-Draft XR Fragments February 2024 separate content into separate files, or show/hide things using a complex logiclayer like javascript. - - - - - -van Kammen Expires 12 August 2024 [Page 32] - -Internet-Draft XR Fragments February 2024 - - 19. URI Templates (RFC6570) XR Fragments adopts Level1 URI *Fragment* expansion to provide safe @@ -1828,6 +1838,18 @@ Internet-Draft XR Fragments February 2024 In fact, it is much safer than relying on a scripting language (javascript) which can change URN too. + + + + + + + +van Kammen Expires 19 August 2024 [Page 33] + +Internet-Draft XR Fragments February 2024 + + 21. FAQ *Q:* Why is everything HTTP GET-based, what about POST/PUT/DELETE @@ -1842,14 +1864,6 @@ Internet-Draft XR Fragments February 2024 scope as it unhyperifies hypermedia, and this is up to XR hypermedia browser-extensions. Historically scripting/Javascript seems to been able to turn webpages - - - -van Kammen Expires 12 August 2024 [Page 33] - -Internet-Draft XR Fragments February 2024 - - from hypermedia documents into its opposite (hyperscripted nonhypermedia documents). In order to prevent this backward-movement (hypermedia tends to @@ -1872,7 +1886,7 @@ Internet-Draft XR Fragments February 2024 22. authors * Leon van Kammen (@lvk@mastodon.online) - * Jens Finkhäuser (@jens@social.finkhaeuser.de) + * Jens Finkhäuser (@jens@social.finkhaeuser.de) 23. IANA Considerations @@ -1884,28 +1898,23 @@ Internet-Draft XR Fragments February 2024 * Future of Text (https://futureoftext.org) * visual-meta.info (https://visual-meta.info) * Michiel Leenaars + + + +van Kammen Expires 19 August 2024 [Page 34] + +Internet-Draft XR Fragments February 2024 + + * Gerben van der Broeke * Mauve - * Jens Finkhäuser + * Jens Finkhäuser * Marc Belmont * Tim Gerritsen * Frode Hegland * Brandel Zackernuk * Mark Anderson - - - - - - - - -van Kammen Expires 12 August 2024 [Page 34] - -Internet-Draft XR Fragments February 2024 - - 25. Appendix: Definitions +=================+=============================================+ @@ -1945,6 +1954,14 @@ Internet-Draft XR Fragments February 2024 | spacetime | positions camera, triggers scene-preset/ | | hashtags | time | +-----------------+---------------------------------------------+ + + + +van Kammen Expires 19 August 2024 [Page 35] + +Internet-Draft XR Fragments February 2024 + + | teleportation | repositioning the enduser to a different | | | position (or 3D scene/file) | +-----------------+---------------------------------------------+ @@ -1954,14 +1971,6 @@ Internet-Draft XR Fragments February 2024 | placeholder | a 3D object which with src-metadata (which | | object | will be replaced by the src-data.) | +-----------------+---------------------------------------------+ - - - -van Kammen Expires 12 August 2024 [Page 35] - -Internet-Draft XR Fragments February 2024 - - | src | (HTML-piggybacked) metadata of a 3D object | | | which instances content | +-----------------+---------------------------------------------+ @@ -1991,7 +2000,7 @@ Internet-Draft XR Fragments February 2024 | extrospective | outward sensemaking ("I'm fairly sure John | | | is a person who lives in oklahoma") | +-----------------+---------------------------------------------+ - | ◻ | ascii representation of an 3D object/mesh | + | ◻ | ascii representation of an 3D object/mesh | +-----------------+---------------------------------------------+ | (un)obtrusive | obtrusive: wrapping human text/thought in | | | XML/HTML/JSON obfuscates human text into a | @@ -2001,6 +2010,14 @@ Internet-Draft XR Fragments February 2024 | | plane | +-----------------+---------------------------------------------+ | BibTeX | simple tagging/citing/referencing standard | + + + +van Kammen Expires 19 August 2024 [Page 36] + +Internet-Draft XR Fragments February 2024 + + | | for plaintext | +-----------------+---------------------------------------------+ | BibTag | a BibTeX tag | @@ -2011,13 +2028,6 @@ Internet-Draft XR Fragments February 2024 | | XML | +-----------------+---------------------------------------------+ - - -van Kammen Expires 12 August 2024 [Page 36] - -Internet-Draft XR Fragments February 2024 - - Table 13 @@ -2059,14 +2069,4 @@ Internet-Draft XR Fragments February 2024 - - - - - - - - - - -van Kammen Expires 12 August 2024 [Page 37] +van Kammen Expires 19 August 2024 [Page 37] diff --git a/doc/RFC_XR_Fragments.xml b/doc/RFC_XR_Fragments.xml index e65b895..80523af 100644 --- a/doc/RFC_XR_Fragments.xml +++ b/doc/RFC_XR_Fragments.xml @@ -48,18 +48,18 @@ XR Fragments exploits the fact that all 3D models already contain such metadata: Instead of forcing authors to combine 3D/2D objects programmatically (publishing thru a game-editor e.g.), XR Fragments integrates all which allows a universal viewing experience.
    - +───────────────────────────────────────────────────────────────────────────────────────────────+ │ │ │ U R N │ │ U R L | │ │ | |-----------------+--------| │ │ +--------------------------------------------------| │ │ | │ - │ + https://foo.com/some/foo/scene.glb#someview <-- http URI (=URL and has URN) │ + │ + https://foo.com/some/foo/scene.glb#someview <-- http URI (=URL and has URN) │ │ | │ - │ + ipfs://cfe0987ec9r9098ecr/cats.fbx#someview <-- an IPFS URI (=URL and has URN) │ + │ + ipfs://cfe0987ec9r9098ecr/cats.fbx#someview <-- an IPFS URI (=URL and has URN) │ │ │ - │ ec09f7e9cf8e7f09c8e7f98e79c09ef89e000efece8f7ecfe9fe <-- an interpeer URI │ + │ ec09f7e9cf8e7f09c8e7f98e79c09ef89e000efece8f7ecfe9fe <-- an interpeer URI │ │ │ │ │ │ |------------------------+-------------------------| │ @@ -68,20 +68,19 @@ Instead of forcing authors to combine 3D/2D objects programmatically (publishing │ │ +───────────────────────────────────────────────────────────────────────────────────────────────+ -]]> Fact: our typical browser URL's are just a possible implementation of URI's (for untapped humancentric potential of URI's see interpeer.io)

    XR Fragments does not look at XR (or the web) thru the lens of HTML or URLs.
    But approaches things from a higherlevel feedbackloop/hypermedia browser-perspective.
    Below you can see how this translates back into good-old URLs: - +───────────────────────────────────────────────────────────────────────────────────────────────+ │ │ │ the soul of any URL: ://macro /meso ?micro #nano │ │ │ │ 2D URL: ://library.com /document ?search #chapter │ │ │ - │ 4D URL: ://park.com /4Dscene.fbx ─> ?other.glb ─> #view ───> hashbus │ + │ 4D URL: ://park.com /4Dscene.fbx ─> ?other.glb ─> #view ───> hashbus │ │ │ #filter │ │ │ │ #tag │ │ │ │ (hypermediatic) #material │ │ @@ -89,14 +88,13 @@ But approaches things from a higherlevel feedbackloop/hypermedia browser-perspec │ │ ( loop ) #texture │ │ │ │ #variable │ │ │ │ │ │ - │ XRWG <─────────────────────<─────────────+ │ + │ XRWG <─────────────────────<─────────────+ │ │ │ │ │ - │ └─ objects ──────────────>─────────────+ │ + │ └─ objects ──────────────>─────────────+ │ │ │ │ │ +───────────────────────────────────────────────────────────────────────────────────────────────+ -]]>
    ?-linked and #-linked navigation allows a Hypermediatic FeedbackLoop (HFL) between external and internal 4D navigation.
    Traditional webbrowsers can become 4D document-ready by: @@ -174,10 +172,9 @@ But approaches things from a higherlevel feedbackloop/hypermedia browser-perspec
    XR Fragment URL Grammar For typical HTTP-like browsers/applications: - +reserved = gen-delims / sub-delims +gen-delims = "#" / "&" +sub-delims = "," / "="
    Example: ://foo.com/my3d.gltf#pos=1,0,0&prio=-5&t=0,100
    @@ -237,8 +234,8 @@ That way, if the link gets shared, the XR Fragments implementation at https: - - + + @@ -358,6 +355,14 @@ That way, if the link gets shared, the XR Fragments implementation at https: + + + + + + + +
    W3C Media Fragments media fragment#t=0,2 #xywhplay/loop 3D animation from 0 seconds till 2 seconds#t=0,2&loopplay (and loop) 3D animation from 0 seconds till 2 seconds
    #foo=bar sets URI Template variable foo to the value #t=0 from existing object metadata (bar:#t=0 e.g.), This allows for reactive URI Template defined in object metadata elsewhere (src:://m.com/cat.mp4#{foo} e.g., to play media using media fragment URI). NOTE: metadata-key should not start with #
    ANIMATION#<tag_or_objectname>=<animationname>string=string#people=walk #people=noanimassign a different animation to object(s)
    @@ -452,13 +457,14 @@ That way, if the link gets shared, the XR Fragments implementation at https: -media parameter (shader uniform) -u:<uniform>=<string -float -vec2 + + ++0.5,+0.5 +scroll instantly by adding 0.5 to the current uv coordinates -
    * = this is extending the W3C media fragments with (missing) playback/viewport-control. Normally #t=0,2 implies setting start/stop-values AND starting playback, whereas #s=0&loop allows pausing a video, speeding up/slowing down media, as well as enabling/disabling looping. +| media parameter (shader uniform) | u:<uniform>=<string|float|vec2|vec3|vec4> | u:color=1,0,0 | set shader uniform value | +
    * = this is extending the W3C media fragments with (missing) playback/viewport-control. Normally #t=0,2 implies setting start/stop-values AND starting playback, whereas #s=0&loop allows pausing a video, speeding up/slowing down media, as well as enabling/disabling looping. The rationale for uv is that the xywh Media Fragment deals with rectangular media, which does not translate well to 3D models (which use triangular polygons, not rectangular) positioned by uv-coordinates.
    Example URI's: @@ -468,11 +474,11 @@ That way, if the link gets shared, the XR Fragments implementation at https:
  • https://shaders.org/plasma.glsl#t=0&u:col1=1,0,0&u:col2=0,1,0 (red-green shader plasma starts playing from time-offset 0)
  • - +──────────────────────────────────────────────────────────+ │ │ │ index.gltf#playall │ │ │ │ - │ ├ # : #t=0&shared=play │ apply default XR Fragment on load (`t` plays global 3D animation timeline) + │ ├ # : #t=0&shared=play │ apply default XR Fragment on load (`t` plays global 3D animation timeline) │ ├ play : #t=l:0,2 │ variable for [URI Templates (RFC6570)](https://www.rfc-editor.org/rfc/rfc6570) │ │ │ │ ├── ◻ plane (with material) │ @@ -482,19 +488,18 @@ That way, if the link gets shared, the XR Fragments implementation at https: │ │ └ src: foo.jpg#suv=l:0,0.1 │ infinite texturescroll `v` of uv·coordinates with 0.1/fps │ │ │ │ ├── ◻ media │ - │ │ └ src: cat.mp4#t=l:2,10&uv=0.5,0.5 │ loop cat.mp4 (or mp3/wav/jpg) between 2 and 10 seconds (uv's shifted with 0.5,0.5) + │ │ └ src: cat.mp4#t=l:2,10&uv=0.5,0.5 │ loop cat.mp4 (or mp3/wav/jpg) between 2 and 10 seconds (uv's shifted with 0.5,0.5) │ │ │ │ └── ◻ wall │ │ ├ href: #color=blue │ updates uniform values (IFS shader e.g.) - │ ├ blue: t=0&u:col=0,0,1 │ variable for [Level1 URI Templates (RFC6570)](https://www.rfc-editor.org/rfc/rfc6570) - │ └ src: ://a.com/art.glsl#{color}&{shared} │ .fs/.vs/.glsl/.wgsl etc shader [Level1 URI Template (RFC6570)](https://www.rfc-editor.org/rfc/rfc6570) + │ ├ blue: t=0&u:col=0,0,1 │ variable for [Level1 URI Templates (RFC6570)](https://www.rfc-editor.org/rfc/rfc6570) + │ └ src: ://a.com/art.glsl#{color}&{shared} │ .fs/.vs/.glsl/.wgsl etc shader [Level1 URI Template (RFC6570)](https://www.rfc-editor.org/rfc/rfc6570) │ │ │ │ +──────────────────────────────────────────────────────────+ -> NOTE: URI Template variables are immutable and respect scope: in other words, the end-user cannot modify `blue` by entering an URL like `#blue=.....` in the browser URL, and `blue` is not accessible by the plane/media-object (however `{play}` would work). +> NOTE: URI Template variables are immutable and respect scope: in other words, the end-user cannot modify `blue` by entering an URL like `#blue=.....` in the browser URL, and `blue` is not accessible by the plane/media-object (however `{play}` would work). -]]> @@ -502,7 +507,7 @@ That way, if the link gets shared, the XR Fragments implementation at https:
    Spatial Referencing 3D XR Fragments assume the following objectname-to-URIFragment mapping: - my.io/scene.fbx +─────────────────────────────+ │ sky │ src: http://my.io/scene.fbx#sky (includes building,mainobject,floor) @@ -518,7 +523,6 @@ That way, if the link gets shared, the XR Fragments implementation at https: │ +─────────────────────────+ │ +─────────────────────────────+ -]]>
    Every 3D fileformat supports named 3D object, and this name allows URLs (fragments) to reference them (and their children objects).
    Clever nested design of 3D scenes allow great ways for re-using content, and/or previewing scenes.
    @@ -564,24 +568,23 @@ For example, to render a portal with a preview-version of the scene, create an 3
  • the Y-coordinate of pos identifies the floorposition. This means that desktop-projections usually need to add 1.5m (average person height) on top (which is done automatically by VR/AR headsets).
  • set the position of the camera accordingly to the vector3 values of #pos
  • rot sets the rotation of the camera (only for non-VR/AR headsets)
  • -
  • t sets the playbackspeed and animation-range of the current scene animation(s) or src-mediacontent (video/audioframes e.g., use t=0,7,7 to 'STOP' at frame 7 e.g.)
  • +
  • t in the top-URL sets the playbackspeed and animation-range of the global scene animation
  • after scene load: in case an href does not mention any pos-coordinate, pos=0,0,0 will be assumed
  • Here's an ascii representation of a 3D scene-graph which contains 3D objects and their metadata: - +────────────────────────────────────────────────────────+ │ │ │ index.gltf │ │ │ │ │ ├── ◻ buttonA │ - │ │ └ href: #pos=1,0,1&t=100,200 │ + │ │ └ href: #pos=1,0,1&t=100,200 │ │ │ │ │ └── ◻ buttonB │ - │ └ href: other.fbx │ <── file─agnostic (can be .gltf .obj etc) + │ └ href: other.fbx │ <── file─agnostic (can be .gltf .obj etc) │ │ +────────────────────────────────────────────────────────+ -]]> An XR Fragment-compatible browser viewing this scene, allows the end-user to interact with the buttonA and buttonB.
    @@ -629,7 +632,7 @@ It instances content (in objects) in the current scene/asset.
    Here's an ascii representation of a 3D scene-graph with 3D objects which embeds remote & local 3D objects with/out using filters: - +────────────────────────────────────────────────────────+ +─────────────────────────+ │ │ │ │ │ index.gltf │ │ ocean.com/aquarium.fbx │ │ │ │ │ ├ room │ @@ -646,7 +649,6 @@ It instances content (in objects) in the current scene/asset.
    │ └ src: #canvas │ │ │ +────────────────────────────────────────────────────────+ -]]> An XR Fragment-compatible browser viewing this scene, lazy-loads and projects painting.png onto the (plane) object called canvas (which is copy-instanced in the bed and livingroom).
    @@ -728,6 +730,8 @@ Resizing will be happen accordingly to its placeholder object aquariumcube
  • href-events should bubble upward the node-tree
  • +
  • the end-user navigator back/forward buttons should repeat a back/forward action until a pos=... primitive is found (the inbetween interaction URI's are only for UX research purposes) +
  • » example implementation
    @@ -965,7 +969,7 @@ XR Fragments does this by detecting Bib(s)Tex, without introducing a new languag Example: - http://y.io/z.fbx | Derived XRWG (expressed as BibTex) ----------------------------------------------------------------------------+-------------------------------------- | @house{castle, +-[src: data:.....]----------------------+ +-[3D mesh]-+ | url = {https://y.io/z.fbx#castle} @@ -983,12 +987,11 @@ XR Fragments does this by detecting Bib(s)Tex, without introducing a new languag | /|\ | | | / \ | | +--------+ | -]]>
    the #john@baroque-bib associates both text John and objectname john, with tag baroque
    Another example: - http://y.io/z.fbx | Derived XRWG (expressed as BibTex) ----------------------------------------------------------------------------+-------------------------------------- | +-[src: data:.....]----------------------+ +-[3D mesh]-+ | @house{castle, @@ -1006,7 +1009,6 @@ XR Fragments does this by detecting Bib(s)Tex, without introducing a new languag | #baroque@todo@house | | /|\ | | | ... | | / \ | | +----------------------------------------+ +--------+ | -]]>
    both #john@baroque-bib and BibTex @baroque{john} result in the same XRWG, however on top of that 2 tages (house and todo) are now associated with text/objectname/tag 'baroque'.
    As seen above, the XRWG can expand bibs (and the whole scene) to BibTeX.
    @@ -1060,22 +1062,21 @@ Some pointers for good UX (but not necessary to be XR Fragment compatible):
    The simplicity of appending metadata (and leveling the metadata-playfield between humans and machines) is also demonstrated by visual-meta in greater detail.
    Fictional chat: - Hey what about this: https://my.com/station.gltf#pos=0,0,1&rot=90,2,0&t=500,1000 - I'm checking it right now - I don't see everything..where's our text from yesterday? - Ah wait, that's tagged with tag 'draft' (and hidden)..hold on, try this: - https://my.com/station.gltf#.draft&pos=0,0,1&rot=90,2,0&t=500,1000 - how about we link the draft to the upcoming YELLO-event? - ok I'm adding #draft@YELLO - Yesterday I also came up with other usefull assocations between other texts in the scene: +<John> Hey what about this: https://my.com/station.gltf#pos=0,0,1&rot=90,2,0&t=500,1000 +<Sarah> I'm checking it right now +<Sarah> I don't see everything..where's our text from yesterday? +<John> Ah wait, that's tagged with tag 'draft' (and hidden)..hold on, try this: +<John> https://my.com/station.gltf#.draft&pos=0,0,1&rot=90,2,0&t=500,1000 +<Sarah> how about we link the draft to the upcoming YELLO-event? +<John> ok I'm adding #draft@YELLO +<Sarah> Yesterday I also came up with other usefull assocations between other texts in the scene: #event#YELLO #2025@YELLO - thanks, added. - Btw. I stumbled upon this spatial book which references station.gltf in some chapters: - https://thecommunity.org/forum/foo/mytrainstory.txt - interesting, I'm importing mytrainstory.txt into station.gltf - ah yes, chapter three points to trainterminal_2A in the scene, cool -]]> +<John> thanks, added. +<Sarah> Btw. I stumbled upon this spatial book which references station.gltf in some chapters: +<Sarah> https://thecommunity.org/forum/foo/mytrainstory.txt +<John> interesting, I'm importing mytrainstory.txt into station.gltf +<John> ah yes, chapter three points to trainterminal_2A in the scene, cool
    Default Data URI mimetype @@ -1111,7 +1112,7 @@ to connect text further with its environment ( setup links between textual/spati
    URL and Data URI - +--------------------------------------------------------------+ +------------------------+ | | | author.com/article.txt | | index.gltf | +------------------------+ | │ | | | @@ -1123,7 +1124,6 @@ to connect text further with its environment ( setup links between textual/spati | | +------------------------+ | | +--------------------------------------------------------------+ -]]> The enduser will only see welcome human and Hello friends rendered verbatim (see mimetype). The beauty is that text in Data URI automatically promotes rich copy-paste (retaining metadata). @@ -1135,14 +1135,14 @@ The XR Fragment-compatible browser can let the enduser access visual-meta(data)-
    XR Text example parser To prime the XRWG with text from plain text src-values, here's an example XR Text (de)multiplexer in javascript (which supports inline bibs & bibtex): -xrtext = { - expandBibs: (text) => { + expandBibs: (text) => { let bibs = { regex: /(#[a-zA-Z0-9_+@\-]+(#)?)/g, tags: {}} - text.replace( bibs.regex , (m,k,v) => { - tok = m.substr(1).split("@") + text.replace( bibs.regex , (m,k,v) => { + tok = m.substr(1).split("@") match = tok.shift() - if( tok.length ) tok.map( (t) => bibs.tags[t] = `@${t}{${match},\n}` ) + if( tok.length ) tok.map( (t) => bibs.tags[t] = `@${t}{${match},\n}` ) else if( match.substr(-1) == '#' ) bibs.tags[match] = `@{${match.replace(/#/,'')}}` else bibs.tags[match] = `@${match}{${match},\n}` @@ -1150,16 +1150,16 @@ The XR Fragment-compatible browser can let the enduser access visual-meta(data)- return text.replace( bibs.regex, '') + Object.values(bibs.tags).join('\n') }, - decode: (str) => { - // bibtex: ↓@ ↓ ↓property ↓end + decode: (str) => { + // bibtex: ↓@ ↓<tag|tag{phrase,|{ruler}> ↓property ↓end let pat = [ /@/, /^\S+[,{}]/, /},/, /}/ ] let tags = [], text='', i=0, prop='' let lines = xrtext.expandBibs(str).replace(/\r?\n/g,'\n').split(/\n/) - for( let i = 0; i < lines.length && !String(lines[i]).match( /^@/ ); i++ ) + for( let i = 0; i < lines.length && !String(lines[i]).match( /^@/ ); i++ ) text += lines[i]+'\n' bibtex = lines.join('\n').substr( text.length ) - bibtex.split( pat[0] ).map( (t) => { + bibtex.split( pat[0] ).map( (t) => { try{ let v = {} if( !(t = t.trim()) ) return @@ -1168,9 +1168,9 @@ The XR Fragment-compatible browser can let the enduser access visual-meta(data)- if( tag.match( /}$/ ) ) return tags.push({k: tag.replace(/}$/,''), v: {}}) t = t.substr( tag.length ) t.split( pat[2] ) - .map( kv => { - if( !(kv = kv.trim()) || kv == "}" ) return - v[ kv.match(/\s?(\S+)\s?=/)[1] ] = kv.substr( kv.indexOf("{")+1 ) + .map( kv => { + if( !(kv = kv.trim()) || kv == "}" ) return + v[ kv.match(/\s?(\S+)\s?=/)[1] ] = kv.substr( kv.indexOf("{")+1 ) }) tags.push( { k:tag, v } ) }catch(e){ console.error(e) } @@ -1178,8 +1178,8 @@ The XR Fragment-compatible browser can let the enduser access visual-meta(data)- return {text, tags} }, - encode: (text,tags) => { - let str = text+"\n" + encode: (text,tags) => { + let str = text+"\n" for( let i in tags ){ let item = tags[i] if( item.ruler ){ @@ -1193,12 +1193,11 @@ The XR Fragment-compatible browser can let the enduser access visual-meta(data)- return str } } -]]> The above functions (de)multiplexe text/metadata, expands bibs, (de)serialize bibtex and vice versa
    above can be used as a startingpoint for LLVM's to translate/steelman to a more formal form/language.
    -str = ` hello world here are some hashtagbibs followed by bibtex: @@ -1211,15 +1210,14 @@ here are some hashtagbibs followed by bibtex: asdf = {23423} }` -var {tags,text} = xrtext.decode(str) // demultiplex text & bibtex -tags.find( (t) => t.k == 'flap{' ).v.asdf = 1 // edit tag +var {tags,text} = xrtext.decode(str) // demultiplex text & bibtex +tags.find( (t) => t.k == 'flap{' ).v.asdf = 1 // edit tag tags.push({ k:'bar{', v:{abc:123} }) // add tag -console.log( xrtext.encode(text,tags) ) // multiplex text & bibtex back together -]]> +console.log( xrtext.encode(text,tags) ) // multiplex text & bibtex back together This expands to the following (hidden by default) BibTex appendix: -hello world here are some hashtagbibs followed by bibtex: @{some-section} @@ -1234,7 +1232,6 @@ here are some hashtagbibs followed by bibtex: @bar{ abc = {123} } -]]>
    when an XR browser updates the human text, a quick scan for nonmatching tags (@book{nonmatchingbook e.g.) should be performed and prompt the enduser for deleting them.
    @@ -1251,7 +1248,7 @@ here are some hashtagbibs followed by bibtex:
    due to the popularity, maturity and extensiveness of HTTP codes for client/server communication, non-HTTP protocols easily map to HTTP codes (ipfs ERR_NOT_FOUND maps to 404 e.g.)
    For example: - +────────────────────────────────────────────────────────+ │ │ │ index.gltf │ │ │ │ @@ -1263,14 +1260,13 @@ here are some hashtagbibs followed by bibtex: │ │ └ href@400: #clienterrortext │ │ │ └ ◻ offlinetext │ │ │ │ - │ └── ◻ embeddedObject <--------- the meshdata inside embeddedObject will (not) + │ └── ◻ embeddedObject <--------- the meshdata inside embeddedObject will (not) │ └ src: https://foo.io/bar.gltf │ be flushed when the request (does not) succeed. │ └ src@404: http://foo.io/bar.gltf │ So worstcase the 3D data (of the time of publishing index.gltf) │ └ src@400: https://archive.org/l2kj43.gltf │ will be displayed. │ │ +────────────────────────────────────────────────────────+ -]]>
    @@ -1302,7 +1298,7 @@ To filter out non-related objects one could take it a step further using filters The following demonstrates a simple video player: - +─────────────────────────────────────────────+ │ │ │ foo.usdz │ @@ -1310,19 +1306,18 @@ The following demonstrates a simple video player: │ │ │ │ ├── ◻ stopbutton │ │ │ ├ #: #-stopbutton │ - │ │ └ href: #player=stop&-stopbutton │ (stop and hide stop-button) + │ │ └ href: #player=stop&-stopbutton │ (stop and hide stop-button) │ │ │ │ └── ◻ plane │ │ ├ play: #t=l:0,10 │ │ ├ stop: #t=0,0 │ - │ ├ href: #player=play&stopbutton │ (play and show stop-button) + │ ├ href: #player=play&stopbutton │ (play and show stop-button) │ └ src: cat.mp4#{player} │ │ │ │ │ +─────────────────────────────────────────────+ -]]>
    diff --git a/doc/RFC_XR_Macros.txt b/doc/RFC_XR_Macros.txt index ab0dc15..556356c 100644 --- a/doc/RFC_XR_Macros.txt +++ b/doc/RFC_XR_Macros.txt @@ -3,7 +3,7 @@ Internet Engineering Task Force L.R. van Kammen -Internet-Draft 8 February 2024 +Internet-Draft 16 February 2024 Intended status: Informational @@ -38,7 +38,7 @@ Status of This Memo time. It is inappropriate to use Internet-Drafts as reference material or to cite them other than as "work in progress." - This Internet-Draft will expire on 11 August 2024. + This Internet-Draft will expire on 19 August 2024. Copyright Notice @@ -53,7 +53,7 @@ Copyright Notice -van Kammen Expires 11 August 2024 [Page 1] +van Kammen Expires 19 August 2024 [Page 1] Internet-Draft XR Macros February 2024 @@ -109,13 +109,13 @@ Table of Contents -van Kammen Expires 11 August 2024 [Page 2] +van Kammen Expires 19 August 2024 [Page 2] Internet-Draft XR Macros February 2024 - 3. Metadata-values can contain the | symbol to 🎲 roundrobin variable - values (!toggleme: fog=0,10|fog=0,1000 e.g.) + 3. Metadata-values can contain the | symbol to 🎲 roundrobin + variable values (!toggleme: fog=0,10|fog=0,1000 e.g.) 4. XR Macros acts as simple eventhandlers for URI Fragments: they are automatically published on the (XR Fragments (https://xrfragment.org)) hashbus, to act as events (so more @@ -165,7 +165,7 @@ Internet-Draft XR Macros February 2024 -van Kammen Expires 11 August 2024 [Page 3] +van Kammen Expires 19 August 2024 [Page 3] Internet-Draft XR Macros February 2024 @@ -221,7 +221,7 @@ Internet-Draft XR Macros February 2024 -van Kammen Expires 11 August 2024 [Page 4] +van Kammen Expires 19 August 2024 [Page 4] Internet-Draft XR Macros February 2024 @@ -277,7 +277,7 @@ Internet-Draft XR Macros February 2024 -van Kammen Expires 11 August 2024 [Page 5] +van Kammen Expires 19 August 2024 [Page 5] Internet-Draft XR Macros February 2024 @@ -333,7 +333,7 @@ click object with (`!clickme`:`!foo|!bar|!flop` e.g.) -van Kammen Expires 11 August 2024 [Page 6] +van Kammen Expires 19 August 2024 [Page 6] Internet-Draft XR Macros February 2024 @@ -375,7 +375,7 @@ Internet-Draft XR Macros February 2024 | | (games,VR,AR e.g.), should be as high as | | | possible | +---------------+---------------------------------------------------+ - | ◻ | ascii representation of an 3D object/mesh | + | ◻ | ascii representation of an 3D object/mesh | +---------------+---------------------------------------------------+ | (un)obtrusive | obtrusive: wrapping human text/thought in | | | XML/HTML/JSON obfuscates human text into | @@ -389,4 +389,4 @@ Internet-Draft XR Macros February 2024 -van Kammen Expires 11 August 2024 [Page 7] +van Kammen Expires 19 August 2024 [Page 7] diff --git a/doc/RFC_XR_Macros.xml b/doc/RFC_XR_Macros.xml index 0150e79..b08a4f2 100644 --- a/doc/RFC_XR_Macros.xml +++ b/doc/RFC_XR_Macros.xml @@ -282,17 +282,16 @@ Macros also act as events, so more serious scripting languages can react to them
    Event Bubble-flow click object with (!clickme:AR or !clickme: !reset e.g.) - ◻ │ └── does current object contain this property-key (`AR` or `!reset` e.g.)? └── no: is there any (root)object containing property `AR` └── yes: evaluate its (roundrobin) XR macro-value(s) (and exit) └── no: trigger URL: #AR -]]> click object with (!clickme:#AR|#VR e.g.) - ◻ │ └── apply the roundrobin (rotate the options, value `#AR` becomes `#VR` upon next click) └── is there any object with property-key (`#AR` e.g.)? @@ -300,7 +299,6 @@ Macros also act as events, so more serious scripting languages can react to them └── yes: apply its value to the scene, and update the URL to `#AR` click object with (`!clickme`:`!foo|!bar|!flop` e.g.) -]]> ◻ │
    diff --git a/src/3rd/js/aframe/index.js b/src/3rd/js/aframe/index.js index b46f015..f52f6fd 100644 --- a/src/3rd/js/aframe/index.js +++ b/src/3rd/js/aframe/index.js @@ -100,6 +100,7 @@ window.AFRAME.registerComponent('xrf', { let isLocal = url.match(/^#/) let hasPos = url.match(/pos=/) let fastFadeMs = 200 + if( !AFRAME.fade ) return p.resolve() if( isLocal ){ if( hasPos ){ diff --git a/src/3rd/js/pubsub.js b/src/3rd/js/pubsub.js index 37a753a..499226f 100644 --- a/src/3rd/js/pubsub.js +++ b/src/3rd/js/pubsub.js @@ -25,9 +25,10 @@ xrf.addEventListener = function(eventName, callback, opts) { this._listeners[eventName].push(callback); // sort this._listeners[eventName] = this._listeners[eventName].sort( (a,b) => a.opts.weight > b.opts.weight ) - return () => { + callback.unlisten = () => { this._listeners[eventName] = this._listeners[eventName].filter( (c) => c != callback ) } + return callback.unlisten }; xrf.emit = function(eventName, data){ @@ -76,7 +77,8 @@ xrf.emit.promise = function(e, opts){ } xrf.addEventListener('reset', () => { -// *TODO* do this nicely -// xrf._listeners['renderPost'] = [] -// xrf._listeners['render'] = [] + let events = ['renderPost'] + events.map( (e) => { + if( xrf._listeners[e] ) xrf._listeners[e].map( (r) => r.unlisten && r.unlisten() ) + }) }) diff --git a/src/3rd/js/three/util/optimize.js b/src/3rd/js/three/util/optimize.js index 18a4764..4a6a9a7 100644 --- a/src/3rd/js/three/util/optimize.js +++ b/src/3rd/js/three/util/optimize.js @@ -11,7 +11,7 @@ xrf.optimize = (opts) => { // check unused animations xrf.optimize.checkAnimations = (opts) => { - console.log("TODO: fix freezeUnAnimatedObjects for SRC's") + if( xrf.debug ) console.log("TODO: fix freezeUnAnimatedObjects for SRC's") return xrf.optimize let {model} = opts @@ -35,7 +35,7 @@ xrf.optimize.checkAnimations = (opts) => { } xrf.optimize.freezeUnAnimatedObjects = (opts) => { - console.log("TODO: fix freezeUnAnimatedObjects for SRC's") + if( xrf.todo ) console.log("TODO: fix freezeUnAnimatedObjects for SRC's") return xrf.optimize let {model} = opts diff --git a/src/3rd/js/three/xrf/#.js b/src/3rd/js/three/xrf/#.js index a940b99..bbadc76 100644 --- a/src/3rd/js/three/xrf/#.js +++ b/src/3rd/js/three/xrf/#.js @@ -4,12 +4,7 @@ xrf.frag.defaultPredefinedViews = (opts) => { let {scene,model} = opts; scene.traverse( (n) => { if( n.userData && n.userData['#'] ){ - let frag = xrf.URI.parse( n.userData['#'] ) - if( !n.parent && document.location.hash.length < 2){ - xrf.navigator.to( n.userData['#'] ) // evaluate default XR fragments (global-level) - }else{ - xrf.hashbus.pub( n.userData['#'], n ) // evaluate default XR fragments (node-level) - } + xrf.hashbus.pub( n.userData['#'], n ) // evaluate default XR fragments without affecting URL } }) } diff --git a/src/3rd/js/three/xrf/dynamic/filter.js b/src/3rd/js/three/xrf/dynamic/filter.js index ae834a5..771f8ac 100644 --- a/src/3rd/js/three/xrf/dynamic/filter.js +++ b/src/3rd/js/three/xrf/dynamic/filter.js @@ -10,7 +10,7 @@ const doFilter = (opts) => { xrf.addEventListener('dynamicKey', doFilter ) xrf.addEventListener('dynamicKeyValue', (opts) => { - console.log("*TODO* filter integers only") + if( xrf.debug ) console.log("*TODO* filter integers only") // doFilter(opts) }) diff --git a/src/3rd/js/three/xrf/src/fbx.js b/src/3rd/js/three/xrf/src/fbx.js new file mode 100644 index 0000000..7c79857 --- /dev/null +++ b/src/3rd/js/three/xrf/src/fbx.js @@ -0,0 +1,27 @@ +/* + * mimetype: model/gltf+json + */ + +xrf.frag.src.type['fbx'] = function( url, opts ){ + return new Promise( async (resolve,reject) => { + let {mesh,src} = opts + let {urlObj,dir,file,hash,ext} = xrf.parseUrl(url) + let loader + + //let {THREE} = await import('https://unpkg.com/three@0.161.0/build/three.module.js') + //let { FBXLoader } = await import('three/addons/loaders/FBXLoader.js') + + //const Loader = xrf.loaders[ext] + //if( !Loader ) throw 'xrfragment: no loader passed to xrfragment for extension .'+ext + //if( !dir.match("://") ){ // force relative path + // dir = dir[0] == './' ? dir : `./${dir}` + // loader = new Loader().setPath( dir ) + //}else loader = new Loader() + + //loader.load(url, (model) => { + // model.isSRC = true + // resolve(model) + //}) + }) +} + diff --git a/test/generated/test.js b/test/generated/test.js index 5800957..e9dd629 100644 --- a/test/generated/test.js +++ b/test/generated/test.js @@ -1,6 +1,13 @@ 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 $estr = function() { return js_Boot.__string_rec(this,''); },$hxEnums = $hxEnums || {},$_; +function $extend(from, fields) { + var proto = Object.create(from); + for (var name in fields) proto[name] = fields[name]; + if( fields.toString !== Object.prototype.toString ) proto.toString = fields.toString; + return proto; +} var EReg = function(r,opt) { this.r = new RegExp(r,opt.split("u").join("")); }; @@ -14,6 +21,26 @@ EReg.prototype = { this.r.s = s; return this.r.m != null; } + ,matched: function(n) { + if(this.r.m != null && n >= 0 && n < this.r.m.length) { + return this.r.m[n]; + } else { + throw haxe_Exception.thrown("EReg::matched"); + } + } + ,matchedRight: function() { + if(this.r.m == null) { + throw haxe_Exception.thrown("No string matched"); + } + var sz = this.r.m.index + this.r.m[0].length; + return HxOverrides.substr(this.r.s,sz,this.r.s.length - sz); + } + ,matchedPos: function() { + if(this.r.m == null) { + throw haxe_Exception.thrown("No string matched"); + } + return { pos : this.r.m.index, len : this.r.m[0].length}; + } ,split: function(s) { var d = "#__delim__#"; return s.replace(this.r,d).split(d); @@ -53,6 +80,25 @@ Reflect.field = function(o,field) { return null; } }; +Reflect.getProperty = function(o,field) { + var tmp; + if(o == null) { + return null; + } else { + var tmp1; + if(o.__properties__) { + tmp = o.__properties__["get_" + field]; + tmp1 = tmp; + } else { + tmp1 = false; + } + if(tmp1) { + return o[tmp](); + } else { + return o[field]; + } + } +}; Reflect.fields = function(o) { var a = []; if(o != null) { @@ -65,6 +111,21 @@ Reflect.fields = function(o) { } return a; }; +Reflect.isObject = function(v) { + if(v == null) { + return false; + } + var t = typeof(v); + if(!(t == "string" || t == "object" && v.__enum__ == null)) { + if(t == "function") { + return (v.__name__ || v.__ename__) != null; + } else { + return false; + } + } else { + return true; + } +}; Reflect.deleteField = function(o,field) { if(!Object.prototype.hasOwnProperty.call(o,field)) { return false; @@ -111,6 +172,10 @@ Std.parseInt = function(x) { } return null; }; +var StringBuf = function() { + this.b = ""; +}; +StringBuf.__name__ = true; var StringTools = function() { }; StringTools.__name__ = true; StringTools.isSpace = function(s,pos) { @@ -144,20 +209,24 @@ StringTools.rtrim = function(s) { StringTools.trim = function(s) { return StringTools.ltrim(StringTools.rtrim(s)); }; +StringTools.replace = function(s,sub,by) { + return s.split(sub).join(by); +}; var Test = function() { }; Test.__name__ = true; Test.main = function() { Test.test("url.json",[{ fn : "url", expect : { fn : "testPredefinedView", input : "mypredefinedview", out : true}, label : "test predefined view executed", data : "http://foo.com?foo=1#mypredefinedview"},{ fn : "url", expect : { fn : "testPredefinedView", input : "another", out : true}, label : "test predefined view executed (multiple)", data : "http://foo.com?foo=1#mypredefinedview&another"},{ fn : "url", expect : { fn : "testPredefinedView", input : "mypredefinedview", out : true}, label : "test predefined view executed (multiple)", data : "http://foo.com?foo=1#mypredefinedview&another"},{ fn : "url", expect : { fn : "testParsed", input : "mycustom", out : true}, label : "test custom property", data : "http://foo.com?foo=1#mycustom=foo"}]); Test.test("pos.json",[{ fn : "url", expect : { fn : "equal.string", input : "pos", out : "1.2,2.2"}, label : "equal.string", data : "http://foo.com?foo=1#pos=1.2,2.2"},{ fn : "url", expect : { fn : "equal.xyz", input : "pos", out : "1.2,2.2,3"}, label : "equal.xyz", data : "http://foo.com?foo=1#pos=1.2,2.2,3"},{ fn : "url", expect : { fn : "equal.xyz", input : "pos", out : "1,2,3"}, label : "pos equal.xyz", data : "http://foo.com?foo=1#pos=1,2,3"},{ fn : "url", expect : { fn : "equal.string", input : "pos", out : "world2"}, label : "pos equal.xyz", data : "http://foo.com?foo=1#pos=world2"}]); - Test.test("t.json",[{ fn : "url", expect : { fn : "equal.x", input : "t", out : "1"}, label : "a equal.x", data : "http://foo.com?foo=1#t=1"},{ fn : "url", expect : { fn : "equal.x", input : "t", out : "-1"}, label : "a equal.x", data : "http://foo.com?foo=1#t=-1"},{ fn : "url", expect : { fn : "equal.x", input : "t", out : "-1.02"}, label : "a equal.x", data : "http://foo.com?foo=1#t=-1.02"},{ fn : "url", expect : { fn : "equal.xy", input : "t", out : "1,2"}, label : "a equal.xy", data : "http://foo.com?foo=1#t=1,2,3"},{ fn : "url", expect : { fn : "equal.xyz", input : "t", out : "1,2,3"}, label : "a equal.xyz", data : "http://foo.com?foo=1#t=1,2,3"},{ fn : "url", expect : { fn : "equal.xyz", input : "t", out : "1,-2,3"}, label : "a equal.xyz", data : "http://foo.com?foo=1#t=1,-2,3"},{ fn : "url", expect : { fn : "equal.xy", input : "t", out : "1,100"}, label : "a equal.xy", data : "http://foo.com?foo=1#t=1,100"},{ fn : "url", expect : { fn : "testBrowserOverride", input : "t", out : true}, label : "browser URI can override t (defined in asset)", data : "http://foo.com?foo=1#t=2,500"},{ fn : "url", expect : { fn : "equal.mediafragmentT", input : "3", out : "500"}, label : "a equal.mediafragment", data : "http://foo.com?foo=1#t=1,100,400,500*1.2,2.3"},{ fn : "url", expect : { fn : "equal.mediafragmentTSpd", input : "1", out : "2.3"}, label : "a equal.mediafragmentSpeed", data : "http://foo.com?foo=1#t=1,100,400,500*1.2,2.3"}]); - Test.test("xywh.json",[{ fn : "url", expect : { fn : "equal.mediafragmentXYWH", input : "2", out : "1"}, label : "xywh", data : "http://foo.com?foo=1#xywh=0,0,1,1"},{ fn : "url", expect : { fn : "equal.mediafragmentXYWH", input : "3", out : "500"}, label : "a equal.mediafragment", data : "http://foo.com?foo=1#xywh=1,100,400,500*1.2,2.3"},{ fn : "url", expect : { fn : "equal.mediafragmentXYWHSpd", input : "1", out : "2.3"}, label : "a equal.mediafragmentSpeed", data : "http://foo.com?foo=1#xywh=1,100,400,500*1.2,2.3"}]); - Test.test("filter.selectors.json",[{ fn : "url", expect : { fn : "testParsed", input : "myid", out : true}, label : "myid exists", data : "http://foo.com?foo=1#foo*&-sometag&-someid&myid"},{ fn : "url", expect : { fn : "testParsed", input : "tag", out : true}, label : "tag exists", data : "http://foo.com?foo=1#tag=bar"},{ fn : "url", expect : { fn : "testParsed", input : "tag", out : true}, label : "tag exists", data : "http://foo.com?foo=1#-tag=bar"},{ fn : "url", expect : { fn : "testParsed", input : "price", out : true}, label : "filter test", data : "http://foo.com?foo=1#price=>2"},{ fn : "filter", expect : { fn : "testProperty", input : ["tag","bar"], out : true}, data : "tag=bar"},{ fn : "filter", expect : { fn : "testProperty", input : ["tag","foo"], out : false}, data : "-tag=foo"},{ fn : "filter", expect : { fn : "testProperty", input : ["tag","foo"], out : false}, data : "-tag*=foo"},{ fn : "filter", expect : { fn : "testProperty", input : ["tag","3"], out : false}, data : "-tag=>2"},{ fn : "filter", expect : { fn : "testProperty", input : ["price","1"], out : false}, data : "price=>2"},{ fn : "filter", expect : { fn : "testProperty", input : ["price","5"], out : false}, data : "price=<2"},{ fn : "filter", expect : { fn : "testProperty", input : ["price","1"], out : true}, data : "price=<2"},{ fn : "url", expect : { fn : "testFilterDeep", input : ["foo"], out : 1}, label : "foo should be deep", data : "#foo*"},{ fn : "url", expect : { fn : "testFilterDeep", input : ["foo"], out : 2}, label : "foo should be deep incl. embeds", data : "#foo**"}]); + Test.test("t.json",[{ fn : "url", expect : { fn : "equal.x", input : "t", out : "1"}, label : "a equal.x", data : "http://foo.com?foo=1#t=1"},{ fn : "url", expect : { fn : "equal.x", input : "t", out : "-1"}, label : "a equal.x", data : "http://foo.com?foo=1#t=-1"},{ fn : "url", expect : { fn : "equal.x", input : "t", out : "-1.02"}, label : "a equal.x", data : "http://foo.com?foo=1#t=-1.02"},{ fn : "url", expect : { fn : "equal.xy", input : "t", out : "1,2"}, label : "a equal.xy", data : "http://foo.com?foo=1#t=1,2,3"},{ fn : "url", expect : { fn : "equal.xyz", input : "t", out : "1,2,3"}, label : "a equal.xyz", data : "http://foo.com?foo=1#t=1,2,3"},{ fn : "url", expect : { fn : "equal.xyz", input : "t", out : "1,-2,3"}, label : "a equal.xyz", data : "http://foo.com?foo=1#t=1,-2,3"},{ fn : "url", expect : { fn : "equal.xy", input : "t", out : "1,100"}, label : "a equal.xy", data : "http://foo.com?foo=1#t=1,100"},{ fn : "url", expect : { fn : "testBrowserOverride", input : "t", out : true}, label : "browser URI can override t (defined in asset)", data : "http://foo.com?foo=1#t=2,500"},{ fn : "url", expect : { fn : "equal.mediafragmentT", input : "3", out : "500"}, label : "a equal.mediafragment", data : "http://foo.com?foo=1#t=1,100,400,500"},{ fn : "url", expect : { fn : "equal.mediafragmentT", input : "3", out : "500"}, label : "a equal.mediafragment loop", data : "http://foo.com?foo=1#t=l:1,100,400,500"}]); + Test.test("s.json",[{ fn : "url", expect : { fn : "equal.x", input : "s", out : "1"}, label : "playback speed", data : "http://foo.com?foo=1#s=1"},{ fn : "url", expect : { fn : "equal.x", input : "s", out : "0.5"}, label : "playback speed", data : "http://foo.com?foo=1#s=0.5"},{ fn : "url", expect : { fn : "equal.x", input : "s", out : "-0.5"}, label : "playback speed", data : "http://foo.com?foo=1#s=-0.5"}]); + Test.test("uv.json",[{ fn : "url", expect : { fn : "equal.xy", input : "uv", out : "1.2,2.2"}, label : "equal.string uv", data : "http://foo.com?foo=1#uv=1.2,2.2"},{ fn : "url", expect : { fn : "equal.xyz", input : "uv", out : "1.2,2.2,1"}, label : "equal.string uv", data : "http://foo.com?foo=1#uv=1.2,2.2,+1,+1"}]); + Test.test("filter.selectors.json",[{ fn : "url", expect : { fn : "testParsed", input : "myid", out : true}, label : "myid exists", data : "http://foo.com?foo=1#foo*&-sometag&-someid&myid"},{ fn : "url", expect : { fn : "testParsed", input : "tag", out : true}, label : "tag exists", data : "http://foo.com?foo=1#tag=bar"},{ fn : "url", expect : { fn : "testParsed", input : "tag", out : true}, label : "tag exists", data : "http://foo.com?foo=1#-tag=bar"},{ fn : "url", expect : { fn : "testParsed", input : "price", out : true}, label : "filter test", data : "http://foo.com?foo=1#price=>2"},{ fn : "filter", expect : { fn : "testProperty", input : ["tag","bar"], out : true}, data : "tag=bar"},{ fn : "filter", expect : { fn : "testProperty", input : ["tag","foo"], out : false}, data : "-tag=foo"},{ fn : "filter", expect : { fn : "testProperty", input : ["tag","foo"], out : false}, data : "-tag*=foo"},{ fn : "filter", expect : { fn : "testProperty", input : ["tag","3"], out : false}, data : "-tag=>2"},{ fn : "filter", expect : { fn : "testProperty", input : ["price","1"], out : false}, data : "price=>2"},{ fn : "filter", expect : { fn : "testProperty", input : ["price","5"], out : false}, data : "price=<2"},{ fn : "filter", expect : { fn : "testProperty", input : ["price","1"], out : true}, data : "price=<2"},{ fn : "url", expect : { fn : "testFilterDeep", input : ["foo"], out : 1}, label : "foo should be deep", data : "#foo*"}]); if(Test.errors > 1) { - console.log("src/Test.hx:24:","\n-----\n[ ❌] " + Test.errors + " errors :/"); + console.log("src/Test.hx:25:","\n-----\n[ ❌] " + Test.errors + " errors :/"); } }; Test.test = function(topic,spec) { - console.log("src/Test.hx:28:","\n[.] running " + topic); + console.log("src/Test.hx:29:","\n[ . ] running " + topic); var Filter = xrfragment_Filter; var _g = 0; var _g1 = spec.length; @@ -215,14 +284,11 @@ Test.test = function(topic,spec) { if(item.expect.fn == "equal.mediafragmentT") { valid = Test.equalMediaFragment(res,item,"t"); } - if(item.expect.fn == "equal.mediafragmentXYWH") { - valid = Test.equalMediaFragment(res,item,"xywh"); + if(item.expect.fn == "equal.mediafragmentUV") { + valid = Test.equalMediaFragment(res,item,"uv"); } - if(item.expect.fn == "equal.mediafragmentTSpd") { - valid = Test.equalMediaFragmentSpd(res,item,"t"); - } - if(item.expect.fn == "equal.mediafragmentXYWHSpd") { - valid = Test.equalMediaFragmentSpd(res,item,"xywh"); + if(item.expect.fn == "equal.mediafragmentS") { + valid = Test.equalMediaFragment(res,item,"s"); } if(item.expect.fn == "testFilterRoot") { valid = Object.prototype.hasOwnProperty.call(res,item.expect.input[0]) && res[item.expect.input[0]].filter.get().root == item.expect.out; @@ -268,16 +334,55 @@ Test.equalMediaFragment = function(res,item,key) { if(!item.expect.out && !res[item.expect.input]) { return true; } else { - return res[key].floats[Std.parseInt(item.expect.input)] == Std.parseInt(item.expect.out); + return res[key].floats[Std.parseInt(item.expect.input)] == parseFloat(item.expect.out); } }; -Test.equalMediaFragmentSpd = function(res,item,key) { - if(!item.expect.out && !res[item.expect.input]) { - return true; +var haxe_Exception = function(message,previous,native) { + Error.call(this,message); + this.message = message; + this.__previousException = previous; + this.__nativeException = native != null ? native : this; +}; +haxe_Exception.__name__ = true; +haxe_Exception.caught = function(value) { + if(((value) instanceof haxe_Exception)) { + return value; + } else if(((value) instanceof Error)) { + return new haxe_Exception(value.message,null,value); } else { - return res[key].speed[Std.parseInt(item.expect.input)] == parseFloat(item.expect.out); + return new haxe_ValueException(value,null,value); } }; +haxe_Exception.thrown = function(value) { + if(((value) instanceof haxe_Exception)) { + return value.get_native(); + } else if(((value) instanceof Error)) { + return value; + } else { + var e = new haxe_ValueException(value); + return e; + } +}; +haxe_Exception.__super__ = Error; +haxe_Exception.prototype = $extend(Error.prototype,{ + unwrap: function() { + return this.__nativeException; + } + ,get_native: function() { + return this.__nativeException; + } + ,__properties__: {get_native:"get_native"} +}); +var haxe__$Template_TemplateExpr = $hxEnums["haxe._Template.TemplateExpr"] = { __ename__:true,__constructs__:null + ,OpVar: ($_=function(v) { return {_hx_index:0,v:v,__enum__:"haxe._Template.TemplateExpr",toString:$estr}; },$_._hx_name="OpVar",$_.__params__ = ["v"],$_) + ,OpExpr: ($_=function(expr) { return {_hx_index:1,expr:expr,__enum__:"haxe._Template.TemplateExpr",toString:$estr}; },$_._hx_name="OpExpr",$_.__params__ = ["expr"],$_) + ,OpIf: ($_=function(expr,eif,eelse) { return {_hx_index:2,expr:expr,eif:eif,eelse:eelse,__enum__:"haxe._Template.TemplateExpr",toString:$estr}; },$_._hx_name="OpIf",$_.__params__ = ["expr","eif","eelse"],$_) + ,OpStr: ($_=function(str) { return {_hx_index:3,str:str,__enum__:"haxe._Template.TemplateExpr",toString:$estr}; },$_._hx_name="OpStr",$_.__params__ = ["str"],$_) + ,OpBlock: ($_=function(l) { return {_hx_index:4,l:l,__enum__:"haxe._Template.TemplateExpr",toString:$estr}; },$_._hx_name="OpBlock",$_.__params__ = ["l"],$_) + ,OpForeach: ($_=function(expr,loop) { return {_hx_index:5,expr:expr,loop:loop,__enum__:"haxe._Template.TemplateExpr",toString:$estr}; },$_._hx_name="OpForeach",$_.__params__ = ["expr","loop"],$_) + ,OpMacro: ($_=function(name,params) { return {_hx_index:6,name:name,params:params,__enum__:"haxe._Template.TemplateExpr",toString:$estr}; },$_._hx_name="OpMacro",$_.__params__ = ["name","params"],$_) +}; +haxe__$Template_TemplateExpr.__constructs__ = [haxe__$Template_TemplateExpr.OpVar,haxe__$Template_TemplateExpr.OpExpr,haxe__$Template_TemplateExpr.OpIf,haxe__$Template_TemplateExpr.OpStr,haxe__$Template_TemplateExpr.OpBlock,haxe__$Template_TemplateExpr.OpForeach,haxe__$Template_TemplateExpr.OpMacro]; var haxe_iterators_ArrayIterator = function(array) { this.current = 0; this.array = array; @@ -291,6 +396,587 @@ haxe_iterators_ArrayIterator.prototype = { return this.array[this.current++]; } }; +var haxe_Template = function(str) { + var tokens = this.parseTokens(str); + this.expr = this.parseBlock(tokens); + if(!tokens.isEmpty()) { + throw haxe_Exception.thrown("Unexpected '" + Std.string(tokens.first().s) + "'"); + } +}; +haxe_Template.__name__ = true; +haxe_Template.prototype = { + execute: function(context,macros) { + this.macros = macros == null ? { } : macros; + this.context = context; + this.stack = new haxe_ds_List(); + this.buf = new StringBuf(); + this.run(this.expr); + return this.buf.b; + } + ,resolve: function(v) { + if(v == "__current__") { + return this.context; + } + if(Reflect.isObject(this.context)) { + var value = Reflect.getProperty(this.context,v); + if(value != null || Object.prototype.hasOwnProperty.call(this.context,v)) { + return value; + } + } + var _g_head = this.stack.h; + while(_g_head != null) { + var val = _g_head.item; + _g_head = _g_head.next; + var ctx = val; + var value = Reflect.getProperty(ctx,v); + if(value != null || Object.prototype.hasOwnProperty.call(ctx,v)) { + return value; + } + } + return Reflect.field(haxe_Template.globals,v); + } + ,parseTokens: function(data) { + var tokens = new haxe_ds_List(); + while(haxe_Template.splitter.match(data)) { + var p = haxe_Template.splitter.matchedPos(); + if(p.pos > 0) { + tokens.add({ p : HxOverrides.substr(data,0,p.pos), s : true, l : null}); + } + if(HxOverrides.cca(data,p.pos) == 58) { + tokens.add({ p : HxOverrides.substr(data,p.pos + 2,p.len - 4), s : false, l : null}); + data = haxe_Template.splitter.matchedRight(); + continue; + } + var parp = p.pos + p.len; + var npar = 1; + var params = []; + var part = ""; + while(true) { + var c = HxOverrides.cca(data,parp); + ++parp; + if(c == 40) { + ++npar; + } else if(c == 41) { + --npar; + if(npar <= 0) { + break; + } + } else if(c == null) { + throw haxe_Exception.thrown("Unclosed macro parenthesis"); + } + if(c == 44 && npar == 1) { + params.push(part); + part = ""; + } else { + part += String.fromCodePoint(c); + } + } + params.push(part); + tokens.add({ p : haxe_Template.splitter.matched(2), s : false, l : params}); + data = HxOverrides.substr(data,parp,data.length - parp); + } + if(data.length > 0) { + tokens.add({ p : data, s : true, l : null}); + } + return tokens; + } + ,parseBlock: function(tokens) { + var l = new haxe_ds_List(); + while(true) { + var t = tokens.first(); + if(t == null) { + break; + } + if(!t.s && (t.p == "end" || t.p == "else" || HxOverrides.substr(t.p,0,7) == "elseif ")) { + break; + } + l.add(this.parse(tokens)); + } + if(l.length == 1) { + return l.first(); + } + return haxe__$Template_TemplateExpr.OpBlock(l); + } + ,parse: function(tokens) { + var t = tokens.pop(); + var p = t.p; + if(t.s) { + return haxe__$Template_TemplateExpr.OpStr(p); + } + if(t.l != null) { + var pe = new haxe_ds_List(); + var _g = 0; + var _g1 = t.l; + while(_g < _g1.length) { + var p1 = _g1[_g]; + ++_g; + pe.add(this.parseBlock(this.parseTokens(p1))); + } + return haxe__$Template_TemplateExpr.OpMacro(p,pe); + } + var kwdEnd = function(kwd) { + var pos = -1; + var length = kwd.length; + if(HxOverrides.substr(p,0,length) == kwd) { + pos = length; + var _g_offset = 0; + var _g_s = HxOverrides.substr(p,length,null); + while(_g_offset < _g_s.length) { + var c = _g_s.charCodeAt(_g_offset++); + if(c == 32) { + ++pos; + } else { + break; + } + } + } + return pos; + }; + var pos = kwdEnd("if"); + if(pos > 0) { + p = HxOverrides.substr(p,pos,p.length - pos); + var e = this.parseExpr(p); + var eif = this.parseBlock(tokens); + var t = tokens.first(); + var eelse; + if(t == null) { + throw haxe_Exception.thrown("Unclosed 'if'"); + } + if(t.p == "end") { + tokens.pop(); + eelse = null; + } else if(t.p == "else") { + tokens.pop(); + eelse = this.parseBlock(tokens); + t = tokens.pop(); + if(t == null || t.p != "end") { + throw haxe_Exception.thrown("Unclosed 'else'"); + } + } else { + t.p = HxOverrides.substr(t.p,4,t.p.length - 4); + eelse = this.parse(tokens); + } + return haxe__$Template_TemplateExpr.OpIf(e,eif,eelse); + } + var pos = kwdEnd("foreach"); + if(pos >= 0) { + p = HxOverrides.substr(p,pos,p.length - pos); + var e = this.parseExpr(p); + var efor = this.parseBlock(tokens); + var t = tokens.pop(); + if(t == null || t.p != "end") { + throw haxe_Exception.thrown("Unclosed 'foreach'"); + } + return haxe__$Template_TemplateExpr.OpForeach(e,efor); + } + if(haxe_Template.expr_splitter.match(p)) { + return haxe__$Template_TemplateExpr.OpExpr(this.parseExpr(p)); + } + return haxe__$Template_TemplateExpr.OpVar(p); + } + ,parseExpr: function(data) { + var l = new haxe_ds_List(); + var expr = data; + while(haxe_Template.expr_splitter.match(data)) { + var p = haxe_Template.expr_splitter.matchedPos(); + var k = p.pos + p.len; + if(p.pos != 0) { + l.add({ p : HxOverrides.substr(data,0,p.pos), s : true}); + } + var p1 = haxe_Template.expr_splitter.matched(0); + l.add({ p : p1, s : p1.indexOf("\"") >= 0}); + data = haxe_Template.expr_splitter.matchedRight(); + } + if(data.length != 0) { + var _g_offset = 0; + var _g_s = data; + while(_g_offset < _g_s.length) { + var _g1_key = _g_offset; + var _g1_value = _g_s.charCodeAt(_g_offset++); + var i = _g1_key; + var c = _g1_value; + if(c != 32) { + l.add({ p : HxOverrides.substr(data,i,null), s : true}); + break; + } + } + } + var e; + try { + e = this.makeExpr(l); + if(!l.isEmpty()) { + throw haxe_Exception.thrown(l.first().p); + } + } catch( _g ) { + var _g1 = haxe_Exception.caught(_g).unwrap(); + if(typeof(_g1) == "string") { + var s = _g1; + throw haxe_Exception.thrown("Unexpected '" + s + "' in " + expr); + } else { + throw _g; + } + } + return function() { + try { + return e(); + } catch( _g ) { + var exc = haxe_Exception.caught(_g).unwrap(); + throw haxe_Exception.thrown("Error : " + Std.string(exc) + " in " + expr); + } + }; + } + ,makeConst: function(v) { + haxe_Template.expr_trim.match(v); + v = haxe_Template.expr_trim.matched(1); + if(HxOverrides.cca(v,0) == 34) { + var str = HxOverrides.substr(v,1,v.length - 2); + return function() { + return str; + }; + } + if(haxe_Template.expr_int.match(v)) { + var i = Std.parseInt(v); + return function() { + return i; + }; + } + if(haxe_Template.expr_float.match(v)) { + var f = parseFloat(v); + return function() { + return f; + }; + } + var me = this; + return function() { + return me.resolve(v); + }; + } + ,makePath: function(e,l) { + var p = l.first(); + if(p == null || p.p != ".") { + return e; + } + l.pop(); + var field = l.pop(); + if(field == null || !field.s) { + throw haxe_Exception.thrown(field.p); + } + var f = field.p; + haxe_Template.expr_trim.match(f); + f = haxe_Template.expr_trim.matched(1); + return this.makePath(function() { + return Reflect.field(e(),f); + },l); + } + ,makeExpr: function(l) { + return this.makePath(this.makeExpr2(l),l); + } + ,skipSpaces: function(l) { + var p = l.first(); + while(p != null) { + var _g_offset = 0; + var _g_s = p.p; + while(_g_offset < _g_s.length) { + var c = _g_s.charCodeAt(_g_offset++); + if(c != 32) { + return; + } + } + l.pop(); + p = l.first(); + } + } + ,makeExpr2: function(l) { + this.skipSpaces(l); + var p = l.pop(); + this.skipSpaces(l); + if(p == null) { + throw haxe_Exception.thrown(""); + } + if(p.s) { + return this.makeConst(p.p); + } + switch(p.p) { + case "!": + var e = this.makeExpr(l); + return function() { + var v = e(); + if(v != null) { + return v == false; + } else { + return true; + } + }; + case "(": + this.skipSpaces(l); + var e1 = this.makeExpr(l); + this.skipSpaces(l); + var p1 = l.pop(); + if(p1 == null || p1.s) { + throw haxe_Exception.thrown(p1); + } + if(p1.p == ")") { + return e1; + } + this.skipSpaces(l); + var e2 = this.makeExpr(l); + this.skipSpaces(l); + var p2 = l.pop(); + this.skipSpaces(l); + if(p2 == null || p2.p != ")") { + throw haxe_Exception.thrown(p2); + } + switch(p1.p) { + case "!=": + return function() { + return e1() != e2(); + }; + case "&&": + return function() { + return e1() && e2(); + }; + case "*": + return function() { + return e1() * e2(); + }; + case "+": + return function() { + return e1() + e2(); + }; + case "-": + return function() { + return e1() - e2(); + }; + case "/": + return function() { + return e1() / e2(); + }; + case "<": + return function() { + return e1() < e2(); + }; + case "<=": + return function() { + return e1() <= e2(); + }; + case "==": + return function() { + return e1() == e2(); + }; + case ">": + return function() { + return e1() > e2(); + }; + case ">=": + return function() { + return e1() >= e2(); + }; + case "||": + return function() { + return e1() || e2(); + }; + default: + throw haxe_Exception.thrown("Unknown operation " + p1.p); + } + break; + case "-": + var e3 = this.makeExpr(l); + return function() { + return -e3(); + }; + } + throw haxe_Exception.thrown(p.p); + } + ,run: function(e) { + switch(e._hx_index) { + case 0: + var v = e.v; + var _this = this.buf; + var x = Std.string(this.resolve(v)); + _this.b += Std.string(x); + break; + case 1: + var e1 = e.expr; + var _this = this.buf; + var x = Std.string(e1()); + _this.b += Std.string(x); + break; + case 2: + var e1 = e.expr; + var eif = e.eif; + var eelse = e.eelse; + var v = e1(); + if(v == null || v == false) { + if(eelse != null) { + this.run(eelse); + } + } else { + this.run(eif); + } + break; + case 3: + var str = e.str; + this.buf.b += str == null ? "null" : "" + str; + break; + case 4: + var l = e.l; + var _g_head = l.h; + while(_g_head != null) { + var val = _g_head.item; + _g_head = _g_head.next; + var e1 = val; + this.run(e1); + } + break; + case 5: + var e1 = e.expr; + var loop = e.loop; + var v = e1(); + try { + var x = $getIterator(v); + if(x.hasNext == null) { + throw haxe_Exception.thrown(null); + } + v = x; + } catch( _g ) { + try { + if(v.hasNext == null) { + throw haxe_Exception.thrown(null); + } + } catch( _g1 ) { + throw haxe_Exception.thrown("Cannot iter on " + Std.string(v)); + } + } + this.stack.push(this.context); + var v1 = v; + var ctx = v1; + while(ctx.hasNext()) { + var ctx1 = ctx.next(); + this.context = ctx1; + this.run(loop); + } + this.context = this.stack.pop(); + break; + case 6: + var m = e.name; + var params = e.params; + var v = Reflect.field(this.macros,m); + var pl = []; + var old = this.buf; + pl.push($bind(this,this.resolve)); + var _g_head = params.h; + while(_g_head != null) { + var val = _g_head.item; + _g_head = _g_head.next; + var p = val; + if(p._hx_index == 0) { + var v1 = p.v; + pl.push(this.resolve(v1)); + } else { + this.buf = new StringBuf(); + this.run(p); + pl.push(this.buf.b); + } + } + this.buf = old; + try { + var _this = this.buf; + var x = Std.string(v.apply(this.macros,pl)); + _this.b += Std.string(x); + } catch( _g ) { + var e = haxe_Exception.caught(_g).unwrap(); + var plstr; + try { + plstr = pl.join(","); + } catch( _g1 ) { + plstr = "???"; + } + var msg = "Macro call " + m + "(" + plstr + ") failed (" + Std.string(e) + ")"; + throw haxe_Exception.thrown(msg); + } + break; + } + } +}; +var haxe_ValueException = function(value,previous,native) { + haxe_Exception.call(this,String(value),previous,native); + this.value = value; +}; +haxe_ValueException.__name__ = true; +haxe_ValueException.__super__ = haxe_Exception; +haxe_ValueException.prototype = $extend(haxe_Exception.prototype,{ + unwrap: function() { + return this.value; + } +}); +var haxe_ds_List = function() { + this.length = 0; +}; +haxe_ds_List.__name__ = true; +haxe_ds_List.prototype = { + add: function(item) { + var x = new haxe_ds__$List_ListNode(item,null); + if(this.h == null) { + this.h = x; + } else { + this.q.next = x; + } + this.q = x; + this.length++; + } + ,push: function(item) { + var x = new haxe_ds__$List_ListNode(item,this.h); + this.h = x; + if(this.q == null) { + this.q = x; + } + this.length++; + } + ,first: function() { + if(this.h == null) { + return null; + } else { + return this.h.item; + } + } + ,pop: function() { + if(this.h == null) { + return null; + } + var x = this.h.item; + this.h = this.h.next; + if(this.h == null) { + this.q = null; + } + this.length--; + return x; + } + ,isEmpty: function() { + return this.h == null; + } + ,toString: function() { + var s_b = ""; + var first = true; + var l = this.h; + s_b += "{"; + while(l != null) { + if(first) { + first = false; + } else { + s_b += ", "; + } + s_b += Std.string(Std.string(l.item)); + l = l.next; + } + s_b += "}"; + return s_b; + } +}; +var haxe_ds__$List_ListNode = function(item,next) { + this.item = item; + this.next = next; +}; +haxe_ds__$List_ListNode.__name__ = true; var js_Boot = function() { }; js_Boot.__name__ = true; js_Boot.__string_rec = function(o,s) { @@ -308,6 +994,34 @@ js_Boot.__string_rec = function(o,s) { case "function": return ""; case "object": + if(o.__enum__) { + var e = $hxEnums[o.__enum__]; + var con = e.__constructs__[o._hx_index]; + var n = con._hx_name; + if(con.__params__) { + s = s + "\t"; + return n + "(" + ((function($this) { + var $r; + var _g = []; + { + var _g1 = 0; + var _g2 = con.__params__; + while(true) { + if(!(_g1 < _g2.length)) { + break; + } + var p = _g2[_g1]; + _g1 = _g1 + 1; + _g.push(js_Boot.__string_rec(o[p],s)); + } + } + $r = _g; + return $r; + }(this))).join(",") + ")"; + } else { + return n; + } + } if(((o) instanceof Array)) { var str = "["; s += "\t"; @@ -492,26 +1206,32 @@ var xrfragment_Parser = $hx_exports["xrfragment"]["Parser"] = function() { }; xrfragment_Parser.__name__ = true; xrfragment_Parser.parse = function(key,value,store,index) { var Frag_h = Object.create(null); - Frag_h["#"] = xrfragment_XRF.ASSET | xrfragment_XRF.T_PREDEFINED_VIEW | xrfragment_XRF.PV_EXECUTE; - Frag_h["src"] = xrfragment_XRF.ASSET | xrfragment_XRF.T_URL; - Frag_h["href"] = xrfragment_XRF.ASSET | xrfragment_XRF.T_URL | xrfragment_XRF.T_PREDEFINED_VIEW; - Frag_h["tag"] = xrfragment_XRF.ASSET | xrfragment_XRF.T_STRING; + Frag_h["#"] = xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_PREDEFINED_VIEW | xrfragment_XRF.PV_EXECUTE; + Frag_h["src"] = xrfragment_XRF.T_URL; + Frag_h["href"] = xrfragment_XRF.T_URL | xrfragment_XRF.T_PREDEFINED_VIEW; + Frag_h["tag"] = xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_STRING; Frag_h["pos"] = xrfragment_XRF.PV_OVERRIDE | xrfragment_XRF.T_VECTOR3 | xrfragment_XRF.T_STRING | xrfragment_XRF.METADATA | xrfragment_XRF.NAVIGATOR; Frag_h["rot"] = xrfragment_XRF.QUERY_OPERATOR | xrfragment_XRF.PV_OVERRIDE | xrfragment_XRF.T_VECTOR3 | xrfragment_XRF.METADATA | xrfragment_XRF.NAVIGATOR; - Frag_h["t"] = xrfragment_XRF.ASSET | xrfragment_XRF.PV_OVERRIDE | xrfragment_XRF.T_FLOAT | xrfragment_XRF.T_VECTOR2 | xrfragment_XRF.T_MEDIAFRAG | xrfragment_XRF.NAVIGATOR | xrfragment_XRF.METADATA; - Frag_h["xywh"] = xrfragment_XRF.ASSET | xrfragment_XRF.PV_OVERRIDE | xrfragment_XRF.T_FLOAT | xrfragment_XRF.T_VECTOR2 | xrfragment_XRF.T_MEDIAFRAG | xrfragment_XRF.NAVIGATOR | xrfragment_XRF.METADATA; - 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.METADATA | xrfragment_XRF.PROMPT; + Frag_h["t"] = xrfragment_XRF.PV_OVERRIDE | xrfragment_XRF.T_FLOAT | xrfragment_XRF.T_VECTOR2 | xrfragment_XRF.NAVIGATOR | xrfragment_XRF.METADATA; + Frag_h["s"] = xrfragment_XRF.PV_OVERRIDE | xrfragment_XRF.T_MEDIAFRAG; + Frag_h["loop"] = xrfragment_XRF.PV_OVERRIDE; + Frag_h["uv"] = xrfragment_XRF.T_VECTOR2 | xrfragment_XRF.T_MEDIAFRAG; + Frag_h["namespace"] = xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_STRING; + Frag_h["SPDX"] = xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_STRING; + Frag_h["unit"] = xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_STRING; + Frag_h["description"] = xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_STRING; var keyStripped = key.replace(xrfragment_XRF.operators.r,""); var isPVDynamic = key.length > 0 && !Object.prototype.hasOwnProperty.call(Frag_h,key); - var isPVDefault = value.length == 0 && key.length > 0 && key == "#"; if(isPVDynamic) { var v = new xrfragment_XRF(key,xrfragment_XRF.PV_EXECUTE | xrfragment_XRF.NAVIGATOR,index); v.validate(value); - v.flags = xrfragment_XRF.set(xrfragment_XRF.T_DYNAMIC,v.flags); + v.flags = xrfragment_XRF.set(xrfragment_XRF.T_DYNAMICKEY,v.flags); + if(!Object.prototype.hasOwnProperty.call(Frag_h,key)) { + v.flags = xrfragment_XRF.set(xrfragment_XRF.CUSTOMFRAG,v.flags); + } + if(value.length == 0) { + v.flags = xrfragment_XRF.set(xrfragment_XRF.T_DYNAMICKEYVALUE,v.flags); + } store[keyStripped] = v; return true; } @@ -529,7 +1249,7 @@ xrfragment_Parser.parse = function(key,value,store,index) { if(typeof(value) == "string") { v.guessType(v,value); } - v.noXRF = true; + v.flags = xrfragment_XRF.set(xrfragment_XRF.CUSTOMFRAG,v.flags); store[keyStripped] = v; } return true; @@ -552,8 +1272,12 @@ xrfragment_URI.parse = function(url,filter) { var key = splitByEqual[0]; var value = ""; if(splitByEqual.length > 1) { - var s = regexPlus.split(splitByEqual[1]).join(" "); - value = decodeURIComponent(s.split("+").join(" ")); + if(xrfragment_XRF.isVector.match(splitByEqual[1])) { + value = splitByEqual[1]; + } else { + var s = regexPlus.split(splitByEqual[1]).join(" "); + value = decodeURIComponent(s.split("+").join(" ")); + } } var ok = xrfragment_Parser.parse(key,value,store,i); } @@ -571,9 +1295,22 @@ xrfragment_URI.parse = function(url,filter) { } return store; }; +xrfragment_URI.template = function(uri,vars) { + var parts = uri.split("#"); + if(parts.length == 1) { + return uri; + } + var frag = parts[1]; + frag = StringTools.replace(frag,"{","::"); + frag = StringTools.replace(frag,"}","::"); + frag = new haxe_Template(frag).execute(vars); + frag = StringTools.replace(frag,"null",""); + parts[1] = frag; + return parts.join("#"); +}; var xrfragment_XRF = $hx_exports["xrfragment"]["XRF"] = function(_fragment,_flags,_index) { - this.speed = []; this.floats = []; + this.shift = []; this.fragment = _fragment; this.flags = _flags; this.index = _index; @@ -606,10 +1343,21 @@ xrfragment_XRF.prototype = { } ,guessType: function(v,str) { v.string = str; + if(xrfragment_XRF.isReset.match(v.fragment)) { + v.reset = true; + } + if(v.fragment == "loop") { + v.loop = true; + } if(typeof(str) != "string") { return; } if(str.length > 0) { + if(xrfragment_XRF.isXRFScheme.match(str)) { + v.xrfScheme = true; + str = str.replace(xrfragment_XRF.isXRFScheme.r,""); + v.string = str; + } if(str.split(",").length > 1) { var xyzn = str.split(","); if(xyzn.length > 0) { @@ -625,7 +1373,8 @@ xrfragment_XRF.prototype = { var _g1 = xyzn.length; while(_g < _g1) { var i = _g++; - v.floats.push(parseFloat(xyzn[i])); + v.shift.push(xrfragment_XRF.isShift.match(xyzn[i])); + v.floats.push(parseFloat(xyzn[i].replace(xrfragment_XRF.isShift.r,""))); } } if(xrfragment_XRF.isColor.match(str)) { @@ -638,19 +1387,7 @@ xrfragment_XRF.prototype = { if(xrfragment_XRF.isInt.match(str)) { v.int = Std.parseInt(str); v.x = v.int; - } - if(xrfragment_XRF.isMediaFrag.match(str)) { - var speed = str.split("*"); - v.speed = []; - if(speed.length > 1) { - var values = speed[1].split(","); - var _g = 0; - var _g1 = values.length; - while(_g < _g1) { - var i = _g++; - v.speed.push(parseFloat(values[i])); - } - } + v.floats.push(v.x); } v.filter = new xrfragment_Filter(v.fragment + "=" + v.string); } else { @@ -658,20 +1395,32 @@ xrfragment_XRF.prototype = { } } }; +function $getIterator(o) { if( o instanceof Array ) return new haxe_iterators_ArrayIterator(o); else return o.iterator(); } +function $bind(o,m) { if( m == null ) return null; if( m.__id__ == null ) m.__id__ = $global.$haxeUID++; var f; if( o.hx__closures__ == null ) o.hx__closures__ = {}; else f = o.hx__closures__[m.__id__]; if( f == null ) { f = m.bind(o); o.hx__closures__[m.__id__] = f; } return f; } +$global.$haxeUID |= 0; if(typeof(performance) != "undefined" ? typeof(performance.now) == "function" : false) { HxOverrides.now = performance.now.bind(performance); } +if( String.fromCodePoint == null ) String.fromCodePoint = function(c) { return c < 0x10000 ? String.fromCharCode(c) : String.fromCharCode((c>>10)+0xD7C0)+String.fromCharCode((c&0x3FF)+0xDC00); } String.__name__ = true; Array.__name__ = true; js_Boot.__toStr = ({ }).toString; Test.errors = 0; +haxe_Template.splitter = new EReg("(::[A-Za-z0-9_ ()&|!+=/><*.\"-]+::|\\$\\$([A-Za-z0-9_-]+)\\()",""); +haxe_Template.expr_splitter = new EReg("(\\(|\\)|[ \r\n\t]*\"[^\"]*\"[ \r\n\t]*|[!+=/><*.&|-]+)",""); +haxe_Template.expr_trim = new EReg("^[ ]*([^ ]+)[ ]*$",""); +haxe_Template.expr_int = new EReg("^[0-9]+$",""); +haxe_Template.expr_float = new EReg("^([+-]?)(?=\\d|,\\d)\\d*(,\\d*)?([Ee]([+-]?\\d+))?$",""); +haxe_Template.globals = { }; +haxe_Template.hxKeepArrayIterator = new haxe_iterators_ArrayIterator([]); xrfragment_Parser.error = ""; xrfragment_Parser.debug = false; -xrfragment_XRF.ASSET = 1; +xrfragment_URI.__meta__ = { statics : { template : { keep : null}}}; +xrfragment_XRF.IMMUTABLE = 1; xrfragment_XRF.PROP_BIND = 2; xrfragment_XRF.QUERY_OPERATOR = 4; xrfragment_XRF.PROMPT = 8; -xrfragment_XRF.ROUNDROBIN = 16; +xrfragment_XRF.CUSTOMFRAG = 16; xrfragment_XRF.NAVIGATOR = 32; xrfragment_XRF.METADATA = 64; xrfragment_XRF.PV_OVERRIDE = 128; @@ -685,7 +1434,8 @@ xrfragment_XRF.T_URL = 262144; xrfragment_XRF.T_PREDEFINED_VIEW = 524288; xrfragment_XRF.T_STRING = 1048576; xrfragment_XRF.T_MEDIAFRAG = 2097152; -xrfragment_XRF.T_DYNAMIC = 4194304; +xrfragment_XRF.T_DYNAMICKEY = 4194304; +xrfragment_XRF.T_DYNAMICKEYVALUE = 8388608; 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]+$",""); @@ -693,12 +1443,15 @@ xrfragment_XRF.isVector = new EReg("([,]+|\\w)",""); xrfragment_XRF.isUrl = new EReg("(://)?\\..*",""); xrfragment_XRF.isUrlOrPretypedView = new EReg("(^#|://)?\\..*",""); xrfragment_XRF.isString = new EReg(".*",""); -xrfragment_XRF.operators = new EReg("(^-|[\\*]+)",""); +xrfragment_XRF.operators = new EReg("(^[-]|^[!]|[\\*]$)","g"); xrfragment_XRF.isProp = new EReg("^.*=[><=]?",""); xrfragment_XRF.isExclude = new EReg("^-",""); xrfragment_XRF.isDeep = new EReg("\\*",""); xrfragment_XRF.isNumber = new EReg("^[0-9\\.]+$",""); -xrfragment_XRF.isMediaFrag = new EReg("^[0-9\\.,\\*]+$",""); +xrfragment_XRF.isMediaFrag = new EReg("^([0-9\\.,\\*+-]+)$",""); +xrfragment_XRF.isReset = new EReg("^!",""); +xrfragment_XRF.isShift = new EReg("^(\\+|--)",""); +xrfragment_XRF.isXRFScheme = new EReg("^xrf://",""); Test.main(); -})({}); +})(typeof window != "undefined" ? window : typeof global != "undefined" ? global : typeof self != "undefined" ? self : this); var xrfragment = $hx_exports["xrfragment"]; diff --git a/test/generated/test.py b/test/generated/test.py index 6ee835a..928eb2a 100644 --- a/test/generated/test.py +++ b/test/generated/test.py @@ -3,10 +3,11 @@ import sys import math as python_lib_Math import math as Math import inspect as python_lib_Inspect +import re as python_lib_Re import sys as python_lib_Sys import functools as python_lib_Functools -import re as python_lib_Re import traceback as python_lib_Traceback +from io import StringIO as python_lib_io_StringIO import urllib.parse as python_lib_urllib_Parse @@ -54,6 +55,7 @@ class Enum: else: return self.tag + '(' + (', '.join(str(v) for v in self.params)) + ')' +Enum._hx_class = Enum class Class: pass @@ -130,17 +132,53 @@ class EReg: replace = _hx_local_0 return python_lib_Re.sub(self.pattern,replace,s,(0 if (self._hx_global) else 1)) +EReg._hx_class = EReg class Reflect: _hx_class_name = "Reflect" __slots__ = () - _hx_statics = ["field", "deleteField", "copy"] + _hx_statics = ["field", "getProperty", "callMethod", "isObject", "deleteField", "copy"] @staticmethod def field(o,field): return python_Boot.field(o,field) + @staticmethod + def getProperty(o,field): + if (o is None): + return None + if (field in python_Boot.keywords): + field = ("_hx_" + field) + elif ((((len(field) > 2) and ((ord(field[0]) == 95))) and ((ord(field[1]) == 95))) and ((ord(field[(len(field) - 1)]) != 95))): + field = ("_hx_" + field) + if isinstance(o,_hx_AnonObject): + return Reflect.field(o,field) + tmp = Reflect.field(o,("get_" + ("null" if field is None else field))) + if ((tmp is not None) and callable(tmp)): + return tmp() + else: + return Reflect.field(o,field) + + @staticmethod + def callMethod(o,func,args): + if callable(func): + return func(*args) + else: + return None + + @staticmethod + def isObject(v): + _g = Type.typeof(v) + tmp = _g.index + if (tmp == 4): + return True + elif (tmp == 6): + _g1 = _g.params[0] + return True + else: + return False + @staticmethod def deleteField(o,field): if (field in python_Boot.keywords): @@ -165,6 +203,7 @@ class Reflect: value = Reflect.field(o,f) setattr(o2,(("_hx_" + f) if ((f in python_Boot.keywords)) else (("_hx_" + f) if (((((len(f) > 2) and ((ord(f[0]) == 95))) and ((ord(f[1]) == 95))) and ((ord(f[(len(f) - 1)]) != 95)))) else f)),value) return o2 +Reflect._hx_class = Reflect class Std: @@ -359,6 +398,7 @@ class Std: if (r1 != x): return Std.parseFloat(r1) return Math.NaN +Std._hx_class = Std class Float: pass @@ -373,10 +413,21 @@ class Bool: pass class Dynamic: pass +class StringBuf: + _hx_class_name = "StringBuf" + __slots__ = ("b",) + _hx_fields = ["b"] + + def __init__(self): + self.b = python_lib_io_StringIO() + +StringBuf._hx_class = StringBuf + + class StringTools: _hx_class_name = "StringTools" __slots__ = () - _hx_statics = ["isSpace", "ltrim", "rtrim", "trim"] + _hx_statics = ["isSpace", "ltrim", "rtrim", "trim", "replace"] @staticmethod def isSpace(s,pos): @@ -414,25 +465,32 @@ class StringTools: def trim(s): return StringTools.ltrim(StringTools.rtrim(s)) + @staticmethod + def replace(s,sub,by): + _this = (list(s) if ((sub == "")) else s.split(sub)) + return by.join([python_Boot.toString1(x1,'') for x1 in _this]) +StringTools._hx_class = StringTools + class Test: _hx_class_name = "Test" __slots__ = () - _hx_statics = ["errors", "main", "test", "equalX", "equalXY", "equalXYZ", "equalMediaFragment", "equalMediaFragmentSpd"] + _hx_statics = ["errors", "main", "test", "equalX", "equalXY", "equalXYZ", "equalMediaFragment"] @staticmethod def main(): Test.test("url.json",[_hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testPredefinedView", 'input': "mypredefinedview", 'out': True}), 'label': "test predefined view executed", 'data': "http://foo.com?foo=1#mypredefinedview"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testPredefinedView", 'input': "another", 'out': True}), 'label': "test predefined view executed (multiple)", 'data': "http://foo.com?foo=1#mypredefinedview&another"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testPredefinedView", 'input': "mypredefinedview", 'out': True}), 'label': "test predefined view executed (multiple)", 'data': "http://foo.com?foo=1#mypredefinedview&another"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testParsed", 'input': "mycustom", 'out': True}), 'label': "test custom property", 'data': "http://foo.com?foo=1#mycustom=foo"})]) Test.test("pos.json",[_hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.string", 'input': "pos", 'out': "1.2,2.2"}), 'label': "equal.string", 'data': "http://foo.com?foo=1#pos=1.2,2.2"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xyz", 'input': "pos", 'out': "1.2,2.2,3"}), 'label': "equal.xyz", 'data': "http://foo.com?foo=1#pos=1.2,2.2,3"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xyz", 'input': "pos", 'out': "1,2,3"}), 'label': "pos equal.xyz", 'data': "http://foo.com?foo=1#pos=1,2,3"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.string", 'input': "pos", 'out': "world2"}), 'label': "pos equal.xyz", 'data': "http://foo.com?foo=1#pos=world2"})]) - Test.test("t.json",[_hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.x", 'input': "t", 'out': "1"}), 'label': "a equal.x", 'data': "http://foo.com?foo=1#t=1"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.x", 'input': "t", 'out': "-1"}), 'label': "a equal.x", 'data': "http://foo.com?foo=1#t=-1"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.x", 'input': "t", 'out': "-1.02"}), 'label': "a equal.x", 'data': "http://foo.com?foo=1#t=-1.02"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xy", 'input': "t", 'out': "1,2"}), 'label': "a equal.xy", 'data': "http://foo.com?foo=1#t=1,2,3"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xyz", 'input': "t", 'out': "1,2,3"}), 'label': "a equal.xyz", 'data': "http://foo.com?foo=1#t=1,2,3"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xyz", 'input': "t", 'out': "1,-2,3"}), 'label': "a equal.xyz", 'data': "http://foo.com?foo=1#t=1,-2,3"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xy", 'input': "t", 'out': "1,100"}), 'label': "a equal.xy", 'data': "http://foo.com?foo=1#t=1,100"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testBrowserOverride", 'input': "t", 'out': True}), 'label': "browser URI can override t (defined in asset)", 'data': "http://foo.com?foo=1#t=2,500"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.mediafragmentT", 'input': "3", 'out': "500"}), 'label': "a equal.mediafragment", 'data': "http://foo.com?foo=1#t=1,100,400,500*1.2,2.3"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.mediafragmentTSpd", 'input': "1", 'out': "2.3"}), 'label': "a equal.mediafragmentSpeed", 'data': "http://foo.com?foo=1#t=1,100,400,500*1.2,2.3"})]) - Test.test("xywh.json",[_hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.mediafragmentXYWH", 'input': "2", 'out': "1"}), 'label': "xywh", 'data': "http://foo.com?foo=1#xywh=0,0,1,1"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.mediafragmentXYWH", 'input': "3", 'out': "500"}), 'label': "a equal.mediafragment", 'data': "http://foo.com?foo=1#xywh=1,100,400,500*1.2,2.3"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.mediafragmentXYWHSpd", 'input': "1", 'out': "2.3"}), 'label': "a equal.mediafragmentSpeed", 'data': "http://foo.com?foo=1#xywh=1,100,400,500*1.2,2.3"})]) - Test.test("filter.selectors.json",[_hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testParsed", 'input': "myid", 'out': True}), 'label': "myid exists", 'data': "http://foo.com?foo=1#foo*&-sometag&-someid&myid"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testParsed", 'input': "tag", 'out': True}), 'label': "tag exists", 'data': "http://foo.com?foo=1#tag=bar"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testParsed", 'input': "tag", 'out': True}), 'label': "tag exists", 'data': "http://foo.com?foo=1#-tag=bar"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testParsed", 'input': "price", 'out': True}), 'label': "filter test", 'data': "http://foo.com?foo=1#price=>2"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["tag", "bar"], 'out': True}), 'data': "tag=bar"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["tag", "foo"], 'out': False}), 'data': "-tag=foo"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["tag", "foo"], 'out': False}), 'data': "-tag*=foo"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["tag", "3"], 'out': False}), 'data': "-tag=>2"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["price", "1"], 'out': False}), 'data': "price=>2"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["price", "5"], 'out': False}), 'data': "price=<2"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["price", "1"], 'out': True}), 'data': "price=<2"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testFilterDeep", 'input': ["foo"], 'out': 1}), 'label': "foo should be deep", 'data': "#foo*"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testFilterDeep", 'input': ["foo"], 'out': 2}), 'label': "foo should be deep incl. embeds", 'data': "#foo**"})]) + Test.test("t.json",[_hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.x", 'input': "t", 'out': "1"}), 'label': "a equal.x", 'data': "http://foo.com?foo=1#t=1"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.x", 'input': "t", 'out': "-1"}), 'label': "a equal.x", 'data': "http://foo.com?foo=1#t=-1"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.x", 'input': "t", 'out': "-1.02"}), 'label': "a equal.x", 'data': "http://foo.com?foo=1#t=-1.02"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xy", 'input': "t", 'out': "1,2"}), 'label': "a equal.xy", 'data': "http://foo.com?foo=1#t=1,2,3"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xyz", 'input': "t", 'out': "1,2,3"}), 'label': "a equal.xyz", 'data': "http://foo.com?foo=1#t=1,2,3"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xyz", 'input': "t", 'out': "1,-2,3"}), 'label': "a equal.xyz", 'data': "http://foo.com?foo=1#t=1,-2,3"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xy", 'input': "t", 'out': "1,100"}), 'label': "a equal.xy", 'data': "http://foo.com?foo=1#t=1,100"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testBrowserOverride", 'input': "t", 'out': True}), 'label': "browser URI can override t (defined in asset)", 'data': "http://foo.com?foo=1#t=2,500"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.mediafragmentT", 'input': "3", 'out': "500"}), 'label': "a equal.mediafragment", 'data': "http://foo.com?foo=1#t=1,100,400,500"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.mediafragmentT", 'input': "3", 'out': "500"}), 'label': "a equal.mediafragment loop", 'data': "http://foo.com?foo=1#t=l:1,100,400,500"})]) + Test.test("s.json",[_hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.x", 'input': "s", 'out': "1"}), 'label': "playback speed", 'data': "http://foo.com?foo=1#s=1"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.x", 'input': "s", 'out': "0.5"}), 'label': "playback speed", 'data': "http://foo.com?foo=1#s=0.5"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.x", 'input': "s", 'out': "-0.5"}), 'label': "playback speed", 'data': "http://foo.com?foo=1#s=-0.5"})]) + Test.test("uv.json",[_hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xy", 'input': "uv", 'out': "1.2,2.2"}), 'label': "equal.string uv", 'data': "http://foo.com?foo=1#uv=1.2,2.2"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "equal.xyz", 'input': "uv", 'out': "1.2,2.2,1"}), 'label': "equal.string uv", 'data': "http://foo.com?foo=1#uv=1.2,2.2,+1,+1"})]) + Test.test("filter.selectors.json",[_hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testParsed", 'input': "myid", 'out': True}), 'label': "myid exists", 'data': "http://foo.com?foo=1#foo*&-sometag&-someid&myid"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testParsed", 'input': "tag", 'out': True}), 'label': "tag exists", 'data': "http://foo.com?foo=1#tag=bar"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testParsed", 'input': "tag", 'out': True}), 'label': "tag exists", 'data': "http://foo.com?foo=1#-tag=bar"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testParsed", 'input': "price", 'out': True}), 'label': "filter test", 'data': "http://foo.com?foo=1#price=>2"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["tag", "bar"], 'out': True}), 'data': "tag=bar"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["tag", "foo"], 'out': False}), 'data': "-tag=foo"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["tag", "foo"], 'out': False}), 'data': "-tag*=foo"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["tag", "3"], 'out': False}), 'data': "-tag=>2"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["price", "1"], 'out': False}), 'data': "price=>2"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["price", "5"], 'out': False}), 'data': "price=<2"}), _hx_AnonObject({'fn': "filter", 'expect': _hx_AnonObject({'fn': "testProperty", 'input': ["price", "1"], 'out': True}), 'data': "price=<2"}), _hx_AnonObject({'fn': "url", 'expect': _hx_AnonObject({'fn': "testFilterDeep", 'input': ["foo"], 'out': 1}), 'label': "foo should be deep", 'data': "#foo*"})]) if (Test.errors > 1): print(str((("\n-----\n[ ❌] " + Std.string(Test.errors)) + " errors :/"))) @staticmethod def test(topic,spec): - print(str(("\n[.] running " + ("null" if topic is None else topic)))) + print(str(("\n[ . ] running " + ("null" if topic is None else topic)))) Filter = xrfragment_Filter _g = 0 _g1 = len(spec) @@ -473,12 +531,10 @@ class Test: valid = Test.equalXYZ(res,item) if (Reflect.field(Reflect.field(item,"expect"),"fn") == "equal.mediafragmentT"): valid = Test.equalMediaFragment(res,item,"t") - if (Reflect.field(Reflect.field(item,"expect"),"fn") == "equal.mediafragmentXYWH"): - valid = Test.equalMediaFragment(res,item,"xywh") - if (Reflect.field(Reflect.field(item,"expect"),"fn") == "equal.mediafragmentTSpd"): - valid = Test.equalMediaFragmentSpd(res,item,"t") - if (Reflect.field(Reflect.field(item,"expect"),"fn") == "equal.mediafragmentXYWHSpd"): - valid = Test.equalMediaFragmentSpd(res,item,"xywh") + if (Reflect.field(Reflect.field(item,"expect"),"fn") == "equal.mediafragmentUV"): + valid = Test.equalMediaFragment(res,item,"uv") + if (Reflect.field(Reflect.field(item,"expect"),"fn") == "equal.mediafragmentS"): + valid = Test.equalMediaFragment(res,item,"s") if (Reflect.field(Reflect.field(item,"expect"),"fn") == "testFilterRoot"): valid = (python_Boot.hasField(res,HxOverrides.arrayGet(Reflect.field(Reflect.field(item,"expect"),"input"), 0)) and (HxOverrides.eq(Reflect.field(Reflect.field(Reflect.field(Reflect.field(res,HxOverrides.arrayGet(Reflect.field(Reflect.field(item,"expect"),"input"), 0)),"filter"),"get")(),"root"),Reflect.field(Reflect.field(item,"expect"),"out")))) if (Reflect.field(Reflect.field(item,"expect"),"fn") == "testFilterDeep"): @@ -523,27 +579,75 @@ class Test: if ((not Reflect.field(Reflect.field(item,"expect"),"out")) and (not Reflect.field(res,Reflect.field(Reflect.field(item,"expect"),"input")))): return True else: - return (HxOverrides.arrayGet(Reflect.field(Reflect.field(res,key),"floats"), Std.parseInt(Reflect.field(Reflect.field(item,"expect"),"input"))) == Std.parseInt(Reflect.field(Reflect.field(item,"expect"),"out"))) + return (HxOverrides.arrayGet(Reflect.field(Reflect.field(res,key),"floats"), Std.parseInt(Reflect.field(Reflect.field(item,"expect"),"input"))) == Std.parseFloat(Reflect.field(Reflect.field(item,"expect"),"out"))) +Test._hx_class = Test + +class ValueType(Enum): + __slots__ = () + _hx_class_name = "ValueType" + _hx_constructs = ["TNull", "TInt", "TFloat", "TBool", "TObject", "TFunction", "TClass", "TEnum", "TUnknown"] @staticmethod - def equalMediaFragmentSpd(res,item,key): - if ((not Reflect.field(Reflect.field(item,"expect"),"out")) and (not Reflect.field(res,Reflect.field(Reflect.field(item,"expect"),"input")))): - return True + def TClass(c): + return ValueType("TClass", 6, (c,)) + + @staticmethod + def TEnum(e): + return ValueType("TEnum", 7, (e,)) +ValueType.TNull = ValueType("TNull", 0, ()) +ValueType.TInt = ValueType("TInt", 1, ()) +ValueType.TFloat = ValueType("TFloat", 2, ()) +ValueType.TBool = ValueType("TBool", 3, ()) +ValueType.TObject = ValueType("TObject", 4, ()) +ValueType.TFunction = ValueType("TFunction", 5, ()) +ValueType.TUnknown = ValueType("TUnknown", 8, ()) +ValueType._hx_class = ValueType + + +class Type: + _hx_class_name = "Type" + __slots__ = () + _hx_statics = ["typeof"] + + @staticmethod + def typeof(v): + if (v is None): + return ValueType.TNull + elif isinstance(v,bool): + return ValueType.TBool + elif isinstance(v,int): + return ValueType.TInt + elif isinstance(v,float): + return ValueType.TFloat + elif isinstance(v,str): + return ValueType.TClass(str) + elif isinstance(v,list): + return ValueType.TClass(list) + elif (isinstance(v,_hx_AnonObject) or python_lib_Inspect.isclass(v)): + return ValueType.TObject + elif isinstance(v,Enum): + return ValueType.TEnum(v.__class__) + elif (isinstance(v,type) or hasattr(v,"_hx_class")): + return ValueType.TClass(v.__class__) + elif callable(v): + return ValueType.TFunction else: - return (HxOverrides.arrayGet(Reflect.field(Reflect.field(res,key),"speed"), Std.parseInt(Reflect.field(Reflect.field(item,"expect"),"input"))) == Std.parseFloat(Reflect.field(Reflect.field(item,"expect"),"out"))) + return ValueType.TUnknown +Type._hx_class = Type class haxe_IMap: _hx_class_name = "haxe.IMap" __slots__ = () +haxe_IMap._hx_class = haxe_IMap class haxe_Exception(Exception): _hx_class_name = "haxe.Exception" - __slots__ = ("_hx___nativeStack", "_hx___nativeException", "_hx___previousException") - _hx_fields = ["__nativeStack", "__nativeException", "__previousException"] - _hx_methods = ["unwrap"] - _hx_statics = ["caught"] + __slots__ = ("_hx___nativeStack", "_hx___skipStack", "_hx___nativeException", "_hx___previousException") + _hx_fields = ["__nativeStack", "__skipStack", "__nativeException", "__previousException"] + _hx_methods = ["unwrap", "get_native"] + _hx_statics = ["caught", "thrown"] _hx_interfaces = [] _hx_super = Exception @@ -552,6 +656,7 @@ class haxe_Exception(Exception): self._hx___previousException = None self._hx___nativeException = None self._hx___nativeStack = None + self._hx___skipStack = 0 super().__init__(message) self._hx___previousException = previous if ((native is not None) and Std.isOfType(native,BaseException)): @@ -568,6 +673,9 @@ class haxe_Exception(Exception): def unwrap(self): return self._hx___nativeException + def get_native(self): + return self._hx___nativeException + @staticmethod def caught(value): if Std.isOfType(value,haxe_Exception): @@ -577,6 +685,18 @@ class haxe_Exception(Exception): else: return haxe_ValueException(value,None,value) + @staticmethod + def thrown(value): + if Std.isOfType(value,haxe_Exception): + return value.get_native() + elif Std.isOfType(value,BaseException): + return value + else: + e = haxe_ValueException(value) + e._hx___skipStack = (e._hx___skipStack + 1) + return e + +haxe_Exception._hx_class = haxe_Exception class haxe_NativeStackTrace: @@ -597,37 +717,41 @@ class haxe_NativeStackTrace: return infos else: return [] +haxe_NativeStackTrace._hx_class = haxe_NativeStackTrace +class haxe__Template_TemplateExpr(Enum): + __slots__ = () + _hx_class_name = "haxe._Template.TemplateExpr" + _hx_constructs = ["OpVar", "OpExpr", "OpIf", "OpStr", "OpBlock", "OpForeach", "OpMacro"] -class haxe_ValueException(haxe_Exception): - _hx_class_name = "haxe.ValueException" - __slots__ = ("value",) - _hx_fields = ["value"] - _hx_methods = ["unwrap"] - _hx_statics = [] - _hx_interfaces = [] - _hx_super = haxe_Exception + @staticmethod + def OpVar(v): + return haxe__Template_TemplateExpr("OpVar", 0, (v,)) + @staticmethod + def OpExpr(expr): + return haxe__Template_TemplateExpr("OpExpr", 1, (expr,)) - def __init__(self,value,previous = None,native = None): - self.value = None - super().__init__(Std.string(value),previous,native) - self.value = value + @staticmethod + def OpIf(expr,eif,eelse): + return haxe__Template_TemplateExpr("OpIf", 2, (expr,eif,eelse)) - def unwrap(self): - return self.value + @staticmethod + def OpStr(str): + return haxe__Template_TemplateExpr("OpStr", 3, (str,)) + @staticmethod + def OpBlock(l): + return haxe__Template_TemplateExpr("OpBlock", 4, (l,)) + @staticmethod + def OpForeach(expr,loop): + return haxe__Template_TemplateExpr("OpForeach", 5, (expr,loop)) -class haxe_ds_StringMap: - _hx_class_name = "haxe.ds.StringMap" - __slots__ = ("h",) - _hx_fields = ["h"] - _hx_interfaces = [haxe_IMap] - - def __init__(self): - self.h = dict() - + @staticmethod + def OpMacro(name,params): + return haxe__Template_TemplateExpr("OpMacro", 6, (name,params)) +haxe__Template_TemplateExpr._hx_class = haxe__Template_TemplateExpr class haxe_iterators_ArrayIterator: @@ -653,6 +777,613 @@ class haxe_iterators_ArrayIterator: return python_internal_ArrayImpl._get(self.array, _hx_local_2()) return _hx_local_3() +haxe_iterators_ArrayIterator._hx_class = haxe_iterators_ArrayIterator + + +class haxe_Template: + _hx_class_name = "haxe.Template" + __slots__ = ("expr", "context", "macros", "stack", "buf") + _hx_fields = ["expr", "context", "macros", "stack", "buf"] + _hx_methods = ["execute", "resolve", "parseTokens", "parseBlock", "parse", "parseExpr", "makeConst", "makePath", "makeExpr", "skipSpaces", "makeExpr2", "run"] + _hx_statics = ["splitter", "expr_splitter", "expr_trim", "expr_int", "expr_float", "globals", "hxKeepArrayIterator"] + + def __init__(self,_hx_str): + self.buf = None + self.stack = None + self.macros = None + self.context = None + self.expr = None + tokens = self.parseTokens(_hx_str) + self.expr = self.parseBlock(tokens) + if (not tokens.isEmpty()): + raise haxe_Exception.thrown((("Unexpected '" + Std.string(tokens.first().s)) + "'")) + + def execute(self,context,macros = None): + self.macros = (_hx_AnonObject({}) if ((macros is None)) else macros) + self.context = context + self.stack = haxe_ds_List() + self.buf = StringBuf() + self.run(self.expr) + return self.buf.b.getvalue() + + def resolve(self,v): + if (v == "__current__"): + return self.context + if Reflect.isObject(self.context): + value = Reflect.getProperty(self.context,v) + if ((value is not None) or python_Boot.hasField(self.context,v)): + return value + _g_head = self.stack.h + while (_g_head is not None): + val = _g_head.item + _g_head = _g_head.next + ctx = val + value = Reflect.getProperty(ctx,v) + if ((value is not None) or python_Boot.hasField(ctx,v)): + return value + return Reflect.field(haxe_Template.globals,v) + + def parseTokens(self,data): + tokens = haxe_ds_List() + while True: + _this = haxe_Template.splitter + _this.matchObj = python_lib_Re.search(_this.pattern,data) + if (not ((_this.matchObj is not None))): + break + _this1 = haxe_Template.splitter + p_pos = _this1.matchObj.start() + p_len = (_this1.matchObj.end() - _this1.matchObj.start()) + if (p_pos > 0): + tokens.add(_hx_AnonObject({'p': HxString.substr(data,0,p_pos), 's': True, 'l': None})) + if (HxString.charCodeAt(data,p_pos) == 58): + tokens.add(_hx_AnonObject({'p': HxString.substr(data,(p_pos + 2),(p_len - 4)), 's': False, 'l': None})) + _this2 = haxe_Template.splitter + data = HxString.substr(_this2.matchObj.string,_this2.matchObj.end(),None) + continue + parp = (p_pos + p_len) + npar = 1 + params = [] + part = "" + while True: + c = HxString.charCodeAt(data,parp) + parp = (parp + 1) + if (c == 40): + npar = (npar + 1) + elif (c == 41): + npar = (npar - 1) + if (npar <= 0): + break + elif (c is None): + raise haxe_Exception.thrown("Unclosed macro parenthesis") + if ((c == 44) and ((npar == 1))): + params.append(part) + part = "" + else: + part = (("null" if part is None else part) + HxOverrides.stringOrNull("".join(map(chr,[c])))) + params.append(part) + tokens.add(_hx_AnonObject({'p': haxe_Template.splitter.matchObj.group(2), 's': False, 'l': params})) + data = HxString.substr(data,parp,(len(data) - parp)) + if (len(data) > 0): + tokens.add(_hx_AnonObject({'p': data, 's': True, 'l': None})) + return tokens + + def parseBlock(self,tokens): + l = haxe_ds_List() + while True: + t = tokens.first() + if (t is None): + break + if ((not t.s) and ((((t.p == "end") or ((t.p == "else"))) or ((HxString.substr(t.p,0,7) == "elseif "))))): + break + l.add(self.parse(tokens)) + if (l.length == 1): + return l.first() + return haxe__Template_TemplateExpr.OpBlock(l) + + def parse(self,tokens): + t = tokens.pop() + p = t.p + if t.s: + return haxe__Template_TemplateExpr.OpStr(p) + if (t.l is not None): + pe = haxe_ds_List() + _g = 0 + _g1 = t.l + while (_g < len(_g1)): + p1 = (_g1[_g] if _g >= 0 and _g < len(_g1) else None) + _g = (_g + 1) + pe.add(self.parseBlock(self.parseTokens(p1))) + return haxe__Template_TemplateExpr.OpMacro(p,pe) + def _hx_local_2(kwd): + pos = -1 + length = len(kwd) + if (HxString.substr(p,0,length) == kwd): + pos = length + _g_offset = 0 + _g_s = HxString.substr(p,length,None) + while (_g_offset < len(_g_s)): + index = _g_offset + _g_offset = (_g_offset + 1) + c = ord(_g_s[index]) + if (c == 32): + pos = (pos + 1) + else: + break + return pos + kwdEnd = _hx_local_2 + pos = kwdEnd("if") + if (pos > 0): + p = HxString.substr(p,pos,(len(p) - pos)) + e = self.parseExpr(p) + eif = self.parseBlock(tokens) + t = tokens.first() + eelse = None + if (t is None): + raise haxe_Exception.thrown("Unclosed 'if'") + if (t.p == "end"): + tokens.pop() + eelse = None + elif (t.p == "else"): + tokens.pop() + eelse = self.parseBlock(tokens) + t = tokens.pop() + if ((t is None) or ((t.p != "end"))): + raise haxe_Exception.thrown("Unclosed 'else'") + else: + t.p = HxString.substr(t.p,4,(len(t.p) - 4)) + eelse = self.parse(tokens) + return haxe__Template_TemplateExpr.OpIf(e,eif,eelse) + pos = kwdEnd("foreach") + if (pos >= 0): + p = HxString.substr(p,pos,(len(p) - pos)) + e = self.parseExpr(p) + efor = self.parseBlock(tokens) + t = tokens.pop() + if ((t is None) or ((t.p != "end"))): + raise haxe_Exception.thrown("Unclosed 'foreach'") + return haxe__Template_TemplateExpr.OpForeach(e,efor) + _this = haxe_Template.expr_splitter + _this.matchObj = python_lib_Re.search(_this.pattern,p) + if (_this.matchObj is not None): + return haxe__Template_TemplateExpr.OpExpr(self.parseExpr(p)) + return haxe__Template_TemplateExpr.OpVar(p) + + def parseExpr(self,data): + l = haxe_ds_List() + expr = data + while True: + _this = haxe_Template.expr_splitter + _this.matchObj = python_lib_Re.search(_this.pattern,data) + if (not ((_this.matchObj is not None))): + break + _this1 = haxe_Template.expr_splitter + p_pos = _this1.matchObj.start() + p_len = (_this1.matchObj.end() - _this1.matchObj.start()) + k = (p_pos + p_len) + if (p_pos != 0): + l.add(_hx_AnonObject({'p': HxString.substr(data,0,p_pos), 's': True})) + p = haxe_Template.expr_splitter.matchObj.group(0) + startIndex = None + l.add(_hx_AnonObject({'p': p, 's': (((p.find("\"") if ((startIndex is None)) else HxString.indexOfImpl(p,"\"",startIndex))) >= 0)})) + _this2 = haxe_Template.expr_splitter + data = HxString.substr(_this2.matchObj.string,_this2.matchObj.end(),None) + if (len(data) != 0): + _g_offset = 0 + _g_s = data + while (_g_offset < len(_g_s)): + _g1_key = _g_offset + s = _g_s + index = _g_offset + _g_offset = (_g_offset + 1) + _g1_value = (-1 if ((index >= len(s))) else ord(s[index])) + i = _g1_key + c = _g1_value + if (c != 32): + l.add(_hx_AnonObject({'p': HxString.substr(data,i,None), 's': True})) + break + e = None + try: + e = self.makeExpr(l) + if (not l.isEmpty()): + raise haxe_Exception.thrown(l.first().p) + except BaseException as _g: + None + _g1 = haxe_Exception.caught(_g).unwrap() + if Std.isOfType(_g1,str): + s = _g1 + raise haxe_Exception.thrown(((("Unexpected '" + ("null" if s is None else s)) + "' in ") + ("null" if expr is None else expr))) + else: + raise _g + def _hx_local_0(): + try: + return e() + except BaseException as _g: + None + exc = haxe_Exception.caught(_g).unwrap() + raise haxe_Exception.thrown(((("Error : " + Std.string(exc)) + " in ") + ("null" if expr is None else expr))) + return _hx_local_0 + + def makeConst(self,v): + _this = haxe_Template.expr_trim + _this.matchObj = python_lib_Re.search(_this.pattern,v) + v = haxe_Template.expr_trim.matchObj.group(1) + if (HxString.charCodeAt(v,0) == 34): + _hx_str = HxString.substr(v,1,(len(v) - 2)) + def _hx_local_0(): + return _hx_str + return _hx_local_0 + _this = haxe_Template.expr_int + _this.matchObj = python_lib_Re.search(_this.pattern,v) + if (_this.matchObj is not None): + i = Std.parseInt(v) + def _hx_local_1(): + return i + return _hx_local_1 + _this = haxe_Template.expr_float + _this.matchObj = python_lib_Re.search(_this.pattern,v) + if (_this.matchObj is not None): + f = Std.parseFloat(v) + def _hx_local_2(): + return f + return _hx_local_2 + me = self + def _hx_local_3(): + return me.resolve(v) + return _hx_local_3 + + def makePath(self,e,l): + p = l.first() + if ((p is None) or ((p.p != "."))): + return e + l.pop() + field = l.pop() + if ((field is None) or (not field.s)): + raise haxe_Exception.thrown(field.p) + f = field.p + _this = haxe_Template.expr_trim + _this.matchObj = python_lib_Re.search(_this.pattern,f) + f = haxe_Template.expr_trim.matchObj.group(1) + def _hx_local_1(): + def _hx_local_0(): + return Reflect.field(e(),f) + return self.makePath(_hx_local_0,l) + return _hx_local_1() + + def makeExpr(self,l): + return self.makePath(self.makeExpr2(l),l) + + def skipSpaces(self,l): + p = l.first() + while (p is not None): + _g_offset = 0 + _g_s = p.p + while (_g_offset < len(_g_s)): + index = _g_offset + _g_offset = (_g_offset + 1) + c = ord(_g_s[index]) + if (c != 32): + return + l.pop() + p = l.first() + + def makeExpr2(self,l): + self.skipSpaces(l) + p = l.pop() + self.skipSpaces(l) + if (p is None): + raise haxe_Exception.thrown("") + if p.s: + return self.makeConst(p.p) + _g = p.p + if (_g == "!"): + e = self.makeExpr(l) + def _hx_local_0(): + v = e() + if (v is not None): + return (v == False) + else: + return True + return _hx_local_0 + elif (_g == "("): + self.skipSpaces(l) + e1 = self.makeExpr(l) + self.skipSpaces(l) + p1 = l.pop() + if ((p1 is None) or p1.s): + raise haxe_Exception.thrown(p1) + if (p1.p == ")"): + return e1 + self.skipSpaces(l) + e2 = self.makeExpr(l) + self.skipSpaces(l) + p2 = l.pop() + self.skipSpaces(l) + if ((p2 is None) or ((p2.p != ")"))): + raise haxe_Exception.thrown(p2) + _g = p1.p + _hx_local_1 = len(_g) + if (_hx_local_1 == 1): + if (_g == "*"): + def _hx_local_2(): + return (e1() * e2()) + return _hx_local_2 + elif (_g == "+"): + def _hx_local_3(): + return python_Boot._add_dynamic(e1(),e2()) + return _hx_local_3 + elif (_g == "-"): + def _hx_local_4(): + return (e1() - e2()) + return _hx_local_4 + elif (_g == "/"): + def _hx_local_5(): + return (e1() / e2()) + return _hx_local_5 + elif (_g == "<"): + def _hx_local_6(): + return (e1() < e2()) + return _hx_local_6 + elif (_g == ">"): + def _hx_local_7(): + return (e1() > e2()) + return _hx_local_7 + else: + raise haxe_Exception.thrown(("Unknown operation " + HxOverrides.stringOrNull(p1.p))) + elif (_hx_local_1 == 2): + if (_g == "!="): + def _hx_local_8(): + return not HxOverrides.eq(e1(),e2()) + return _hx_local_8 + elif (_g == "&&"): + def _hx_local_9(): + return (e1() and e2()) + return _hx_local_9 + elif (_g == "<="): + def _hx_local_10(): + return (e1() <= e2()) + return _hx_local_10 + elif (_g == "=="): + def _hx_local_11(): + return HxOverrides.eq(e1(),e2()) + return _hx_local_11 + elif (_g == ">="): + def _hx_local_12(): + return (e1() >= e2()) + return _hx_local_12 + elif (_g == "||"): + def _hx_local_13(): + return (e1() or e2()) + return _hx_local_13 + else: + raise haxe_Exception.thrown(("Unknown operation " + HxOverrides.stringOrNull(p1.p))) + else: + raise haxe_Exception.thrown(("Unknown operation " + HxOverrides.stringOrNull(p1.p))) + elif (_g == "-"): + e3 = self.makeExpr(l) + def _hx_local_14(): + return -e3() + return _hx_local_14 + else: + pass + raise haxe_Exception.thrown(p.p) + + def run(self,e): + tmp = e.index + if (tmp == 0): + v = e.params[0] + _this = self.buf + s = Std.string(Std.string(self.resolve(v))) + _this.b.write(s) + elif (tmp == 1): + e1 = e.params[0] + _this = self.buf + s = Std.string(Std.string(e1())) + _this.b.write(s) + elif (tmp == 2): + e1 = e.params[0] + eif = e.params[1] + eelse = e.params[2] + v = e1() + if ((v is None) or ((v == False))): + if (eelse is not None): + self.run(eelse) + else: + self.run(eif) + elif (tmp == 3): + _hx_str = e.params[0] + _this = self.buf + s = Std.string(_hx_str) + _this.b.write(s) + elif (tmp == 4): + l = e.params[0] + _g_head = l.h + while (_g_head is not None): + val = _g_head.item + _g_head = _g_head.next + e1 = val + self.run(e1) + elif (tmp == 5): + e1 = e.params[0] + loop = e.params[1] + v = e1() + try: + x = Reflect.field(v,"iterator")() + if (Reflect.field(x,"hasNext") is None): + raise haxe_Exception.thrown(None) + v = x + except BaseException as _g: + None + try: + if (Reflect.field(v,"hasNext") is None): + raise haxe_Exception.thrown(None) + except BaseException as _g: + raise haxe_Exception.thrown(("Cannot iter on " + Std.string(v))) + self.stack.push(self.context) + v1 = v + ctx = v1 + while ctx.hasNext(): + ctx1 = ctx.next() + self.context = ctx1 + self.run(loop) + self.context = self.stack.pop() + elif (tmp == 6): + m = e.params[0] + params = e.params[1] + v = Reflect.field(self.macros,m) + pl = list() + old = self.buf + pl.append(self.resolve) + _g_head = params.h + while (_g_head is not None): + val = _g_head.item + _g_head = _g_head.next + p = val + if (p.index == 0): + v1 = p.params[0] + x = self.resolve(v1) + pl.append(x) + else: + self.buf = StringBuf() + self.run(p) + x1 = self.buf.b.getvalue() + pl.append(x1) + self.buf = old + try: + _this = self.buf + s = Std.string(Std.string(Reflect.callMethod(self.macros,v,pl))) + _this.b.write(s) + except BaseException as _g: + None + e = haxe_Exception.caught(_g).unwrap() + plstr = None + try: + plstr = ",".join([python_Boot.toString1(x1,'') for x1 in pl]) + except BaseException as _g: + plstr = "???" + msg = (((((("Macro call " + ("null" if m is None else m)) + "(") + ("null" if plstr is None else plstr)) + ") failed (") + Std.string(e)) + ")") + raise haxe_Exception.thrown(msg) + else: + pass + +haxe_Template._hx_class = haxe_Template + + +class haxe_ValueException(haxe_Exception): + _hx_class_name = "haxe.ValueException" + __slots__ = ("value",) + _hx_fields = ["value"] + _hx_methods = ["unwrap"] + _hx_statics = [] + _hx_interfaces = [] + _hx_super = haxe_Exception + + + def __init__(self,value,previous = None,native = None): + self.value = None + super().__init__(Std.string(value),previous,native) + self.value = value + + def unwrap(self): + return self.value + +haxe_ValueException._hx_class = haxe_ValueException + + +class haxe_ds_List: + _hx_class_name = "haxe.ds.List" + __slots__ = ("h", "q", "length") + _hx_fields = ["h", "q", "length"] + _hx_methods = ["add", "push", "first", "pop", "isEmpty", "toString"] + + def __init__(self): + self.q = None + self.h = None + self.length = 0 + + def add(self,item): + x = haxe_ds__List_ListNode(item,None) + if (self.h is None): + self.h = x + else: + self.q.next = x + self.q = x + _hx_local_0 = self + _hx_local_1 = _hx_local_0.length + _hx_local_0.length = (_hx_local_1 + 1) + _hx_local_1 + + def push(self,item): + x = haxe_ds__List_ListNode(item,self.h) + self.h = x + if (self.q is None): + self.q = x + _hx_local_0 = self + _hx_local_1 = _hx_local_0.length + _hx_local_0.length = (_hx_local_1 + 1) + _hx_local_1 + + def first(self): + if (self.h is None): + return None + else: + return self.h.item + + def pop(self): + if (self.h is None): + return None + x = self.h.item + self.h = self.h.next + if (self.h is None): + self.q = None + _hx_local_0 = self + _hx_local_1 = _hx_local_0.length + _hx_local_0.length = (_hx_local_1 - 1) + _hx_local_1 + return x + + def isEmpty(self): + return (self.h is None) + + def toString(self): + s_b = python_lib_io_StringIO() + first = True + l = self.h + s_b.write("{") + while (l is not None): + if first: + first = False + else: + s_b.write(", ") + s_b.write(Std.string(Std.string(l.item))) + l = l.next + s_b.write("}") + return s_b.getvalue() + +haxe_ds_List._hx_class = haxe_ds_List + + +class haxe_ds__List_ListNode: + _hx_class_name = "haxe.ds._List.ListNode" + __slots__ = ("item", "next") + _hx_fields = ["item", "next"] + + def __init__(self,item,next): + self.item = item + self.next = next + +haxe_ds__List_ListNode._hx_class = haxe_ds__List_ListNode + + +class haxe_ds_StringMap: + _hx_class_name = "haxe.ds.StringMap" + __slots__ = ("h",) + _hx_fields = ["h"] + _hx_interfaces = [haxe_IMap] + + def __init__(self): + self.h = dict() + +haxe_ds_StringMap._hx_class = haxe_ds_StringMap class haxe_iterators_ArrayKeyValueIterator: @@ -678,12 +1409,21 @@ class haxe_iterators_ArrayKeyValueIterator: return _hx_AnonObject({'value': python_internal_ArrayImpl._get(self.array, self.current), 'key': _hx_local_2()}) return _hx_local_3() +haxe_iterators_ArrayKeyValueIterator._hx_class = haxe_iterators_ArrayKeyValueIterator class python_Boot: _hx_class_name = "python.Boot" __slots__ = () - _hx_statics = ["keywords", "toString1", "fields", "simpleField", "hasField", "field", "getInstanceFields", "getSuperClass", "getClassFields", "prefixLength", "unhandleKeywords"] + _hx_statics = ["keywords", "_add_dynamic", "toString1", "fields", "simpleField", "hasField", "field", "getInstanceFields", "getSuperClass", "getClassFields", "prefixLength", "unhandleKeywords"] + + @staticmethod + def _add_dynamic(a,b): + if (isinstance(a,str) and isinstance(b,str)): + return (a + b) + if (isinstance(a,str) or isinstance(b,str)): + return (python_Boot.toString1(a,"") + python_Boot.toString1(b,"")) + return (a + b) @staticmethod def toString1(o,s): @@ -1099,6 +1839,7 @@ class python_Boot: if (real in python_Boot.keywords): return real return name +python_Boot._hx_class = python_Boot class python_HaxeIterator: @@ -1134,12 +1875,13 @@ class python_HaxeIterator: self.checked = True return self.has +python_HaxeIterator._hx_class = python_HaxeIterator class python_internal_ArrayImpl: _hx_class_name = "python.internal.ArrayImpl" __slots__ = () - _hx_statics = ["get_length", "concat", "copy", "iterator", "keyValueIterator", "indexOf", "lastIndexOf", "join", "toString", "pop", "push", "unshift", "remove", "contains", "shift", "slice", "sort", "splice", "map", "filter", "insert", "reverse", "_get"] + _hx_statics = ["get_length", "concat", "copy", "iterator", "keyValueIterator", "indexOf", "lastIndexOf", "join", "toString", "pop", "push", "unshift", "remove", "contains", "shift", "slice", "sort", "splice", "map", "filter", "insert", "reverse", "_get", "_set"] @staticmethod def get_length(x): @@ -1275,11 +2017,30 @@ class python_internal_ArrayImpl: else: return None + @staticmethod + def _set(x,idx,v): + l = len(x) + while (l < idx): + x.append(None) + l = (l + 1) + if (l == idx): + x.append(v) + else: + x[idx] = v + return v +python_internal_ArrayImpl._hx_class = python_internal_ArrayImpl + class HxOverrides: _hx_class_name = "HxOverrides" __slots__ = () - _hx_statics = ["eq", "stringOrNull", "filter", "length", "arrayGet"] + _hx_statics = ["iterator", "eq", "stringOrNull", "filter", "length", "arrayGet"] + + @staticmethod + def iterator(x): + if isinstance(x,list): + return haxe_iterators_ArrayIterator(x) + return x.iterator() @staticmethod def eq(a,b): @@ -1318,6 +2079,7 @@ class HxOverrides: return None else: return a[i] +HxOverrides._hx_class = HxOverrides class python_internal_MethodClosure: @@ -1333,6 +2095,7 @@ class python_internal_MethodClosure: def __call__(self,*args): return self.func(self.obj,*args) +python_internal_MethodClosure._hx_class = python_internal_MethodClosure class HxString: @@ -1447,6 +2210,7 @@ class HxString: if (startIndex < 0): startIndex = 0 return s[startIndex:(startIndex + _hx_len)] +HxString._hx_class = HxString class xrfragment_Filter: @@ -1555,8 +2319,8 @@ class xrfragment_Filter: fails = 0 qualify = 0 def _hx_local_2(expr): - nonlocal conds nonlocal fails + nonlocal conds conds = (conds + 1) fails = (fails + (0 if expr else 1)) return expr @@ -1581,6 +2345,7 @@ class xrfragment_Filter: qualify = (qualify + 1) return (qualify > 0) +xrfragment_Filter._hx_class = xrfragment_Filter class xrfragment_Parser: @@ -1591,26 +2356,30 @@ class xrfragment_Parser: @staticmethod def parse(key,value,store,index = None): Frag = haxe_ds_StringMap() - Frag.h["#"] = ((xrfragment_XRF.ASSET | xrfragment_XRF.T_PREDEFINED_VIEW) | xrfragment_XRF.PV_EXECUTE) - Frag.h["src"] = (xrfragment_XRF.ASSET | xrfragment_XRF.T_URL) - Frag.h["href"] = ((xrfragment_XRF.ASSET | xrfragment_XRF.T_URL) | xrfragment_XRF.T_PREDEFINED_VIEW) - Frag.h["tag"] = (xrfragment_XRF.ASSET | xrfragment_XRF.T_STRING) + Frag.h["#"] = ((xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_PREDEFINED_VIEW) | xrfragment_XRF.PV_EXECUTE) + Frag.h["src"] = xrfragment_XRF.T_URL + Frag.h["href"] = (xrfragment_XRF.T_URL | xrfragment_XRF.T_PREDEFINED_VIEW) + Frag.h["tag"] = (xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_STRING) Frag.h["pos"] = ((((xrfragment_XRF.PV_OVERRIDE | xrfragment_XRF.T_VECTOR3) | xrfragment_XRF.T_STRING) | xrfragment_XRF.METADATA) | xrfragment_XRF.NAVIGATOR) Frag.h["rot"] = ((((xrfragment_XRF.QUERY_OPERATOR | xrfragment_XRF.PV_OVERRIDE) | xrfragment_XRF.T_VECTOR3) | xrfragment_XRF.METADATA) | xrfragment_XRF.NAVIGATOR) - Frag.h["t"] = ((((((xrfragment_XRF.ASSET | xrfragment_XRF.PV_OVERRIDE) | xrfragment_XRF.T_FLOAT) | xrfragment_XRF.T_VECTOR2) | xrfragment_XRF.T_MEDIAFRAG) | xrfragment_XRF.NAVIGATOR) | xrfragment_XRF.METADATA) - Frag.h["xywh"] = ((((((xrfragment_XRF.ASSET | xrfragment_XRF.PV_OVERRIDE) | xrfragment_XRF.T_FLOAT) | xrfragment_XRF.T_VECTOR2) | xrfragment_XRF.T_MEDIAFRAG) | xrfragment_XRF.NAVIGATOR) | xrfragment_XRF.METADATA) - 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.METADATA) | xrfragment_XRF.PROMPT) + Frag.h["t"] = ((((xrfragment_XRF.PV_OVERRIDE | xrfragment_XRF.T_FLOAT) | xrfragment_XRF.T_VECTOR2) | xrfragment_XRF.NAVIGATOR) | xrfragment_XRF.METADATA) + Frag.h["s"] = (xrfragment_XRF.PV_OVERRIDE | xrfragment_XRF.T_MEDIAFRAG) + Frag.h["loop"] = xrfragment_XRF.PV_OVERRIDE + Frag.h["uv"] = (xrfragment_XRF.T_VECTOR2 | xrfragment_XRF.T_MEDIAFRAG) + Frag.h["namespace"] = (xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_STRING) + Frag.h["SPDX"] = (xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_STRING) + Frag.h["unit"] = (xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_STRING) + Frag.h["description"] = (xrfragment_XRF.IMMUTABLE | xrfragment_XRF.T_STRING) keyStripped = xrfragment_XRF.operators.replace(key,"") isPVDynamic = ((len(key) > 0) and (not (key in Frag.h))) - isPVDefault = (((len(value) == 0) and ((len(key) > 0))) and ((key == "#"))) if isPVDynamic: v = xrfragment_XRF(key,(xrfragment_XRF.PV_EXECUTE | xrfragment_XRF.NAVIGATOR),index) v.validate(value) - v.flags = xrfragment_XRF.set(xrfragment_XRF.T_DYNAMIC,v.flags) + v.flags = xrfragment_XRF.set(xrfragment_XRF.T_DYNAMICKEY,v.flags) + if (not (key in Frag.h)): + v.flags = xrfragment_XRF.set(xrfragment_XRF.CUSTOMFRAG,v.flags) + if (len(value) == 0): + v.flags = xrfragment_XRF.set(xrfragment_XRF.T_DYNAMICKEYVALUE,v.flags) setattr(store,(("_hx_" + keyStripped) if ((keyStripped in python_Boot.keywords)) else (("_hx_" + keyStripped) if (((((len(keyStripped) > 2) and ((ord(keyStripped[0]) == 95))) and ((ord(keyStripped[1]) == 95))) and ((ord(keyStripped[(len(keyStripped) - 1)]) != 95)))) else keyStripped)),v) return True v = xrfragment_XRF(key,Frag.h.get(key,None),index) @@ -1624,15 +2393,16 @@ class xrfragment_Parser: else: if Std.isOfType(value,str): v.guessType(v,value) - v.noXRF = True + v.flags = xrfragment_XRF.set(xrfragment_XRF.CUSTOMFRAG,v.flags) setattr(store,(("_hx_" + keyStripped) if ((keyStripped in python_Boot.keywords)) else (("_hx_" + keyStripped) if (((((len(keyStripped) > 2) and ((ord(keyStripped[0]) == 95))) and ((ord(keyStripped[1]) == 95))) and ((ord(keyStripped[(len(keyStripped) - 1)]) != 95)))) else keyStripped)),v) return True +xrfragment_Parser._hx_class = xrfragment_Parser class xrfragment_URI: _hx_class_name = "xrfragment.URI" __slots__ = () - _hx_statics = ["parse"] + _hx_statics = ["__meta__", "parse", "template"] @staticmethod def parse(url,_hx_filter): @@ -1659,8 +2429,13 @@ class xrfragment_URI: key = (splitByEqual[0] if 0 < len(splitByEqual) else None) value = "" if (len(splitByEqual) > 1): - _this1 = regexPlus.split((splitByEqual[1] if 1 < len(splitByEqual) else None)) - value = python_lib_urllib_Parse.unquote(" ".join([python_Boot.toString1(x1,'') for x1 in _this1])) + _this1 = xrfragment_XRF.isVector + _this1.matchObj = python_lib_Re.search(_this1.pattern,(splitByEqual[1] if 1 < len(splitByEqual) else None)) + if (_this1.matchObj is not None): + value = (splitByEqual[1] if 1 < len(splitByEqual) else None) + else: + _this2 = regexPlus.split((splitByEqual[1] if 1 < len(splitByEqual) else None)) + value = python_lib_urllib_Parse.unquote(" ".join([python_Boot.toString1(x1,'') for x1 in _this2])) ok = xrfragment_Parser.parse(key,value,store,i) if ((_hx_filter is not None) and ((_hx_filter != 0))): _g = 0 @@ -1673,16 +2448,32 @@ class xrfragment_URI: Reflect.deleteField(store,key) return store + @staticmethod + def template(uri,vars): + parts = uri.split("#") + if (len(parts) == 1): + return uri + frag = (parts[1] if 1 < len(parts) else None) + frag = StringTools.replace(frag,"{","::") + frag = StringTools.replace(frag,"}","::") + frag = haxe_Template(frag).execute(vars) + frag = StringTools.replace(frag,"null","") + python_internal_ArrayImpl._set(parts, 1, frag) + return "#".join([python_Boot.toString1(x1,'') for x1 in parts]) +xrfragment_URI._hx_class = xrfragment_URI + class xrfragment_XRF: _hx_class_name = "xrfragment.XRF" - __slots__ = ("fragment", "flags", "index", "x", "y", "z", "floats", "speed", "color", "string", "int", "float", "filter", "noXRF") - _hx_fields = ["fragment", "flags", "index", "x", "y", "z", "floats", "speed", "color", "string", "int", "float", "filter", "noXRF"] + __slots__ = ("fragment", "flags", "index", "x", "y", "z", "shift", "floats", "color", "string", "int", "float", "filter", "reset", "loop", "xrfScheme") + _hx_fields = ["fragment", "flags", "index", "x", "y", "z", "shift", "floats", "color", "string", "int", "float", "filter", "reset", "loop", "xrfScheme"] _hx_methods = ["is", "validate", "guessType"] - _hx_statics = ["ASSET", "PROP_BIND", "QUERY_OPERATOR", "PROMPT", "ROUNDROBIN", "NAVIGATOR", "METADATA", "PV_OVERRIDE", "PV_EXECUTE", "T_COLOR", "T_INT", "T_FLOAT", "T_VECTOR2", "T_VECTOR3", "T_URL", "T_PREDEFINED_VIEW", "T_STRING", "T_MEDIAFRAG", "T_DYNAMIC", "isColor", "isInt", "isFloat", "isVector", "isUrl", "isUrlOrPretypedView", "isString", "operators", "isProp", "isExclude", "isDeep", "isNumber", "isMediaFrag", "set", "unset"] + _hx_statics = ["IMMUTABLE", "PROP_BIND", "QUERY_OPERATOR", "PROMPT", "CUSTOMFRAG", "NAVIGATOR", "METADATA", "PV_OVERRIDE", "PV_EXECUTE", "T_COLOR", "T_INT", "T_FLOAT", "T_VECTOR2", "T_VECTOR3", "T_URL", "T_PREDEFINED_VIEW", "T_STRING", "T_MEDIAFRAG", "T_DYNAMICKEY", "T_DYNAMICKEYVALUE", "isColor", "isInt", "isFloat", "isVector", "isUrl", "isUrlOrPretypedView", "isString", "operators", "isProp", "isExclude", "isDeep", "isNumber", "isMediaFrag", "isReset", "isShift", "isXRFScheme", "set", "unset"] def __init__(self,_fragment,_flags,_index = None): - self.noXRF = None + self.xrfScheme = None + self.loop = None + self.reset = None self.filter = None self.float = None self.int = None @@ -1691,8 +2482,8 @@ class xrfragment_XRF: self.z = None self.y = None self.x = None - self.speed = list() self.floats = list() + self.shift = list() self.fragment = _fragment self.flags = _flags self.index = _index @@ -1713,9 +2504,21 @@ class xrfragment_XRF: def guessType(self,v,_hx_str): v.string = _hx_str + _this = xrfragment_XRF.isReset + _this.matchObj = python_lib_Re.search(_this.pattern,v.fragment) + if (_this.matchObj is not None): + v.reset = True + if (v.fragment == "loop"): + v.loop = True if (not Std.isOfType(_hx_str,str)): return if (len(_hx_str) > 0): + _this = xrfragment_XRF.isXRFScheme + _this.matchObj = python_lib_Re.search(_this.pattern,_hx_str) + if (_this.matchObj is not None): + v.xrfScheme = True + _hx_str = xrfragment_XRF.isXRFScheme.replace(_hx_str,"") + v.string = _hx_str if (len(_hx_str.split(",")) > 1): xyzn = _hx_str.split(",") if (len(xyzn) > 0): @@ -1729,9 +2532,14 @@ class xrfragment_XRF: while (_g < _g1): i = _g _g = (_g + 1) - _this = v.floats - x = Std.parseFloat((xyzn[i] if i >= 0 and i < len(xyzn) else None)) + _this = v.shift + _this1 = xrfragment_XRF.isShift + _this1.matchObj = python_lib_Re.search(_this1.pattern,(xyzn[i] if i >= 0 and i < len(xyzn) else None)) + x = (_this1.matchObj is not None) _this.append(x) + _this2 = v.floats + x1 = Std.parseFloat(xrfragment_XRF.isShift.replace((xyzn[i] if i >= 0 and i < len(xyzn) else None),"")) + _this2.append(x1) _this = xrfragment_XRF.isColor _this.matchObj = python_lib_Re.search(_this.pattern,_hx_str) if (_this.matchObj is not None): @@ -1746,22 +2554,9 @@ class xrfragment_XRF: if (_this.matchObj is not None): v.int = Std.parseInt(_hx_str) v.x = v.int - _this = xrfragment_XRF.isMediaFrag - _this.matchObj = python_lib_Re.search(_this.pattern,_hx_str) - if (_this.matchObj is not None): - speed = _hx_str.split("*") - v.speed = list() - if (len(speed) > 1): - _this = (speed[1] if 1 < len(speed) else None) - values = _this.split(",") - _g = 0 - _g1 = len(values) - while (_g < _g1): - i = _g - _g = (_g + 1) - _this = v.speed - x = Std.parseFloat((values[i] if i >= 0 and i < len(values) else None)) - _this.append(x) + _this = v.floats + x = v.x + _this.append(x) v.filter = xrfragment_Filter(((HxOverrides.stringOrNull(v.fragment) + "=") + HxOverrides.stringOrNull(v.string))) else: v.filter = xrfragment_Filter(v.fragment) @@ -1774,6 +2569,7 @@ class xrfragment_XRF: def unset(flag,flags): return (flags & ~flag) +xrfragment_XRF._hx_class = xrfragment_XRF Math.NEGATIVE_INFINITY = float("-inf") Math.POSITIVE_INFINITY = float("inf") @@ -1781,15 +2577,23 @@ Math.NaN = float("nan") Math.PI = python_lib_Math.pi Test.errors = 0 +haxe_Template.splitter = EReg("(::[A-Za-z0-9_ ()&|!+=/><*.\"-]+::|\\$\\$([A-Za-z0-9_-]+)\\()","") +haxe_Template.expr_splitter = EReg("(\\(|\\)|[ \r\n\t]*\"[^\"]*\"[ \r\n\t]*|[!+=/><*.&|-]+)","") +haxe_Template.expr_trim = EReg("^[ ]*([^ ]+)[ ]*$","") +haxe_Template.expr_int = EReg("^[0-9]+$","") +haxe_Template.expr_float = EReg("^([+-]?)(?=\\d|,\\d)\\d*(,\\d*)?([Ee]([+-]?\\d+))?$","") +haxe_Template.globals = _hx_AnonObject({}) +haxe_Template.hxKeepArrayIterator = haxe_iterators_ArrayIterator([]) python_Boot.keywords = set(["and", "del", "from", "not", "with", "as", "elif", "global", "or", "yield", "assert", "else", "if", "pass", "None", "break", "except", "import", "raise", "True", "class", "exec", "in", "return", "False", "continue", "finally", "is", "try", "def", "for", "lambda", "while"]) python_Boot.prefixLength = len("_hx_") xrfragment_Parser.error = "" xrfragment_Parser.debug = False -xrfragment_XRF.ASSET = 1 +xrfragment_URI.__meta__ = _hx_AnonObject({'statics': _hx_AnonObject({'template': _hx_AnonObject({'keep': None})})}) +xrfragment_XRF.IMMUTABLE = 1 xrfragment_XRF.PROP_BIND = 2 xrfragment_XRF.QUERY_OPERATOR = 4 xrfragment_XRF.PROMPT = 8 -xrfragment_XRF.ROUNDROBIN = 16 +xrfragment_XRF.CUSTOMFRAG = 16 xrfragment_XRF.NAVIGATOR = 32 xrfragment_XRF.METADATA = 64 xrfragment_XRF.PV_OVERRIDE = 128 @@ -1803,7 +2607,8 @@ xrfragment_XRF.T_URL = 262144 xrfragment_XRF.T_PREDEFINED_VIEW = 524288 xrfragment_XRF.T_STRING = 1048576 xrfragment_XRF.T_MEDIAFRAG = 2097152 -xrfragment_XRF.T_DYNAMIC = 4194304 +xrfragment_XRF.T_DYNAMICKEY = 4194304 +xrfragment_XRF.T_DYNAMICKEYVALUE = 8388608 xrfragment_XRF.isColor = EReg("^#([A-Fa-f0-9]{6}|[A-Fa-f0-9]{3})$","") xrfragment_XRF.isInt = EReg("^[-0-9]+$","") xrfragment_XRF.isFloat = EReg("^[-0-9]+\\.[0-9]+$","") @@ -1811,11 +2616,14 @@ xrfragment_XRF.isVector = EReg("([,]+|\\w)","") xrfragment_XRF.isUrl = EReg("(://)?\\..*","") xrfragment_XRF.isUrlOrPretypedView = EReg("(^#|://)?\\..*","") xrfragment_XRF.isString = EReg(".*","") -xrfragment_XRF.operators = EReg("(^-|[\\*]+)","") +xrfragment_XRF.operators = EReg("(^[-]|^[!]|[\\*]$)","g") xrfragment_XRF.isProp = EReg("^.*=[><=]?","") xrfragment_XRF.isExclude = EReg("^-","") xrfragment_XRF.isDeep = EReg("\\*","") xrfragment_XRF.isNumber = EReg("^[0-9\\.]+$","") -xrfragment_XRF.isMediaFrag = EReg("^[0-9\\.,\\*]+$","") +xrfragment_XRF.isMediaFrag = EReg("^([0-9\\.,\\*+-]+)$","") +xrfragment_XRF.isReset = EReg("^!","") +xrfragment_XRF.isShift = EReg("^(\\+|--)","") +xrfragment_XRF.isXRFScheme = EReg("^xrf://","") Test.main()