new: yew-based front-end

This commit is contained in:
Matthew Ryan Dillon 2021-12-09 12:08:28 -07:00
parent abadb2ec7d
commit d64938e8e5
20 changed files with 680 additions and 6263 deletions

View file

@ -1,70 +0,0 @@
name: ci
on:
push:
branches: [ main ]
pull_request:
branches: [ main ]
env:
CARGO_TERM_COLOR: always
jobs:
test:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- uses: actions/cache@v2
with:
path: |
~/.cargo/registry
~/.cargo/git
target
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
- name: lint
run: cargo fmt -- --check
- name: install wasm-pack
run: curl https://rustwasm.github.io/wasm-pack/installer/init.sh -sSf | sh
- run: cargo check
- run: cargo test
- run: wasm-pack test --headless --firefox
- run: wasm-pack test --headless --chrome
build:
runs-on: ubuntu-latest
needs: test
steps:
- uses: actions/checkout@v2
- uses: actions/cache@v2
with:
path: |
~/.cargo/registry
~/.cargo/git
target
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
- name: install wasm-pack
run: curl https://rustwasm.github.io/wasm-pack/installer/init.sh -sSf | sh
- name: build
run: |
wasm-pack build
cd www
npm install
npm run build
- name: deploy
uses: peaceiris/actions-gh-pages@v3
if: github.ref == 'refs/heads/main'
with:
github_token: ${{ secrets.GITHUB_TOKEN }}
publish_dir: www/dist
cname: gpx.thermokar.st

1
.gitignore vendored
View file

@ -1,5 +1,6 @@
/target
**/*.rs.bk
bin/
dist/
pkg/
wasm-pack.log

316
Cargo.lock generated
View file

@ -1,5 +1,7 @@
# This file is automatically @generated by Cargo.
# It is not intended for manual editing.
version = 3
[[package]]
name = "addr2line"
version = "0.14.0"
@ -41,6 +43,12 @@ dependencies = [
"rustc-demangle",
]
[[package]]
name = "boolinator"
version = "2.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "cfa8873f51c92e232f9bac4065cddef41b714152812bfc5f7672ba16d6ef8cd9"
[[package]]
name = "bumpalo"
version = "3.4.0"
@ -106,6 +114,113 @@ version = "0.23.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f6503fe142514ca4799d4c26297c4248239fe8838d827db6bd6065c6ed29a6ce"
[[package]]
name = "gloo"
version = "0.4.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "23947965eee55e3e97a5cd142dd4c10631cc349b48cecca0ed230fd296f568cd"
dependencies = [
"gloo-console",
"gloo-dialogs",
"gloo-events",
"gloo-file",
"gloo-render",
"gloo-storage",
"gloo-timers",
"gloo-utils",
]
[[package]]
name = "gloo-console"
version = "0.2.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3907f786f65bbb4f419e918b0c5674175ef1c231ecda93b2dbd65fd1e8882637"
dependencies = [
"js-sys",
"serde",
"wasm-bindgen",
"web-sys",
]
[[package]]
name = "gloo-dialogs"
version = "0.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "4ffb557a2ea2ed283f1334423d303a336fad55fb8572d51ae488f828b1464b40"
dependencies = [
"wasm-bindgen",
"web-sys",
]
[[package]]
name = "gloo-events"
version = "0.1.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "088514ec8ef284891c762c88a66b639b3a730134714692ee31829765c5bc814f"
dependencies = [
"wasm-bindgen",
"web-sys",
]
[[package]]
name = "gloo-file"
version = "0.2.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d31ba1f51868ae10a0b665c6dccd5ed967486e7c17055d1c889596ee983be493"
dependencies = [
"gloo-events",
"js-sys",
"wasm-bindgen",
"web-sys",
]
[[package]]
name = "gloo-render"
version = "0.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "5b4cda6e149df3bb4a3c6a343873903e5bcc2448a9877d61bb8274806ad67f6e"
dependencies = [
"wasm-bindgen",
"web-sys",
]
[[package]]
name = "gloo-storage"
version = "0.2.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1b5057761927af1b1929d02b1f49cf83553dd347a473ee7c8bb08420f2673ffc"
dependencies = [
"gloo-utils",
"js-sys",
"serde",
"serde_json",
"thiserror",
"wasm-bindgen",
"web-sys",
]
[[package]]
name = "gloo-timers"
version = "0.2.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "47204a46aaff920a1ea58b11d03dec6f704287d27561724a4631e450654a891f"
dependencies = [
"js-sys",
"wasm-bindgen",
"web-sys",
]
[[package]]
name = "gloo-utils"
version = "0.1.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3d77d28d9a6f7c384d9e40293fa11f05558bf928a993208e12528ee6633cb415"
dependencies = [
"js-sys",
"wasm-bindgen",
"web-sys",
]
[[package]]
name = "gpx"
version = "0.8.1"
@ -124,12 +239,37 @@ name = "gpx-web-utils"
version = "0.0.1"
dependencies = [
"console_error_panic_hook",
"gloo-file",
"gpx",
"js-sys",
"wasm-bindgen",
"wasm-bindgen-test",
"web-sys",
"wee_alloc",
"yew",
]
[[package]]
name = "hashbrown"
version = "0.11.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ab5ef0d4909ef3724cc8cce6ccc8572c5c817592e9285f5464f8e86f8bd3726e"
[[package]]
name = "indexmap"
version = "1.7.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "bc633605454125dec4b66843673f01c7df2b89479b32e0ed634e43a91cff62a5"
dependencies = [
"autocfg",
"hashbrown",
]
[[package]]
name = "itoa"
version = "0.4.8"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b71991ff56294aa922b450139ee08b3bfc70982c6b2c7562771375cf73542dd4"
[[package]]
name = "js-sys"
version = "0.3.45"
@ -160,6 +300,12 @@ dependencies = [
"cfg-if 0.1.10",
]
[[package]]
name = "memory_units"
version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8452105ba047068f40ff7093dd1d9da90898e63dd61736462e9cdda6a90ad3c3"
[[package]]
name = "miniz_oxide"
version = "0.4.3"
@ -195,6 +341,30 @@ version = "0.22.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8d3b63360ec3cb337817c2dbd47ab4a0f170d285d8e5a2064600f3def1402397"
[[package]]
name = "proc-macro-error"
version = "1.0.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "da25490ff9892aab3fcf7c36f08cfb902dd3e71ca0f9f9517bea02a73a5ce38c"
dependencies = [
"proc-macro-error-attr",
"proc-macro2",
"quote",
"syn",
"version_check",
]
[[package]]
name = "proc-macro-error-attr"
version = "1.0.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a1be40180e52ecc98ad80b184934baf3d0d29f979574e439af5a55274b35f869"
dependencies = [
"proc-macro2",
"quote",
"version_check",
]
[[package]]
name = "proc-macro2"
version = "1.0.24"
@ -220,10 +390,53 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "6e3bad0ee36814ca07d7968269dd4b7ec89ec2da10c4bb613928d3077083c232"
[[package]]
name = "scoped-tls"
version = "1.0.0"
name = "ryu"
version = "1.0.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ea6a9290e3c9cf0f18145ef7ffa62d68ee0bf5fcd651017e586dc7fd5da448c2"
checksum = "71d301d4193d031abdd79ff7e3dd721168a9572ef3fe51a1517aba235bd8f86e"
[[package]]
name = "scoped-tls-hkt"
version = "0.1.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c2e9d7eaddb227e8fbaaa71136ae0e1e913ca159b86c7da82f3e8f0044ad3a63"
[[package]]
name = "serde"
version = "1.0.118"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "06c64263859d87aa2eb554587e2d23183398d617427327cf2b3d0ed8c69e4800"
dependencies = [
"serde_derive",
]
[[package]]
name = "serde_derive"
version = "1.0.118"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c84d3526699cd55261af4b941e4e725444df67aa4f9e6a3564f18030d12672df"
dependencies = [
"proc-macro2",
"quote",
"syn",
]
[[package]]
name = "serde_json"
version = "1.0.72"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d0ffa0837f2dfa6fb90868c2b5468cad482e175f7dad97e7421951e663f2b527"
dependencies = [
"itoa",
"ryu",
"serde",
]
[[package]]
name = "slab"
version = "0.4.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9def91fd1e018fe007022791f865d0ccc9b3a0d5001e01aabb8b40e46000afb5"
[[package]]
name = "syn"
@ -236,6 +449,26 @@ dependencies = [
"unicode-xid",
]
[[package]]
name = "thiserror"
version = "1.0.30"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "854babe52e4df1653706b98fcfc05843010039b406875930a70e4d9644e5c417"
dependencies = [
"thiserror-impl",
]
[[package]]
name = "thiserror-impl"
version = "1.0.30"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "aa32fd3f627f367fe16f893e2597ae3c05020f8bba2666a4e6ea73d377e5714b"
dependencies = [
"proc-macro2",
"quote",
"syn",
]
[[package]]
name = "time"
version = "0.1.44"
@ -253,6 +486,12 @@ version = "0.2.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f7fe0bb3479651439c9112f72b6c505038574c9fbb575ed1bf3b797fa39dd564"
[[package]]
name = "version_check"
version = "0.9.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "5fecdca9a5291cc2b8dcf7dc02453fee791a280f3743cb0905f8822ae463b3fe"
[[package]]
name = "wasi"
version = "0.10.0+wasi-snapshot-preview1"
@ -266,6 +505,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1ac64ead5ea5f05873d7c12b545865ca2b8d28adfc50a49b84770a3a97265d42"
dependencies = [
"cfg-if 0.1.10",
"serde",
"serde_json",
"wasm-bindgen-macro",
]
@ -325,30 +566,6 @@ version = "0.2.68"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1d649a3145108d7d3fbcde896a468d1bd636791823c9921135218ad89be08307"
[[package]]
name = "wasm-bindgen-test"
version = "0.3.18"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "34d1cdc8b98a557f24733d50a1199c4b0635e465eecba9c45b214544da197f64"
dependencies = [
"console_error_panic_hook",
"js-sys",
"scoped-tls",
"wasm-bindgen",
"wasm-bindgen-futures",
"wasm-bindgen-test-macro",
]
[[package]]
name = "wasm-bindgen-test-macro"
version = "0.3.18"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e8fb9c67be7439ee8ab1b7db502a49c05e51e2835b66796c705134d9b8e1a585"
dependencies = [
"proc-macro2",
"quote",
]
[[package]]
name = "web-sys"
version = "0.3.45"
@ -359,6 +576,18 @@ dependencies = [
"wasm-bindgen",
]
[[package]]
name = "wee_alloc"
version = "0.4.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "dbb3b5a6b2bb17cb6ad44a2e68a43e8d2722c997da10e928665c72ec6c0a0b8e"
dependencies = [
"cfg-if 0.1.10",
"libc",
"memory_units",
"winapi",
]
[[package]]
name = "winapi"
version = "0.3.9"
@ -386,3 +615,34 @@ name = "xml-rs"
version = "0.8.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b07db065a5cf61a7e4ba64f29e67db906fb1787316516c4e6e5ff0fea1efcd8a"
[[package]]
name = "yew"
version = "0.18.0"
source = "git+https://github.com/yewstack/yew.git#996bf5b41ac30d7ada14fdc2f419de7659e94613"
dependencies = [
"console_error_panic_hook",
"gloo",
"gloo-utils",
"indexmap",
"js-sys",
"scoped-tls-hkt",
"slab",
"wasm-bindgen",
"wasm-bindgen-futures",
"web-sys",
"yew-macro",
]
[[package]]
name = "yew-macro"
version = "0.18.0"
source = "git+https://github.com/yewstack/yew.git#996bf5b41ac30d7ada14fdc2f419de7659e94613"
dependencies = [
"boolinator",
"lazy_static",
"proc-macro-error",
"proc-macro2",
"quote",
"syn",
]

View file

@ -4,25 +4,25 @@ version = "0.0.1"
authors = ["Matthew Dillon <matthewrdillon@gmail.com>"]
edition = "2018"
description = "just some gpx-related tools that i want to use"
repository = "https://github.com/thermokarst/gpx-web-utils"
repository = "git://pingo.thermokar.st/gpx-web-utils"
license = "MIT"
[lib]
crate-type = ["cdylib", "rlib"]
[features]
default = ["console_error_panic_hook"]
[dependencies]
wasm-bindgen = "0.2.63"
js-sys = "0.3.45"
wasm-bindgen = "0.2"
js-sys = "0.3"
gpx = "0.8.1"
# TODO: still waiting on a public release
# yew = "0.19"
yew = { git = "https://github.com/yewstack/yew.git" }
gloo-file = "0.2"
wee_alloc = "0.4"
console_error_panic_hook = { version = "0.1.6", optional = true }
[dev-dependencies]
wasm-bindgen-test = "0.3.13"
[profile.release]
opt-level = 3
lto = true
[dependencies.web-sys]
version = "0.3"
features = ["File", "Blob", "Element", "MouseEvent", "EventTarget", "Url", "Event"]

View file

@ -1,11 +1,21 @@
# gpx-web-utils
![ci](https://github.com/thermokarst/gpx-web-utils/workflows/ci/badge.svg)
## development
## quickstart
```bash
cargo install --locked trunk
cargo install wasm-bindgen-cli
rustup target add wasm32-unknown-unknown
trunk serve
```
1. install [wasm-pack](https://rustwasm.github.io/wasm-pack/installer/)
2. `wasm-pack build`
3. `cd www`
4. `npm install`
5. `npm run start`
## deployment
```bash
trunk build --release
git switch deploy
rm index-*
cp dist/* .
git commit -am 'foo'
git push <DOKKU INSTANCE> deploy
```

25
index.html Normal file
View file

@ -0,0 +1,25 @@
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8" />
<title>gpx.thermokar.st</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<style>
body {
max-width: 35em;
margin: 0 auto;
line-height: 1.5;
font-family: sans-serif;
font-size: large;
}
</style>
</head>
<body>
<noscript>
<mark>
This page contains webassembly and javascript content, please enable
javascript in your browser to use this tool.
</mark>
</noscript>
</body>
</html>

81
src/app.rs Normal file
View file

@ -0,0 +1,81 @@
use super::loader::Loader;
use web_sys::Node;
use yew::virtual_dom::VNode;
use yew::{function_component, html, Html};
#[function_component(App)]
pub fn app() -> Html {
html! {
<>
<h1>
<a href="/">
{ "gpx.thermokar.st" }
</a>
</h1>
<p>
{ "a client-side tool for merging " }
<a href="https://www.topografix.com/gpx.asp">
{ "gpx files" }
</a>
</p>
<Loader />
<hr/>
<Footer />
</>
}
}
#[function_component(Footer)]
fn footer() -> Html {
let notes = Vec::from([
// note 1
"this has only been tested on GPX files produced by \
<a href=\"https://strava.com\" target=\"_blank\">strava</a> and \
<a href=\"https://garmin.com\" target=\"_blank\">garmin</a>",
// note 2
"all third-party extension info \
<a href=\"https://github.com/georust/gpx/issues/8\" target=\"_blank\">\
is stripped</a>",
// note 3
"if the app breaks, try refreshing the page",
"source (public access): git://pingo.thermokar.st/gpx-web-utils",
// note 4
"source (mirror): \
<a href=\"https://github.com/thermokarst/gpx-web-utils\" target=\"_blank\">\
https://github.com/thermokarst/gpx-web-utils</a>",
]);
html! {
<div>
<ul>
{ for notes.iter().map(|n| inner_html_enabled_li(n)) }
</ul>
<span>
<small>
{ "\u{000A9} matthew ryan dillon, 2021" }
</small>
</span>
</div>
}
}
fn inner_html_enabled_li(data: &str) -> Html {
let li = web_sys::window()
.unwrap()
.document()
.unwrap()
.create_element("li")
.unwrap();
li.set_inner_html(data);
let node = Node::from(li);
VNode::VRef(node)
}

88
src/gpx.rs Normal file
View file

@ -0,0 +1,88 @@
use std::error::Error;
use wasm_bindgen::prelude::*;
use web_sys::{Blob, MouseEvent, Url};
fn join_gpx_files(files: &[String]) -> Result<gpx::Gpx, Box<dyn Error>> {
let mut merged_gpx: gpx::Gpx = Default::default();
let mut merged_track: gpx::Track = gpx::Track::new();
for file in files.iter() {
let buffer = std::io::BufReader::new(file.as_bytes());
let mut parsed_gpx: gpx::Gpx = gpx::read(buffer)?;
// consolidate all track segements into one single track.
for track in parsed_gpx.tracks {
for segment in track.segments {
merged_track.segments.push(segment);
}
}
merged_gpx.waypoints.append(&mut parsed_gpx.waypoints);
}
merged_gpx.tracks.push(merged_track);
let link = gpx::Link {
href: String::from("https://gpx.thermokar.st"),
..Default::default()
};
let author = gpx::Person {
link: Some(link),
..Default::default()
};
let metadata = gpx::Metadata {
name: Some(String::from("merged")),
author: Some(author),
..Default::default()
};
merged_gpx.metadata = Some(metadata);
merged_gpx.version = gpx::GpxVersion::Gpx11;
Ok(merged_gpx)
}
fn write_gpx_to_buffer(gpx: gpx::Gpx) -> Result<js_sys::Array, Box<dyn Error>> {
let mut buffer = Vec::new();
gpx::write(&gpx, &mut buffer)?;
let uint8arr = js_sys::Uint8Array::new(&unsafe { js_sys::Uint8Array::view(&buffer) }.into());
let array = js_sys::Array::new();
array.push(&uint8arr.buffer());
Ok(array)
}
pub fn merge(files: &[String]) -> Result<Blob, Box<dyn Error>> {
let merged: gpx::Gpx = join_gpx_files(files)?;
let out_vec = write_gpx_to_buffer(merged)?;
let result = Blob::new_with_u8_array_sequence(&out_vec).map_err(|e| e.as_string().unwrap())?;
Ok(result)
}
pub fn download(merged: Blob) -> Result<(), Box<dyn Error>> {
let window = web_sys::window().ok_or("no global `window` exists")?;
let document = window
.document()
.ok_or("should have a document on window")?;
let err_handler = |e: JsValue| e.as_string().unwrap();
let anchor_element = document.create_element("a").map_err(err_handler)?;
let url = Url::create_object_url_with_blob(&merged).map_err(err_handler)?;
anchor_element
.set_attribute("href", &url)
.map_err(err_handler)?;
anchor_element
.set_attribute("download", "merged.gpx")
.map_err(err_handler)?;
let event = MouseEvent::new("click").map_err(err_handler)?;
anchor_element.dispatch_event(&event).map_err(err_handler)?;
Ok(())
}
#[wasm_bindgen]
extern "C" {
pub fn alert(s: &str);
}

View file

@ -1,17 +0,0 @@
mod utils;
use wasm_bindgen::prelude::*;
#[cfg(feature = "wee_alloc")]
#[global_allocator]
static ALLOC: wee_alloc::WeeAlloc = wee_alloc::WeeAlloc::INIT;
#[wasm_bindgen]
pub fn merge(files: js_sys::Array) -> wasm_bindgen::JsValue {
utils::set_panic_hook();
let files: Vec<String> = utils::translate_js_to_rust(files);
let merged: gpx::Gpx = utils::join_gpx_files(files);
let out_vec: Vec<u8> = utils::write_gpx_to_buffer(merged);
JsValue::from_str(&String::from_utf8(out_vec).unwrap())
}

155
src/loader.rs Normal file
View file

@ -0,0 +1,155 @@
use std::collections::HashMap;
use gloo_file::callbacks::FileReader;
use gloo_file::File;
use web_sys::{Blob, Event, HtmlInputElement};
use yew::{html, html::TargetCast, Component, Context, Html};
use super::gpx;
pub enum Msg {
FileLoaded(String, String),
StartLoad(Vec<File>),
FilesLoaded,
Download(Blob),
Reset,
}
pub struct Loader {
readers: HashMap<String, FileReader>,
files: Vec<String>,
count: usize,
is_loading: bool,
// This field is to handle resetting the native HTML widget's state on error
field_value: &'static str,
}
impl Component for Loader {
type Message = Msg;
type Properties = ();
fn create(_ctx: &Context<Self>) -> Self {
Self {
readers: HashMap::default(),
files: vec![],
count: 0,
is_loading: false,
field_value: "",
}
}
fn update(&mut self, ctx: &Context<Self>, msg: Self::Message) -> bool {
match msg {
Msg::FileLoaded(filename, data) => {
self.files.push(data);
self.readers.remove(&filename);
if self.files.len() == self.count {
ctx.link().send_message(Msg::FilesLoaded);
}
true
}
Msg::StartLoad(files) => {
self.count = files.len();
if self.count < 2 {
gpx::alert("must load two or more files");
ctx.link().send_message(Msg::Reset);
return true;
}
self.is_loading = true;
for file in files.into_iter() {
let file_name = file.name();
let task = {
let file_name = file_name.clone();
let link = ctx.link().clone();
gloo_file::callbacks::read_as_text(&file, move |res| {
link.send_message(Msg::FileLoaded(
file_name,
res.unwrap_or_else(|e| e.to_string()),
))
})
};
self.readers.insert(file_name, task);
}
true
}
Msg::FilesLoaded => {
let link = ctx.link();
let merged = match gpx::merge(&self.files) {
Ok(result) => result,
Err(err) => {
gpx::alert(&err.to_string());
link.send_message(Msg::Reset);
return true;
}
};
link.send_message(Msg::Download(merged));
true
}
Msg::Download(merged) => {
let link = ctx.link();
match gpx::download(merged) {
Ok(_) => (),
Err(err) => {
gpx::alert(&err.to_string());
link.send_message(Msg::Reset);
return true;
}
}
link.send_message(Msg::Reset);
true
}
Msg::Reset => {
self.readers = HashMap::default();
self.files = vec![];
self.count = 0;
self.is_loading = false;
self.field_value = "";
true
}
}
}
fn view(&self, ctx: &Context<Self>) -> Html {
let cb = move |e: Event| {
let mut result = Vec::new();
let input: HtmlInputElement = e.target_unchecked_into();
if let Some(files) = input.files() {
let files = js_sys::try_iter(&files)
.unwrap()
.unwrap()
.map(|v| web_sys::File::from(v.unwrap()))
.map(File::from);
result.extend(files);
}
Msg::StartLoad(result)
};
html! {
if self.is_loading {
<span><strong>{"processing..."}</strong></span>
} else {
<input
type="file"
value={self.field_value}
multiple=true
onchange={ctx.link().callback(cb)}
/>
}
}
}
}

13
src/main.rs Normal file
View file

@ -0,0 +1,13 @@
extern crate wee_alloc;
mod app;
mod gpx;
mod loader;
// Use `wee_alloc` as the global allocator.
#[global_allocator]
static ALLOC: wee_alloc::WeeAlloc = wee_alloc::WeeAlloc::INIT;
fn main() {
yew::start_app::<app::App>();
}

View file

@ -1,55 +0,0 @@
pub fn set_panic_hook() {
#[cfg(feature = "console_error_panic_hook")]
console_error_panic_hook::set_once();
}
pub fn translate_js_to_rust(files: js_sys::Array) -> Vec<String> {
// https://github.com/rustwasm/wasm-bindgen/issues/111
files.iter().map(|f| f.as_string().unwrap()).collect()
}
pub fn join_gpx_files(files: Vec<String>) -> gpx::Gpx {
let mut merged_gpx: gpx::Gpx = Default::default();
let mut merged_track: gpx::Track = gpx::Track::new();
for file in files.iter() {
let buffer = std::io::BufReader::new(file.as_bytes());
let mut parsed_gpx: gpx::Gpx = gpx::read(buffer).expect("invalid gpx");
// consolidate all track segements into one single track.
for track in parsed_gpx.tracks {
for segment in track.segments {
merged_track.segments.push(segment);
}
}
merged_gpx.waypoints.append(&mut parsed_gpx.waypoints);
}
merged_gpx.tracks.push(merged_track);
let link = gpx::Link {
href: String::from("https://gpx.thermokar.st"),
..Default::default()
};
let author = gpx::Person {
link: Some(link),
..Default::default()
};
let metadata = gpx::Metadata {
name: Some(String::from("merged")),
author: Some(author),
..Default::default()
};
merged_gpx.metadata = Some(metadata);
merged_gpx.version = gpx::GpxVersion::Gpx11;
merged_gpx
}
pub fn write_gpx_to_buffer(gpx: gpx::Gpx) -> Vec<u8> {
let mut buffer = Vec::new();
gpx::write(&gpx, &mut buffer).unwrap();
buffer
}

View file

@ -1,93 +0,0 @@
//! Test suite for the Web and headless browsers.
#![cfg(target_arch = "wasm32")]
extern crate wasm_bindgen_test;
use wasm_bindgen_test::*;
wasm_bindgen_test_configure!(run_in_browser);
#[wasm_bindgen_test]
fn basic_merge() {
// arrange
let array: js_sys::Array = js_sys::Array::new();
let file1 = wasm_bindgen::JsValue::from_str(
"<?xml version='1.0' encoding='utf-8'?>
<gpx version='1.0' encoding='UTF-8'>
<trk>
<name>file1 tracks</name>
<type>1</type>
<trkseg>
<trkpt lat='35.466388' lon='-111.640076'>
<ele>2152.8</ele>
<time>2020-09-27T15:39:27+00:00</time>
</trkpt>
</trkseg>
</trk>
</gpx>",
);
let file2 = wasm_bindgen::JsValue::from_str(
"<?xml version='1.0' encoding='utf-8'?>
<gpx version='1.0' encoding='UTF-8'>
<trk>
<name>file2 tracks</name>
<type>1</type>
<trkseg>
<trkpt lat='35.339854' lon='-111.737165'>
<ele>2556.8</ele>
<time>2020-09-26T19:07:14+00:00</time>
</trkpt>
</trkseg>
</trk>
</gpx>",
);
array.push(&file1);
array.push(&file2);
let exp = wasm_bindgen::JsValue::from_str(
"<?xml version=\"1.0\" encoding=\"utf-8\"?>
<gpx version=\"1.1\" creator=\"https://github.com/georust/gpx\">
<metadata>
<name>merged</name>
<author>
<link href=\"https://gpx.thermokar.st\" />
</author>
</metadata>
<trk>
<name>file1 tracks</name>
<type>1</type>
<trkseg>
<trkpt lat=\"35.466388\" lon=\"-111.640076\">
<ele>2152.8</ele>
<time>2020-09-27T15:39:27+00:00</time>
</trkpt>
</trkseg>
</trk>
<trk>
<name>file2 tracks</name>
<type>1</type>
<trkseg>
<trkpt lat=\"35.339854\" lon=\"-111.737165\">
<ele>2556.8</ele>
<time>2020-09-26T19:07:14+00:00</time>
</trkpt>
</trkseg>
</trk>
<rte />
</gpx>",
);
// act
let obs = gpx_web_utils::merge(array);
// assert
assert_eq!(obs, exp);
}
// TODO: https://github.com/rustwasm/wasm-bindgen/issues/2286
// #[wasm_bindgen_test]
// #[should_panic]
// fn invalid_inputs() {
// let array: js_sys::Array = js_sys::Array::new_with_length(10);
// let obs = gpx_web_utils::merge(array);
// }

2
www/.gitignore vendored
View file

@ -1,2 +0,0 @@
node_modules
dist

5
www/bootstrap.js vendored
View file

@ -1,5 +0,0 @@
// A dependency graph that contains any wasm must all be imported
// asynchronously. This `bootstrap.js` file does the single async import, so
// that no one else needs to worry about it again.
import("./index.js")
.catch(e => console.error("Error importing `index.js`:", e));

View file

@ -1,46 +0,0 @@
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>gpx.thermokar.st</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<style>
body {
max-width: 35em;
margin: 0 auto;
line-height: 1.5;
font-family: sans-serif;
font-size: large;
}
</style>
</head>
<body>
<h1>
<a href="https://gpx.thermokar.st">gpx.thermokar.st</a>
</h1>
<noscript>
<mark>
This page contains webassembly and javascript content, please enable
javascript in your browser.
</mark>
</noscript>
<p>
This client-side tool is for merging
<a href="https://www.topografix.com/gpx.asp">GPX files</a>.
Please note, this has only been tested on GPX files produced by
<a href="https://www.garmin.com">Garmin</a> and
<a href="https://www.strava.com">Strava</a> - your mileage may vary.
</p>
<form>
<input id="gpxInput" type="file" multiple accept="text/gpx,.gpx">
</form>
<hr>
<p>
<small>
<a href="https://github.com/thermokarst/gpx-web-utils">
https://github.com/thermokarst/gpx-web-utils</a>
</small>
</p>
<script src="./bootstrap.js"></script>
</body>
</html>

View file

@ -1,37 +0,0 @@
import * as gpx from "gpx-web-utils";
const inputElement = document.getElementById("gpxInput");
const loadingElement = document.createElement("span");
loadingElement.innerHTML = "<strong>processing...</strong>";
inputElement.value = "";
inputElement.addEventListener("change", readFiles, false);
function readFiles() {
if (inputElement.files.length < 2) { alert("open two or more files"); return; }
inputElement.replaceWith(loadingElement);
const files = Array.from(inputElement.files);
const promises = files.map(f => f.text());
Promise.all(promises).then(gpxes => {
try {
const merged = gpx.merge(gpxes);
writeOutput(merged);
} catch {
alert("there was a problem, please check the console.");
} finally {
inputElement.value = "";
loadingElement.replaceWith(inputElement);
}
});
}
function writeOutput(file) {
const blob = new Blob([file], {type: "text/gpx"});
const anchorElement = document.createElement("a");
anchorElement.href = URL.createObjectURL(blob);
anchorElement.download = "merged.gpx";
anchorElement.click();
URL.revokeObjectURL(anchorElement.href);
}

5842
www/package-lock.json generated

File diff suppressed because it is too large Load diff

View file

@ -1,35 +0,0 @@
{
"name": "gpx-web-utils",
"version": "0.0.1",
"description": "just some gpx-related tools that i want to use.",
"main": "index.js",
"scripts": {
"build": "webpack --config webpack.config.js",
"start": "webpack-dev-server"
},
"repository": {
"type": "git",
"url": "git+https://github.com/thermokarst/gpx-web-utils.git"
},
"keywords": [
"webassembly",
"wasm",
"rust",
"webpack"
],
"author": "Matthew Ryan Dillon <matthewrdillon@gmail.com>",
"license": "MIT",
"bugs": {
"url": "https://github.com/thermokarst/gpx-web-utils/issues"
},
"homepage": "https://github.com/thermokarst/gpx-web-util#readme",
"dependencies": {
"gpx-web-utils": "file:../pkg"
},
"devDependencies": {
"webpack": "^4.29.3",
"webpack-cli": "^3.1.0",
"webpack-dev-server": "^3.1.5",
"copy-webpack-plugin": "^5.0.0"
}
}

View file

@ -1,14 +0,0 @@
const CopyWebpackPlugin = require("copy-webpack-plugin");
const path = require('path');
module.exports = {
entry: "./bootstrap.js",
output: {
path: path.resolve(__dirname, "dist"),
filename: "bootstrap.js",
},
mode: "development",
plugins: [
new CopyWebpackPlugin(['index.html'])
],
};