+ <% end %>
+<% end %>
+
+<% content_for :actions do %>
+ <%= link_to t(".bulk_edit"), edit_models_path((@additional_filters || {}).merge(@filter&.to_params)), class: "btn btn-secondary" if policy(:model).edit? %>
+<% end %>
+
+<% content_for :sidebar do %>
+ <%= render "filters_card" %>
+ <%= render "tags_card" %>
+<% end %>
diff --git a/manyfold/usr/src/app/app/views/models/_tags_info.html.erb b/manyfold/usr/src/app/app/views/models/_tags_info.html.erb
index d7d2e2d..a9611d5 100644
--- a/manyfold/usr/src/app/app/views/models/_tags_info.html.erb
+++ b/manyfold/usr/src/app/app/views/models/_tags_info.html.erb
@@ -47,4 +47,21 @@
Set this tag for experiences (meditation, personal productivity space) which are better off without multiple users.
+
+ This shows avatars of users participating in the space.
+ Enable this if you want metaverse-ish kind of experiences in which user-locations are important.
+
-
-
- This is a Godot project which wraps your (3D file) experience.
- Godot is a Free and Opensource Game engine.
- The Godot project is basically its own XR Fragment browser (which you can extend).
+
+
+ This is a Godot project which wraps your (3D file) experience.
+ Godot is a Free and Opensource Game engine.
+ The Godot project is basically its own XR Fragment browser (which you can extend).
+ <% end %>
<%= card :secondary, t("layouts.card_list_page.actions_heading") do %>
<%= render Components::ReportButton.new(object: @model, path: new_model_report_path(@model)) %>
diff --git a/manyfold/usr/src/app/config/initializers/content_security_policy.rb b/manyfold/usr/src/app/config/initializers/content_security_policy.rb
index 6912f1a..b8d4b0b 100644
--- a/manyfold/usr/src/app/config/initializers/content_security_policy.rb
+++ b/manyfold/usr/src/app/config/initializers/content_security_policy.rb
@@ -8,3 +8,20 @@ Rails.application.configure do
end
end
end
+
+Rails.application.config.after_initialize do
+ Rails.application.configure do
+ config.content_security_policy do |policy|
+ # Default policy for all content
+ policy.default_src :self, :https, :http
+
+ # *** ADD THE DOMAIN(S) FOR YOUR IFRAME CONTENT HERE ***
+ policy.frame_src :self, ENV['FEDERATE_DRIVE_HOST'], ENV['SERVER_CORS'], ENV['SERVER_JANUS']
+ policy.script_src :self, ENV['FEDERATE_DRIVE_HOST'], ENV['SERVER_CORS'], ENV['SERVER_JANUS']
+ policy.style_src :self, ENV['FEDERATE_DRIVE_HOST'], ENV['SERVER_CORS'], ENV['SERVER_JANUS']
+ policy.connect_src :self, ENV['FEDERATE_DRIVE_HOST'], ENV['SERVER_CORS'], ENV['SERVER_JANUS']
+
+ # ...
+ end
+ end
+end
diff --git a/manyfold/usr/src/app/config/initializers/xrforge.rb b/manyfold/usr/src/app/config/initializers/xrforge.rb
index bc616df..ec1a8e1 100644
--- a/manyfold/usr/src/app/config/initializers/xrforge.rb
+++ b/manyfold/usr/src/app/config/initializers/xrforge.rb
@@ -15,7 +15,8 @@ Rails.application.config.to_prepare do
def run_cli_hooks
- file = "#{self.model.library.path}/#{self.path_within_library()}"
+ file = "#{self.model.library.path || ENV['UPLOAD_PATH'] || "/mnt/experiences"}"
+ file = file.strip() + "/" + self.path_within_library().strip()
if File.exist?(file) && (ENV['BOOT_SCAN'].present? || file.match?("datapackage") )
cache_key = "ttl:file:cli_hook:#{self.id}#{Digest::MD5.file(file).hexdigest}"
diff --git a/manyfold/usr/src/app/config/routes/proxyanywhere.rb b/manyfold/usr/src/app/config/routes/proxyanywhere.rb
new file mode 100644
index 0000000..ec34059
--- /dev/null
+++ b/manyfold/usr/src/app/config/routes/proxyanywhere.rb
@@ -0,0 +1,3 @@
+if ENV['CORS_PROXY']
+ match '/cors/*target_url_segment', to: 'cors_proxy#proxy', via: :all, format: false
+end
diff --git a/manyfold/usr/src/app/public/assets/howto.png b/manyfold/usr/src/app/public/assets/howto.png
new file mode 100644
index 0000000..8444d3e
Binary files /dev/null and b/manyfold/usr/src/app/public/assets/howto.png differ
diff --git a/manyfold/usr/src/app/public/assets/lobby.png b/manyfold/usr/src/app/public/assets/lobby.png
deleted file mode 100644
index fbee904..0000000
Binary files a/manyfold/usr/src/app/public/assets/lobby.png and /dev/null differ
diff --git a/manyfold/usr/src/app/public/assets/xrforge.css b/manyfold/usr/src/app/public/assets/xrforge.css
index 6934af0..ce80781 100644
--- a/manyfold/usr/src/app/public/assets/xrforge.css
+++ b/manyfold/usr/src/app/public/assets/xrforge.css
@@ -32,3 +32,53 @@ input[type="checkbox"]:checked + .hidden-tooltip {
.col-form-label{
width:111px;
}
+
+/* shimmer loader */
+
+.shimcontainer {
+ width: 100%;
+ box-shadow: 0px 4px 8px rgba(0, 0, 0, 0.1);
+ background: #000;
+ border-radius: 10px;
+}
+
+article {
+ background: #000;
+ width: 100%;
+ position: relative;
+ box-sizing: border-box;
+ border-radius: 10px;
+ overflow: hidden;
+ text-align:center;
+ color:#AAA;
+}
+
+.shimmer {
+ position: absolute;
+ top: 0;
+ left: 0;
+ width: 50%;
+ height: 100%;
+ background: linear-gradient(
+ 100deg,
+ rgba(255, 255, 255, 0) 20%,
+ rgba(255, 255, 255, 0.2) 50%,
+ rgba(255, 255, 255, 0) 80%
+ );
+ animation: shimmer 2s infinite linear;
+}
+
+@keyframes shimmer {
+ from {
+ transform: translateX(-200%);
+ }
+ to {
+ transform: translateX(200%);
+ }
+}
+
+.shimcontainer img {
+ height: 175px;
+ margin-top: 37px;
+ display:inline-block;
+}
diff --git a/manyfold/usr/src/app/public/view/AR.js b/manyfold/usr/src/app/public/view/AR.js
new file mode 100644
index 0000000..c1457dc
--- /dev/null
+++ b/manyfold/usr/src/app/public/view/AR.js
@@ -0,0 +1,49 @@
+room.onLoad = function () {
+
+ buttonPrompt();
+
+}
+
+function buttonPrompt(){
+
+ let style = document.createElement("style")
+ style.type = "text/css"
+ style.innerHTML = `
+ body{
+ background: #555;
+ height: 100%;
+ margin: 0;
+ overflow: hidden;
+ padding: 0;
+ width: 100%;
+ }
+
+ img#btn_play:hover {
+ opacity: 1;
+ }
+ img#btn_play {
+ position: fixed;
+ top: 50%;
+ left: 50%;
+ width:30.000002%;
+ max-width:400px;
+ transform: translate(-50%,-50%);
+ cursor: pointer;
+ opacity: 0.6;
+ transition: 0.3s;
+ z-index: 999;
+ }
+ `)
+ document.body.appendChild(style);
+
+ let playBtn = document.createElement("img")
+ playBtn.src = "play.svg"
+ playBtn.id = "btn_play"
+ playBtn.onclick = function () {
+ document.body.removeChild(promptDiv);
+ janus.engine.client.startXR('immersive-ar', { requiredFeatures: [], optionalFeatures: ['local-floor', 'plane-detection'] });
+ room.skybox = false;
+ };
+ document.body.appendChild(playBtn);
+}
+
diff --git a/manyfold/usr/src/app/public/view/aframe.html b/manyfold/usr/src/app/public/view/aframe.html
new file mode 100644
index 0000000..edad99a
--- /dev/null
+++ b/manyfold/usr/src/app/public/view/aframe.html
@@ -0,0 +1,51 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/manyfold/usr/src/app/public/view/avatar/avatar.xml b/manyfold/usr/src/app/public/view/avatar/avatar.xml
new file mode 100644
index 0000000..c4a5b8c
--- /dev/null
+++ b/manyfold/usr/src/app/public/view/avatar/avatar.xml
@@ -0,0 +1,10 @@
+
+
+
+
+
+
+
+
+
+
diff --git a/manyfold/usr/src/app/public/view/avatar/janus-avatar-animations-sit.glb b/manyfold/usr/src/app/public/view/avatar/janus-avatar-animations-sit.glb
new file mode 100644
index 0000000..69b08f4
Binary files /dev/null and b/manyfold/usr/src/app/public/view/avatar/janus-avatar-animations-sit.glb differ
diff --git a/manyfold/usr/src/app/public/view/avatar/janus-avatar-animations.glb b/manyfold/usr/src/app/public/view/avatar/janus-avatar-animations.glb
new file mode 100644
index 0000000..daf2213
Binary files /dev/null and b/manyfold/usr/src/app/public/view/avatar/janus-avatar-animations.glb differ
diff --git a/manyfold/usr/src/app/public/view/avatar/janus-avatar-base.glb b/manyfold/usr/src/app/public/view/avatar/janus-avatar-base.glb
new file mode 100644
index 0000000..66969db
Binary files /dev/null and b/manyfold/usr/src/app/public/view/avatar/janus-avatar-base.glb differ
diff --git a/manyfold/usr/src/app/public/view/blackskybox.png b/manyfold/usr/src/app/public/view/blackskybox.png
new file mode 100644
index 0000000..3cf27f4
Binary files /dev/null and b/manyfold/usr/src/app/public/view/blackskybox.png differ
diff --git a/manyfold/usr/src/app/public/view/element/LICENSE.txt b/manyfold/usr/src/app/public/view/element/LICENSE.txt
new file mode 100644
index 0000000..f288702
--- /dev/null
+++ b/manyfold/usr/src/app/public/view/element/LICENSE.txt
@@ -0,0 +1,674 @@
+ GNU GENERAL PUBLIC LICENSE
+ Version 3, 29 June 2007
+
+ Copyright (C) 2007 Free Software Foundation, Inc.
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+ Preamble
+
+ The GNU General Public License is a free, copyleft license for
+software and other kinds of works.
+
+ The licenses for most software and other practical works are designed
+to take away your freedom to share and change the works. By contrast,
+the GNU General Public License is intended to guarantee your freedom to
+share and change all versions of a program--to make sure it remains free
+software for all its users. We, the Free Software Foundation, use the
+GNU General Public License for most of our software; it applies also to
+any other work released this way by its authors. You can apply it to
+your programs, too.
+
+ When we speak of free software, we are referring to freedom, not
+price. Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+them if you wish), that you receive source code or can get it if you
+want it, that you can change the software or use pieces of it in new
+free programs, and that you know you can do these things.
+
+ To protect your rights, we need to prevent others from denying you
+these rights or asking you to surrender the rights. Therefore, you have
+certain responsibilities if you distribute copies of the software, or if
+you modify it: responsibilities to respect the freedom of others.
+
+ For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must pass on to the recipients the same
+freedoms that you received. You must make sure that they, too, receive
+or can get the source code. And you must show them these terms so they
+know their rights.
+
+ Developers that use the GNU GPL protect your rights with two steps:
+(1) assert copyright on the software, and (2) offer you this License
+giving you legal permission to copy, distribute and/or modify it.
+
+ For the developers' and authors' protection, the GPL clearly explains
+that there is no warranty for this free software. For both users' and
+authors' sake, the GPL requires that modified versions be marked as
+changed, so that their problems will not be attributed erroneously to
+authors of previous versions.
+
+ Some devices are designed to deny users access to install or run
+modified versions of the software inside them, although the manufacturer
+can do so. This is fundamentally incompatible with the aim of
+protecting users' freedom to change the software. The systematic
+pattern of such abuse occurs in the area of products for individuals to
+use, which is precisely where it is most unacceptable. Therefore, we
+have designed this version of the GPL to prohibit the practice for those
+products. If such problems arise substantially in other domains, we
+stand ready to extend this provision to those domains in future versions
+of the GPL, as needed to protect the freedom of users.
+
+ Finally, every program is threatened constantly by software patents.
+States should not allow patents to restrict development and use of
+software on general-purpose computers, but in those that do, we wish to
+avoid the special danger that patents applied to a free program could
+make it effectively proprietary. To prevent this, the GPL assures that
+patents cannot be used to render the program non-free.
+
+ The precise terms and conditions for copying, distribution and
+modification follow.
+
+ TERMS AND CONDITIONS
+
+ 0. Definitions.
+
+ "This License" refers to version 3 of the GNU General Public License.
+
+ "Copyright" also means copyright-like laws that apply to other kinds of
+works, such as semiconductor masks.
+
+ "The Program" refers to any copyrightable work licensed under this
+License. Each licensee is addressed as "you". "Licensees" and
+"recipients" may be individuals or organizations.
+
+ To "modify" a work means to copy from or adapt all or part of the work
+in a fashion requiring copyright permission, other than the making of an
+exact copy. The resulting work is called a "modified version" of the
+earlier work or a work "based on" the earlier work.
+
+ A "covered work" means either the unmodified Program or a work based
+on the Program.
+
+ To "propagate" a work means to do anything with it that, without
+permission, would make you directly or secondarily liable for
+infringement under applicable copyright law, except executing it on a
+computer or modifying a private copy. Propagation includes copying,
+distribution (with or without modification), making available to the
+public, and in some countries other activities as well.
+
+ To "convey" a work means any kind of propagation that enables other
+parties to make or receive copies. Mere interaction with a user through
+a computer network, with no transfer of a copy, is not conveying.
+
+ An interactive user interface displays "Appropriate Legal Notices"
+to the extent that it includes a convenient and prominently visible
+feature that (1) displays an appropriate copyright notice, and (2)
+tells the user that there is no warranty for the work (except to the
+extent that warranties are provided), that licensees may convey the
+work under this License, and how to view a copy of this License. If
+the interface presents a list of user commands or options, such as a
+menu, a prominent item in the list meets this criterion.
+
+ 1. Source Code.
+
+ The "source code" for a work means the preferred form of the work
+for making modifications to it. "Object code" means any non-source
+form of a work.
+
+ A "Standard Interface" means an interface that either is an official
+standard defined by a recognized standards body, or, in the case of
+interfaces specified for a particular programming language, one that
+is widely used among developers working in that language.
+
+ The "System Libraries" of an executable work include anything, other
+than the work as a whole, that (a) is included in the normal form of
+packaging a Major Component, but which is not part of that Major
+Component, and (b) serves only to enable use of the work with that
+Major Component, or to implement a Standard Interface for which an
+implementation is available to the public in source code form. A
+"Major Component", in this context, means a major essential component
+(kernel, window system, and so on) of the specific operating system
+(if any) on which the executable work runs, or a compiler used to
+produce the work, or an object code interpreter used to run it.
+
+ The "Corresponding Source" for a work in object code form means all
+the source code needed to generate, install, and (for an executable
+work) run the object code and to modify the work, including scripts to
+control those activities. However, it does not include the work's
+System Libraries, or general-purpose tools or generally available free
+programs which are used unmodified in performing those activities but
+which are not part of the work. For example, Corresponding Source
+includes interface definition files associated with source files for
+the work, and the source code for shared libraries and dynamically
+linked subprograms that the work is specifically designed to require,
+such as by intimate data communication or control flow between those
+subprograms and other parts of the work.
+
+ The Corresponding Source need not include anything that users
+can regenerate automatically from other parts of the Corresponding
+Source.
+
+ The Corresponding Source for a work in source code form is that
+same work.
+
+ 2. Basic Permissions.
+
+ All rights granted under this License are granted for the term of
+copyright on the Program, and are irrevocable provided the stated
+conditions are met. This License explicitly affirms your unlimited
+permission to run the unmodified Program. The output from running a
+covered work is covered by this License only if the output, given its
+content, constitutes a covered work. This License acknowledges your
+rights of fair use or other equivalent, as provided by copyright law.
+
+ You may make, run and propagate covered works that you do not
+convey, without conditions so long as your license otherwise remains
+in force. You may convey covered works to others for the sole purpose
+of having them make modifications exclusively for you, or provide you
+with facilities for running those works, provided that you comply with
+the terms of this License in conveying all material for which you do
+not control copyright. Those thus making or running the covered works
+for you must do so exclusively on your behalf, under your direction
+and control, on terms that prohibit them from making any copies of
+your copyrighted material outside their relationship with you.
+
+ Conveying under any other circumstances is permitted solely under
+the conditions stated below. Sublicensing is not allowed; section 10
+makes it unnecessary.
+
+ 3. Protecting Users' Legal Rights From Anti-Circumvention Law.
+
+ No covered work shall be deemed part of an effective technological
+measure under any applicable law fulfilling obligations under article
+11 of the WIPO copyright treaty adopted on 20 December 1996, or
+similar laws prohibiting or restricting circumvention of such
+measures.
+
+ When you convey a covered work, you waive any legal power to forbid
+circumvention of technological measures to the extent such circumvention
+is effected by exercising rights under this License with respect to
+the covered work, and you disclaim any intention to limit operation or
+modification of the work as a means of enforcing, against the work's
+users, your or third parties' legal rights to forbid circumvention of
+technological measures.
+
+ 4. Conveying Verbatim Copies.
+
+ You may convey verbatim copies of the Program's source code as you
+receive it, in any medium, provided that you conspicuously and
+appropriately publish on each copy an appropriate copyright notice;
+keep intact all notices stating that this License and any
+non-permissive terms added in accord with section 7 apply to the code;
+keep intact all notices of the absence of any warranty; and give all
+recipients a copy of this License along with the Program.
+
+ You may charge any price or no price for each copy that you convey,
+and you may offer support or warranty protection for a fee.
+
+ 5. Conveying Modified Source Versions.
+
+ You may convey a work based on the Program, or the modifications to
+produce it from the Program, in the form of source code under the
+terms of section 4, provided that you also meet all of these conditions:
+
+ a) The work must carry prominent notices stating that you modified
+ it, and giving a relevant date.
+
+ b) The work must carry prominent notices stating that it is
+ released under this License and any conditions added under section
+ 7. This requirement modifies the requirement in section 4 to
+ "keep intact all notices".
+
+ c) You must license the entire work, as a whole, under this
+ License to anyone who comes into possession of a copy. This
+ License will therefore apply, along with any applicable section 7
+ additional terms, to the whole of the work, and all its parts,
+ regardless of how they are packaged. This License gives no
+ permission to license the work in any other way, but it does not
+ invalidate such permission if you have separately received it.
+
+ d) If the work has interactive user interfaces, each must display
+ Appropriate Legal Notices; however, if the Program has interactive
+ interfaces that do not display Appropriate Legal Notices, your
+ work need not make them do so.
+
+ A compilation of a covered work with other separate and independent
+works, which are not by their nature extensions of the covered work,
+and which are not combined with it such as to form a larger program,
+in or on a volume of a storage or distribution medium, is called an
+"aggregate" if the compilation and its resulting copyright are not
+used to limit the access or legal rights of the compilation's users
+beyond what the individual works permit. Inclusion of a covered work
+in an aggregate does not cause this License to apply to the other
+parts of the aggregate.
+
+ 6. Conveying Non-Source Forms.
+
+ You may convey a covered work in object code form under the terms
+of sections 4 and 5, provided that you also convey the
+machine-readable Corresponding Source under the terms of this License,
+in one of these ways:
+
+ a) Convey the object code in, or embodied in, a physical product
+ (including a physical distribution medium), accompanied by the
+ Corresponding Source fixed on a durable physical medium
+ customarily used for software interchange.
+
+ b) Convey the object code in, or embodied in, a physical product
+ (including a physical distribution medium), accompanied by a
+ written offer, valid for at least three years and valid for as
+ long as you offer spare parts or customer support for that product
+ model, to give anyone who possesses the object code either (1) a
+ copy of the Corresponding Source for all the software in the
+ product that is covered by this License, on a durable physical
+ medium customarily used for software interchange, for a price no
+ more than your reasonable cost of physically performing this
+ conveying of source, or (2) access to copy the
+ Corresponding Source from a network server at no charge.
+
+ c) Convey individual copies of the object code with a copy of the
+ written offer to provide the Corresponding Source. This
+ alternative is allowed only occasionally and noncommercially, and
+ only if you received the object code with such an offer, in accord
+ with subsection 6b.
+
+ d) Convey the object code by offering access from a designated
+ place (gratis or for a charge), and offer equivalent access to the
+ Corresponding Source in the same way through the same place at no
+ further charge. You need not require recipients to copy the
+ Corresponding Source along with the object code. If the place to
+ copy the object code is a network server, the Corresponding Source
+ may be on a different server (operated by you or a third party)
+ that supports equivalent copying facilities, provided you maintain
+ clear directions next to the object code saying where to find the
+ Corresponding Source. Regardless of what server hosts the
+ Corresponding Source, you remain obligated to ensure that it is
+ available for as long as needed to satisfy these requirements.
+
+ e) Convey the object code using peer-to-peer transmission, provided
+ you inform other peers where the object code and Corresponding
+ Source of the work are being offered to the general public at no
+ charge under subsection 6d.
+
+ A separable portion of the object code, whose source code is excluded
+from the Corresponding Source as a System Library, need not be
+included in conveying the object code work.
+
+ A "User Product" is either (1) a "consumer product", which means any
+tangible personal property which is normally used for personal, family,
+or household purposes, or (2) anything designed or sold for incorporation
+into a dwelling. In determining whether a product is a consumer product,
+doubtful cases shall be resolved in favor of coverage. For a particular
+product received by a particular user, "normally used" refers to a
+typical or common use of that class of product, regardless of the status
+of the particular user or of the way in which the particular user
+actually uses, or expects or is expected to use, the product. A product
+is a consumer product regardless of whether the product has substantial
+commercial, industrial or non-consumer uses, unless such uses represent
+the only significant mode of use of the product.
+
+ "Installation Information" for a User Product means any methods,
+procedures, authorization keys, or other information required to install
+and execute modified versions of a covered work in that User Product from
+a modified version of its Corresponding Source. The information must
+suffice to ensure that the continued functioning of the modified object
+code is in no case prevented or interfered with solely because
+modification has been made.
+
+ If you convey an object code work under this section in, or with, or
+specifically for use in, a User Product, and the conveying occurs as
+part of a transaction in which the right of possession and use of the
+User Product is transferred to the recipient in perpetuity or for a
+fixed term (regardless of how the transaction is characterized), the
+Corresponding Source conveyed under this section must be accompanied
+by the Installation Information. But this requirement does not apply
+if neither you nor any third party retains the ability to install
+modified object code on the User Product (for example, the work has
+been installed in ROM).
+
+ The requirement to provide Installation Information does not include a
+requirement to continue to provide support service, warranty, or updates
+for a work that has been modified or installed by the recipient, or for
+the User Product in which it has been modified or installed. Access to a
+network may be denied when the modification itself materially and
+adversely affects the operation of the network or violates the rules and
+protocols for communication across the network.
+
+ Corresponding Source conveyed, and Installation Information provided,
+in accord with this section must be in a format that is publicly
+documented (and with an implementation available to the public in
+source code form), and must require no special password or key for
+unpacking, reading or copying.
+
+ 7. Additional Terms.
+
+ "Additional permissions" are terms that supplement the terms of this
+License by making exceptions from one or more of its conditions.
+Additional permissions that are applicable to the entire Program shall
+be treated as though they were included in this License, to the extent
+that they are valid under applicable law. If additional permissions
+apply only to part of the Program, that part may be used separately
+under those permissions, but the entire Program remains governed by
+this License without regard to the additional permissions.
+
+ When you convey a copy of a covered work, you may at your option
+remove any additional permissions from that copy, or from any part of
+it. (Additional permissions may be written to require their own
+removal in certain cases when you modify the work.) You may place
+additional permissions on material, added by you to a covered work,
+for which you have or can give appropriate copyright permission.
+
+ Notwithstanding any other provision of this License, for material you
+add to a covered work, you may (if authorized by the copyright holders of
+that material) supplement the terms of this License with terms:
+
+ a) Disclaiming warranty or limiting liability differently from the
+ terms of sections 15 and 16 of this License; or
+
+ b) Requiring preservation of specified reasonable legal notices or
+ author attributions in that material or in the Appropriate Legal
+ Notices displayed by works containing it; or
+
+ c) Prohibiting misrepresentation of the origin of that material, or
+ requiring that modified versions of such material be marked in
+ reasonable ways as different from the original version; or
+
+ d) Limiting the use for publicity purposes of names of licensors or
+ authors of the material; or
+
+ e) Declining to grant rights under trademark law for use of some
+ trade names, trademarks, or service marks; or
+
+ f) Requiring indemnification of licensors and authors of that
+ material by anyone who conveys the material (or modified versions of
+ it) with contractual assumptions of liability to the recipient, for
+ any liability that these contractual assumptions directly impose on
+ those licensors and authors.
+
+ All other non-permissive additional terms are considered "further
+restrictions" within the meaning of section 10. If the Program as you
+received it, or any part of it, contains a notice stating that it is
+governed by this License along with a term that is a further
+restriction, you may remove that term. If a license document contains
+a further restriction but permits relicensing or conveying under this
+License, you may add to a covered work material governed by the terms
+of that license document, provided that the further restriction does
+not survive such relicensing or conveying.
+
+ If you add terms to a covered work in accord with this section, you
+must place, in the relevant source files, a statement of the
+additional terms that apply to those files, or a notice indicating
+where to find the applicable terms.
+
+ Additional terms, permissive or non-permissive, may be stated in the
+form of a separately written license, or stated as exceptions;
+the above requirements apply either way.
+
+ 8. Termination.
+
+ You may not propagate or modify a covered work except as expressly
+provided under this License. Any attempt otherwise to propagate or
+modify it is void, and will automatically terminate your rights under
+this License (including any patent licenses granted under the third
+paragraph of section 11).
+
+ However, if you cease all violation of this License, then your
+license from a particular copyright holder is reinstated (a)
+provisionally, unless and until the copyright holder explicitly and
+finally terminates your license, and (b) permanently, if the copyright
+holder fails to notify you of the violation by some reasonable means
+prior to 60 days after the cessation.
+
+ Moreover, your license from a particular copyright holder is
+reinstated permanently if the copyright holder notifies you of the
+violation by some reasonable means, this is the first time you have
+received notice of violation of this License (for any work) from that
+copyright holder, and you cure the violation prior to 30 days after
+your receipt of the notice.
+
+ Termination of your rights under this section does not terminate the
+licenses of parties who have received copies or rights from you under
+this License. If your rights have been terminated and not permanently
+reinstated, you do not qualify to receive new licenses for the same
+material under section 10.
+
+ 9. Acceptance Not Required for Having Copies.
+
+ You are not required to accept this License in order to receive or
+run a copy of the Program. Ancillary propagation of a covered work
+occurring solely as a consequence of using peer-to-peer transmission
+to receive a copy likewise does not require acceptance. However,
+nothing other than this License grants you permission to propagate or
+modify any covered work. These actions infringe copyright if you do
+not accept this License. Therefore, by modifying or propagating a
+covered work, you indicate your acceptance of this License to do so.
+
+ 10. Automatic Licensing of Downstream Recipients.
+
+ Each time you convey a covered work, the recipient automatically
+receives a license from the original licensors, to run, modify and
+propagate that work, subject to this License. You are not responsible
+for enforcing compliance by third parties with this License.
+
+ An "entity transaction" is a transaction transferring control of an
+organization, or substantially all assets of one, or subdividing an
+organization, or merging organizations. If propagation of a covered
+work results from an entity transaction, each party to that
+transaction who receives a copy of the work also receives whatever
+licenses to the work the party's predecessor in interest had or could
+give under the previous paragraph, plus a right to possession of the
+Corresponding Source of the work from the predecessor in interest, if
+the predecessor has it or can get it with reasonable efforts.
+
+ You may not impose any further restrictions on the exercise of the
+rights granted or affirmed under this License. For example, you may
+not impose a license fee, royalty, or other charge for exercise of
+rights granted under this License, and you may not initiate litigation
+(including a cross-claim or counterclaim in a lawsuit) alleging that
+any patent claim is infringed by making, using, selling, offering for
+sale, or importing the Program or any portion of it.
+
+ 11. Patents.
+
+ A "contributor" is a copyright holder who authorizes use under this
+License of the Program or a work on which the Program is based. The
+work thus licensed is called the contributor's "contributor version".
+
+ A contributor's "essential patent claims" are all patent claims
+owned or controlled by the contributor, whether already acquired or
+hereafter acquired, that would be infringed by some manner, permitted
+by this License, of making, using, or selling its contributor version,
+but do not include claims that would be infringed only as a
+consequence of further modification of the contributor version. For
+purposes of this definition, "control" includes the right to grant
+patent sublicenses in a manner consistent with the requirements of
+this License.
+
+ Each contributor grants you a non-exclusive, worldwide, royalty-free
+patent license under the contributor's essential patent claims, to
+make, use, sell, offer for sale, import and otherwise run, modify and
+propagate the contents of its contributor version.
+
+ In the following three paragraphs, a "patent license" is any express
+agreement or commitment, however denominated, not to enforce a patent
+(such as an express permission to practice a patent or covenant not to
+sue for patent infringement). To "grant" such a patent license to a
+party means to make such an agreement or commitment not to enforce a
+patent against the party.
+
+ If you convey a covered work, knowingly relying on a patent license,
+and the Corresponding Source of the work is not available for anyone
+to copy, free of charge and under the terms of this License, through a
+publicly available network server or other readily accessible means,
+then you must either (1) cause the Corresponding Source to be so
+available, or (2) arrange to deprive yourself of the benefit of the
+patent license for this particular work, or (3) arrange, in a manner
+consistent with the requirements of this License, to extend the patent
+license to downstream recipients. "Knowingly relying" means you have
+actual knowledge that, but for the patent license, your conveying the
+covered work in a country, or your recipient's use of the covered work
+in a country, would infringe one or more identifiable patents in that
+country that you have reason to believe are valid.
+
+ If, pursuant to or in connection with a single transaction or
+arrangement, you convey, or propagate by procuring conveyance of, a
+covered work, and grant a patent license to some of the parties
+receiving the covered work authorizing them to use, propagate, modify
+or convey a specific copy of the covered work, then the patent license
+you grant is automatically extended to all recipients of the covered
+work and works based on it.
+
+ A patent license is "discriminatory" if it does not include within
+the scope of its coverage, prohibits the exercise of, or is
+conditioned on the non-exercise of one or more of the rights that are
+specifically granted under this License. You may not convey a covered
+work if you are a party to an arrangement with a third party that is
+in the business of distributing software, under which you make payment
+to the third party based on the extent of your activity of conveying
+the work, and under which the third party grants, to any of the
+parties who would receive the covered work from you, a discriminatory
+patent license (a) in connection with copies of the covered work
+conveyed by you (or copies made from those copies), or (b) primarily
+for and in connection with specific products or compilations that
+contain the covered work, unless you entered into that arrangement,
+or that patent license was granted, prior to 28 March 2007.
+
+ Nothing in this License shall be construed as excluding or limiting
+any implied license or other defenses to infringement that may
+otherwise be available to you under applicable patent law.
+
+ 12. No Surrender of Others' Freedom.
+
+ If conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License. If you cannot convey a
+covered work so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you may
+not convey it at all. For example, if you agree to terms that obligate you
+to collect a royalty for further conveying from those to whom you convey
+the Program, the only way you could satisfy both those terms and this
+License would be to refrain entirely from conveying the Program.
+
+ 13. Use with the GNU Affero General Public License.
+
+ Notwithstanding any other provision of this License, you have
+permission to link or combine any covered work with a work licensed
+under version 3 of the GNU Affero General Public License into a single
+combined work, and to convey the resulting work. The terms of this
+License will continue to apply to the part which is the covered work,
+but the special requirements of the GNU Affero General Public License,
+section 13, concerning interaction through a network will apply to the
+combination as such.
+
+ 14. Revised Versions of this License.
+
+ The Free Software Foundation may publish revised and/or new versions of
+the GNU General Public License from time to time. Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+ Each version is given a distinguishing version number. If the
+Program specifies that a certain numbered version of the GNU General
+Public License "or any later version" applies to it, you have the
+option of following the terms and conditions either of that numbered
+version or of any later version published by the Free Software
+Foundation. If the Program does not specify a version number of the
+GNU General Public License, you may choose any version ever published
+by the Free Software Foundation.
+
+ If the Program specifies that a proxy can decide which future
+versions of the GNU General Public License can be used, that proxy's
+public statement of acceptance of a version permanently authorizes you
+to choose that version for the Program.
+
+ Later license versions may give you additional or different
+permissions. However, no additional obligations are imposed on any
+author or copyright holder as a result of your choosing to follow a
+later version.
+
+ 15. Disclaimer of Warranty.
+
+ THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
+APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
+HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
+OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
+THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
+IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
+ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
+
+ 16. Limitation of Liability.
+
+ IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
+THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
+GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
+USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
+DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
+PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
+EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
+SUCH DAMAGES.
+
+ 17. Interpretation of Sections 15 and 16.
+
+ If the disclaimer of warranty and limitation of liability provided
+above cannot be given local legal effect according to their terms,
+reviewing courts shall apply local law that most closely approximates
+an absolute waiver of all civil liability in connection with the
+Program, unless a warranty or assumption of liability accompanies a
+copy of the Program in return for a fee.
+
+ END OF TERMS AND CONDITIONS
+
+ How to Apply These Terms to Your New Programs
+
+ If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+ To do so, attach the following notices to the program. It is safest
+to attach them to the start of each source file to most effectively
+state the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+
+ Copyright (C)
+
+ This program is free software: you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program. If not, see .
+
+Also add information on how to contact you by electronic and paper mail.
+
+ If the program does terminal interaction, make it output a short
+notice like this when it starts in an interactive mode:
+
+ Copyright (C)
+ This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+ This is free software, and you are welcome to redistribute it
+ under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License. Of course, your program's commands
+might be different; for a GUI interface, you would use an "about box".
+
+ You should also get your employer (if you work as a programmer) or school,
+if any, to sign a "copyright disclaimer" for the program, if necessary.
+For more information on this, and how to apply and follow the GNU GPL, see
+.
+
+ The GNU General Public License does not permit incorporating your program
+into proprietary programs. If your program is a subroutine library, you
+may consider it more useful to permit linking proprietary applications with
+the library. If this is what you want to do, use the GNU Lesser General
+Public License instead of this License. But first, please read
+.
diff --git a/manyfold/usr/src/app/public/view/element/README.md b/manyfold/usr/src/app/public/view/element/README.md
new file mode 100644
index 0000000..5f0dabc
--- /dev/null
+++ b/manyfold/usr/src/app/public/view/element/README.md
@@ -0,0 +1,7 @@
+opensource libraries:
+
+https://github.com/jbaicoianu/janus-script-ui
+https://codeberg.org/coderofsalvation/janus-script-peertube
+ |- https://codeberg.org/coderofsalvation/janus-script-jjq
+ |- https://codeberg.org/coderofsalvation/janus-script-dialog
+ |- https://codeberg.org/coderofsalvation/janus-script-tween
diff --git a/manyfold/usr/src/app/public/view/element/janus-script-fetch.js b/manyfold/usr/src/app/public/view/element/janus-script-fetch.js
new file mode 100644
index 0000000..231dfbe
--- /dev/null
+++ b/manyfold/usr/src/app/public/view/element/janus-script-fetch.js
@@ -0,0 +1,44 @@
+room.registerElement("fetch",{
+ interval: 0,
+ to: "",
+ url: "",
+ selector: "",
+ append: false,
+ prefix: "",
+ loader: "loading..",
+
+ createChildren: function(){
+ let el = this.children[0]
+ if( !this.to ) return console.warn(" has no 'to' attr")
+ if( !this.children.length ) return console.warn(" needs one child element")
+ if( !el[this.to] ) el[this.to] = this.loader
+ if( this.append ) this.prefix = el[ this.to ]
+ this.fetch()
+
+ if( this.interval ) setInterval( () => this.fetch, this.interval * 1000 )
+ },
+
+ fetch: function(){
+ const finalUrl = `${elation.engine.assets.corsproxy||''}${this.url}`
+ fetch( finalUrl )
+ .then( (res) => res.text() )
+ .then( (text) => {
+ this.text = text
+ this.updateChild()
+ })
+ },
+
+ updateChild: function(){
+ let el = this.children[0]
+ let result = this.text
+ if( this.selector ){
+ let div = document.createElement("div")
+ div.innerHTML = this.text
+ let partial = div.querySelector(this.selector)
+ if( partial ){ result = partial.outerHTML
+ }else result = "oops..remote content not available"
+ }
+ el[ this.to ] = this.append ? this.prefix + result : result
+ }
+
+})
diff --git a/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.click.mp3 b/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.click.mp3
new file mode 100644
index 0000000..852f5fd
Binary files /dev/null and b/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.click.mp3 differ
diff --git a/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.glb b/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.glb
new file mode 100644
index 0000000..0c20185
Binary files /dev/null and b/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.glb differ
diff --git a/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.hover.mp3 b/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.hover.mp3
new file mode 100644
index 0000000..949ee29
Binary files /dev/null and b/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.hover.mp3 differ
diff --git a/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.icon.glb b/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.icon.glb
new file mode 100644
index 0000000..13818e2
Binary files /dev/null and b/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/button.icon.glb differ
diff --git a/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/icon.arrow-left.glb b/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/icon.arrow-left.glb
new file mode 100644
index 0000000..bc5128e
Binary files /dev/null and b/manyfold/usr/src/app/public/view/element/janus-script-peertube/asset/icon.arrow-left.glb differ
diff --git a/manyfold/usr/src/app/public/view/element/janus-script-peertube/janus-script-peertube.js b/manyfold/usr/src/app/public/view/element/janus-script-peertube/janus-script-peertube.js
new file mode 100644
index 0000000..73552b5
--- /dev/null
+++ b/manyfold/usr/src/app/public/view/element/janus-script-peertube/janus-script-peertube.js
@@ -0,0 +1,1753 @@
+room.registerElement('pushbutton', {
+ width: .5,
+ length: .5,
+ height: .5,
+ clicking: false,
+ state: 'off',
+ accesskey: false,
+
+ onbuttondown: new CustomEvent('buttondown'),
+ onbuttonpress: new CustomEvent('buttonpress'),
+ onbuttonup: new CustomEvent('buttonup'),
+ onclick: new CustomEvent('click'),
+
+ createChildren: function() {
+ this.base = room.createObject('Object', {
+ id: 'cube',
+ collision_id: 'cube',
+ col: V(.6,.6,.6),
+ scale: V(this.length,this.height / 2,this.width),
+ pos: V(0,this.height/4,0)
+ }, this);
+ this.button = room.createObject('Object', {
+ id: 'cube',
+ collision_id: 'cube',
+ col: this.col,
+ scale: V(this.length * .8, this.height / 4, this.width * .8),
+ }, this);
+ this.sounds = {
+ clickin: room.createObject('Sound', { id: 'pushbutton-click-in', }, this),
+ clickon: room.createObject('Sound', { id: 'pushbutton-click-on', }, this),
+ clickoff: room.createObject('Sound', { id: 'pushbutton-click-off', }, this),
+ };
+
+ this.button.addEventListener('mousedown', this.onMouseDown);
+ window.addEventListener('mouseup', this.onMouseUp);
+ window.addEventListener('touchend', this.onMouseUp);
+ this.button.addEventListener('click', this.onClick);
+
+ if (this.accesskey) {
+ room.onKeyDown = this.onKeyDown;
+ room.onKeyUp = this.onKeyUp;
+ }
+
+ this.setButtonPos();
+ },
+ press: function() {
+ this.executeCallback(this.onbuttonpress);
+ },
+ onClick: function(ev) {
+ if (this.onclick) {
+ this.executeCallback(this.onclick);
+ }
+ },
+ onMouseDown: function(ev) {
+ this.button.sync = true;
+ this.clicking = true;
+ this.activate();
+
+ this.setState('in');
+ this.executeCallback(this.onbuttondown);
+
+ if (ev) {
+ ev.stopPropagation();
+ }
+ },
+ onMouseUp: function(ev) {
+ if (this.clicking) {
+ this.setState('off');
+ this.button.sync = true;
+ this.clicking = false;
+ this.executeCallback(this.onbuttonup);
+
+ if (ev) {
+ ev.stopPropagation();
+ }
+ }
+ },
+ onKeyDown: function(ev) {
+ if (ev.keyCode == this.accesskey) {
+ this.onMouseDown(ev);
+ }
+ },
+ onKeyUp: function(ev) {
+ if (ev.keyCode == this.accesskey) {
+ this.onMouseUp(ev);
+ }
+ },
+ setActive: function(active) {
+ this.setState(active ? 'on' : 'off');
+ },
+ setButtonPos: function(pos) {
+ if (typeof pos == 'undefined') pos = (this.active ? .6 : 0);
+ var percent = Math.min(1, Math.max(0, pos));
+ var defaultpos = this.height / 2;
+ this.button.pos = V(0, defaultpos - ((this.height / 8) * percent) + (this.height / 16), 0);
+ },
+ setState: function(state) {
+ if (state != this.state) {
+ this.state = state;
+ if (state == 'in') {
+ this.setButtonPos(1);
+ this.sounds.clickin.play();
+ } else if (state == 'on') {
+ this.setButtonPos(.6);
+ this.sounds.clickon.play();
+ } else if (state == 'off') {
+ this.setButtonPos(0);
+ this.sounds.clickoff.play();
+ }
+ }
+ }
+
+});
+
+room.registerElement('slider', {
+ grabbing: false,
+ value: 1,
+ min: 0,
+ max: 1,
+ length: 1,
+ width: .25,
+ height: .2,
+
+ ongrabstart: new CustomEvent('grabstart'),
+ onchange: new CustomEvent('change'),
+ ongrabend: new CustomEvent('grabend'),
+
+ createChildren: function() {
+ this.base = room.createObject('Object', {
+ id: 'cube',
+ collision_id: 'cube',
+ col: V(.6,.6,.6),
+ scale: V(this.length,this.height/2,this.width),
+ pos: V(0,this.height/4,0)
+ }, this);
+ this.track = room.createObject('Object', {
+ id: 'cube',
+ collision_id: 'cube',
+ col: V(.02,.02,.02),
+ scale: V(this.length * .9, this.height / 4, this.width / 5),
+ pos: V(0, this.base.pos.y + this.height / 4,0)
+ }, this);
+ this.handle = room.createObject('Object', {
+ id: 'cube',
+ collision_id: 'cube',
+ col: this.col,
+ scale: V(this.width / 4, this.height / 4, this.width * .75),
+ pos: V(0,this.base.pos.y,0)
+ }, this);
+ this.sounds = {
+ clickin: room.createObject('Sound', { id: 'pushbutton-click-in', }, this),
+ clickon: room.createObject('Sound', { id: 'pushbutton-click-on', }, this),
+ clickoff: room.createObject('Sound', { id: 'pushbutton-click-off', }, this),
+ };
+
+ this.handle.addEventListener('mousedown', this.onMouseDown);
+ this.track.addEventListener('mousedown', this.onMouseDown);
+ // FIXME - bind these in mousedown, for maximum efficiency
+ window.addEventListener('mouseup', this.onMouseUp);
+ window.addEventListener('touchend', this.onMouseUp);
+ window.addEventListener('mousemove', this.onMouseMove);
+ window.addEventListener('touchmove', this.onMouseMove);
+ this.handle.addEventListener('click', this.onClick);
+
+ this.setValue(this.value);
+ },
+ setValue: function(value, skipchange) {
+ var realvalue = Math.min(this.max, Math.max(this.min, value));
+ var percent = realvalue / (this.max - this.min);
+ var pos = (percent - .5) * this.length * .9;
+ this.handle.pos = V(pos, this.base.pos.y + this.height * 3 / 8 , 0);
+ this.handle.sync = true;
+ this.value = realvalue;
+
+ if (this.onchange && !skipchange) {
+ this.executeCallback(this.onchange);
+ }
+ },
+ updateValueFromCursorPos: function() {
+ // FIXME - vector proxies are currently broken
+ var sliderpos = this.parent.localToWorld(this.pos._target.clone());
+ var dist = player.head_pos.distanceTo(sliderpos);
+
+ // Cast a ray from my head to the cursor position
+ var dir = V(player.cursor_pos.x - player.head_pos.x, player.cursor_pos.y - player.head_pos.y, player.cursor_pos.z - player.head_pos.z);
+
+ // Then project the ray into the same plane as the slider I'm manipulating
+ var pos = translate(player.head_pos, scalarMultiply(normalized(dir), dist));
+ this.worldToLocal(pos);
+
+ // The slider position is now determined based on the x position (left/right) in the slider's own coordinate system, and the length of the slider
+ var foo = pos.x / (this.length * .9) + .5;
+
+ this.setValue(foo * (this.max - this.min));
+ },
+ onMouseDown: function(ev) {
+ this.grabbing = true;
+ this.updateValueFromCursorPos();
+ this.sounds.clickin.play();
+ if (this.ongrabstart) this.executeCallback(this.ongrabstart);
+ ev.preventDefault();
+ ev.stopPropagation();
+ },
+ onMouseMove: function(ev) {
+ if (this.grabbing) {
+ this.updateValueFromCursorPos();
+ ev.stopPropagation();
+ }
+ },
+ onMouseUp: function(ev) {
+ if (this.grabbing) {
+ this.grabbing = false;
+ this.sounds.clickoff.play();
+ if (this.ongrabend) this.executeCallback(this.ongrabend);
+ ev.stopPropagation();
+ }
+ },
+ onClick: function(ev) {
+ }
+
+});
+
+room.registerElement('togglebutton', {
+ width: .5,
+ length: .5,
+ height: .5,
+ clicking: false,
+ active: false,
+
+ onbuttondown: new CustomEvent('buttondown'),
+ onbuttonpress: new CustomEvent('buttonpress'),
+ onbuttonup: new CustomEvent('buttonup'),
+ onactivate: new CustomEvent('activate'),
+ ondeactivate: new CustomEvent('deactivate'),
+
+ createChildren: function() {
+ this.base = room.createObject('Object', {
+ id: 'cube',
+ collision_id: 'cube',
+ col: V(.6,.6,.6),
+ scale: V(this.length,this.height / 2,this.width),
+ pos: V(0,this.height/4,0)
+ }, this);
+ this.button = room.createObject('Object', {
+ id: 'cube',
+ collision_id: 'cube',
+ col: this.col,
+ scale: V(this.length * .8, this.height / 4, this.width * .8),
+ }, this);
+ this.sounds = {
+ clickin: this.createObject('Sound', { id: 'pushbutton-click-in', }),
+ clickon: this.createObject('Sound', { id: 'pushbutton-click-on', }),
+ clickoff: this.createObject('Sound', { id: 'pushbutton-click-off', }),
+ };
+
+ this.button.addEventListener('mousedown', this.onMouseDown);
+ window.addEventListener('mouseup', this.onMouseUp);
+ window.addEventListener('touchend', this.onMouseUp);
+ this.button.addEventListener('click', this.onClick);
+
+ this.setButtonPos();
+ },
+ press: function() {
+ this.active = !this.active;
+
+ if (this.active && this.onactivate) {
+ this.executeCallback(this.onactivate);
+ } else if (!this.active && this.ondeactivate) {
+ this.executeCallback(this.ondeactivate);
+ }
+ },
+ onClick: function(ev) {
+ if (this.onclick) {
+ this.executeCallback(this.onclick);
+ }
+ },
+ onMouseDown: function(ev) {
+ this.button.sync = true;
+ this.clicking = true;
+ this.press();
+
+ this.setState('in');
+
+ ev.stopPropagation();
+ },
+ onMouseUp: function(ev) {
+ if (this.clicking) {
+ if (this.active) {
+ this.setState('on');
+ } else {
+ this.setState('off');
+ }
+ this.button.sync = true;
+ this.clicking = false;
+
+ ev.stopPropagation();
+ }
+ },
+ setActive: function(active) {
+ this.active = active;
+ this.setState(active ? 'on' : 'off');
+ },
+ setButtonPos: function(pos) {
+ if (typeof pos == 'undefined') pos = (this.active ? .6 : 0);
+ var percent = Math.min(1, Math.max(0, pos));
+ var defaultpos = this.height / 2;
+ this.button.pos = V(0, defaultpos - ((this.height / 8) * percent) + (this.height / 16), 0);
+ },
+ setState: function(state) {
+ if (state != this.state) {
+ this.state = state;
+ if (state == 'in') {
+ this.setButtonPos(1);
+ this.sounds.clickin.play();
+ } else if (state == 'on') {
+ this.setButtonPos(.6);
+ this.sounds.clickon.play();
+ } else if (state == 'off') {
+ this.setButtonPos(0);
+ this.sounds.clickoff.play();
+ }
+ }
+ }
+
+});
+
+/*
+ * ## JJQuery
+ * see https://codeberg.org/coderofsalvation/janus-script-jjq
+ */
+
+$ = function(sel,opts){
+ out = $$(sel,opts)[0]
+ return out || false
+}
+
+$$ = function(sel, opts){
+ opts = opts || {}
+ if( !opts.in ) opts.in =['tag','js_id','name','id']
+ out = []
+ db = {children: room.objects}
+ if( typeof sel !== 'string' && !(sel instanceof RegExp) ){
+ if( sel?.length ) db.children = sel
+ if( sel?.children ) db.children = sel.children
+ sel = /.*/
+ }
+ if( opts?.from ) db.children = opts.from
+ $$.traverse( function(child){
+ for( let prop in opts.in ){
+ found = child[ opts.in[prop] ]
+ if( found ){
+ let match = ( typeof sel == 'string' ? String(found).toLowerCase() == sel.toLowerCase()
+ : String(found).match(sel) )
+ if( match && typeof child != 'function' ){
+ out.push( child )
+ break;
+ }
+ }
+ }
+ },db)
+ return $$.decorate(out)
+}
+
+$$.decorate = function(obj){
+ for( let i in $$ ) obj[i] = $$[i].bind(obj)
+ return obj
+}
+
+$$.each = function(cb){
+ this.map( cb )
+ return this
+}
+
+$$.traverse = function(cb,me){
+ me = me || room
+ if( !me.children ) return
+ for( let i in me.children ){
+ proceed = cb( me.children[i] )
+ if( proceed === false ) return
+ $$.traverse(cb, me.children[i] )
+ }
+ return this
+}
+
+$$.get = function(k, fallback){
+ if( !k ){ // return first object
+ return this[0] || false
+ }
+ res = []
+ this.each( function(el){
+ res.push(
+ $.get( el, k )
+ )
+ })
+ //res = res.filter( v => v != undefined) // delete empties
+
+ return $$.decorate(res)
+}
+
+$.get = function(el,path){
+ return new Function('el',`return el.${path.replace(/\./,'?.')}`)(el)
+}
+
+$.set = function set(obj, path, value) {
+ var last
+ var o = obj
+ path = String(path)
+ var vars = path.split(".")
+ var lastVar = vars[vars.length - 1]
+ vars.map(function(v) {
+ if (lastVar == v) return
+ o = (new Function("o","return o." + v)(o) || new Function("o","return o."+v+" = {}")(o))
+ last = v
+ })
+ new Function("o","v","o." + lastVar + " = v")(o, value)
+}
+
+$$.set = function(k,v,opts){
+
+ this.each( function(el){ $.set( el, k, v ) })
+ return this
+}
+
+$$.wrap = function wrap(obj, method, handler, context) {
+ var org = obj[method];
+ // Unpatch first if already patched.
+ if (org.unwrap) {
+ org = org.unwrap();
+ }
+ // Patch the function.
+ obj[method] = function() {
+ var ctx = context || obj;
+ var args = [].slice.call(arguments);
+ args.unshift(org.bind(ctx));
+ return handler.apply(ctx, args);
+ };
+ // Provide "unpatch" function.
+ obj[method].unwrap = function() {
+ obj[method] = org;
+ return org;
+ };
+ // Return the original.
+ return org;
+}
+
+$$.dispatchEvent = function(event){
+ this.each( (o) => {
+ if( o.dispatchEvent ){
+ o.dispatchEvent(event)
+ }
+ })
+ return this
+}
+
+$$.addEventListener = function(eventname, cb){
+ this.each( (o) => {
+ if( o.addEventListener ){
+ o.addEventListener(eventname, cb)
+ }
+ })
+ return this
+}
+
+$$.when = function( condition ){
+ res = this.filter( condition )
+ return $$.decorate(res)
+}
+
+$$.then = function( cb ){
+ this.each( cb )
+ return this
+}
+
+$$.proxy = function(extra){
+ let newtargets = []
+ this.each( (el) => newtargets.push( $.proxy(extra,el) ) )
+ return $$.decorate( newtargets )
+}
+
+$.proxy = (extra,el) => new Proxy(el,{
+ get(me,k){ return me[k]; },
+ set(me,k,v){
+ if( extra[k] ){
+ return extra[k].apply(me,[() => me[k] = v])
+ }else me[k] = v
+ return true
+ },
+})
+
+if( typeof room != 'undefined' ) room.registerElement("$$",{}) // just to notify dependencies
+/* tiny (recursive) tween function for Janus objects with auto-garbage collection
+ *
+ * usage: const speed = 0.05 // closer to zero == smoother/slower
+ * logo.addEventListener("tween_done", alert )
+ * logo.addEventListener("tween_prop_done", alert )
+ * tween( logo, {"pos.y": [-0.1,speed], "pos.x": [-0.05,speed]})
+ * room.update = $.tween.update // call tween.update() every frame
+ *
+ * dependency: https://codeberg.org/coderofsalvation/janus-script-jjq
+ */
+
+mountTween = function(){
+
+ $.tween = function(el, opts){
+ $.tween.remove = (el) => {
+ delete el.tween
+ el.tween = undefined
+ $.tween.items = $.tween.items.filter( (el) => el.tween ? el : false )
+ }
+ // This is our Linear Interpolation method. It takes 3 parameters:
+ // a: The starting value
+ // b: The destination value
+ // n: The normal value (between 0 and 1) to control the Linear Interpolation
+ $.tween.lerp = function(a, b, n) {
+ return (1 - n) * a + n * b;
+ }
+ $.tween.update = () => {
+ $.tween.items.map( (el) => {
+ for( let prop in el.tween ){
+ let diff = 0
+ const to = el.tween[prop][0]
+ const speed = el.tween[prop][1]
+ let children = 0
+ const lerp = (child) => {
+ if( $.get(child,prop) == undefined ) return
+ const v = $.tween.lerp( $.get(child,prop), to, speed )
+ $.set(child, prop, v)
+ diff += Math.abs(to-v)
+ children++
+ }
+ lerp(el)
+ $$(el).each( lerp )
+ // garbage collect
+ if( diff < $.tween.items.resolution ){
+ if( el.dispatchEvent ){
+ el.dispatchEvent({type:"tween_prop_done",prop,tween:el.tween})
+ }
+ delete el.tween[prop]
+ }
+ }
+ if( Object.keys(el.tween).length == 0 ){
+ $.tween.remove(el)
+ if( el.dispatchEvent ){
+ setTimeout( () => {
+ el.dispatchEvent({type:"tween_done",tween:el.tween})
+ },50)
+ }
+ }
+ })
+ }
+ $.tween.items = $.tween.items || []
+ $.tween.items.resolution = 0.01 // decides when to flush tweens
+
+ if( el.tween ) $.tween.remove(el) // overwrite previous
+ el.tween = opts
+ $.tween.items.push(el)
+ }
+
+}
+
+// be direct or patient
+if( typeof $$ != 'undefined' ) mountTween()
+else{
+ room.addEventListener("registerelement", function(e){
+ if( e.data == "$$") mountTween()
+ })
+}
+function register(){
+
+ if( register.triggered ) return // ignore on-the-fly-loaded assetscripts
+ register.triggered = true
+
+ room.registerElement('button', {
+ use:"button",
+ lighting: "false",
+ text: "",
+ locked:true,
+ wireframe: false,
+ scale: "1 1 1",
+ scale_back: "1 1 1",
+ text_scale: 0.6,
+ font_scale: false,
+ font_size: 0.08,
+ col: "0 0 0",
+ col_text: "1 1 1",
+
+ createChildren: function(){
+ if( this.billboard ){
+ console.warn("billboarding on multiple transparent dialogs will tank the framerate :/")
+ }
+ this.sounds = {
+ click: room.createObject('Sound', { id: 'button-click', }, this),
+ hover: room.createObject('Sound', { id: 'button-hover', }, this),
+ };
+ if( this.text ){
+ this.label = this.createObject('text',{
+ xdir: "-1 0 -0.000002",
+ locked:true,
+ pos: "0 -0.022 0.01",
+ scale: `${this.scale.x*this.text_scale} ${this.scale.y*this.text_scale} 0.0001`, // dereference
+ font_scale: this.font_scale,
+ font_size: this.font_size,
+ text: this.text,
+ col: `{this.col_text.r} ${this.col_text.g} ${this.col_text.b}` // dereference
+ })
+ }
+ this.button = this.createObject('object',{
+ id: this.use,
+ collision_id: 'button',
+ wireframe: this.wireframe,
+ col: `{this.col.r} ${this.col.g} ${this.col.b}`, // dereference
+ locked:true,
+ lighting: this.lighting ? "true" : "false",
+ scale: this.scale_back //`${this.scale_back.x} ${this.scale_back.y} ${this.scale_back.z}`
+ })
+
+ const onClick = () => {
+ this.sounds.click.play()
+ setTimeout( () => { this.button.visible = false }, 1 ) // wait for theme-change
+ setTimeout( () => { this.button.visible = true; }, 70 )
+ room.dispatchEvent({type:"dialog_button_click", button:this})
+ }
+ const onHover = () => {
+ this.sounds.hover.play()
+ room.dispatchEvent({type:"dialog_button_hover", button:this})
+ }
+ this.button.addEventListener('activate', onClick)
+ this.button.addEventListener('click', onClick)
+ this.button.addEventListener('gazeenter', onHover)
+ this.button.addEventListener('mouseover', onHover)
+ }
+
+ })
+
+ room.registerElement('dialog', {
+ "content_id":"",
+ "backportal": false,
+ "debug": false,
+ "theme_default": "light",
+ "reroute": true, // reroute window.alert e.g.
+ "billboard": false,
+ watch: "",
+
+ theme:{
+ "current":"dark",
+ "dark":{ /* generated from "light" via lightToDarkTheme() */ },
+ "light":{
+ css: (opts) => `
+ .paragraphcontainer{
+ max-width:unset;
+ min-height:unset;
+ text-align: center;
+ }
+ .br { height: 1em; }
+ .hr { margin: .5em 0; border: 1px inset #ccc; height: 0px; }
+ .paragraphcontent{
+ overflow:hidden;
+ text-align:left;
+ height: ${opts.height} !important; /* important for crossbrowser compatibility */
+ max-height: ${opts.height} !important; /* important for crossbrowser compatibility */
+ width: ${opts.width} !important; /* important for crossbrowser compatibility */
+ max-width: ${opts.width} !important; /* important for crossbrowser compatibility */
+ background:#FFFFFFCC;
+ border-radius:10px;
+ padding:7px 15px;
+ display:inline-block;
+ }
+ blockquote {
+ margin-left: 0;
+ padding: 5px 15px;
+ border-left: 4px solid #000;
+ }
+ pre {
+ font-face: monospace;
+ padding:15px;
+ margin:0;
+ background:#00000044;
+ border-radius:7px;
+ display:inline-block;
+ }
+ `,
+ jml: {
+ "*": {
+ },
+ object: {
+ back_alpha: 0
+ },
+ paragraph: {
+ scale: "2 2 2",
+ back_alpha: "0",
+ lighting: "false"
+ }
+ },
+ custom: (opts) => {
+ $$('text', opts.db).set('col', opts.invert ? '1 1 1' : '0 0 0')
+ $$('paragraph', opts.db).set('col', opts.invert ? '1 1 1' : '0 0 0')
+ $$('object', opts.db).set('col', opts.invert ? '0 0 0' : '1 1 1')
+ $$('object', opts.db).set('lighting', 'false')
+ $$('object', opts.db).set('transparent','false')
+ }
+ }
+ },
+
+ //onbuttondown: new CustomEvent('buttondown'),
+
+ getTheme: function(){
+ if( !room.theme ){ // setup proxy so we can listen to global themechanges
+ room.theme = new Proxy( this.theme,{
+ get(me,k){ return me[k] },
+ set(me,k,v){
+ me[k] = v
+ if( k == 'current' ){
+ $$('dialog').each( (dialog) => dialog.render() )
+ }
+ return true
+ }
+ })
+ room.dispatchEvent({type:"dialog_theme_init"})
+ }
+ return room.theme[ room.theme.current ]
+ },
+
+ createChildren: function() {
+ this.lightToDarkTheme() // generate extra dark theme
+ // attach theme to room.theme
+ this._theme = this.getTheme()
+ // set default theme
+ room.theme.current = this.theme_default
+
+ // adjust for billboard
+ this.xdir = this.billboard ? "1 0 0" : "-1 0 -0.000002"
+ this.zdir = this.billboard ? "0 0 1" : "0.000002 0 -1"
+
+ this.getContent()
+ this.setupWatchers()
+ this.render()
+ this.rerouteWindow()
+ this.optimizeAR()
+
+ // fix janus.navigateBack() janus.navigateForward() userposition
+ room.addEventListener('room_active', () => {
+ this.optimizeAR(true)
+ })
+ },
+
+ removeContent: function(js_id){
+ let content = this.children[0]
+ if( content ){
+ this.removeChild(content)
+ room.objects[ content.js_id ] = content
+ room.appendChild(content)
+ room.dispatchEvent({type:"dialog_content_remove", content, dialog:this})
+ }
+ },
+
+ getContent: function(js_id){
+ if( js_id ){
+ this.removeContent()
+ this.content_id = js_id
+ }
+ let content = this.children[0] || $(this.content_id)
+ if( !content ) return
+ content.visible = true
+ content.pos = "0 0 0"
+ content.parentBackup = content.parent
+ content.parent.removeChild(content)
+ this.appendChild( content )
+ },
+
+ render: function(js_id){
+ if( this.rendering ) return // prevent recursion
+ this.rendering = true
+ this._theme = this.getTheme()
+ room.dispatchEvent({type:"dialog_content_render", dialog:this})
+ try{
+ if( js_id ) this.getContent(js_id)
+ room.dispatchEvent({type:"dialog_theme_apply", dialog:this})
+ this.processParagraphs()
+ $$(this.children).set('collidable', this.collidable ? "true" : "false")
+ $$(this.children).set('pickable', this.collidable ? "true" : "false")
+ this.applyTheme()
+ }catch(e){
+ console.error(e)
+ }
+ this.rendering = false
+ },
+
+ applyTheme: function(){
+ let theme = this._theme = this.getTheme()
+ $$( this.children )
+ .traverse( (child) => {
+ if( !child.tag ) return
+ let tag = String(child.tag).toLowerCase()
+ for( let i in theme.jml['*'] ){ // default
+ child[i] = theme.jml['*'][i]
+ if( this.debug ) console.log(`child.${tag} = theme.jml['*'][${i}] = ${theme.jml['*'][i]}`)
+ }
+ if( theme.jml[tag] ){ // tag-specific
+ for( let i in theme.jml[tag] ){
+ if( this.debug ) console.log(`child.${tag} = theme.jml[${tag}][${i}]`)
+ child[i] = theme.jml[tag][i]
+ }
+ }
+ })
+
+ if( !this.children[0]?.children ) return
+ const db = {in:['tag'], from: this.children[0].children }
+ theme.custom({db}) // apply only to children of dialog
+ // invert colors of objects which have
+ $$(/.*/, db).each( (el) => {
+ if( el.invert || el?.args?.properties?.invert ){
+ el.col.r = 1.0 - el.col.r
+ el.col.g = 1.0 - el.col.g
+ el.col.b = 1.0 - el.col.b
+ }
+ })
+ return this
+ },
+
+ processParagraphs(){
+ // get all elements
+ $$('paragraph', {in:['tag'], from:this.children})
+ .each( el => {
+ // *FIXME* is el.args a janusweb thing?
+ const height = el.args.properties.height || '200px' // annoying default to urge dev to set it explicitly
+ const width = el.args.properties.width || '300px' // annoying default to urge dev to set it explicitly
+ const cssClass = el.cssClass
+ let tag = el.tag
+ el.textBackup = el.textBackup || el.text
+ let text = el.textBackup // reset to original template
+ text = this.processTemplateVars(text)
+ text = this.processMarkdown(text)
+ text = this.wrapCSS(text, this._theme.css({height,width}), cssClass )
+ el.text = text
+ room.dispatchEvent({type:"dialog_paragraph_render", paragraph: el, dialog:this})
+ if( this.debug ) console.log(el.text)
+ })
+ return this
+ },
+
+ processMarkdown: function(text){
+ return text
+ .replace(/(\n|^|\s+)#(?!#) *(.*)/g, "