Compare commits
No commits in common. "main" and "dev" have entirely different histories.
25
.github/workflows/main.yml
vendored
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
name: main
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [ main ]
|
||||||
|
schedule:
|
||||||
|
- cron: 0 0 1 * *
|
||||||
|
|
||||||
|
env:
|
||||||
|
CARGO_TERM_COLOR: always
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
cargo-test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout source
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- uses: Swatinem/rust-cache@v1
|
||||||
|
|
||||||
|
- name: cargo test
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: test
|
||||||
|
args: --workspace --all-features
|
35
.github/workflows/pr.yml
vendored
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
name: PR
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches: [ main ]
|
||||||
|
|
||||||
|
env:
|
||||||
|
CARGO_TERM_COLOR: always
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
cargo-test-and-lint:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout source
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- uses: Swatinem/rust-cache@v1
|
||||||
|
|
||||||
|
- name: cargo test
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: test
|
||||||
|
args: --workspace --all-features
|
||||||
|
|
||||||
|
- name: rustfmt
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: fmt
|
||||||
|
args: --all -- --check
|
||||||
|
|
||||||
|
- name: clippy
|
||||||
|
uses: actions-rs/clippy-check@v1
|
||||||
|
with:
|
||||||
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
args: --all --all-features --tests -- -D warnings
|
2326
Cargo.lock
generated
Normal file
52
Cargo.toml
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
[package]
|
||||||
|
name = "yewprint-app"
|
||||||
|
version = "0.1.0"
|
||||||
|
authors = ["Gabriel Tofvesson <contact@w1zzrd.dev>"]
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
crate-type = ["cdylib"]
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
# The `console_error_panic_hook` crate provides better debugging of panics by
|
||||||
|
# logging them with `console.error`. This is great for development, but requires
|
||||||
|
# all the `std::fmt` and `std::panicking` infrastructure, so isn't great for
|
||||||
|
# code size when deploying.
|
||||||
|
console_error_panic_hook = { version = "0.1.6", optional = true }
|
||||||
|
|
||||||
|
wasm-bindgen = "0.2"
|
||||||
|
web-sys = { version = "0.3", features = ["Window", "MediaQueryList", "DomTokenList"] }
|
||||||
|
js-sys = "0.3"
|
||||||
|
|
||||||
|
# `wee_alloc` is a tiny allocator for wasm that is only ~1K in code size
|
||||||
|
# compared to the default allocator's ~10K. It is slower than the default
|
||||||
|
# allocator, however.
|
||||||
|
#
|
||||||
|
# Unfortunately, `wee_alloc` requires nightly Rust when targeting wasm for now.
|
||||||
|
wee_alloc = { version = "0.4.5", optional = false }
|
||||||
|
|
||||||
|
yew = { version = "0.20.0", features = ["csr"] }
|
||||||
|
yew-router = { version = "0.17.0" }
|
||||||
|
#yewprint = "0.4.0"
|
||||||
|
gloo = "0.8.0"
|
||||||
|
gloo-events = "0.1"
|
||||||
|
futures = "0.3.25"
|
||||||
|
|
||||||
|
# Resource access
|
||||||
|
gloo-net = "0.2"
|
||||||
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
|
wasm-bindgen-futures = "0.4"
|
||||||
|
|
||||||
|
comrak = "0.15"
|
||||||
|
|
||||||
|
[profile.release]
|
||||||
|
# Tell `rustc` to optimize for small code size.
|
||||||
|
# opt-level = "s"
|
||||||
|
lto = true
|
||||||
|
panic = "abort"
|
||||||
|
opt-level = "z"
|
||||||
|
|
||||||
|
[workspace]
|
||||||
|
members = ["xtask"]
|
201
LICENSE.Apache-2.0
Normal file
@ -0,0 +1,201 @@
|
|||||||
|
Apache License
|
||||||
|
Version 2.0, January 2004
|
||||||
|
http://www.apache.org/licenses/
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||||
|
|
||||||
|
1. Definitions.
|
||||||
|
|
||||||
|
"License" shall mean the terms and conditions for use, reproduction,
|
||||||
|
and distribution as defined by Sections 1 through 9 of this document.
|
||||||
|
|
||||||
|
"Licensor" shall mean the copyright owner or entity authorized by
|
||||||
|
the copyright owner that is granting the License.
|
||||||
|
|
||||||
|
"Legal Entity" shall mean the union of the acting entity and all
|
||||||
|
other entities that control, are controlled by, or are under common
|
||||||
|
control with that entity. For the purposes of this definition,
|
||||||
|
"control" means (i) the power, direct or indirect, to cause the
|
||||||
|
direction or management of such entity, whether by contract or
|
||||||
|
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
|
||||||
|
"You" (or "Your") shall mean an individual or Legal Entity
|
||||||
|
exercising permissions granted by this License.
|
||||||
|
|
||||||
|
"Source" form shall mean the preferred form for making modifications,
|
||||||
|
including but not limited to software source code, documentation
|
||||||
|
source, and configuration files.
|
||||||
|
|
||||||
|
"Object" form shall mean any form resulting from mechanical
|
||||||
|
transformation or translation of a Source form, including but
|
||||||
|
not limited to compiled object code, generated documentation,
|
||||||
|
and conversions to other media types.
|
||||||
|
|
||||||
|
"Work" shall mean the work of authorship, whether in Source or
|
||||||
|
Object form, made available under the License, as indicated by a
|
||||||
|
copyright notice that is included in or attached to the work
|
||||||
|
(an example is provided in the Appendix below).
|
||||||
|
|
||||||
|
"Derivative Works" shall mean any work, whether in Source or Object
|
||||||
|
form, that is based on (or derived from) the Work and for which the
|
||||||
|
editorial revisions, annotations, elaborations, or other modifications
|
||||||
|
represent, as a whole, an original work of authorship. For the purposes
|
||||||
|
of this License, Derivative Works shall not include works that remain
|
||||||
|
separable from, or merely link (or bind by name) to the interfaces of,
|
||||||
|
the Work and Derivative Works thereof.
|
||||||
|
|
||||||
|
"Contribution" shall mean any work of authorship, including
|
||||||
|
the original version of the Work and any modifications or additions
|
||||||
|
to that Work or Derivative Works thereof, that is intentionally
|
||||||
|
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||||
|
or by an individual or Legal Entity authorized to submit on behalf of
|
||||||
|
the copyright owner. For the purposes of this definition, "submitted"
|
||||||
|
means any form of electronic, verbal, or written communication sent
|
||||||
|
to the Licensor or its representatives, including but not limited to
|
||||||
|
communication on electronic mailing lists, source code control systems,
|
||||||
|
and issue tracking systems that are managed by, or on behalf of, the
|
||||||
|
Licensor for the purpose of discussing and improving the Work, but
|
||||||
|
excluding communication that is conspicuously marked or otherwise
|
||||||
|
designated in writing by the copyright owner as "Not a Contribution."
|
||||||
|
|
||||||
|
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||||
|
on behalf of whom a Contribution has been received by Licensor and
|
||||||
|
subsequently incorporated within the Work.
|
||||||
|
|
||||||
|
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
copyright license to reproduce, prepare Derivative Works of,
|
||||||
|
publicly display, publicly perform, sublicense, and distribute the
|
||||||
|
Work and such Derivative Works in Source or Object form.
|
||||||
|
|
||||||
|
3. Grant of Patent License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
(except as stated in this section) patent license to make, have made,
|
||||||
|
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||||
|
where such license applies only to those patent claims licensable
|
||||||
|
by such Contributor that are necessarily infringed by their
|
||||||
|
Contribution(s) alone or by combination of their Contribution(s)
|
||||||
|
with the Work to which such Contribution(s) was submitted. If You
|
||||||
|
institute patent litigation against any entity (including a
|
||||||
|
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||||
|
or a Contribution incorporated within the Work constitutes direct
|
||||||
|
or contributory patent infringement, then any patent licenses
|
||||||
|
granted to You under this License for that Work shall terminate
|
||||||
|
as of the date such litigation is filed.
|
||||||
|
|
||||||
|
4. Redistribution. You may reproduce and distribute copies of the
|
||||||
|
Work or Derivative Works thereof in any medium, with or without
|
||||||
|
modifications, and in Source or Object form, provided that You
|
||||||
|
meet the following conditions:
|
||||||
|
|
||||||
|
(a) You must give any other recipients of the Work or
|
||||||
|
Derivative Works a copy of this License; and
|
||||||
|
|
||||||
|
(b) You must cause any modified files to carry prominent notices
|
||||||
|
stating that You changed the files; and
|
||||||
|
|
||||||
|
(c) You must retain, in the Source form of any Derivative Works
|
||||||
|
that You distribute, all copyright, patent, trademark, and
|
||||||
|
attribution notices from the Source form of the Work,
|
||||||
|
excluding those notices that do not pertain to any part of
|
||||||
|
the Derivative Works; and
|
||||||
|
|
||||||
|
(d) If the Work includes a "NOTICE" text file as part of its
|
||||||
|
distribution, then any Derivative Works that You distribute must
|
||||||
|
include a readable copy of the attribution notices contained
|
||||||
|
within such NOTICE file, excluding those notices that do not
|
||||||
|
pertain to any part of the Derivative Works, in at least one
|
||||||
|
of the following places: within a NOTICE text file distributed
|
||||||
|
as part of the Derivative Works; within the Source form or
|
||||||
|
documentation, if provided along with the Derivative Works; or,
|
||||||
|
within a display generated by the Derivative Works, if and
|
||||||
|
wherever such third-party notices normally appear. The contents
|
||||||
|
of the NOTICE file are for informational purposes only and
|
||||||
|
do not modify the License. You may add Your own attribution
|
||||||
|
notices within Derivative Works that You distribute, alongside
|
||||||
|
or as an addendum to the NOTICE text from the Work, provided
|
||||||
|
that such additional attribution notices cannot be construed
|
||||||
|
as modifying the License.
|
||||||
|
|
||||||
|
You may add Your own copyright statement to Your modifications and
|
||||||
|
may provide additional or different license terms and conditions
|
||||||
|
for use, reproduction, or distribution of Your modifications, or
|
||||||
|
for any such Derivative Works as a whole, provided Your use,
|
||||||
|
reproduction, and distribution of the Work otherwise complies with
|
||||||
|
the conditions stated in this License.
|
||||||
|
|
||||||
|
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||||
|
any Contribution intentionally submitted for inclusion in the Work
|
||||||
|
by You to the Licensor shall be under the terms and conditions of
|
||||||
|
this License, without any additional terms or conditions.
|
||||||
|
Notwithstanding the above, nothing herein shall supersede or modify
|
||||||
|
the terms of any separate license agreement you may have executed
|
||||||
|
with Licensor regarding such Contributions.
|
||||||
|
|
||||||
|
6. Trademarks. This License does not grant permission to use the trade
|
||||||
|
names, trademarks, service marks, or product names of the Licensor,
|
||||||
|
except as required for reasonable and customary use in describing the
|
||||||
|
origin of the Work and reproducing the content of the NOTICE file.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||||
|
agreed to in writing, Licensor provides the Work (and each
|
||||||
|
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
implied, including, without limitation, any warranties or conditions
|
||||||
|
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||||
|
appropriateness of using or redistributing the Work and assume any
|
||||||
|
risks associated with Your exercise of permissions under this License.
|
||||||
|
|
||||||
|
8. Limitation of Liability. In no event and under no legal theory,
|
||||||
|
whether in tort (including negligence), contract, or otherwise,
|
||||||
|
unless required by applicable law (such as deliberate and grossly
|
||||||
|
negligent acts) or agreed to in writing, shall any Contributor be
|
||||||
|
liable to You for damages, including any direct, indirect, special,
|
||||||
|
incidental, or consequential damages of any character arising as a
|
||||||
|
result of this License or out of the use or inability to use the
|
||||||
|
Work (including but not limited to damages for loss of goodwill,
|
||||||
|
work stoppage, computer failure or malfunction, or any and all
|
||||||
|
other commercial damages or losses), even if such Contributor
|
||||||
|
has been advised of the possibility of such damages.
|
||||||
|
|
||||||
|
9. Accepting Warranty or Additional Liability. While redistributing
|
||||||
|
the Work or Derivative Works thereof, You may choose to offer,
|
||||||
|
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||||
|
or other liability obligations and/or rights consistent with this
|
||||||
|
License. However, in accepting such obligations, You may act only
|
||||||
|
on Your own behalf and on Your sole responsibility, not on behalf
|
||||||
|
of any other Contributor, and only if You agree to indemnify,
|
||||||
|
defend, and hold each Contributor harmless for any liability
|
||||||
|
incurred by, or claims asserted against, such Contributor by reason
|
||||||
|
of your accepting any such warranty or additional liability.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
APPENDIX: How to apply the Apache License to your work.
|
||||||
|
|
||||||
|
To apply the Apache License to your work, attach the following
|
||||||
|
boilerplate notice, with the fields enclosed by brackets "[]"
|
||||||
|
replaced with your own identifying information. (Don't include
|
||||||
|
the brackets!) The text should be enclosed in the appropriate
|
||||||
|
comment syntax for the file format. We also recommend that a
|
||||||
|
file or class name and description of purpose be included on the
|
||||||
|
same "printed page" as the copyright notice for easier
|
||||||
|
identification within third-party archives.
|
||||||
|
|
||||||
|
Copyright 2023 Gabriel Tofvesson <contact@w1zzrd.dev>
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
21
LICENSE.MIT
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2023 Gabriel Tofvesson <contact@w1zzrd.dev>
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
15
README.md
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
# yewprint-app
|
||||||
|
|
||||||
|
## Development Server
|
||||||
|
|
||||||
|
```
|
||||||
|
cargo xtask start
|
||||||
|
```
|
||||||
|
|
||||||
|
You can now go to http://localhost:8000
|
||||||
|
|
||||||
|
## Production Build
|
||||||
|
|
||||||
|
```
|
||||||
|
cargo xtask dist
|
||||||
|
```
|
941
app.js
@ -1,941 +0,0 @@
|
|||||||
|
|
||||||
let wasm;
|
|
||||||
|
|
||||||
const heap = new Array(32).fill(undefined);
|
|
||||||
|
|
||||||
heap.push(undefined, null, true, false);
|
|
||||||
|
|
||||||
function getObject(idx) { return heap[idx]; }
|
|
||||||
|
|
||||||
let heap_next = heap.length;
|
|
||||||
|
|
||||||
function dropObject(idx) {
|
|
||||||
if (idx < 36) return;
|
|
||||||
heap[idx] = heap_next;
|
|
||||||
heap_next = idx;
|
|
||||||
}
|
|
||||||
|
|
||||||
function takeObject(idx) {
|
|
||||||
const ret = getObject(idx);
|
|
||||||
dropObject(idx);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
function addHeapObject(obj) {
|
|
||||||
if (heap_next === heap.length) heap.push(heap.length + 1);
|
|
||||||
const idx = heap_next;
|
|
||||||
heap_next = heap[idx];
|
|
||||||
|
|
||||||
heap[idx] = obj;
|
|
||||||
return idx;
|
|
||||||
}
|
|
||||||
|
|
||||||
let WASM_VECTOR_LEN = 0;
|
|
||||||
|
|
||||||
let cachedUint8Memory0 = new Uint8Array();
|
|
||||||
|
|
||||||
function getUint8Memory0() {
|
|
||||||
if (cachedUint8Memory0.byteLength === 0) {
|
|
||||||
cachedUint8Memory0 = new Uint8Array(wasm.memory.buffer);
|
|
||||||
}
|
|
||||||
return cachedUint8Memory0;
|
|
||||||
}
|
|
||||||
|
|
||||||
const cachedTextEncoder = new TextEncoder('utf-8');
|
|
||||||
|
|
||||||
const encodeString = (typeof cachedTextEncoder.encodeInto === 'function'
|
|
||||||
? function (arg, view) {
|
|
||||||
return cachedTextEncoder.encodeInto(arg, view);
|
|
||||||
}
|
|
||||||
: function (arg, view) {
|
|
||||||
const buf = cachedTextEncoder.encode(arg);
|
|
||||||
view.set(buf);
|
|
||||||
return {
|
|
||||||
read: arg.length,
|
|
||||||
written: buf.length
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
function passStringToWasm0(arg, malloc, realloc) {
|
|
||||||
|
|
||||||
if (realloc === undefined) {
|
|
||||||
const buf = cachedTextEncoder.encode(arg);
|
|
||||||
const ptr = malloc(buf.length);
|
|
||||||
getUint8Memory0().subarray(ptr, ptr + buf.length).set(buf);
|
|
||||||
WASM_VECTOR_LEN = buf.length;
|
|
||||||
return ptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
let len = arg.length;
|
|
||||||
let ptr = malloc(len);
|
|
||||||
|
|
||||||
const mem = getUint8Memory0();
|
|
||||||
|
|
||||||
let offset = 0;
|
|
||||||
|
|
||||||
for (; offset < len; offset++) {
|
|
||||||
const code = arg.charCodeAt(offset);
|
|
||||||
if (code > 0x7F) break;
|
|
||||||
mem[ptr + offset] = code;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (offset !== len) {
|
|
||||||
if (offset !== 0) {
|
|
||||||
arg = arg.slice(offset);
|
|
||||||
}
|
|
||||||
ptr = realloc(ptr, len, len = offset + arg.length * 3);
|
|
||||||
const view = getUint8Memory0().subarray(ptr + offset, ptr + len);
|
|
||||||
const ret = encodeString(arg, view);
|
|
||||||
|
|
||||||
offset += ret.written;
|
|
||||||
}
|
|
||||||
|
|
||||||
WASM_VECTOR_LEN = offset;
|
|
||||||
return ptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
function isLikeNone(x) {
|
|
||||||
return x === undefined || x === null;
|
|
||||||
}
|
|
||||||
|
|
||||||
let cachedInt32Memory0 = new Int32Array();
|
|
||||||
|
|
||||||
function getInt32Memory0() {
|
|
||||||
if (cachedInt32Memory0.byteLength === 0) {
|
|
||||||
cachedInt32Memory0 = new Int32Array(wasm.memory.buffer);
|
|
||||||
}
|
|
||||||
return cachedInt32Memory0;
|
|
||||||
}
|
|
||||||
|
|
||||||
const cachedTextDecoder = new TextDecoder('utf-8', { ignoreBOM: true, fatal: true });
|
|
||||||
|
|
||||||
cachedTextDecoder.decode();
|
|
||||||
|
|
||||||
function getStringFromWasm0(ptr, len) {
|
|
||||||
return cachedTextDecoder.decode(getUint8Memory0().subarray(ptr, ptr + len));
|
|
||||||
}
|
|
||||||
|
|
||||||
let cachedFloat64Memory0 = new Float64Array();
|
|
||||||
|
|
||||||
function getFloat64Memory0() {
|
|
||||||
if (cachedFloat64Memory0.byteLength === 0) {
|
|
||||||
cachedFloat64Memory0 = new Float64Array(wasm.memory.buffer);
|
|
||||||
}
|
|
||||||
return cachedFloat64Memory0;
|
|
||||||
}
|
|
||||||
|
|
||||||
function debugString(val) {
|
|
||||||
// primitive types
|
|
||||||
const type = typeof val;
|
|
||||||
if (type == 'number' || type == 'boolean' || val == null) {
|
|
||||||
return `${val}`;
|
|
||||||
}
|
|
||||||
if (type == 'string') {
|
|
||||||
return `"${val}"`;
|
|
||||||
}
|
|
||||||
if (type == 'symbol') {
|
|
||||||
const description = val.description;
|
|
||||||
if (description == null) {
|
|
||||||
return 'Symbol';
|
|
||||||
} else {
|
|
||||||
return `Symbol(${description})`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (type == 'function') {
|
|
||||||
const name = val.name;
|
|
||||||
if (typeof name == 'string' && name.length > 0) {
|
|
||||||
return `Function(${name})`;
|
|
||||||
} else {
|
|
||||||
return 'Function';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// objects
|
|
||||||
if (Array.isArray(val)) {
|
|
||||||
const length = val.length;
|
|
||||||
let debug = '[';
|
|
||||||
if (length > 0) {
|
|
||||||
debug += debugString(val[0]);
|
|
||||||
}
|
|
||||||
for(let i = 1; i < length; i++) {
|
|
||||||
debug += ', ' + debugString(val[i]);
|
|
||||||
}
|
|
||||||
debug += ']';
|
|
||||||
return debug;
|
|
||||||
}
|
|
||||||
// Test for built-in
|
|
||||||
const builtInMatches = /\[object ([^\]]+)\]/.exec(toString.call(val));
|
|
||||||
let className;
|
|
||||||
if (builtInMatches.length > 1) {
|
|
||||||
className = builtInMatches[1];
|
|
||||||
} else {
|
|
||||||
// Failed to match the standard '[object ClassName]'
|
|
||||||
return toString.call(val);
|
|
||||||
}
|
|
||||||
if (className == 'Object') {
|
|
||||||
// we're a user defined class or Object
|
|
||||||
// JSON.stringify avoids problems with cycles, and is generally much
|
|
||||||
// easier than looping through ownProperties of `val`.
|
|
||||||
try {
|
|
||||||
return 'Object(' + JSON.stringify(val) + ')';
|
|
||||||
} catch (_) {
|
|
||||||
return 'Object';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// errors
|
|
||||||
if (val instanceof Error) {
|
|
||||||
return `${val.name}: ${val.message}\n${val.stack}`;
|
|
||||||
}
|
|
||||||
// TODO we could test for more things here, like `Set`s and `Map`s.
|
|
||||||
return className;
|
|
||||||
}
|
|
||||||
|
|
||||||
function makeMutClosure(arg0, arg1, dtor, f) {
|
|
||||||
const state = { a: arg0, b: arg1, cnt: 1, dtor };
|
|
||||||
const real = (...args) => {
|
|
||||||
// First up with a closure we increment the internal reference
|
|
||||||
// count. This ensures that the Rust closure environment won't
|
|
||||||
// be deallocated while we're invoking it.
|
|
||||||
state.cnt++;
|
|
||||||
const a = state.a;
|
|
||||||
state.a = 0;
|
|
||||||
try {
|
|
||||||
return f(a, state.b, ...args);
|
|
||||||
} finally {
|
|
||||||
if (--state.cnt === 0) {
|
|
||||||
wasm.__wbindgen_export_2.get(state.dtor)(a, state.b);
|
|
||||||
|
|
||||||
} else {
|
|
||||||
state.a = a;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
real.original = state;
|
|
||||||
|
|
||||||
return real;
|
|
||||||
}
|
|
||||||
|
|
||||||
let stack_pointer = 32;
|
|
||||||
|
|
||||||
function addBorrowedObject(obj) {
|
|
||||||
if (stack_pointer == 1) throw new Error('out of js stack');
|
|
||||||
heap[--stack_pointer] = obj;
|
|
||||||
return stack_pointer;
|
|
||||||
}
|
|
||||||
function __wbg_adapter_32(arg0, arg1, arg2) {
|
|
||||||
try {
|
|
||||||
wasm._dyn_core__ops__function__FnMut___A____Output___R_as_wasm_bindgen__closure__WasmClosure___describe__invoke__hd9fcefc6bb5eb4be(arg0, arg1, addBorrowedObject(arg2));
|
|
||||||
} finally {
|
|
||||||
heap[stack_pointer++] = undefined;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function __wbg_adapter_35(arg0, arg1, arg2) {
|
|
||||||
wasm._dyn_core__ops__function__FnMut__A____Output___R_as_wasm_bindgen__closure__WasmClosure___describe__invoke__hecf8e7fbcf7e70f4(arg0, arg1, addHeapObject(arg2));
|
|
||||||
}
|
|
||||||
|
|
||||||
let cachedUint32Memory0 = new Uint32Array();
|
|
||||||
|
|
||||||
function getUint32Memory0() {
|
|
||||||
if (cachedUint32Memory0.byteLength === 0) {
|
|
||||||
cachedUint32Memory0 = new Uint32Array(wasm.memory.buffer);
|
|
||||||
}
|
|
||||||
return cachedUint32Memory0;
|
|
||||||
}
|
|
||||||
|
|
||||||
function getArrayJsValueFromWasm0(ptr, len) {
|
|
||||||
const mem = getUint32Memory0();
|
|
||||||
const slice = mem.subarray(ptr / 4, ptr / 4 + len);
|
|
||||||
const result = [];
|
|
||||||
for (let i = 0; i < slice.length; i++) {
|
|
||||||
result.push(takeObject(slice[i]));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
function handleError(f, args) {
|
|
||||||
try {
|
|
||||||
return f.apply(this, args);
|
|
||||||
} catch (e) {
|
|
||||||
wasm.__wbindgen_exn_store(addHeapObject(e));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/**
|
|
||||||
*/
|
|
||||||
export function run_app() {
|
|
||||||
wasm.run_app();
|
|
||||||
}
|
|
||||||
|
|
||||||
async function load(module, imports) {
|
|
||||||
if (typeof Response === 'function' && module instanceof Response) {
|
|
||||||
if (typeof WebAssembly.instantiateStreaming === 'function') {
|
|
||||||
try {
|
|
||||||
return await WebAssembly.instantiateStreaming(module, imports);
|
|
||||||
|
|
||||||
} catch (e) {
|
|
||||||
if (module.headers.get('Content-Type') != 'application/wasm') {
|
|
||||||
console.warn("`WebAssembly.instantiateStreaming` failed because your server does not serve wasm with `application/wasm` MIME type. Falling back to `WebAssembly.instantiate` which is slower. Original error:\n", e);
|
|
||||||
|
|
||||||
} else {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const bytes = await module.arrayBuffer();
|
|
||||||
return await WebAssembly.instantiate(bytes, imports);
|
|
||||||
|
|
||||||
} else {
|
|
||||||
const instance = await WebAssembly.instantiate(module, imports);
|
|
||||||
|
|
||||||
if (instance instanceof WebAssembly.Instance) {
|
|
||||||
return { instance, module };
|
|
||||||
|
|
||||||
} else {
|
|
||||||
return instance;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function getImports() {
|
|
||||||
const imports = {};
|
|
||||||
imports.wbg = {};
|
|
||||||
imports.wbg.__wbg_location_8cc8ccf27e342c0a = function(arg0) {
|
|
||||||
const ret = getObject(arg0).location;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_history_2a104346a1208269 = function() { return handleError(function (arg0) {
|
|
||||||
const ret = getObject(arg0).history;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_querySelector_3628dc2c3319e7e0 = function() { return handleError(function (arg0, arg1, arg2) {
|
|
||||||
const ret = getObject(arg0).querySelector(getStringFromWasm0(arg1, arg2));
|
|
||||||
return isLikeNone(ret) ? 0 : addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_href_90ff36b5040e3b76 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).href;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_matchMedia_0b5dc8aaf445df72 = function() { return handleError(function (arg0, arg1, arg2) {
|
|
||||||
const ret = getObject(arg0).matchMedia(getStringFromWasm0(arg1, arg2));
|
|
||||||
return isLikeNone(ret) ? 0 : addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_matches_0ffc2232d99a6034 = function(arg0) {
|
|
||||||
const ret = getObject(arg0).matches;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_new_abda76e883ba8a5f = function() {
|
|
||||||
const ret = new Error();
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_stack_658279fe44541cf6 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).stack;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_error_f851667af71bcfc6 = function(arg0, arg1) {
|
|
||||||
try {
|
|
||||||
console.error(getStringFromWasm0(arg0, arg1));
|
|
||||||
} finally {
|
|
||||||
wasm.__wbindgen_free(arg0, arg1);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_object_drop_ref = function(arg0) {
|
|
||||||
takeObject(arg0);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_new_0b9bfdd97583284e = function() {
|
|
||||||
const ret = new Object();
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_number_new = function(arg0) {
|
|
||||||
const ret = arg0;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_set_ce5827ace4c694dc = function(arg0, arg1, arg2) {
|
|
||||||
getObject(arg0)[takeObject(arg1)] = takeObject(arg2);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_state_4896ba54c2e3301e = function() { return handleError(function (arg0) {
|
|
||||||
const ret = getObject(arg0).state;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbindgen_is_object = function(arg0) {
|
|
||||||
const val = getObject(arg0);
|
|
||||||
const ret = typeof(val) === 'object' && val !== null;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_get_723f83ba0c34871a = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg0)[takeObject(arg1)];
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_is_string = function(arg0) {
|
|
||||||
const ret = typeof(getObject(arg0)) === 'string';
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_entries_65a76a413fc91037 = function(arg0) {
|
|
||||||
const ret = Object.entries(getObject(arg0));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_length_6e3bbe7c8bd4dbd8 = function(arg0) {
|
|
||||||
const ret = getObject(arg0).length;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_get_57245cc7d7c7619d = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg0)[arg1 >>> 0];
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_string_get = function(arg0, arg1) {
|
|
||||||
const obj = getObject(arg1);
|
|
||||||
const ret = typeof(obj) === 'string' ? obj : undefined;
|
|
||||||
var ptr0 = isLikeNone(ret) ? 0 : passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
var len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_isSafeInteger_dfa0593e8d7ac35a = function(arg0) {
|
|
||||||
const ret = Number.isSafeInteger(getObject(arg0));
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_search_4aac147f005678e5 = function() { return handleError(function (arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).search;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_hash_8565e7b1ae1f2be4 = function() { return handleError(function (arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).hash;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_newwithbase_41b4a8c94dd8c467 = function() { return handleError(function (arg0, arg1, arg2, arg3) {
|
|
||||||
const ret = new URL(getStringFromWasm0(arg0, arg1), getStringFromWasm0(arg2, arg3));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_hash_5ca9e2d439e2b3e1 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).hash;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_object_clone_ref = function(arg0) {
|
|
||||||
const ret = getObject(arg0);
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_removeEventListener_b10f1a66647f3aa0 = function() { return handleError(function (arg0, arg1, arg2, arg3, arg4) {
|
|
||||||
getObject(arg0).removeEventListener(getStringFromWasm0(arg1, arg2), getObject(arg3), arg4 !== 0);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_instanceof_Error_56b496a10a56de66 = function(arg0) {
|
|
||||||
let result;
|
|
||||||
try {
|
|
||||||
result = getObject(arg0) instanceof Error;
|
|
||||||
} catch {
|
|
||||||
result = false;
|
|
||||||
}
|
|
||||||
const ret = result;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_name_48eda3ae6aa697ca = function(arg0) {
|
|
||||||
const ret = getObject(arg0).name;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_message_fe2af63ccc8985bc = function(arg0) {
|
|
||||||
const ret = getObject(arg0).message;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_toString_73c9b562dccf34bd = function(arg0) {
|
|
||||||
const ret = getObject(arg0).toString();
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_new_2d0053ee81e4dd2a = function() { return handleError(function () {
|
|
||||||
const ret = new Headers();
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_new_ca4d3a3eca340210 = function() { return handleError(function () {
|
|
||||||
const ret = new URLSearchParams();
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbindgen_string_new = function(arg0, arg1) {
|
|
||||||
const ret = getStringFromWasm0(arg0, arg1);
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_document_3ead31dbcad65886 = function(arg0) {
|
|
||||||
const ret = getObject(arg0).document;
|
|
||||||
return isLikeNone(ret) ? 0 : addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_self_6d479506f72c6a71 = function() { return handleError(function () {
|
|
||||||
const ret = self.self;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_window_f2557cc78490aceb = function() { return handleError(function () {
|
|
||||||
const ret = window.window;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_globalThis_7f206bda628d5286 = function() { return handleError(function () {
|
|
||||||
const ret = globalThis.globalThis;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_global_ba75c50d1cf384f4 = function() { return handleError(function () {
|
|
||||||
const ret = global.global;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbindgen_is_undefined = function(arg0) {
|
|
||||||
const ret = getObject(arg0) === undefined;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_newnoargs_b5b063fc6c2f0376 = function(arg0, arg1) {
|
|
||||||
const ret = new Function(getStringFromWasm0(arg0, arg1));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_call_97ae9d8645dc388b = function() { return handleError(function (arg0, arg1) {
|
|
||||||
const ret = getObject(arg0).call(getObject(arg1));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_is_40a66842732708e7 = function(arg0, arg1) {
|
|
||||||
const ret = Object.is(getObject(arg0), getObject(arg1));
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_toString_7be108a12ef03bc2 = function(arg0) {
|
|
||||||
const ret = getObject(arg0).toString();
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_length_9e1ae1900cb0fbd5 = function(arg0) {
|
|
||||||
const ret = getObject(arg0).length;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_memory = function() {
|
|
||||||
const ret = wasm.memory;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_buffer_3f3d764d4747d564 = function(arg0) {
|
|
||||||
const ret = getObject(arg0).buffer;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_new_8c3f0052272a457a = function(arg0) {
|
|
||||||
const ret = new Uint8Array(getObject(arg0));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_set_83db9690f9353e79 = function(arg0, arg1, arg2) {
|
|
||||||
getObject(arg0).set(getObject(arg1), arg2 >>> 0);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_get_765201544a2b6869 = function() { return handleError(function (arg0, arg1) {
|
|
||||||
const ret = Reflect.get(getObject(arg0), getObject(arg1));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_set_bf3f89b92d5a34bf = function() { return handleError(function (arg0, arg1, arg2) {
|
|
||||||
const ret = Reflect.set(getObject(arg0), getObject(arg1), getObject(arg2));
|
|
||||||
return ret;
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbindgen_is_null = function(arg0) {
|
|
||||||
const ret = getObject(arg0) === null;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_boolean_get = function(arg0) {
|
|
||||||
const v = getObject(arg0);
|
|
||||||
const ret = typeof(v) === 'boolean' ? (v ? 1 : 0) : 2;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_instanceof_Uint8Array_971eeda69eb75003 = function(arg0) {
|
|
||||||
let result;
|
|
||||||
try {
|
|
||||||
result = getObject(arg0) instanceof Uint8Array;
|
|
||||||
} catch {
|
|
||||||
result = false;
|
|
||||||
}
|
|
||||||
const ret = result;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_instanceof_ArrayBuffer_e5e48f4762c5610b = function(arg0) {
|
|
||||||
let result;
|
|
||||||
try {
|
|
||||||
result = getObject(arg0) instanceof ArrayBuffer;
|
|
||||||
} catch {
|
|
||||||
result = false;
|
|
||||||
}
|
|
||||||
const ret = result;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_number_get = function(arg0, arg1) {
|
|
||||||
const obj = getObject(arg1);
|
|
||||||
const ret = typeof(obj) === 'number' ? obj : undefined;
|
|
||||||
getFloat64Memory0()[arg0 / 8 + 1] = isLikeNone(ret) ? 0 : ret;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = !isLikeNone(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_new_8d2af00bc1e329ee = function(arg0, arg1) {
|
|
||||||
const ret = new Error(getStringFromWasm0(arg0, arg1));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_debug_string = function(arg0, arg1) {
|
|
||||||
const ret = debugString(getObject(arg1));
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_throw = function(arg0, arg1) {
|
|
||||||
throw new Error(getStringFromWasm0(arg0, arg1));
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_then_cedad20fbbd9418a = function(arg0, arg1, arg2) {
|
|
||||||
const ret = getObject(arg0).then(getObject(arg1), getObject(arg2));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_resolve_99fe17964f31ffc0 = function(arg0) {
|
|
||||||
const ret = Promise.resolve(getObject(arg0));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_cb_drop = function(arg0) {
|
|
||||||
const obj = takeObject(arg0).original;
|
|
||||||
if (obj.cnt-- == 1) {
|
|
||||||
obj.a = 0;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
const ret = false;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_then_11f7a54d67b4bfad = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg0).then(getObject(arg1));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_createElement_976dbb84fe1661b5 = function() { return handleError(function (arg0, arg1, arg2) {
|
|
||||||
const ret = getObject(arg0).createElement(getStringFromWasm0(arg1, arg2));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_pathname_78a642e573bf8169 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).pathname;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_search_afb25c63fe262036 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).search;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_new_7d95b89914e4d377 = function() { return handleError(function (arg0, arg1) {
|
|
||||||
const ret = new URL(getStringFromWasm0(arg0, arg1));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_setvalue_df64bc6794c098f2 = function(arg0, arg1, arg2) {
|
|
||||||
getObject(arg0).value = getStringFromWasm0(arg1, arg2);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_instanceof_Window_acc97ff9f5d2c7b4 = function(arg0) {
|
|
||||||
let result;
|
|
||||||
try {
|
|
||||||
result = getObject(arg0) instanceof Window;
|
|
||||||
} catch {
|
|
||||||
result = false;
|
|
||||||
}
|
|
||||||
const ret = result;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_setchecked_f1e1f3e62cdca8e7 = function(arg0, arg1) {
|
|
||||||
getObject(arg0).checked = arg1 !== 0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_setvalue_e5b519cca37d82a7 = function(arg0, arg1, arg2) {
|
|
||||||
getObject(arg0).value = getStringFromWasm0(arg1, arg2);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_removeAttribute_beaed7727852af78 = function() { return handleError(function (arg0, arg1, arg2) {
|
|
||||||
getObject(arg0).removeAttribute(getStringFromWasm0(arg1, arg2));
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_setAttribute_d8436c14a59ab1af = function() { return handleError(function (arg0, arg1, arg2, arg3, arg4) {
|
|
||||||
getObject(arg0).setAttribute(getStringFromWasm0(arg1, arg2), getStringFromWasm0(arg3, arg4));
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_addEventListener_1fc744729ac6dc27 = function() { return handleError(function (arg0, arg1, arg2, arg3, arg4) {
|
|
||||||
getObject(arg0).addEventListener(getStringFromWasm0(arg1, arg2), getObject(arg3), getObject(arg4));
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_nextSibling_62338ec2a05607b4 = function(arg0) {
|
|
||||||
const ret = getObject(arg0).nextSibling;
|
|
||||||
return isLikeNone(ret) ? 0 : addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_insertBefore_9f2d2defb9471006 = function() { return handleError(function (arg0, arg1, arg2) {
|
|
||||||
const ret = getObject(arg0).insertBefore(getObject(arg1), getObject(arg2));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_removeChild_6751e9ca5d9aaf00 = function() { return handleError(function (arg0, arg1) {
|
|
||||||
const ret = getObject(arg0).removeChild(getObject(arg1));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_pushState_38917fb88b4add30 = function() { return handleError(function (arg0, arg1, arg2, arg3, arg4, arg5) {
|
|
||||||
getObject(arg0).pushState(getObject(arg1), getStringFromWasm0(arg2, arg3), arg4 === 0 ? undefined : getStringFromWasm0(arg4, arg5));
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_href_bbb11e0e61ea410e = function() { return handleError(function (arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).href;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_pathname_4441d4d8fc4aba51 = function() { return handleError(function (arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).pathname;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_namespaceURI_e19c7be2c60e5b5c = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).namespaceURI;
|
|
||||||
var ptr0 = isLikeNone(ret) ? 0 : passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
var len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_createElementNS_1561aca8ee3693c0 = function() { return handleError(function (arg0, arg1, arg2, arg3, arg4) {
|
|
||||||
const ret = getObject(arg0).createElementNS(arg1 === 0 ? undefined : getStringFromWasm0(arg1, arg2), getStringFromWasm0(arg3, arg4));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_parentElement_0cffb3ceb0f107bd = function(arg0) {
|
|
||||||
const ret = getObject(arg0).parentElement;
|
|
||||||
return isLikeNone(ret) ? 0 : addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_parentNode_e397bbbe28be7b28 = function(arg0) {
|
|
||||||
const ret = getObject(arg0).parentNode;
|
|
||||||
return isLikeNone(ret) ? 0 : addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_instanceof_ShadowRoot_76b32ccdae10a710 = function(arg0) {
|
|
||||||
let result;
|
|
||||||
try {
|
|
||||||
result = getObject(arg0) instanceof ShadowRoot;
|
|
||||||
} catch {
|
|
||||||
result = false;
|
|
||||||
}
|
|
||||||
const ret = result;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_host_57eec05a2624bc1b = function(arg0) {
|
|
||||||
const ret = getObject(arg0).host;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_composedPath_160ed014dc4d787f = function(arg0) {
|
|
||||||
const ret = getObject(arg0).composedPath();
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_cachekey_b61393159c57fd7b = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).__yew_subtree_cache_key;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = isLikeNone(ret) ? 0 : ret;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = !isLikeNone(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_subtreeid_e348577f7ef777e3 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).__yew_subtree_id;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = isLikeNone(ret) ? 0 : ret;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = !isLikeNone(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_bubbles_03eed164b4feeaf1 = function(arg0) {
|
|
||||||
const ret = getObject(arg0).bubbles;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_setsubtreeid_d32e6327eef1f7fc = function(arg0, arg1) {
|
|
||||||
getObject(arg0).__yew_subtree_id = arg1 >>> 0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_setcachekey_80183b7cfc421143 = function(arg0, arg1) {
|
|
||||||
getObject(arg0).__yew_subtree_cache_key = arg1 >>> 0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_cancelBubble_8c0bdf21c08f1717 = function(arg0) {
|
|
||||||
const ret = getObject(arg0).cancelBubble;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_listenerid_12315eee21527820 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).__yew_listener_id;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = isLikeNone(ret) ? 0 : ret;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = !isLikeNone(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_setinnerHTML_32081d8a164e6dc4 = function(arg0, arg1, arg2) {
|
|
||||||
getObject(arg0).innerHTML = getStringFromWasm0(arg1, arg2);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_children_67776b4810f38b6a = function(arg0) {
|
|
||||||
const ret = getObject(arg0).children;
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_from_7ce3cb27cb258569 = function(arg0) {
|
|
||||||
const ret = Array.from(getObject(arg0));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_instanceof_Element_33bd126d58f2021b = function(arg0) {
|
|
||||||
let result;
|
|
||||||
try {
|
|
||||||
result = getObject(arg0) instanceof Element;
|
|
||||||
} catch {
|
|
||||||
result = false;
|
|
||||||
}
|
|
||||||
const ret = result;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_createTextNode_300f845fab76642f = function(arg0, arg1, arg2) {
|
|
||||||
const ret = getObject(arg0).createTextNode(getStringFromWasm0(arg1, arg2));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_appendChild_e513ef0e5098dfdd = function() { return handleError(function (arg0, arg1) {
|
|
||||||
const ret = getObject(arg0).appendChild(getObject(arg1));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_error_71d6845bf00a930f = function(arg0, arg1) {
|
|
||||||
var v0 = getArrayJsValueFromWasm0(arg0, arg1).slice();
|
|
||||||
wasm.__wbindgen_free(arg0, arg1 * 4);
|
|
||||||
console.error(...v0);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_setnodeValue_4077cafeefd0725e = function(arg0, arg1, arg2) {
|
|
||||||
getObject(arg0).nodeValue = arg1 === 0 ? undefined : getStringFromWasm0(arg1, arg2);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_value_b2a620d34c663701 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).value;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_value_ccb32485ee1b3928 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).value;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_setlistenerid_3183aae8fa5840fb = function(arg0, arg1) {
|
|
||||||
getObject(arg0).__yew_listener_id = arg1 >>> 0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_log_4cf38bb073e9467a = function(arg0, arg1) {
|
|
||||||
console.log(getStringFromWasm0(arg0, arg1));
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_text_1169d752cc697903 = function() { return handleError(function (arg0) {
|
|
||||||
const ret = getObject(arg0).text();
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_newwithstr_fdce36db91ec5f92 = function() { return handleError(function (arg0, arg1) {
|
|
||||||
const ret = new Request(getStringFromWasm0(arg0, arg1));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_url_1c013f0875e97715 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).url;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_setsearch_40007c2a91333011 = function(arg0, arg1, arg2) {
|
|
||||||
getObject(arg0).search = getStringFromWasm0(arg1, arg2);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_newwithstrandinit_05d7180788420c40 = function() { return handleError(function (arg0, arg1, arg2) {
|
|
||||||
const ret = new Request(getStringFromWasm0(arg0, arg1), getObject(arg2));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
}, arguments) };
|
|
||||||
imports.wbg.__wbg_fetch_0fe04905cccfc2aa = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg0).fetch(getObject(arg1));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_instanceof_WorkerGlobalScope_16bb97a4549a3f21 = function(arg0) {
|
|
||||||
let result;
|
|
||||||
try {
|
|
||||||
result = getObject(arg0) instanceof WorkerGlobalScope;
|
|
||||||
} catch {
|
|
||||||
result = false;
|
|
||||||
}
|
|
||||||
const ret = result;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_fetch_749a56934f95c96c = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg0).fetch(getObject(arg1));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_instanceof_Response_eaa426220848a39e = function(arg0) {
|
|
||||||
let result;
|
|
||||||
try {
|
|
||||||
result = getObject(arg0) instanceof Response;
|
|
||||||
} catch {
|
|
||||||
result = false;
|
|
||||||
}
|
|
||||||
const ret = result;
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_getElementsByTagName_617098e0ae766c3e = function(arg0, arg1, arg2) {
|
|
||||||
const ret = getObject(arg0).getElementsByTagName(getStringFromWasm0(arg1, arg2));
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_getwithindex_5571ba24207565a4 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg0)[arg1 >>> 0];
|
|
||||||
return isLikeNone(ret) ? 0 : addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_body_3cb4b4042b9a632b = function(arg0) {
|
|
||||||
const ret = getObject(arg0).body;
|
|
||||||
return isLikeNone(ret) ? 0 : addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_lastChild_a2f5ed739809bb31 = function(arg0) {
|
|
||||||
const ret = getObject(arg0).lastChild;
|
|
||||||
return isLikeNone(ret) ? 0 : addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_sethash_d35570df091aa47e = function(arg0, arg1, arg2) {
|
|
||||||
getObject(arg0).hash = getStringFromWasm0(arg1, arg2);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbg_href_9b462d09b5f8b378 = function(arg0, arg1) {
|
|
||||||
const ret = getObject(arg1).href;
|
|
||||||
const ptr0 = passStringToWasm0(ret, wasm.__wbindgen_malloc, wasm.__wbindgen_realloc);
|
|
||||||
const len0 = WASM_VECTOR_LEN;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 1] = len0;
|
|
||||||
getInt32Memory0()[arg0 / 4 + 0] = ptr0;
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_closure_wrapper397 = function(arg0, arg1, arg2) {
|
|
||||||
const ret = makeMutClosure(arg0, arg1, 68, __wbg_adapter_32);
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
imports.wbg.__wbindgen_closure_wrapper2691 = function(arg0, arg1, arg2) {
|
|
||||||
const ret = makeMutClosure(arg0, arg1, 68, __wbg_adapter_35);
|
|
||||||
return addHeapObject(ret);
|
|
||||||
};
|
|
||||||
|
|
||||||
return imports;
|
|
||||||
}
|
|
||||||
|
|
||||||
function initMemory(imports, maybe_memory) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
function finalizeInit(instance, module) {
|
|
||||||
wasm = instance.exports;
|
|
||||||
init.__wbindgen_wasm_module = module;
|
|
||||||
cachedFloat64Memory0 = new Float64Array();
|
|
||||||
cachedInt32Memory0 = new Int32Array();
|
|
||||||
cachedUint32Memory0 = new Uint32Array();
|
|
||||||
cachedUint8Memory0 = new Uint8Array();
|
|
||||||
|
|
||||||
wasm.__wbindgen_start();
|
|
||||||
return wasm;
|
|
||||||
}
|
|
||||||
|
|
||||||
function initSync(module) {
|
|
||||||
const imports = getImports();
|
|
||||||
|
|
||||||
initMemory(imports);
|
|
||||||
|
|
||||||
if (!(module instanceof WebAssembly.Module)) {
|
|
||||||
module = new WebAssembly.Module(module);
|
|
||||||
}
|
|
||||||
|
|
||||||
const instance = new WebAssembly.Instance(module, imports);
|
|
||||||
|
|
||||||
return finalizeInit(instance, module);
|
|
||||||
}
|
|
||||||
|
|
||||||
async function init(input) {
|
|
||||||
|
|
||||||
const imports = getImports();
|
|
||||||
|
|
||||||
if (typeof input === 'string' || (typeof Request === 'function' && input instanceof Request) || (typeof URL === 'function' && input instanceof URL)) {
|
|
||||||
input = fetch(input);
|
|
||||||
}
|
|
||||||
|
|
||||||
initMemory(imports);
|
|
||||||
|
|
||||||
const { instance, module } = await load(await input, imports);
|
|
||||||
|
|
||||||
return finalizeInit(instance, module);
|
|
||||||
}
|
|
||||||
|
|
||||||
export { initSync }
|
|
||||||
export default init;
|
|
8357
blueprint.css
3
netlify.toml
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
[build]
|
||||||
|
publish = "target/release/dist"
|
||||||
|
command = "cargo xtask dist --release"
|
87
src/app.rs
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
use gloo::utils::window;
|
||||||
|
use yew::prelude::*;
|
||||||
|
use yew_router::Routable;
|
||||||
|
use yew_router::prelude::Navigator;
|
||||||
|
use yew_router::prelude::use_navigator;
|
||||||
|
use crate::component::fa_icon::FontawesomeIcon;
|
||||||
|
use crate::page::Page;
|
||||||
|
use crate::page::PageRouter;
|
||||||
|
use crate::page::Pages;
|
||||||
|
use crate::theme::ThemeContext;
|
||||||
|
use crate::theme::ThemeMsg;
|
||||||
|
use crate::theme::ThemeProvider;
|
||||||
|
use crate::component::actionbar::{Actionbar, ActionbarOption};
|
||||||
|
use crate::theme::ThemeState;
|
||||||
|
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn AppRoot() -> Html {
|
||||||
|
html! {
|
||||||
|
<ThemeProvider>
|
||||||
|
<PageRouter>
|
||||||
|
<ThemedApp />
|
||||||
|
</PageRouter>
|
||||||
|
</ThemeProvider>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
fn ThemedApp() -> Html {
|
||||||
|
let navigator = use_navigator().unwrap();
|
||||||
|
|
||||||
|
// Helper function for generating tabs
|
||||||
|
fn page_option(current: &Page, page: Page, navigator: Navigator, state: UseStateSetter<Page>) -> ActionbarOption {
|
||||||
|
let is_current = *current == page;
|
||||||
|
ActionbarOption::new(
|
||||||
|
page.name(),
|
||||||
|
match page {
|
||||||
|
Page::Home => FontawesomeIcon::House,
|
||||||
|
Page::Projects => FontawesomeIcon::Code,
|
||||||
|
Page::Socials => FontawesomeIcon::ShareNodes,
|
||||||
|
Page::Gallery => FontawesomeIcon::Camera,
|
||||||
|
Page::Contact => FontawesomeIcon::Envelope,
|
||||||
|
},
|
||||||
|
Callback::from(move |_| {
|
||||||
|
navigator.push(&page);
|
||||||
|
state.set(page);
|
||||||
|
}),
|
||||||
|
is_current
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
let ctx = use_context::<ThemeContext>().expect("No theme context supplied");
|
||||||
|
let theme_icon = if ctx.theme == ThemeState::Dark { FontawesomeIcon::Sun } else { FontawesomeIcon::Moon };
|
||||||
|
let page_state = use_state_eq(||
|
||||||
|
window()
|
||||||
|
.location()
|
||||||
|
.pathname()
|
||||||
|
.ok()
|
||||||
|
.and_then(|p| Page::recognize(&p))
|
||||||
|
.or(Some(Page::Home))
|
||||||
|
.unwrap()
|
||||||
|
);
|
||||||
|
let current_page = *page_state;
|
||||||
|
|
||||||
|
html! {
|
||||||
|
<>
|
||||||
|
<div class="main">
|
||||||
|
<Pages />
|
||||||
|
</div>
|
||||||
|
<Actionbar mobile_title="Menu" items={vec![
|
||||||
|
page_option(¤t_page, Page::Home, navigator.clone(), page_state.setter()),
|
||||||
|
page_option(¤t_page, Page::Projects, navigator.clone(), page_state.setter()),
|
||||||
|
page_option(¤t_page, Page::Socials, navigator.clone(), page_state.setter()),
|
||||||
|
page_option(¤t_page, Page::Gallery, navigator.clone(), page_state.setter()),
|
||||||
|
page_option(¤t_page, Page::Contact, navigator, page_state.setter()),
|
||||||
|
ActionbarOption::new_opt(
|
||||||
|
None,
|
||||||
|
Some(theme_icon),
|
||||||
|
Some(Callback::from(move |_| ctx.dispatch(ThemeMsg::Toggle))),
|
||||||
|
true,
|
||||||
|
false,
|
||||||
|
false
|
||||||
|
)
|
||||||
|
]}/>
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
}
|
100
src/component/actionbar.rs
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
use yew::prelude::*;
|
||||||
|
use yew::virtual_dom::VNode;
|
||||||
|
|
||||||
|
use super::fa_icon::{FAIcon, FontawesomeIcon};
|
||||||
|
|
||||||
|
#[derive(PartialEq, Clone)]
|
||||||
|
pub struct ActionbarOption {
|
||||||
|
name: Option<String>,
|
||||||
|
icon: Option<FontawesomeIcon>,
|
||||||
|
onclick: Option<Callback<MouseEvent>>,
|
||||||
|
center_content: bool,
|
||||||
|
selected: bool,
|
||||||
|
fill: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Clone)]
|
||||||
|
struct ActionbarOptionState {
|
||||||
|
inner: ActionbarOption,
|
||||||
|
callback: UseStateSetter<bool>
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<VNode> for ActionbarOption {
|
||||||
|
fn into(self) -> VNode {
|
||||||
|
/*
|
||||||
|
let classes = if self.center_content { classes!() } else { classes!("ab-navbutton") };
|
||||||
|
html! {
|
||||||
|
<Button fill=true large=true class={classes} icon={self.icon} onclick={self.onclick.unwrap_or_default()} active={self.selected}>{self.name}</Button>
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
panic!("Stateless actionbar option")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ActionbarOption {
|
||||||
|
pub fn new_opt(name: Option<String>, icon: Option<FontawesomeIcon>, onclick: Option<Callback<MouseEvent>>, center_content: bool, selected: bool, fill: bool) -> ActionbarOption {
|
||||||
|
ActionbarOption { name, icon, onclick, center_content, selected, fill }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn new(name: &str, icon: FontawesomeIcon, onclick: Callback<MouseEvent>, selected: bool) -> ActionbarOption {
|
||||||
|
ActionbarOption::new_opt(Some(name.to_owned()), Some(icon), Some(onclick), false, selected, true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<VNode> for ActionbarOptionState {
|
||||||
|
fn into(self) -> VNode {
|
||||||
|
let onclick = self.inner.onclick.unwrap_or_default();
|
||||||
|
html! {
|
||||||
|
<div
|
||||||
|
class={classes!(
|
||||||
|
if self.inner.selected { "current" } else { "" },
|
||||||
|
if self.inner.icon.is_some() && self.inner.name.is_some() { "" } else { "small" }
|
||||||
|
)}
|
||||||
|
onclick={Callback::from(move |e| {
|
||||||
|
self.callback.set(false);
|
||||||
|
onclick.emit(e)
|
||||||
|
})}>
|
||||||
|
<div>
|
||||||
|
{if let Some(icon) = self.inner.icon { html!{ <FAIcon icon={icon} /> } } else { html!{} }}
|
||||||
|
<span>{self.inner.name}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Properties, PartialEq)]
|
||||||
|
pub struct ActionbarProps {
|
||||||
|
#[prop_or_default]
|
||||||
|
pub mobile_title: Option<String>,
|
||||||
|
pub items: Vec<ActionbarOption>
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn Actionbar(props: &ActionbarProps) -> Html {
|
||||||
|
let expanded = use_state_eq(|| false);
|
||||||
|
let expanded_setter = expanded.setter();
|
||||||
|
let is_expanded = *expanded;
|
||||||
|
|
||||||
|
html! {
|
||||||
|
<div class="navbar">
|
||||||
|
<div style={if props.mobile_title.is_some() { "gap: 5px;" } else { "" }} onclick={Callback::from(move |_| expanded.set(!*expanded))}>
|
||||||
|
<FAIcon
|
||||||
|
icon={if is_expanded { FontawesomeIcon::ChevronUp } else { FontawesomeIcon::ChevronDown }} />
|
||||||
|
{
|
||||||
|
if let Some(ref title) = props.mobile_title {
|
||||||
|
html! {
|
||||||
|
<span>{title}</span>
|
||||||
|
}
|
||||||
|
} else { html! {} }
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
<div class={if is_expanded { "open" } else { "" }}></div>
|
||||||
|
<div>
|
||||||
|
{props.items.iter()
|
||||||
|
.map(move |opt| ActionbarOptionState { inner: (*opt).clone(), callback: expanded_setter.clone() })
|
||||||
|
.collect::<Html>()}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
30
src/component/card.rs
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
use web_sys::MouseEvent;
|
||||||
|
use yew::{Html, function_component, html, Children, Properties, Classes, Callback, classes};
|
||||||
|
|
||||||
|
|
||||||
|
#[derive(Properties, PartialEq)]
|
||||||
|
pub struct CardProps {
|
||||||
|
#[prop_or_default]
|
||||||
|
pub children: Children,
|
||||||
|
|
||||||
|
#[prop_or_default]
|
||||||
|
pub style: String,
|
||||||
|
|
||||||
|
#[prop_or_default]
|
||||||
|
pub class: Classes,
|
||||||
|
|
||||||
|
#[prop_or_default]
|
||||||
|
pub onclick: Callback<MouseEvent>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn Card(props: &CardProps) -> Html {
|
||||||
|
html! {
|
||||||
|
<div
|
||||||
|
class={classes!("card", props.class.clone())}
|
||||||
|
style={props.style.clone()}
|
||||||
|
onclick={props.onclick.clone()}>
|
||||||
|
{props.children.clone()}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
377
src/component/fa_icon.rs
Normal file
@ -0,0 +1,377 @@
|
|||||||
|
use std::fmt::Display;
|
||||||
|
|
||||||
|
use web_sys::MouseEvent;
|
||||||
|
use yew::{Properties, function_component, Html, html, Callback, Classes, classes};
|
||||||
|
|
||||||
|
#[derive(PartialEq, Clone)]
|
||||||
|
pub enum FontawesomeIcon {
|
||||||
|
Rust,
|
||||||
|
GitHub,
|
||||||
|
Camera,
|
||||||
|
ShareNodes,
|
||||||
|
Sun,
|
||||||
|
Moon,
|
||||||
|
House,
|
||||||
|
Code,
|
||||||
|
Envelope,
|
||||||
|
ChevronDown,
|
||||||
|
ChevronUp,
|
||||||
|
ChevronLeft,
|
||||||
|
ChevronRight,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for FontawesomeIcon {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
f.write_str(match self {
|
||||||
|
Self::Rust => "fa-brands fa-rust",
|
||||||
|
Self::GitHub => "fa-brands fa-github",
|
||||||
|
Self::Camera => "fa-camera",
|
||||||
|
Self::ShareNodes => "fa-share-nodes",
|
||||||
|
Self::Sun => "fa-sun",
|
||||||
|
Self::Moon => "fa-moon",
|
||||||
|
Self::House => "fa-house",
|
||||||
|
Self::Code => "fa-code",
|
||||||
|
Self::Envelope => "fa-envelope",
|
||||||
|
Self::ChevronDown => "fa-chevron-down",
|
||||||
|
Self::ChevronUp => "fa-chevron-up",
|
||||||
|
Self::ChevronLeft => "fa-chevron-left",
|
||||||
|
Self::ChevronRight => "fa-chevron-right",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Default)]
|
||||||
|
pub enum FontawesomeStyle {
|
||||||
|
#[default]
|
||||||
|
//Regular,
|
||||||
|
Solid,
|
||||||
|
Light,
|
||||||
|
DuoTone,
|
||||||
|
Thin,
|
||||||
|
Sharp,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for FontawesomeStyle {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
f.write_str(match self {
|
||||||
|
//Self::Regular => "fa-regular",
|
||||||
|
Self::Solid => "fa-solid",
|
||||||
|
Self::Light => "fa-light",
|
||||||
|
Self::DuoTone => "fa-duotone",
|
||||||
|
Self::Thin => "fa-thin",
|
||||||
|
Self::Sharp => "fa-sharp",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Default)]
|
||||||
|
pub enum FontawesomeSize {
|
||||||
|
XXS,
|
||||||
|
XS,
|
||||||
|
S,
|
||||||
|
#[default]
|
||||||
|
Regular,
|
||||||
|
L,
|
||||||
|
XL,
|
||||||
|
XXL
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for FontawesomeSize {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
f.write_str(match self {
|
||||||
|
Self::XXS => "fa-2xs",
|
||||||
|
Self::XS => "fa-xs",
|
||||||
|
Self::S => "fa-sm",
|
||||||
|
Self::Regular => "",
|
||||||
|
Self::L => "fa-lg",
|
||||||
|
Self::XL => "fa-xl",
|
||||||
|
Self::XXL => "fa-2xl"
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Default)]
|
||||||
|
pub enum FontawesomeSpinType {
|
||||||
|
#[default]
|
||||||
|
Regular,
|
||||||
|
Pulse
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Default)]
|
||||||
|
pub enum FontawesomeSpinDirection {
|
||||||
|
#[default]
|
||||||
|
Clockwise,
|
||||||
|
CounterClockwise
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq)]
|
||||||
|
pub enum FontawesomeAnimationType {
|
||||||
|
Beat {
|
||||||
|
scale: f32,
|
||||||
|
},
|
||||||
|
Fade {
|
||||||
|
opacity: f32,
|
||||||
|
},
|
||||||
|
BeatFade {
|
||||||
|
scale: f32,
|
||||||
|
opacity: f32,
|
||||||
|
},
|
||||||
|
Bounce {
|
||||||
|
rebound: f32,
|
||||||
|
height: f32,
|
||||||
|
squish_scale_x: f32,
|
||||||
|
squish_scale_y: f32,
|
||||||
|
jump_scale_x: f32,
|
||||||
|
jump_scale_y: f32,
|
||||||
|
land_scale_x: f32,
|
||||||
|
land_scale_y: f32,
|
||||||
|
},
|
||||||
|
Flip {
|
||||||
|
x: f32,
|
||||||
|
y: f32,
|
||||||
|
z: f32,
|
||||||
|
angle: f32,
|
||||||
|
},
|
||||||
|
Shake,
|
||||||
|
Spin {
|
||||||
|
spin_type: FontawesomeSpinType,
|
||||||
|
direction: FontawesomeSpinDirection
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FontawesomeAnimationType {
|
||||||
|
fn class(&self) -> &'static str {
|
||||||
|
match self {
|
||||||
|
Self::Beat { scale: _ } => "fa-beat",
|
||||||
|
Self::Fade { opacity: _ } => "fa-fade",
|
||||||
|
Self::BeatFade { scale: _, opacity: _ } => "fa-beat-fade",
|
||||||
|
Self::Bounce { rebound: _, height: _, squish_scale_x: _, squish_scale_y: _, jump_scale_x: _, jump_scale_y: _, land_scale_x: _, land_scale_y: _ } => "fa-bounce",
|
||||||
|
Self::Flip { x: _, y: _, z: _, angle: _ } => "fa-flip",
|
||||||
|
Self::Shake => "fa-shake",
|
||||||
|
Self::Spin { spin_type, direction } =>
|
||||||
|
match direction {
|
||||||
|
FontawesomeSpinDirection::Clockwise =>
|
||||||
|
match spin_type {
|
||||||
|
FontawesomeSpinType::Regular => "fa-spin",
|
||||||
|
FontawesomeSpinType::Pulse => "fa-spin-pulse",
|
||||||
|
},
|
||||||
|
FontawesomeSpinDirection::CounterClockwise =>
|
||||||
|
match spin_type {
|
||||||
|
FontawesomeSpinType::Regular => "fa-spin fa-spin-reverse",
|
||||||
|
FontawesomeSpinType::Pulse => "fa-spin-pulse fa-spin-reverse",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn style(&self) -> String {
|
||||||
|
match self {
|
||||||
|
Self::Beat { scale } => format!("--fa-beat-scale: {scale};"),
|
||||||
|
Self::Fade { opacity } => format!("--fa-fade-opacity: {opacity};"),
|
||||||
|
Self::BeatFade { scale, opacity } => format!("--fa-beat-fade-opacity: {opacity}; --fa-beat-fade-scale: {scale};"),
|
||||||
|
Self::Bounce {
|
||||||
|
rebound,
|
||||||
|
height,
|
||||||
|
squish_scale_x,
|
||||||
|
squish_scale_y,
|
||||||
|
jump_scale_x,
|
||||||
|
jump_scale_y,
|
||||||
|
land_scale_x,
|
||||||
|
land_scale_y
|
||||||
|
} => format!("--fa-bounce-rebound: {rebound}; --fa-bounce-height: {height}; --fa-bounce-start-scale-x: {squish_scale_x}; --fa-bounce-start-scale-y: {squish_scale_y}; --fa-bounce-jump-scale-x: {jump_scale_x}; --fa-bounce-jump-scale-y: {jump_scale_y}; --fa-bounce-land-scale-x: {land_scale_x}; --fa-bounce-land-scale-y: {land_scale_y};"),
|
||||||
|
Self::Flip { x, y, z, angle } => format!("--fa-flip-x: {x}; --fa-flip-y: {y}; --fa-flip-z: {z}; --fa-flip-angle: {angle};"),
|
||||||
|
_ => "".to_owned(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq)]
|
||||||
|
pub enum AnimationTime {
|
||||||
|
Seconds(f32),
|
||||||
|
Milliseconds(f32),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AnimationTime {
|
||||||
|
fn style_unit(&self) -> String {
|
||||||
|
match self {
|
||||||
|
Self::Seconds(seconds) => format!("{seconds}s"),
|
||||||
|
Self::Milliseconds(millis) => format!("{millis}ms"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for AnimationTime {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self::Seconds(0.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for AnimationTime {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
f.write_str(&self.style_unit())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq)]
|
||||||
|
pub enum AnimationDuration {
|
||||||
|
Time(AnimationTime),
|
||||||
|
Infinite
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for AnimationDuration {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self::Time(Default::default())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for AnimationDuration {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
f.write_str(&match self {
|
||||||
|
Self::Time(time) => time.style_unit(),
|
||||||
|
Self::Infinite => "infinite".to_owned(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Default)]
|
||||||
|
pub enum AnimationDirection {
|
||||||
|
#[default]
|
||||||
|
Normal,
|
||||||
|
Reverse,
|
||||||
|
Alternate,
|
||||||
|
AlternateReverse,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for AnimationDirection {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
f.write_str(match self {
|
||||||
|
Self::Normal => "normal",
|
||||||
|
Self::Reverse => "reverse",
|
||||||
|
Self::Alternate => "alternate",
|
||||||
|
Self::AlternateReverse => "alternate-reverse"
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq)]
|
||||||
|
pub enum AnimationStepType {
|
||||||
|
JumpStart,
|
||||||
|
JumpEnd,
|
||||||
|
JumpNone,
|
||||||
|
JumpBoth,
|
||||||
|
Start,
|
||||||
|
End
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for AnimationStepType {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
f.write_str(match self {
|
||||||
|
Self::JumpStart => "jump-start",
|
||||||
|
Self::JumpEnd => "jump-end",
|
||||||
|
Self::JumpNone => "jump-none",
|
||||||
|
Self::JumpBoth => "jump-both",
|
||||||
|
Self::Start => "start",
|
||||||
|
Self::End => "end",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Default)]
|
||||||
|
pub enum AnimationTiming {
|
||||||
|
#[default]
|
||||||
|
Ease,
|
||||||
|
Linear,
|
||||||
|
EaseIn,
|
||||||
|
EaseOut,
|
||||||
|
EaseInOut,
|
||||||
|
CubicBezier(f32, f32, f32, f32),
|
||||||
|
Steps(i32, AnimationStepType),
|
||||||
|
StepStart,
|
||||||
|
StepEnd,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for AnimationTiming {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
f.write_str(&
|
||||||
|
match self {
|
||||||
|
Self::Ease => "ease".to_owned(),
|
||||||
|
Self::Linear => "linear".to_owned(),
|
||||||
|
Self::EaseIn => "ease-in".to_owned(),
|
||||||
|
Self::EaseOut => "ease-out".to_owned(),
|
||||||
|
Self::EaseInOut => "ease-in-out".to_owned(),
|
||||||
|
Self::CubicBezier(p1, p2, p3, p4) => format!("cubic-bezier({p1}, {p2}, {p3}, {p4})"),
|
||||||
|
Self::Steps(count, step_type) => format!("steps({count}, {step_type})"),
|
||||||
|
Self::StepStart => "step-start".to_owned(),
|
||||||
|
Self::StepEnd => "step-end".to_owned(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq)]
|
||||||
|
pub struct FontawesomeAnimation {
|
||||||
|
duration: AnimationDuration,
|
||||||
|
delay: AnimationTime,
|
||||||
|
direction: AnimationDirection,
|
||||||
|
iterations: AnimationDuration,
|
||||||
|
timing: AnimationTiming,
|
||||||
|
anim_type: FontawesomeAnimationType
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FontawesomeAnimation {
|
||||||
|
fn class(&self) -> &'static str {
|
||||||
|
self.anim_type.class()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn style(&self) -> String {
|
||||||
|
format!(
|
||||||
|
"--fa-animation-delay: {}; --fa-animation-direction: {}; --fa-animation-duration: {}; --fa-animation-iteration-count: {}; --fa-animation-timing: {}; {}",
|
||||||
|
self.delay,
|
||||||
|
self.direction,
|
||||||
|
self.duration,
|
||||||
|
self.iterations,
|
||||||
|
self.timing,
|
||||||
|
self.anim_type.style()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#[derive(Properties, PartialEq)]
|
||||||
|
pub struct FontawesomeProperties {
|
||||||
|
pub icon: FontawesomeIcon,
|
||||||
|
#[prop_or_default]
|
||||||
|
pub style: FontawesomeStyle,
|
||||||
|
#[prop_or_default]
|
||||||
|
pub size: FontawesomeSize,
|
||||||
|
#[prop_or_default]
|
||||||
|
pub animation: Option<FontawesomeAnimation>,
|
||||||
|
#[prop_or_default]
|
||||||
|
pub onclick: Callback<MouseEvent>,
|
||||||
|
#[prop_or_default]
|
||||||
|
pub class: Classes,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn FAIcon(props: &FontawesomeProperties) -> Html {
|
||||||
|
html! {
|
||||||
|
<i
|
||||||
|
onclick={props.onclick.clone()}
|
||||||
|
class={classes!(format!(
|
||||||
|
"{} {} {} {}",
|
||||||
|
props.icon,
|
||||||
|
props.style,
|
||||||
|
props.size,
|
||||||
|
match props.animation {
|
||||||
|
None => "",
|
||||||
|
Some(ref animation) => animation.class()
|
||||||
|
}
|
||||||
|
), props.class.clone())}
|
||||||
|
style={format!(
|
||||||
|
"{}",
|
||||||
|
match props.animation {
|
||||||
|
None => "".to_owned(),
|
||||||
|
Some(ref animation) => animation.style()
|
||||||
|
}
|
||||||
|
)}></i>
|
||||||
|
}
|
||||||
|
}
|
217
src/component/image_viewer.rs
Normal file
@ -0,0 +1,217 @@
|
|||||||
|
use web_sys::MouseEvent;
|
||||||
|
use yew::{function_component, Html, html, Callback, Properties, Component};
|
||||||
|
|
||||||
|
use crate::component::{overlay::Overlay, fa_icon::{FAIcon, FontawesomeIcon, FontawesomeSize}};
|
||||||
|
|
||||||
|
#[derive(PartialEq, Clone)]
|
||||||
|
pub struct ImageDescription {
|
||||||
|
link: String,
|
||||||
|
description: Option<&'static str>
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ImageDescription {
|
||||||
|
pub fn new(link: impl Into<String>, description: &'static str) -> Self {
|
||||||
|
Self {
|
||||||
|
link: link.into(),
|
||||||
|
description: Some(description)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn new_blank(link: impl Into<String>) -> Self {
|
||||||
|
Self {
|
||||||
|
link: link.into(),
|
||||||
|
description: None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Properties, PartialEq)]
|
||||||
|
pub struct DefaultImageViewerProps {
|
||||||
|
pub images: Vec<ImageDescription>,
|
||||||
|
pub open: bool,
|
||||||
|
#[prop_or_default]
|
||||||
|
pub infinite: bool,
|
||||||
|
pub onclose: Callback<()>,
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
#[function_component]
|
||||||
|
pub fn DefaultImageViewer(props: &DefaultImageViewerProps) -> Html {
|
||||||
|
let selected_image = use_state_eq(|| 0);
|
||||||
|
let select_next = selected_image.setter();
|
||||||
|
let select_prev = selected_image.setter();
|
||||||
|
|
||||||
|
let has_next = props.infinite || *selected_image < props.images.len() - 1;
|
||||||
|
let has_prev = props.infinite || *selected_image > 0;
|
||||||
|
let next = (*selected_image + props.images.len() + 1) % props.images.len();
|
||||||
|
let prev = (*selected_image + props.images.len() - 1) % props.images.len();
|
||||||
|
let onclose = props.onclose.clone();
|
||||||
|
|
||||||
|
html! {
|
||||||
|
<ImageViewer
|
||||||
|
image={props.images[*selected_image].clone()}
|
||||||
|
open={props.open && !props.images.is_empty()}
|
||||||
|
onclose={Callback::from(move |_| onclose.emit(()))}
|
||||||
|
has_next={has_next}
|
||||||
|
has_prev={has_prev}
|
||||||
|
onnext={Callback::from(move |_| if has_next {
|
||||||
|
select_next.set(next);
|
||||||
|
})}
|
||||||
|
onprev={Callback::from(move |_| if has_prev {
|
||||||
|
select_prev.set(prev);
|
||||||
|
})}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
pub enum DefaultImageViewerMessage {
|
||||||
|
Next,
|
||||||
|
Prev
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct DefaultImageViewer {
|
||||||
|
current: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Component for DefaultImageViewer {
|
||||||
|
type Message = DefaultImageViewerMessage;
|
||||||
|
|
||||||
|
type Properties = DefaultImageViewerProps;
|
||||||
|
|
||||||
|
fn create(_ctx: &yew::Context<Self>) -> Self {
|
||||||
|
Self { current: 0 }
|
||||||
|
}
|
||||||
|
|
||||||
|
fn view(&self, ctx: &yew::Context<Self>) -> Html {
|
||||||
|
let DefaultImageViewerProps {
|
||||||
|
images,
|
||||||
|
open,
|
||||||
|
infinite,
|
||||||
|
onclose,
|
||||||
|
} = ctx.props();
|
||||||
|
|
||||||
|
let has_next = *infinite || self.current < images.len() - 1;
|
||||||
|
let has_prev = *infinite || self.current > 0;
|
||||||
|
let onclose = onclose.clone();
|
||||||
|
|
||||||
|
html! {
|
||||||
|
<ImageViewer
|
||||||
|
image={images[self.current].clone()}
|
||||||
|
open={*open && !images.is_empty()}
|
||||||
|
onclose={Callback::from(move |_| onclose.emit(()))}
|
||||||
|
has_next={has_next}
|
||||||
|
has_prev={has_prev}
|
||||||
|
onnext={if has_next { ctx.link().callback(|_| DefaultImageViewerMessage::Next)} else { Default::default() }}
|
||||||
|
onprev={if has_prev { ctx.link().callback(|_| DefaultImageViewerMessage::Prev)} else { Default::default() }}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn update(&mut self, ctx: &yew::Context<Self>, msg: Self::Message) -> bool {
|
||||||
|
let images = &ctx.props().images;
|
||||||
|
|
||||||
|
self.current = match msg {
|
||||||
|
DefaultImageViewerMessage::Next => {
|
||||||
|
if self.current == images.len() - 1 { 0 } else { self.current + 1 }
|
||||||
|
},
|
||||||
|
DefaultImageViewerMessage::Prev => {
|
||||||
|
if self.current == 0 { images.len() - 1 } else { self.current - 1 }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
images.len() > 1
|
||||||
|
}
|
||||||
|
|
||||||
|
fn changed(&mut self, ctx: &yew::Context<Self>, old_props: &Self::Properties) -> bool {
|
||||||
|
ctx.props() != old_props
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Properties, PartialEq)]
|
||||||
|
pub struct ImageViewerProps {
|
||||||
|
pub image: Option<ImageDescription>,
|
||||||
|
pub open: bool,
|
||||||
|
#[prop_or_default]
|
||||||
|
pub has_prev: bool,
|
||||||
|
#[prop_or_default]
|
||||||
|
pub has_next: bool,
|
||||||
|
pub onclose: Callback<()>,
|
||||||
|
pub onnext: Callback<MouseEvent>,
|
||||||
|
pub onprev: Callback<MouseEvent>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn ImageViewer(props: &ImageViewerProps) -> Html {
|
||||||
|
let onclose = props.onclose.clone();
|
||||||
|
html! {
|
||||||
|
<>
|
||||||
|
<Overlay
|
||||||
|
class="overlay-gallery"
|
||||||
|
open={props.open && props.image.is_some()}
|
||||||
|
onclick={Callback::from(move |_| onclose.emit(()))}>
|
||||||
|
<>
|
||||||
|
<div></div>
|
||||||
|
{
|
||||||
|
if let Some(image_desc) = props.image.as_ref() {
|
||||||
|
let onclose = props.onclose.clone();
|
||||||
|
Some(html! {
|
||||||
|
<>
|
||||||
|
<div>
|
||||||
|
<img
|
||||||
|
onclick={Callback::from(move |_| onclose.emit(()))}
|
||||||
|
src={
|
||||||
|
if image_desc.link.starts_with("https?://") {
|
||||||
|
image_desc.link.to_string()
|
||||||
|
} else {
|
||||||
|
format!("/img/{}", image_desc.link)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
{
|
||||||
|
if let Some(description) = &image_desc.description {
|
||||||
|
Some(html! {
|
||||||
|
<div>{description}</div>
|
||||||
|
})
|
||||||
|
} else { None }
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{
|
||||||
|
// Controls: Next
|
||||||
|
if props.has_next {
|
||||||
|
Some(html! {
|
||||||
|
<div class="overlay-gallery-next">
|
||||||
|
<FAIcon
|
||||||
|
icon={FontawesomeIcon::ChevronRight}
|
||||||
|
size={FontawesomeSize::XL}
|
||||||
|
onclick={props.onnext.clone()}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
})
|
||||||
|
} else { None }
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
// Controls: Prev
|
||||||
|
if props.has_prev {
|
||||||
|
Some(html! {
|
||||||
|
<div class="overlay-gallery-prev">
|
||||||
|
<FAIcon
|
||||||
|
icon={FontawesomeIcon::ChevronLeft}
|
||||||
|
size={FontawesomeSize::XL}
|
||||||
|
onclick={props.onprev.clone()}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
})
|
||||||
|
} else { None }
|
||||||
|
}
|
||||||
|
</>
|
||||||
|
})
|
||||||
|
} else { None }
|
||||||
|
}
|
||||||
|
</>
|
||||||
|
</Overlay>
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
}
|
7
src/component/mod.rs
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
pub mod actionbar;
|
||||||
|
pub mod image_viewer;
|
||||||
|
pub mod cache;
|
||||||
|
pub mod fa_icon;
|
||||||
|
pub mod overlay;
|
||||||
|
pub mod card;
|
||||||
|
pub mod tag;
|
92
src/component/overlay.rs
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
use web_sys::MouseEvent;
|
||||||
|
use yew::{Html, html, Properties, Children, Classes, classes, Callback, Component};
|
||||||
|
|
||||||
|
#[derive(Properties, PartialEq, Debug)]
|
||||||
|
pub struct OverlayProps {
|
||||||
|
pub open: bool,
|
||||||
|
|
||||||
|
#[prop_or_default]
|
||||||
|
pub noshade: bool,
|
||||||
|
|
||||||
|
#[prop_or_default]
|
||||||
|
pub center: bool,
|
||||||
|
|
||||||
|
#[prop_or_default]
|
||||||
|
pub children: Children,
|
||||||
|
|
||||||
|
#[prop_or_default]
|
||||||
|
pub class: Classes,
|
||||||
|
|
||||||
|
#[prop_or_default]
|
||||||
|
pub onclick: Callback<()>
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub enum OverlayMessage {
|
||||||
|
ClickContent,
|
||||||
|
ClickRoot
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Overlay {
|
||||||
|
should_notify: bool,
|
||||||
|
on_root_click: Callback<MouseEvent>,
|
||||||
|
on_child_click: Callback<MouseEvent>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Component for Overlay {
|
||||||
|
type Message = OverlayMessage;
|
||||||
|
type Properties = OverlayProps;
|
||||||
|
|
||||||
|
fn create(ctx: &yew::Context<Self>) -> Self {
|
||||||
|
Self {
|
||||||
|
should_notify: true,
|
||||||
|
on_root_click: ctx.link().callback(|_| OverlayMessage::ClickRoot),
|
||||||
|
on_child_click: ctx.link().callback(|_| OverlayMessage::ClickContent)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn view(&self, ctx: &yew::Context<Self>) -> Html {
|
||||||
|
let OverlayProps {
|
||||||
|
open,
|
||||||
|
noshade,
|
||||||
|
center,
|
||||||
|
children,
|
||||||
|
class,
|
||||||
|
onclick: _,
|
||||||
|
} = ctx.props();
|
||||||
|
html! {
|
||||||
|
<div
|
||||||
|
onclick={self.on_root_click.clone()}
|
||||||
|
class={format!(
|
||||||
|
"overlay{}{}",
|
||||||
|
if *noshade { "" } else { " shade" },
|
||||||
|
if *open { "" } else { " hidden" }
|
||||||
|
)}>
|
||||||
|
<div
|
||||||
|
onclick={self.on_child_click.clone()}
|
||||||
|
class={classes!(if *center { "center" } else { "" }, class.clone())}>{children.clone()}</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn update(&mut self, ctx: &yew::Context<Self>, msg: Self::Message) -> bool {
|
||||||
|
let mut notified = false;
|
||||||
|
match msg {
|
||||||
|
OverlayMessage::ClickContent => self.should_notify = false,
|
||||||
|
OverlayMessage::ClickRoot => {
|
||||||
|
if self.should_notify {
|
||||||
|
ctx.props().onclick.emit(());
|
||||||
|
notified = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.should_notify = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
notified
|
||||||
|
}
|
||||||
|
|
||||||
|
fn changed(&mut self, ctx: &yew::Context<Self>, old_props: &Self::Properties) -> bool {
|
||||||
|
ctx.props() != old_props
|
||||||
|
}
|
||||||
|
}
|
35
src/component/tag.rs
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
use yew::{function_component, Properties, Html, html, classes};
|
||||||
|
|
||||||
|
#[derive(PartialEq, Default)]
|
||||||
|
pub enum TagColor {
|
||||||
|
#[default]
|
||||||
|
Blue,
|
||||||
|
Red,
|
||||||
|
Orange,
|
||||||
|
Green
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TagColor {
|
||||||
|
fn as_class(&self) -> &'static str {
|
||||||
|
match self {
|
||||||
|
Self::Blue => "blue",
|
||||||
|
Self::Red => "red",
|
||||||
|
Self::Orange => "orange",
|
||||||
|
Self::Green => "green",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Properties, PartialEq)]
|
||||||
|
pub struct TagProps {
|
||||||
|
#[prop_or_default]
|
||||||
|
pub color: TagColor,
|
||||||
|
pub text: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn Tag(props: &TagProps) -> Html {
|
||||||
|
html! {
|
||||||
|
<span class={classes!("tag", props.color.as_class())}><span>{props.text.clone()}</span></span>
|
||||||
|
}
|
||||||
|
}
|
22
src/lib.rs
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
mod util;
|
||||||
|
mod app;
|
||||||
|
mod theme;
|
||||||
|
mod component;
|
||||||
|
mod page;
|
||||||
|
|
||||||
|
use wasm_bindgen::prelude::*;
|
||||||
|
|
||||||
|
// When the `wee_alloc` feature is enabled, use `wee_alloc` as the global allocator.
|
||||||
|
// #[cfg(feature = "wee_alloc")]
|
||||||
|
#[global_allocator]
|
||||||
|
static ALLOC: wee_alloc::WeeAlloc = wee_alloc::WeeAlloc::INIT;
|
||||||
|
|
||||||
|
#[wasm_bindgen(start)]
|
||||||
|
pub fn run_app() -> Result<(), JsValue> {
|
||||||
|
#[cfg(feature = "console_error_panic_hook")]
|
||||||
|
console_error_panic_hook::set_once();
|
||||||
|
|
||||||
|
yew::Renderer::<app::AppRoot>::new().render();
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
8
src/page/contact.rs
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
use yew::{function_component, html, Html};
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn Contact() -> Html {
|
||||||
|
html! {
|
||||||
|
<>{"Contact"}</>
|
||||||
|
}
|
||||||
|
}
|
63
src/page/gallery.rs
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
use web_sys::MouseEvent;
|
||||||
|
use yew::{function_component, html, Html, Callback, use_state_eq};
|
||||||
|
|
||||||
|
use crate::component::image_viewer::{ImageDescription, ImageViewer};
|
||||||
|
|
||||||
|
const GALLERY: [&str; 13] = [
|
||||||
|
"gallery-1.jpg",
|
||||||
|
"gallery-2.jpg",
|
||||||
|
"gallery-3.jpg",
|
||||||
|
"gallery-4.jpg",
|
||||||
|
"gallery-5.jpg",
|
||||||
|
"gallery-6.jpg",
|
||||||
|
"gallery-7.jpg",
|
||||||
|
"gallery-8.jpg",
|
||||||
|
"gallery-9.png",
|
||||||
|
"gallery-10.jpg",
|
||||||
|
"gallery-11.jpg",
|
||||||
|
"gallery-12.jpg",
|
||||||
|
"gallery-13.jpg",
|
||||||
|
];
|
||||||
|
|
||||||
|
fn gallery_entry(link: &&str, onclick: Callback<MouseEvent>) -> Html {
|
||||||
|
html! {
|
||||||
|
<div><img src={format!("/img/{link}")} onclick={onclick}/></div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn Gallery() -> Html {
|
||||||
|
let images = GALLERY.iter().map(|it| ImageDescription::new(it.to_string(), it)).collect::<Vec<ImageDescription>>();
|
||||||
|
let is_open = use_state_eq(|| false);
|
||||||
|
let selected_image = use_state_eq(|| 0);
|
||||||
|
let select_next = selected_image.setter();
|
||||||
|
let select_prev = selected_image.setter();
|
||||||
|
let next = (*selected_image + images.len() + 1) % images.len();
|
||||||
|
let prev = (*selected_image + images.len() - 1) % images.len();
|
||||||
|
|
||||||
|
html! {
|
||||||
|
<>
|
||||||
|
<div class="image-gallery">
|
||||||
|
{
|
||||||
|
GALLERY.iter().enumerate().map(|(index, img_str)| {
|
||||||
|
let select = selected_image.clone();
|
||||||
|
let open = is_open.clone();
|
||||||
|
gallery_entry(img_str, Callback::from(move |_| {
|
||||||
|
select.set(index);
|
||||||
|
open.set(true);
|
||||||
|
}))
|
||||||
|
}).collect::<Html>()
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
<ImageViewer
|
||||||
|
image={images[*selected_image].clone()}
|
||||||
|
open={*is_open && !images.is_empty()}
|
||||||
|
onclose={Callback::from(move |_| is_open.set(false))}
|
||||||
|
has_next=true
|
||||||
|
has_prev=true
|
||||||
|
onnext={Callback::from(move |_| select_next.set(next))}
|
||||||
|
onprev={Callback::from(move |_| select_prev.set(prev))}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
}
|
235
src/page/home.rs
Normal file
@ -0,0 +1,235 @@
|
|||||||
|
use comrak::{markdown_to_html, ComrakOptions};
|
||||||
|
use gloo::utils::document;
|
||||||
|
use gloo_net::http::Request;
|
||||||
|
use serde::Deserialize;
|
||||||
|
use yew::{function_component, html, Html, UseStateHandle, use_effect_with_deps, Properties, Children, use_context, Callback, use_state_eq};
|
||||||
|
use crate::{util::log, theme::{ThemeContext, ThemeState}, component::{card::Card, tag::{Tag, TagColor}}};
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
enum TagType {
|
||||||
|
NaturalLanguage, CodeLanguage, Interest
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq)]
|
||||||
|
struct ImageResource {
|
||||||
|
source: String,
|
||||||
|
clickable: bool
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ImageResource {
|
||||||
|
pub fn new_link(link: String, clickable: bool) -> Self {
|
||||||
|
Self {
|
||||||
|
source: link,
|
||||||
|
clickable
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq)]
|
||||||
|
struct DescribedImage {
|
||||||
|
pub image: ImageResource,
|
||||||
|
pub description: &'static str
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Properties, PartialEq)]
|
||||||
|
struct HomeCardProps {
|
||||||
|
#[prop_or_default]
|
||||||
|
pub image: Option<DescribedImage>,
|
||||||
|
pub children: Children
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, PartialEq, Deserialize)]
|
||||||
|
struct GithubEntry {
|
||||||
|
link: String,
|
||||||
|
title: String,
|
||||||
|
description: String
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn Home() -> Html {
|
||||||
|
html! {
|
||||||
|
<>
|
||||||
|
<HomeTitle />
|
||||||
|
<hr />
|
||||||
|
<div class="home-content">
|
||||||
|
<Profile />
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
fn HomeTitle() -> Html {
|
||||||
|
html! {
|
||||||
|
<div class="home-title">
|
||||||
|
<h2>{"Gabriel Tofvesson"}</h2>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_text_resource(file: String, on_result: impl (FnOnce(String) -> ()) + 'static) {
|
||||||
|
wasm_bindgen_futures::spawn_local(async move {
|
||||||
|
log(&format!("Fetching {file}"));
|
||||||
|
let response = Request::get(file.as_str()).send().await;
|
||||||
|
if let Ok(response) = response {
|
||||||
|
if let Ok(text) = response.text().await {
|
||||||
|
on_result(text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_json_resource<T>(file: &'static str, on_result: impl FnOnce(Vec<T>) -> () + 'static) where T: for<'a> Deserialize<'a> {
|
||||||
|
wasm_bindgen_futures::spawn_local(async move {
|
||||||
|
log(&format!("Fetching {file}"));
|
||||||
|
let response = Request::get(file).send().await;
|
||||||
|
|
||||||
|
if let Ok(response) = response {
|
||||||
|
if let Ok(value) = response.json::<Vec<T>>().await {
|
||||||
|
on_result(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
fn ProfileTags() -> Html {
|
||||||
|
let natural_languages: UseStateHandle<Vec<String>> = use_state_eq(|| vec![]);
|
||||||
|
let code_languages: UseStateHandle<Vec<String>> = use_state_eq(|| vec![]);
|
||||||
|
let interests: UseStateHandle<Vec<String>> = use_state_eq(|| vec![]);
|
||||||
|
|
||||||
|
|
||||||
|
// TODO: Cache results
|
||||||
|
{
|
||||||
|
let natural_languages = natural_languages.setter();
|
||||||
|
use_effect_with_deps(
|
||||||
|
move |_| get_json_resource(
|
||||||
|
"/res/languages.json",
|
||||||
|
move |it| natural_languages.set(it)
|
||||||
|
),
|
||||||
|
()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
let code_languages = code_languages.setter();
|
||||||
|
use_effect_with_deps(
|
||||||
|
move |_| get_json_resource(
|
||||||
|
"/res/code.json",
|
||||||
|
move |it| code_languages.set(it)
|
||||||
|
),
|
||||||
|
()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
let interests = interests.setter();
|
||||||
|
use_effect_with_deps(
|
||||||
|
move |_| get_json_resource(
|
||||||
|
"/res/interests.json",
|
||||||
|
move |it| interests.set(it)
|
||||||
|
),
|
||||||
|
()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
let tags = vec![
|
||||||
|
natural_languages.iter().map(|it| (it, TagType::NaturalLanguage)).collect::<Vec<(&String, TagType)>>(),
|
||||||
|
code_languages.iter().map(|it| (it, TagType::CodeLanguage)).collect::<Vec<(&String, TagType)>>(),
|
||||||
|
interests.iter().map(|it| (it, TagType::Interest)).collect::<Vec<(&String, TagType)>>()
|
||||||
|
].into_iter().flatten().map(|(tag, tag_type)| {
|
||||||
|
html! {
|
||||||
|
<Tag
|
||||||
|
text={tag.clone()}
|
||||||
|
color={
|
||||||
|
match tag_type {
|
||||||
|
TagType::NaturalLanguage => TagColor::Blue,
|
||||||
|
TagType::CodeLanguage => TagColor::Orange,
|
||||||
|
TagType::Interest => TagColor::Green
|
||||||
|
}
|
||||||
|
}/>
|
||||||
|
}
|
||||||
|
}).collect::<Html>();
|
||||||
|
|
||||||
|
html! {
|
||||||
|
<div class="profiletags">{tags}</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
fn Profile() -> Html {
|
||||||
|
let profile_text = use_state_eq(|| "".to_owned());
|
||||||
|
{
|
||||||
|
let profile_text = profile_text.setter();
|
||||||
|
use_effect_with_deps(move |_| {
|
||||||
|
get_text_resource("/res/profile.md".to_owned(), move |text| profile_text.set(markdown_to_html(&text, &ComrakOptions::default())));
|
||||||
|
}, ());
|
||||||
|
}
|
||||||
|
|
||||||
|
html! {
|
||||||
|
<HomeCard image={DescribedImage{ image: ImageResource::new_link("profile.jpg".to_owned(), true), description: "About me" }}>
|
||||||
|
<ProfileTags />
|
||||||
|
{Html::from_html_unchecked(profile_text.to_string().into())}
|
||||||
|
</HomeCard>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
fn HomeCard(props: &HomeCardProps) -> Html {
|
||||||
|
html! {
|
||||||
|
<Card class="home-card">
|
||||||
|
{
|
||||||
|
if let Some(image) = &props.image {
|
||||||
|
html! {
|
||||||
|
<div class="home-tag">
|
||||||
|
<img
|
||||||
|
class="home-image circle"
|
||||||
|
src={if image.image.source.starts_with("https?://") {
|
||||||
|
image.image.source.to_string()
|
||||||
|
} else {
|
||||||
|
format!("/img/{}", image.image.source)
|
||||||
|
}} />
|
||||||
|
<b>{image.description}</b>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
} else { html! {} }
|
||||||
|
}
|
||||||
|
<div class="home-info">
|
||||||
|
{props.children.clone()}
|
||||||
|
</div>
|
||||||
|
</Card>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
fn Github() -> Html {
|
||||||
|
let github_entries: UseStateHandle<Vec<GithubEntry>> = use_state_eq(|| vec![]);
|
||||||
|
{
|
||||||
|
let github_entries = github_entries.clone();
|
||||||
|
use_effect_with_deps(move |_| get_json_resource("/res/github.json", move |it| github_entries.set(it)), ());
|
||||||
|
}
|
||||||
|
|
||||||
|
let theme_state = use_context::<ThemeContext>().expect("Theme context");
|
||||||
|
html! {
|
||||||
|
<HomeCard image={ DescribedImage {
|
||||||
|
image: ImageResource::new_link(format!("github/github-mark{}.png", if theme_state.theme == ThemeState::Dark { "-white" } else { "" }), false),
|
||||||
|
description: "GitHub"
|
||||||
|
} }>
|
||||||
|
{
|
||||||
|
github_entries.iter().map(|it| {
|
||||||
|
let link = it.link.clone();
|
||||||
|
html! {
|
||||||
|
<Card onclick={Callback::from(move |_| {
|
||||||
|
if let Err(_) = document().location().unwrap().set_href(link.as_str()) {
|
||||||
|
log("Couldn't change href");
|
||||||
|
}
|
||||||
|
})}>
|
||||||
|
<h3>{it.title.clone()}</h3>
|
||||||
|
<p>{it.description.clone()}</p>
|
||||||
|
</Card>
|
||||||
|
}
|
||||||
|
}).collect::<Html>()
|
||||||
|
}
|
||||||
|
</HomeCard>
|
||||||
|
}
|
||||||
|
}
|
69
src/page/mod.rs
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
use yew::prelude::*;
|
||||||
|
use yew_router::prelude::*;
|
||||||
|
|
||||||
|
mod home;
|
||||||
|
mod projects;
|
||||||
|
mod contact;
|
||||||
|
mod socials;
|
||||||
|
mod gallery;
|
||||||
|
|
||||||
|
#[derive(PartialEq, Copy, Clone, Routable)]
|
||||||
|
pub enum Page {
|
||||||
|
#[at("/")]
|
||||||
|
Home,
|
||||||
|
|
||||||
|
#[at("/projects")]
|
||||||
|
Projects,
|
||||||
|
|
||||||
|
#[at("/socials")]
|
||||||
|
Socials,
|
||||||
|
|
||||||
|
#[at("/photos")]
|
||||||
|
Gallery,
|
||||||
|
|
||||||
|
#[at("/contacts")]
|
||||||
|
Contact,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn switch(page: Page) -> Html {
|
||||||
|
match page {
|
||||||
|
Page::Home => html!{<home::Home />},
|
||||||
|
Page::Projects => html!{<projects::Projects />},
|
||||||
|
Page::Socials => html!{<socials::Socials />},
|
||||||
|
Page::Gallery => html!{<gallery::Gallery />},
|
||||||
|
Page::Contact => html!{<contact::Contact />},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Properties, PartialEq)]
|
||||||
|
pub struct PageRouterProps {
|
||||||
|
pub children: Children
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn PageRouter(props: &PageRouterProps) -> Html {
|
||||||
|
html! {
|
||||||
|
<BrowserRouter>
|
||||||
|
{props.children.clone()}
|
||||||
|
</BrowserRouter>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn Pages() -> Html {
|
||||||
|
html! {
|
||||||
|
<Switch<Page> render={switch} />
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Page {
|
||||||
|
pub fn name(&self) -> &'static str{
|
||||||
|
match self {
|
||||||
|
Page::Home => "Home",
|
||||||
|
Page::Projects => "Projects",
|
||||||
|
Page::Socials => "Social Media",
|
||||||
|
Page::Gallery => "Gallery",
|
||||||
|
Page::Contact => "Contact"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
8
src/page/projects.rs
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
use yew::{function_component, html, Html};
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn Projects() -> Html {
|
||||||
|
html! {
|
||||||
|
<>{"Projects"}</>
|
||||||
|
}
|
||||||
|
}
|
8
src/page/socials.rs
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
use yew::{function_component, html, Html};
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn Socials() -> Html {
|
||||||
|
html! {
|
||||||
|
<>{"Social Media"}</>
|
||||||
|
}
|
||||||
|
}
|
95
src/theme.rs
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
use std::rc::Rc;
|
||||||
|
|
||||||
|
use gloo::utils::document;
|
||||||
|
use yew::{Children, Properties, Html, html};
|
||||||
|
use yew::prelude::*;
|
||||||
|
use crate::util::log;
|
||||||
|
|
||||||
|
fn set_global_dark(is_dark: bool) {
|
||||||
|
let root = document().get_elements_by_tag_name("html").get_with_index(0).expect("Root html tag");
|
||||||
|
|
||||||
|
if is_dark {
|
||||||
|
if let Err(_) = root.set_attribute("data-theme", "dark") {
|
||||||
|
log("Couldn't set attribute 'data-theme' on root");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if let Err(_) = root.remove_attribute("data-theme") {
|
||||||
|
log("Couldn't remove attribute 'data-theme' from root");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, PartialEq, Eq)]
|
||||||
|
pub struct ThemeCtx {
|
||||||
|
pub theme: ThemeState
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Reducible for ThemeCtx {
|
||||||
|
type Action = ThemeMsg;
|
||||||
|
|
||||||
|
fn reduce(self: Rc<Self>, action: Self::Action) -> Rc<Self> {
|
||||||
|
let ctx: Rc<ThemeCtx> = ThemeCtx {
|
||||||
|
theme: match action {
|
||||||
|
//ThemeMsg::Dark => ThemeState::Dark,
|
||||||
|
//ThemeMsg::Light => ThemeState::Light,
|
||||||
|
ThemeMsg::Toggle => match self.theme {
|
||||||
|
ThemeState::Dark => ThemeState::Light,
|
||||||
|
ThemeState::Light => ThemeState::Dark
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}.into();
|
||||||
|
|
||||||
|
set_global_dark(
|
||||||
|
match &ctx.theme {
|
||||||
|
ThemeState::Dark => true,
|
||||||
|
ThemeState::Light => false
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
ctx
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Eq, PartialOrd, Ord, Clone, Debug)]
|
||||||
|
pub enum ThemeState {
|
||||||
|
Dark, Light
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Eq, PartialOrd, Ord, Clone, Debug)]
|
||||||
|
pub enum ThemeMsg {
|
||||||
|
//Dark,
|
||||||
|
//Light,
|
||||||
|
Toggle
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Properties, PartialEq, Debug)]
|
||||||
|
pub struct ThemeProps {
|
||||||
|
#[prop_or_default]
|
||||||
|
pub children: Children
|
||||||
|
}
|
||||||
|
|
||||||
|
pub type ThemeContext = UseReducerHandle<ThemeCtx>;
|
||||||
|
|
||||||
|
#[function_component]
|
||||||
|
pub fn ThemeProvider(props: &ThemeProps) -> Html {
|
||||||
|
let theme = use_reducer(|| {
|
||||||
|
let dark_mode = web_sys::window()
|
||||||
|
.and_then(|x| x.match_media("(prefers-color-scheme: dark)").ok().flatten())
|
||||||
|
.map(|x| x.matches())
|
||||||
|
.unwrap_or(true);
|
||||||
|
set_global_dark(dark_mode);
|
||||||
|
ThemeCtx {
|
||||||
|
theme: if dark_mode {
|
||||||
|
ThemeState::Dark
|
||||||
|
} else {
|
||||||
|
ThemeState::Light
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
html! {
|
||||||
|
<ContextProvider<ThemeContext> context={theme}>
|
||||||
|
{props.children.clone()}
|
||||||
|
</ContextProvider<ThemeContext>>
|
||||||
|
}
|
||||||
|
}
|
7
src/util.rs
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
use wasm_bindgen::prelude::wasm_bindgen;
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
extern "C" {
|
||||||
|
#[wasm_bindgen(js_namespace = console)]
|
||||||
|
pub fn log(message: &str);
|
||||||
|
}
|
Before Width: | Height: | Size: 298 KiB After Width: | Height: | Size: 298 KiB |
Before Width: | Height: | Size: 393 KiB After Width: | Height: | Size: 393 KiB |
Before Width: | Height: | Size: 172 KiB After Width: | Height: | Size: 172 KiB |
Before Width: | Height: | Size: 513 KiB After Width: | Height: | Size: 513 KiB |
Before Width: | Height: | Size: 149 KiB After Width: | Height: | Size: 149 KiB |
Before Width: | Height: | Size: 311 KiB After Width: | Height: | Size: 311 KiB |
Before Width: | Height: | Size: 254 KiB After Width: | Height: | Size: 254 KiB |
Before Width: | Height: | Size: 291 KiB After Width: | Height: | Size: 291 KiB |
Before Width: | Height: | Size: 1.0 MiB After Width: | Height: | Size: 1.0 MiB |
Before Width: | Height: | Size: 4.8 MiB After Width: | Height: | Size: 4.8 MiB |
Before Width: | Height: | Size: 6.5 MiB After Width: | Height: | Size: 6.5 MiB |
Before Width: | Height: | Size: 4.2 MiB After Width: | Height: | Size: 4.2 MiB |
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.5 MiB |
Before Width: | Height: | Size: 4.7 KiB After Width: | Height: | Size: 4.7 KiB |
Before Width: | Height: | Size: 960 B After Width: | Height: | Size: 960 B |
Before Width: | Height: | Size: 6.2 KiB After Width: | Height: | Size: 6.2 KiB |
Before Width: | Height: | Size: 963 B After Width: | Height: | Size: 963 B |
Before Width: | Height: | Size: 779 KiB After Width: | Height: | Size: 779 KiB |
0
static/style/gallery.css
Normal file
10
xtask/Cargo.toml
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
[package]
|
||||||
|
name = "xtask"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
xtask-wasm = "0.1"
|
||||||
|
yewprint-css = "0.4.0"
|
44
xtask/src/main.rs
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
use std::{path::Path, process};
|
||||||
|
use xtask_wasm::{anyhow::Result, clap, DistResult};
|
||||||
|
|
||||||
|
#[derive(clap::Parser)]
|
||||||
|
enum Cli {
|
||||||
|
Dist(xtask_wasm::Dist),
|
||||||
|
Watch(xtask_wasm::Watch),
|
||||||
|
Start(xtask_wasm::DevServer),
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() -> Result<()> {
|
||||||
|
let cli: Cli = clap::Parser::parse();
|
||||||
|
|
||||||
|
match cli {
|
||||||
|
Cli::Dist(args) => {
|
||||||
|
let DistResult { dist_dir, .. } =
|
||||||
|
args.static_dir_path("static").run("yewprint-app")?;
|
||||||
|
|
||||||
|
download_css(&dist_dir)?;
|
||||||
|
}
|
||||||
|
Cli::Watch(args) => {
|
||||||
|
let mut command = process::Command::new("cargo");
|
||||||
|
command.arg("check");
|
||||||
|
|
||||||
|
args.run(command)?;
|
||||||
|
}
|
||||||
|
Cli::Start(args) => {
|
||||||
|
args.arg("dist")
|
||||||
|
.start(xtask_wasm::default_dist_dir(false))?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn download_css(path: &Path) -> Result<()> {
|
||||||
|
let css_path = path.join("blueprint.css");
|
||||||
|
|
||||||
|
if !css_path.exists() {
|
||||||
|
yewprint_css::download_css(&css_path)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|