Compare commits
No commits in common. "main" and "0.3.0" have entirely different histories.
22 changed files with 54 additions and 210 deletions
29
.forgejo/workflows/build.yaml
Normal file
29
.forgejo/workflows/build.yaml
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
on: [push]
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: container
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- uses: pnpm/action-setup@v4
|
||||||
|
name: Install pnpm
|
||||||
|
with:
|
||||||
|
run_install: false
|
||||||
|
|
||||||
|
- name: Install Node.js
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: 20
|
||||||
|
cache: 'pnpm'
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: pnpm install
|
||||||
|
|
||||||
|
- name: Build package (transpile ts => js)
|
||||||
|
run: ng build
|
||||||
|
|
||||||
|
- name: Store build
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Package
|
||||||
|
path: dist/
|
2
.gitignore
vendored
2
.gitignore
vendored
|
@ -44,5 +44,3 @@ testem.log
|
||||||
# System Files
|
# System Files
|
||||||
.DS_Store
|
.DS_Store
|
||||||
Thumbs.db
|
Thumbs.db
|
||||||
|
|
||||||
.npmrc
|
|
||||||
|
|
|
@ -1,50 +0,0 @@
|
||||||
when:
|
|
||||||
- event: manual
|
|
||||||
- event: tag
|
|
||||||
|
|
||||||
#depends_on:
|
|
||||||
#- test
|
|
||||||
|
|
||||||
steps:
|
|
||||||
npm:
|
|
||||||
image: code.philo.ydns.eu/philorg/pnpm-deb
|
|
||||||
environment:
|
|
||||||
ORG: philorg
|
|
||||||
REGISTRY_TOKEN:
|
|
||||||
from_secret: registry_token
|
|
||||||
volumes:
|
|
||||||
- pnpm:/root/.local/share/pnpm
|
|
||||||
commands:
|
|
||||||
- pnpm install --frozen-lockfile
|
|
||||||
- pnpm set "//code.philo.ydns.eu/api/packages/$ORG/npm/:_authToken=$REGISTRY_TOKEN"
|
|
||||||
- pnpm publish --no-git-checks
|
|
||||||
failure: ignore
|
|
||||||
|
|
||||||
build:
|
|
||||||
image: code.philo.ydns.eu/philorg/pnpm-deb
|
|
||||||
commands:
|
|
||||||
- pnpm build
|
|
||||||
|
|
||||||
build_publish:
|
|
||||||
image: quay.io/podman/stable:latest
|
|
||||||
# Caution: This image is built daily. It might fill up your image store quickly.
|
|
||||||
#pull: true
|
|
||||||
volumes:
|
|
||||||
- containers:/var/lib/containers
|
|
||||||
- pnpm:/root/.local/share/pnpm
|
|
||||||
# Fill in the trusted checkbox in Woodpecker's settings as well
|
|
||||||
privileged: true
|
|
||||||
environment:
|
|
||||||
registry: code.philo.ydns.eu
|
|
||||||
org: philorg
|
|
||||||
container_name: treetrail-frontend
|
|
||||||
registry_token:
|
|
||||||
from_secret: registry_token
|
|
||||||
commands:
|
|
||||||
# Login at the registry
|
|
||||||
- podman login -u __token__ --password $registry_token $registry
|
|
||||||
# Build the container image
|
|
||||||
- podman build --volume=/var/lib/containers:/var/lib/containers --tag $registry/$org/$container_name:latest --tag $registry/$org/$container_name:$CI_COMMIT_TAG .
|
|
||||||
# Push the image
|
|
||||||
- podman push $registry/$org/$container_name:latest
|
|
||||||
- podman push $registry/$org/$container_name:$CI_COMMIT_TAG
|
|
|
@ -1,6 +1,7 @@
|
||||||
FROM docker.io/library/nginx:alpine
|
FROM docker.io/library/nginx:alpine
|
||||||
|
MAINTAINER philo email phil.dev@philome.mooo.com
|
||||||
|
|
||||||
EXPOSE 80
|
EXPOSE 80
|
||||||
|
|
||||||
COPY nginx.conf /etc/nginx/nginx.conf
|
COPY nginx.conf /etc/nginx/nginx.conf
|
||||||
COPY dist/treetrail/browser /usr/share/nginx/html
|
COPY treetrail-app/dist/treetrail/browser /usr/share/nginx/html
|
11
README.md
11
README.md
|
@ -1,10 +1 @@
|
||||||
# Treetrail web front-end
|
Front-end for *Tree Trail*, a fun and pedagogic tool to discover the trails and trees around.
|
||||||
|
|
||||||
Front-end for *Tree Trail*, a fun and pedagogic tool
|
|
||||||
to discover the trails and trees around.
|
|
||||||
|
|
||||||
Home page (source, bugs, etc):
|
|
||||||
<https://code.philo.ydns.eu/philorg/treetrail-frontend> .
|
|
||||||
|
|
||||||
The main documentation for Tree Trail is:
|
|
||||||
<https://code.philo.ydns.eu/philorg/treetrail-backend> .
|
|
49
build.yaml
49
build.yaml
|
@ -1,49 +0,0 @@
|
||||||
- name: Build container image
|
|
||||||
hosts: localhost
|
|
||||||
gather_facts: false
|
|
||||||
vars:
|
|
||||||
repository: tiptop:5000
|
|
||||||
force_rm: false
|
|
||||||
cache: false
|
|
||||||
|
|
||||||
tasks:
|
|
||||||
- name: Install dependencies
|
|
||||||
command: pnpm install
|
|
||||||
args:
|
|
||||||
chdir: "{{ playbook_dir }}"
|
|
||||||
|
|
||||||
- name: Build the Angular app
|
|
||||||
command: ng build
|
|
||||||
args:
|
|
||||||
chdir: "{{ playbook_dir }}"
|
|
||||||
|
|
||||||
- name: Get the version from git
|
|
||||||
command: git describe --dirty --tags
|
|
||||||
register: version
|
|
||||||
args:
|
|
||||||
chdir: "{{ playbook_dir }}"
|
|
||||||
|
|
||||||
- name: Update version.json
|
|
||||||
command: pnpm run version
|
|
||||||
args:
|
|
||||||
chdir: "{{ playbook_dir }}"
|
|
||||||
|
|
||||||
- name: Build frontend container
|
|
||||||
containers.podman.podman_image:
|
|
||||||
name: "treetrail-frontend:{{ version.stdout }}"
|
|
||||||
state: build
|
|
||||||
force: true
|
|
||||||
path: "{{ playbook_dir }}"
|
|
||||||
build:
|
|
||||||
format: oci
|
|
||||||
force_rm: "{{ force_rm }}"
|
|
||||||
cache: "{{ cache }}"
|
|
||||||
file: Containerfile
|
|
||||||
extra_args: "--build-arg APP_VERSION={{ version.stdout }}"
|
|
||||||
push: true
|
|
||||||
push_args:
|
|
||||||
dest: "{{ repository }}/treetrail-frontend:{{ version.stdout }}"
|
|
||||||
#quadlet_options:
|
|
||||||
# - |
|
|
||||||
# [Install]
|
|
||||||
# WantedBy=default.target
|
|
66
nginx.conf
66
nginx.conf
|
@ -1,66 +0,0 @@
|
||||||
events {
|
|
||||||
worker_connections 1024; ## Default: 1024
|
|
||||||
}
|
|
||||||
|
|
||||||
http {
|
|
||||||
|
|
||||||
## use mime types
|
|
||||||
include /etc/nginx/mime.types;
|
|
||||||
|
|
||||||
server {
|
|
||||||
|
|
||||||
listen 80 default_server;
|
|
||||||
listen [::]:80 default_server;
|
|
||||||
server_name _;
|
|
||||||
root /usr/share/nginx/html;
|
|
||||||
|
|
||||||
location = / {
|
|
||||||
## Convevience redirect to redirect to the web root
|
|
||||||
return 301 /treetrail/;
|
|
||||||
}
|
|
||||||
|
|
||||||
location = /treetrail {
|
|
||||||
## Convevience redirect to redirect to the web root
|
|
||||||
return 301 /treetrail/;
|
|
||||||
}
|
|
||||||
|
|
||||||
location /treetrail/ {
|
|
||||||
alias /usr/share/nginx/html/;
|
|
||||||
index index.html;
|
|
||||||
try_files $uri $uri/ /index.html;
|
|
||||||
}
|
|
||||||
|
|
||||||
location /treetrail/v1/ {
|
|
||||||
proxy_set_header Host $http_host;
|
|
||||||
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
|
||||||
proxy_set_header X-Forwarded-Proto $scheme;
|
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
|
||||||
proxy_set_header Connection $connection_upgrade;
|
|
||||||
proxy_redirect off;
|
|
||||||
proxy_buffering off;
|
|
||||||
proxy_pass http://127.0.0.1:8081;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
map $http_upgrade $connection_upgrade {
|
|
||||||
default upgrade;
|
|
||||||
'' close;
|
|
||||||
}
|
|
||||||
|
|
||||||
## enable gzip compression
|
|
||||||
gzip on;
|
|
||||||
gzip_vary on;
|
|
||||||
gzip_min_length 256;
|
|
||||||
gzip_proxied any;
|
|
||||||
|
|
||||||
gzip_types
|
|
||||||
## text/html is always compressed : https://nginx.org/en/docs/http/ngx_http_gzip_module.html
|
|
||||||
text/plain
|
|
||||||
text/css
|
|
||||||
text/javascript
|
|
||||||
application/javascript
|
|
||||||
application/x-javascript
|
|
||||||
application/xml
|
|
||||||
application/json
|
|
||||||
application/ld+json;
|
|
||||||
}
|
|
4
package-lock.json
generated
4
package-lock.json
generated
|
@ -1,12 +1,12 @@
|
||||||
{
|
{
|
||||||
"name": "treetrail",
|
"name": "treetrail",
|
||||||
"version": "0.4.5",
|
"version": "0.0.0",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "treetrail",
|
"name": "treetrail",
|
||||||
"version": "0.4.5",
|
"version": "0.0.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@angular/animations": "^18.2.6",
|
"@angular/animations": "^18.2.6",
|
||||||
"@angular/cdk": "^18.2.6",
|
"@angular/cdk": "^18.2.6",
|
||||||
|
|
12
package.json
12
package.json
|
@ -1,14 +1,15 @@
|
||||||
{
|
{
|
||||||
"name": "treetrail-frontend",
|
"name": "treetrail",
|
||||||
"version": "0.4.5",
|
"version": "0.0.0",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"ng": "ng",
|
"ng": "ng",
|
||||||
"start": "ng serve --proxy-config proxy.conf.json --port 4201 --serve-path /treetrail --host 0.0.0.0",
|
"start": "ng serve --proxy-config proxy.conf.json --port 4201",
|
||||||
"build": "ng build",
|
"build": "ng build",
|
||||||
"watch": "ng build --watch --configuration development",
|
"watch": "ng build --watch --configuration development",
|
||||||
"test": "ng test",
|
"test": "ng test",
|
||||||
"openapi-ts": "openapi-ts"
|
"openapi-ts": "openapi-ts"
|
||||||
},
|
},
|
||||||
|
"private": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@angular/animations": "^18.2.6",
|
"@angular/animations": "^18.2.6",
|
||||||
"@angular/cdk": "^18.2.6",
|
"@angular/cdk": "^18.2.6",
|
||||||
|
@ -60,8 +61,5 @@
|
||||||
"karma-jasmine-html-reporter": "^2.1.0",
|
"karma-jasmine-html-reporter": "^2.1.0",
|
||||||
"typescript": "~5.4.5"
|
"typescript": "~5.4.5"
|
||||||
},
|
},
|
||||||
"packageManager": "pnpm@9.12.2",
|
"packageManager": "pnpm@9.12.2"
|
||||||
"publishConfig": {
|
|
||||||
"registry": "http://code.philo.ydns.eu/api/packages/philorg/npm/"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,17 +1,17 @@
|
||||||
{
|
{
|
||||||
"/treetrail/static": {
|
"/static": {
|
||||||
"target": "http://127.0.0.1:5002",
|
"target": "http://127.0.0.1:5002",
|
||||||
"secure": false
|
"secure": false
|
||||||
},
|
},
|
||||||
"/treetrail/v1": {
|
"/v1": {
|
||||||
"target": "http://127.0.0.1:5002",
|
"target": "http://127.0.0.1:5002",
|
||||||
"secure": false
|
"secure": false
|
||||||
},
|
},
|
||||||
"/treetrail/attachment": {
|
"/attachment": {
|
||||||
"target": "http://127.0.0.1:5002",
|
"target": "http://127.0.0.1:5002",
|
||||||
"secure": false
|
"secure": false
|
||||||
},
|
},
|
||||||
"/treetrail/tiles": {
|
"/tiles": {
|
||||||
"target": "http://127.0.0.1:5002",
|
"target": "http://127.0.0.1:5002",
|
||||||
"secure": false
|
"secure": false
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
</mat-card-title>
|
</mat-card-title>
|
||||||
<mat-card-content>
|
<mat-card-content>
|
||||||
<h2>Version</h2>
|
<h2>Version</h2>
|
||||||
<p><span class='h'>Client: </span>{{ version }}></p>
|
<p><span class='h'>Client: </span>{{ (configService.conf | async).bootstrap.client.version }}></p>
|
||||||
<p><span class='h'>Server: </span>{{ (configService.conf | async).bootstrap.server.version }}</p>
|
<p><span class='h'>Server: </span>{{ (configService.conf | async).bootstrap.server.version }}</p>
|
||||||
</mat-card-content>
|
</mat-card-content>
|
||||||
</mat-card>
|
</mat-card>
|
|
@ -1,7 +1,6 @@
|
||||||
import { Component, OnInit } from '@angular/core'
|
import { Component, OnInit } from '@angular/core';
|
||||||
|
|
||||||
import { ConfigService } from '../config.service'
|
import { ConfigService } from '../config.service'
|
||||||
import packageJson from '../../../package.json'
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-about',
|
selector: 'app-about',
|
||||||
|
@ -13,8 +12,6 @@ export class AboutComponent implements OnInit {
|
||||||
public configService: ConfigService,
|
public configService: ConfigService,
|
||||||
) { }
|
) { }
|
||||||
|
|
||||||
public version: string = packageJson.version
|
|
||||||
|
|
||||||
ngOnInit(): void {
|
ngOnInit(): void {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
0
src/app/app.component.scss
Normal file
0
src/app/app.component.scss
Normal file
|
@ -1,7 +1,5 @@
|
||||||
import {
|
import { Component, OnInit,
|
||||||
Component, OnInit,
|
ChangeDetectorRef, ChangeDetectionStrategy } from '@angular/core'
|
||||||
ChangeDetectorRef, ChangeDetectionStrategy
|
|
||||||
} from '@angular/core'
|
|
||||||
import { DataService } from './data.service'
|
import { DataService } from './data.service'
|
||||||
|
|
||||||
import { ActionService } from './action.service'
|
import { ActionService } from './action.service'
|
||||||
|
@ -12,6 +10,7 @@ import { combineLatest } from 'rxjs'
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-root',
|
selector: 'app-root',
|
||||||
templateUrl: './app.component.html',
|
templateUrl: './app.component.html',
|
||||||
|
styleUrls: ['./app.component.scss'],
|
||||||
changeDetection: ChangeDetectionStrategy.OnPush,
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
||||||
})
|
})
|
||||||
export class AppComponent implements OnInit {
|
export class AppComponent implements OnInit {
|
||||||
|
|
|
@ -80,7 +80,7 @@ export class MapComponent implements AfterContentInit, OnInit {
|
||||||
let conf = this.configService.conf.value
|
let conf = this.configService.conf.value
|
||||||
let bms = conf.background
|
let bms = conf.background
|
||||||
if (conf.bootstrap.baseMapStyles.embedded.indexOf(bms) >= 0) {
|
if (conf.bootstrap.baseMapStyles.embedded.indexOf(bms) >= 0) {
|
||||||
this.styleUrl = `tiles/style/${bms}`
|
this.styleUrl = `/tiles/style/${bms}`
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
this.styleUrl = conf.bootstrap.baseMapStyles.external[bms]
|
this.styleUrl = conf.bootstrap.baseMapStyles.external[bms]
|
||||||
|
|
0
src/app/profile/profile.component.scss
Normal file
0
src/app/profile/profile.component.scss
Normal file
|
@ -3,6 +3,7 @@ import { Component } from '@angular/core';
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-profile',
|
selector: 'app-profile',
|
||||||
templateUrl: './profile.component.html',
|
templateUrl: './profile.component.html',
|
||||||
|
styleUrls: ['./profile.component.scss']
|
||||||
})
|
})
|
||||||
export class ProfileComponent {
|
export class ProfileComponent {
|
||||||
|
|
||||||
|
|
|
@ -40,7 +40,7 @@
|
||||||
<div>
|
<div>
|
||||||
<mat-button-toggle-group #background="matButtonToggleGroup"
|
<mat-button-toggle-group #background="matButtonToggleGroup"
|
||||||
[value]="conf.background || conf.map?.background"
|
[value]="conf.background || conf.map?.background"
|
||||||
(change)="configService.setUserPref('background', background.value) ; router.navigate(['/map'])"
|
(change)="configService.setUserPref('background', background.value)"
|
||||||
>
|
>
|
||||||
<mat-button-toggle
|
<mat-button-toggle
|
||||||
*ngFor="let bms of conf.bootstrap.baseMapStyles.embedded"
|
*ngFor="let bms of conf.bootstrap.baseMapStyles.embedded"
|
||||||
|
@ -55,7 +55,7 @@
|
||||||
<mat-label>Types of zones displayed on the map</mat-label>
|
<mat-label>Types of zones displayed on the map</mat-label>
|
||||||
<div class="settings">
|
<div class="settings">
|
||||||
<mat-checkbox *ngFor="let item of conf.showZones | keyvalue" [checked]="item.value"
|
<mat-checkbox *ngFor="let item of conf.showZones | keyvalue" [checked]="item.value"
|
||||||
(change)="configService.setUserPrefValue('showZones', item.key, $event.checked) ; router.navigate(['/map'])">
|
(change)="configService.setUserPrefValue('showZones', item.key, $event.checked)">
|
||||||
{{ item.key }}
|
{{ item.key }}
|
||||||
</mat-checkbox>
|
</mat-checkbox>
|
||||||
</div>
|
</div>
|
||||||
|
@ -108,7 +108,7 @@
|
||||||
<li><span class='h'>Dave Storey</span>: communication</li>
|
<li><span class='h'>Dave Storey</span>: communication</li>
|
||||||
</ul>
|
</ul>
|
||||||
<p><span class='h'>Contact</span>: phil.treetrail at philome.mooo.com</p>
|
<p><span class='h'>Contact</span>: phil.treetrail at philome.mooo.com</p>
|
||||||
<p><span class='h'>Version</span>: {{conf.bootstrap?.client.version }}, frontend: {{ version }}</p>
|
<p><span class='h'>Version</span>: {{conf.bootstrap?.client.version }}</p>
|
||||||
</mat-expansion-panel>
|
</mat-expansion-panel>
|
||||||
|
|
||||||
<mat-expansion-panel>
|
<mat-expansion-panel>
|
||||||
|
|
|
@ -3,7 +3,6 @@ import { Router } from '@angular/router'
|
||||||
|
|
||||||
import { ActionService } from '../action.service'
|
import { ActionService } from '../action.service'
|
||||||
import { ConfigService, Config } from '../config.service'
|
import { ConfigService, Config } from '../config.service'
|
||||||
import packageJson from '../../../package.json'
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-settings',
|
selector: 'app-settings',
|
||||||
|
@ -18,8 +17,6 @@ export class SettingsComponent implements OnInit {
|
||||||
public router: Router,
|
public router: Router,
|
||||||
) { }
|
) { }
|
||||||
|
|
||||||
public version: string = packageJson.version
|
|
||||||
|
|
||||||
ngOnInit(): void {
|
ngOnInit(): void {
|
||||||
this.configService.conf.subscribe(
|
this.configService.conf.subscribe(
|
||||||
conf => this.conf = conf
|
conf => this.conf = conf
|
||||||
|
|
0
src/assets/.gitkeep
Normal file
0
src/assets/.gitkeep
Normal file
0
src/assets/fonts/.gitignore
vendored
Normal file
0
src/assets/fonts/.gitignore
vendored
Normal file
|
@ -15,8 +15,6 @@
|
||||||
"importHelpers": true,
|
"importHelpers": true,
|
||||||
"target": "ES2022",
|
"target": "ES2022",
|
||||||
"module": "es2020",
|
"module": "es2020",
|
||||||
//"allowSyntheticDefaultImports": true,
|
|
||||||
"resolveJsonModule": true,
|
|
||||||
"lib": [
|
"lib": [
|
||||||
"es2019",
|
"es2019",
|
||||||
"dom"
|
"dom"
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue