30 Commits

Author SHA1 Message Date
Andreani Jean
facaa4aa07 Release 2.1.2 2023-09-12 21:37:49 +02:00
Andreani Jean
6830bd5616 Release 2.1.1 2023-09-06 09:47:22 +02:00
Jean Andreani
692408de0f Merge branch 'fix/css-report-button' into 'develop'
fix version

See merge request geovisio/website!88
2023-09-06 07:43:34 +00:00
Jean Andreani
e2dce48730 fix version 2023-09-06 07:43:34 +00:00
Jean Andreani
6e021d1611 Merge branch 'fix/css-report-button' into 'develop'
fix css report button

See merge request geovisio/website!87
2023-09-06 07:12:18 +00:00
Andreani Jean
157bf0f01e fix css report button 2023-09-06 09:09:02 +02:00
Jean Andreani
2bc18bf494 Merge branch 'fix/maxzoom-doc' into 'develop'
Add doc for VITE_MAX_ZOOM env var

See merge request geovisio/website!86
2023-09-05 17:27:30 +00:00
Adrien Pavie
74e82d5352 Add doc for VITE_MAX_ZOOM env var 2023-09-05 17:27:30 +00:00
Jean Andreani
4999142e0d Merge branch 'fix-feat/some-bugs-add-report-button-viewer' into 'develop'
fix-feat/format date - add report button - license from API - maxZoom in ENV var

See merge request geovisio/website!85
2023-09-05 16:18:39 +00:00
Jean Andreani
6955d71752 fix-feat/format date - add report button - license from API - maxZoom in ENV var 2023-09-05 16:18:39 +00:00
Adrien Pavie
d72f01bbe0 Merge branch 'feature/docker2' into 'develop'
Improve Dockerfile

See merge request geovisio/website!83
2023-08-31 08:26:07 +00:00
Adrien Pavie
f9d1d154e0 Improve Dockerfile 2023-08-31 08:26:07 +00:00
Jean Andreani
d3a83f5157 Merge branch 'feature/gvs-213' into 'develop'
Web viewer update to 2.1.3, minor docs updates

See merge request geovisio/website!82
2023-08-30 11:01:52 +00:00
Adrien Pavie
c551ebdb21 Web viewer update to 2.1.3, minor docs updates 2023-08-30 11:01:52 +00:00
Andreani Jean
8f7e35dc43 Release 2.1.0 2023-08-29 14:56:25 +02:00
Jean Andreani
e0b17ae423 Merge branch 'fix-url-upload-responsive-sequences' into 'develop'
fix css my sequences

See merge request geovisio/website!81
2023-08-29 12:13:22 +00:00
Andreani Jean
876d0f1683 fix css my sequences 2023-08-29 14:08:02 +02:00
Jean Andreani
86c0e59c14 Merge branch 'fix-url-upload-responsive-sequences' into 'develop'
fix url upload to /envoyer + fix css responsive

See merge request geovisio/website!80
2023-08-29 10:24:34 +00:00
Andreani Jean
25c5833e26 fix url upload to /envoyer + fix css responsive 2023-08-29 12:21:42 +02:00
Jean Andreani
9e4b1f87f4 Merge branch 'feat-add-sort-file' into 'develop'
Feat add sorting for file upload

See merge request geovisio/website!79
2023-08-29 09:51:05 +00:00
Jean Andreani
35d95a85f8 Feat add sorting for file upload 2023-08-29 09:51:05 +00:00
Andreani Jean
6383fafcbf add fix button 2023-08-28 15:04:32 +02:00
Jean Andreani
c64604e4b7 Merge branch 'fix-authconf' into 'main'
fix-authconf

See merge request geovisio/website!78
2023-08-28 09:37:15 +00:00
Jean Andreani
197cdba0ca fix-authconf 2023-08-28 09:37:14 +00:00
Jean Andreani
456704f449 Merge branch 'feat/upload-interface' into 'main'
Feat/upload interface

See merge request geovisio/website!70
2023-08-28 09:23:20 +00:00
Jean Andreani
7f570e4086 Feat/upload interface 2023-08-28 09:23:19 +00:00
Adrien Pavie
0d39d522a5 Merge branch 'feature/gvs-212' into 'develop'
Update GeoVisio to 2.1.2

See merge request geovisio/website!77
2023-08-20 20:22:57 +00:00
Adrien Pavie
b708543df1 Update GeoVisio to 2.1.2 2023-08-20 22:18:36 +02:00
Adrien Pavie
fe8112356e Merge branch 'feature/docker' into 'main'
Add Dockerfile and CI Docker Hub deployment (fixes #16)

Closes #16

See merge request geovisio/website!74
2023-08-16 08:09:09 +00:00
Adrien Pavie
92fc0302a4 Add Dockerfile and CI Docker Hub deployment (fixes #16) 2023-08-03 09:02:44 +02:00
60 changed files with 2645 additions and 2432 deletions

7
.dockerignore Normal file
View File

@@ -0,0 +1,7 @@
docs/
.git/
.idea/
dist/
node_modules/
*.md
LICENSE

5
.gitignore vendored
View File

@@ -90,5 +90,6 @@ sw.*
*.swp
# Cypress generated screen and videos files
src/tests/cypress/screenshot/
src/tests/cypress/videos/
cypress/screenshot/
cypress/videos/
*.cy.ts.mp4

View File

@@ -5,6 +5,12 @@ stages:
variables:
CYPRESS_CACHE_FOLDER: '$CI_PROJECT_DIR/cache/Cypress'
DOCKER_BUILDKIT: 1 # use buildkit for better performance
DOCKER_DRIVER: overlay2 # better docker driver to avoid copying too many files on each run
GITLAB_REGISTRY: registry.gitlab.com # We use docker.io for official images and gitlab's registry to store temporary images
IMAGE_NAME: geovisio/api
CI_IMAGE_CACHE: $GITLAB_REGISTRY/$IMAGE_NAME:build_cache
DOCKER_TLS_CERTDIR: ""
before_script:
## chmod is unfortunately currently mandatory : https://github.com/nodejs/docker-node/issues/661
@@ -57,3 +63,71 @@ deploy:
script:
- yarn install
- yarn build
deploy:develop:
rules:
# run job only for fork that have the credentials to pull images from the gitlab-registry
# and only for merge on 'develop' branch
- if: $CI_DEPLOY_PASSWORD == null || $CI_DEPLOY_USER == null
when: never
- if: $CI_COMMIT_REF_SLUG == "develop"
stage: Deploy
image: docker:latest
services:
- docker:dind
before_script:
# login to the gitlab docker registry to use the cache and to publish
- echo $CI_DEPLOY_PASSWORD | docker login -u $CI_DEPLOY_USER --password-stdin $GITLAB_REGISTRY
- docker buildx create --use --name "geovisio-image-builder" --driver=docker-container # use docker-container driver to be able to publish a full cache
# login to dockerhub
- echo $CI_REGISTRY_PASSWORD | docker login -u $CI_REGISTRY_USER --password-stdin $CI_REGISTRY
script:
# build image using repository as cache
- docker buildx build
--cache-from "type=registry,ref=$CI_IMAGE_CACHE"
--cache-to "type=registry,mode=max,ref=$CI_IMAGE_CACHE"
--tag "$CI_REGISTRY_IMAGE:develop"
--label "org.opencontainers.image.title=$CI_PROJECT_TITLE"
--label "org.opencontainers.image.url=$CI_PROJECT_URL"
--label "org.opencontainers.image.created=$CI_JOB_STARTED_AT"
--label "org.opencontainers.image.revision=$CI_COMMIT_SHORT_SHA"
--load
--progress=plain
.
# publish image to dockerhub with the develop tag
- docker push "$CI_REGISTRY_IMAGE:develop"
deploy:latest:
# we consider that tag always land on main
# and they always should publish a tagged image and the `latest` docker image
only:
- tags
stage: Deploy
image: docker:latest
services:
- docker:dind
before_script:
# login to the gitlab docker registry to use the cache and to publish
- echo $CI_DEPLOY_PASSWORD | docker login -u $CI_DEPLOY_USER --password-stdin $GITLAB_REGISTRY
- docker buildx create --use --name "geovisio-image-builder" --driver=docker-container # use docker-container driver to be able to publish a full cache
# login to dockerhub
- echo $CI_REGISTRY_PASSWORD | docker login -u $CI_REGISTRY_USER --password-stdin $CI_REGISTRY
script:
# build image using repository as cache
- docker buildx build
--cache-from "type=registry,ref=$CI_IMAGE_CACHE"
--cache-to "type=registry,mode=max,ref=$CI_IMAGE_CACHE"
--tag "$CI_REGISTRY_IMAGE:$CI_COMMIT_REF_NAME"
--tag "$CI_REGISTRY_IMAGE:latest"
--label "org.opencontainers.image.title=$CI_PROJECT_TITLE"
--label "org.opencontainers.image.url=$CI_PROJECT_URL"
--label "org.opencontainers.image.created=$CI_JOB_STARTED_AT"
--label "org.opencontainers.image.revision=$GIT_DESCRIBE"
--load
--progress=plain
.
# publish image to dockerhub
- docker push $CI_REGISTRY_IMAGE:$CI_COMMIT_REF_NAME
- docker push $CI_REGISTRY_IMAGE:latest

View File

@@ -2,14 +2,52 @@
All notable changes to this project will be documented in this file.
Before _0.1.0_ Changelog didn't exist.
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
## [0.1.0] -
Before _0.1.0_, website development was on rolling release, meaning there are no version tags.
## [2.1.2] - 2023-09-12
### Changed
- add a fix to center the map with ENV variables
## [2.1.1] - 2023-09-06
### Changed
- GeoVisio web viewer upgraded to 2.1.4, [with alls its changes embedded](https://gitlab.com/geovisio/web-viewer/-/blob/develop/CHANGELOG.md?ref_type=heads#213-2023-08-30).
- Dockerfile creates smaller and faster containers, using pre-built website and Nginx for HTTP serving.
- In the upload input you can now choose between gallery and camera on mobile IOS
- Some CSS fix for the responsive of one Sequence
- Insert the report button in the viewer by passing a params
### Added
- Get the License with the API route
- Add ENV var for maxZoom params of the viewer
## [2.1.0] - 2023-08-29
### Added
- A new page `/envoyer` to upload picture with an interface ([#13](https://gitlab.com/geovisio/website/-/issues/13)) :
- the user can upload multiples pictures with the interface
- the pictures are sorted by name
- the user can see all the pictures uploaded and all the errors
### Changed
- Website releases now follow the synced `MAJOR.MINOR` API version rule, meaning that any version >= 2.1 of the website will be compatible with corresponding [GeoVisio API](https://gitlab.com/geovisio/api) version.
### Fixed
- fix a bug in the header hidden sub menu when authentication is not with keycloak
## [0.1.0] - 2023-07-04
### Added
- A new page `/mes-sequences` to access to a list of sequences for a logged user ([#14](https://gitlab.com/geovisio/website/-/issues/14)) :
- the user can see all his sequences
- the user can filter sequences
- the user can enter to a specific sequence
@@ -20,10 +58,13 @@ Before _0.1.0_ Changelog didn't exist.
- the user can see all the sequence's photos
- the user can disable and delete one or many photo(s) of the sequence
### Changed
- Header have now a new entry `Mes photos` when the user is logged to access to the sequence list
- The router guard for logged pages has been changed to not call the api to check the token
### Fixed
[Unreleased]: https://gitlab.com/geovisio/website/-/compare/2.1.0...develop
[2.1.1]: https://gitlab.com/geovisio/website/-/compare/2.1.0...2.1.1
[2.1.0]: https://gitlab.com/geovisio/website/-/compare/0.1.0...2.1.0
[0.1.0]: https://gitlab.com/geovisio/website/-/commits/0.1.0

55
Dockerfile Normal file
View File

@@ -0,0 +1,55 @@
#--------------------------------------------------------------
#- Build image
#-
FROM node:18.16.0-alpine AS build
WORKDIR /opt/geovisio
# Import dependencies files
COPY package.json yarn.lock ./
# Install NodeJS dependencies
RUN yarn install --frozen-lockfile
# Import source code
COPY static ./static
COPY src ./src
COPY *.json *.js *.ts *.html ./
# Replace env variables by placeholder for dynamic change on container start
ENV VITE_INSTANCE_NAME=DOCKER_VITE_INSTANCE_NAME
ENV VITE_API_URL=DOCKER_VITE_API_URL
ENV VITE_TILES=DOCKER_VITE_TILES
ENV VITE_MAX_ZOOM=DOCKER_VITE_MAX_ZOOM
# Build code
ENV PORT=3000
RUN yarn deploy
#--------------------------------------------------------------
#- Final image
#-
FROM nginx:1-alpine
RUN apk add bash
# Retrieve files from build
COPY --from=build /opt/geovisio/dist /usr/share/nginx/html
# Add Docker scripts and Nginx conf
COPY docker/nginx.conf /etc/nginx/nginx.conf
COPY docker/docker-entrypoint.sh /etc/nginx/docker-entrypoint.sh
RUN chmod +x /etc/nginx/docker-entrypoint.sh
# Define env variables defaults
ENV VITE_INSTANCE_NAME="GeoVisio/Docker"
ENV VITE_API_URL="https://panoramax.openstreetmap.fr"
ENV VITE_TILES="https://tile-vect.openstreetmap.fr/styles/basic/style.json"
ENV VITE_MAX_ZOOM=""
# Start Nginx
EXPOSE 3000
ENTRYPOINT ["/etc/nginx/docker-entrypoint.sh"]
CMD ["nginx", "-g", "daemon off;"]

View File

@@ -12,11 +12,6 @@ export default defineConfig({
setupNodeEvents(on, config) {
// implement node event listeners here
},
baseUrl: 'http://localhost:5173',
supportFile: 'src/tests/cypress/support/e2e.{js,jsx,ts,tsx}',
specPattern: 'src/tests/cypress/**/*.cy.{js,jsx,ts,tsx}',
fixturesFolder: 'src/tests/cypress/fixtures',
videosFolder: 'src/tests/cypress/videos',
screenshotsFolder: 'src/tests/cypress/screenshot'
baseUrl: 'http://localhost:5173'
}
})

View File

@@ -1,5 +1,5 @@
describe('In the home page', () => {
it('click on the link in the header to go to the utils page', () => {
it('click on the link in the header to go to the upload page', () => {
cy.visit('/')
cy.fixture('home').then((homeData) => {
cy.contains(homeData.textLinkUpload).click()

View File

@@ -1,5 +1,5 @@
{
"addressToSearch": "97 boulevard Voltaire 75011 paris",
"textAddressToSelect": "Boulevard Voltaire, Quartier de la Folie-Méricourt, Paris 11e Arrondissement, Paris, Île-de-France, France métropolitaine, 75011, France",
"textLinkUpload": "Aide"
"textLinkUpload": "À propos"
}

View File

@@ -0,0 +1,16 @@
#!/bin/bash
ROOT_DIR=/usr/share/nginx/html
DOCKER_VARS=(VITE_INSTANCE_NAME VITE_API_URL VITE_TILES VITE_MAX_ZOOM)
echo "Setting env variables in web files"
for file in $ROOT_DIR/assets/*.js $ROOT_DIR/index.html; do
echo "Processing $file...";
for i in ${!DOCKER_VARS[@]}; do
sed -i "s|DOCKER_${DOCKER_VARS[i]}|${!DOCKER_VARS[i]}|g" $file
done
done
echo "GeoVisio website is now ready !"
exec "$@"

22
docker/nginx.conf Normal file
View File

@@ -0,0 +1,22 @@
events {
worker_connections 1024;
}
http {
include mime.types;
sendfile on;
server {
listen 3000;
listen [::]:3000;
resolver 127.0.0.11;
autoindex off;
server_name _;
server_tokens off;
root /usr/share/nginx/html;
gzip_static on;
}
}

View File

@@ -1,6 +1,15 @@
# Setup
## System requirements
GeoVisio website can be installed through classic method, or using Docker.
__Contents__
[[_TOC_]]
## Classic install
### System requirements
**You need to have [Nodejs installed](https://nodejs.org/en/download)**
Node version : >=18.13.0
@@ -9,7 +18,7 @@ Node version : >=18.13.0
You can use npm or [yarn](https://yarnpkg.com/) as package manager
## Install
### Install
The website can be installed locally by retrieving this repository and installing dependencies:
@@ -22,7 +31,7 @@ cd website/
npm install
```
## Build for production
### Build for production
Before building, you need to define a bit of settings. At least, you have to create a `.env` file and edit its content.
@@ -41,6 +50,33 @@ PORT=3000 npm run start
The website is now available at [localhost:3000](http://localhost:3000).
## Docker setup
The [Docker](https://docs.docker.com/get-docker/) deployment is a really convenient way to have a Geovisio website running in an easy and fast way. Note that this setup documentation only covers __GeoVisio front-end__ (website), if you also need an API running, please refer to [Docker API deployment](https://gitlab.com/geovisio/api/-/blob/develop/docs/14_Running_Docker.md).
You can use the provided __Docker Hub__ `geovisio/website:latest` image directly:
```bash
docker run \
-e VITE_API_URL="https://your.geovisio.api/" \
-e VITE_INSTANCE_NAME="My Own GeoVisio" \
-e VITE_TILES="https://your.geovisio.api/vector/tiles/style.json" \
-p 3000:3000 \
--name geovisio-website \
-d \
geovisio/website:latest
```
This will run a container bound on [localhost:3000](http://localhost:3000).
You can also build the image from the local source with:
```bash
docker build -t geovisio/website:latest .
```
## Next steps
You can check out [the available settings for your instance](./03_Settings.md).

View File

@@ -11,6 +11,7 @@ Available parameters are:
- `VITE_API_URL`: the URL to the GeoVisio API (with trailing `/`, example: `https://geovisio.fr/`)
- `VITE_INSTANCE_NAME`: the name of the instance (example: `IGN`)
- `VITE_TILES`: the URL of your tiles : default tiles are the Open Street Map Tiles (example: `https://wxs.ign.fr/essentiels/static/vectorTiles/styles/PLAN.IGN/attenue.json`)
- `VITE_MAX_ZOOM`: the max zoom to use on the map (defaults to 24).
- Settings for the work environment:
- `NPM_CONFIG_PRODUCTION`: is it production environment (`true`, `false`)
- `YARN_PRODUCTION`: same as below, but if you use Yarn instead of NPM

24
docs/90_Releases.md Normal file
View File

@@ -0,0 +1,24 @@
# Make a release
The web site uses [semantic versioning](https://semver.org/) for its release numbers.
__Note__ : make sure that versions are in-sync with other Website components. Each component can have different `PATCH` versions, but compatibility __must__ be ensured between `MAJOR.MINOR` versions.
Run these commands in order to issue a new release:
```bash
git checkout develop
vim package.json # Change version
npm run doc
vim CHANGELOG.md # Replace unreleased to version number and update versions links (at bottom)
git add *
git commit -m "Release x.x.x"
git tag -a x.x.x -m "Release x.x.x"
git push origin develop
git checkout main
git merge develop
git push origin main --tags
```

View File

@@ -1,6 +1,6 @@
{
"name": "geovisio-website",
"version": "1.0.0",
"version": "2.1.2",
"engines": {
"node": "18.16.0"
},
@@ -25,13 +25,13 @@
"axios": "^1.2.3",
"bootstrap": "^5.2.3",
"bootstrap-icons": "^1.10.3",
"geovisio": "2.1.1",
"geovisio": "2.1.4",
"moment": "^2.29.4",
"pinia": "^2.1.4",
"vue": "^3.2.45",
"vue-axios": "^3.5.2",
"vue-eslint-parser": "^9.1.0",
"vue-i18n": "^9.2.2",
"vue-i18n": "9.2.2",
"vue-meta": "^3.0.0-alpha.10",
"vue-router": "^4.1.6",
"vue3-cookies": "^1.0.6",

View File

@@ -3,7 +3,7 @@ import Header from '@/components/Header.vue'
import { RouterView } from 'vue-router'
import { useMeta } from 'vue-meta'
import { useI18n } from 'vue-i18n'
import title from '@/utils/index'
import { title } from '@/utils/index'
import authConfig from './composables/auth'
const { authConf } = authConfig()
const { t } = useI18n()
@@ -26,8 +26,19 @@ useMeta({
<template v-slot:title="{ content }">{{ content }}</template>
</metainfo>
<Header
:auth-enabled="authConf.enabled"
:user-profile-url="authConf.user_profile ? authConf.user_profile.url : ''"
:auth-enabled="
authConf && authConf.auth && authConf.auth.enabled
? authConf.auth.enabled
: true
"
:user-profile-url="
authConf &&
authConf.auth &&
authConf.auth.user_profile &&
authConf.auth.user_profile.url
? authConf.auth.user_profile.url
: ''
"
/>
<RouterView />
</template>

View File

@@ -1,7 +1,7 @@
@mixin text($size) {
@if $size == h1 {
font-weight: normal;
font-size: toRem(3);
font-size: toRem(4);
@media (max-width: toRem(50)) {
font-size: toRem(2.6);
@@ -15,14 +15,6 @@
font-size: toRem(1.8);
}
}
@if $size == h3 {
font-weight: normal;
font-size: toRem(1.8);
@media (max-width: toRem(50)) {
font-size: toRem(1.6);
}
}
@if $size == h4 {
font-weight: normal;
font-size: toRem(1.6);

Binary file not shown.

Before

Width:  |  Height:  |  Size: 17 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 15 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 71 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 15 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 14 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 54 KiB

View File

@@ -40,7 +40,6 @@ h5 {
--blue-semi: rgba(207, 226, 255, 0.5);
--blue-pale: #f9fafd;
--beige: #f5f3ec;
--beige-pale: #ececec80;
--yellow: #fec868;
--orange: #ff6f00;
--green: #59ce8f;

View File

@@ -0,0 +1,3 @@
export function createLink(href: string, text: string): string {
return `<a href='mailto:signalement.ign@panoramax.fr${href}' target='_blank' title='${text}' class='gvs-btn gvs-widget-bg gvs-btn-large' style='font-size: 1.6em;display: block'><i class="bi bi-exclamation-triangle"></i></a>`
}

View File

@@ -6,7 +6,7 @@
</template>
<script lang="ts" setup>
import title from '@/utils/index'
import { title } from '@/utils/index'
</script>
<style lang="scss" scoped>

View File

@@ -7,14 +7,9 @@
<div class="wrapper-logo desktop">
<Link
:image="{ url: 'logo.jpeg', alt: $t('general.header.alt_logo') }"
:text="title($t('general.header.title'))"
:route="{ name: 'home' }"
/>
<span class="title-text"
>{{ $t('general.header.title') }}
<span v-if="instanceName" class="instance-text">{{
instanceName
}}</span></span
>
</div>
<div class="wrapper-logo responsive">
<Link
@@ -114,6 +109,7 @@ import { useCookies } from 'vue3-cookies'
import { useRoute } from 'vue-router'
import { useI18n } from 'vue-i18n'
import { getAuthRoute } from '@/utils/auth'
import { title } from '@/utils/index'
import Link from '@/components/Link.vue'
import BetaText from '@/components/BetaText.vue'
@@ -137,7 +133,6 @@ function toggleMenu(): void {
menuIsClosed.value = !menuIsClosed.value
}
const isLogged = computed((): boolean => !!cookies.get('user_id'))
const instanceName = computed((): string => import.meta.env.VITE_INSTANCE_NAME)
const ariaLabel = computed((): string =>
menuIsClosed.value
? t('general.header.burger_menu_aria_label_open')
@@ -169,16 +164,6 @@ const userName = computed((): string =>
.wrapper-logo {
display: flex;
align-items: center;
color: var(--blue-dark);
}
.title-text {
font-weight: bolder;
}
.instance-text {
font-weight: normal;
border-left: toRem(0.1) solid var(--blue-dark);
padding-left: toRem(0.7);
margin-left: toRem(0.5);
}
.wrapper-entries {
display: flex;
@@ -187,7 +172,7 @@ const userName = computed((): string =>
display: none;
}
.desktop {
display: flex;
display: block;
}
.wrapper-logo p {
@include text(m-r-regular);

View File

@@ -8,7 +8,7 @@
@click="$emit('trigger')"
>
<div
v-if="status.length && status !== 'waiting-for-process'"
v-if="status.length && (status === 'ready' || status === 'hidden')"
class="photo-img-wrapper"
>
<i v-if="status === 'hidden'" class="bi bi-eye-slash icon-hidden"></i>
@@ -20,8 +20,18 @@
class="photo-img"
/>
</div>
<div
v-else-if="status.length && status === 'broken'"
class="waiting-wrapper error"
>
<i class="bi bi-exclamation-octagon icon-waiting"></i>
<span class="waiting info">{{ $t('pages.sequence.broken') }}</span>
</div>
<div v-else class="waiting-wrapper">
<i class="bi bi-card-image icon-waiting"></i>
<span class="waiting info">{{
$t('pages.sequence.waiting_process')
}}</span>
</div>
<div
v-if="selectedOnMap && !selected"
@@ -36,12 +46,10 @@
>
<i class="bi bi-check-lg" />
</div>
<div v-if="status === 'waiting-for-process'" class="photo-info">
<span class="waiting info">{{
$t('pages.sequence.waiting_process')
}}</span>
</div>
<div v-else class="photo-info">
<div
v-if="status.length && (status === 'ready' || status === 'hidden')"
class="photo-info"
>
<span v-if="created" class="info"
><i class="bi bi-clock"></i> {{ created }}</span
>
@@ -72,7 +80,7 @@ defineProps({
status: {
type: String,
validator: (value: string): boolean =>
['waiting-for-process', 'ready', 'hidden', ''].includes(value),
['waiting-for-process', 'ready', 'hidden', 'broken', ''].includes(value),
default: ''
}
})
@@ -117,13 +125,32 @@ defineProps({
.waiting-wrapper {
height: toRem(16);
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
color: var(--blue);
margin-right: toRem(1);
margin-left: toRem(1);
&.error {
color: var(--red);
}
}
.icon-waiting {
height: toRem(4);
font-size: toRem(4);
font-size: toRem(3);
margin-bottom: toRem(1);
}
.waiting-info {
text-align: center;
width: 100%;
color: var(--black);
width: fit-content;
}
.info {
@include text(xs-r-regular);
padding: toRem(0.5) toRem(0.8);
background-color: var(--white);
border-radius: toRem(0.5);
}
.icon-img {
top: toRem(1);
@@ -156,19 +183,6 @@ defineProps({
width: 100%;
}
.waiting {
text-align: center;
width: 100%;
color: var(--black);
}
.info {
padding: toRem(0.5) toRem(0.8);
background-color: var(--white);
border-radius: toRem(0.5);
color: var(--blue);
}
.button-image-item:hover {
.photo-img {
opacity: 0.5;

View File

@@ -77,6 +77,5 @@ function updateValue(value: boolean): void {
.label {
cursor: pointer;
margin-left: toRem(0.5);
@include text(s-regular);
}
</style>

View File

@@ -10,7 +10,6 @@
type="file"
multiple
:accept="accept"
capture="environment"
class="input-file"
@change="changeFile"
/>
@@ -68,9 +67,7 @@ function drop(event: DragEvent): void | boolean {
}
function checkPicturesType(files: FileList): number {
const picturesToUpload = [...files]
.filter((p) => p.type == 'image/jpeg')
.sort((a, b) => a.name.localeCompare(b.name))
const picturesToUpload = [...files].filter((p) => p.type == 'image/jpeg')
return picturesToUpload.length
}
</script>

View File

@@ -61,7 +61,7 @@ const titleImg = computed<string>(() =>
@include text(s-regular);
display: flex;
align-items: center;
color: var(--blue-dark);
color: var(--black);
text-decoration: none;
.icon {
margin-right: toRem(1);
@@ -94,6 +94,10 @@ const titleImg = computed<string>(() =>
background-color: transparent;
text-decoration: underline;
}
.link--grey {
color: var(--grey-semi-dark);
text-decoration: underline;
}
.button--white {
background-color: var(--white);
color: var(--black);

View File

@@ -1,38 +0,0 @@
<template>
<div class="share-card">
<img v-if="image" :src="img(image.url)" :alt="image.alt" class="image" />
<h2 class="title">{{ title }}</h2>
<p class="text">{{ text }}</p>
</div>
</template>
<script lang="ts" setup>
import { img } from '../../utils/image'
import type { PropType } from 'vue'
interface ImageInterface {
url: string
alt: string
}
defineProps({
image: { type: Object as PropType<ImageInterface>, default: null },
title: { type: String, default: '' },
text: { type: String, default: '' }
})
</script>
<style scoped lang="scss">
.share-card {
color: var(--blue-dark);
}
.title {
@include text(h2);
}
.text {
@include text(s-regular);
color: var(--grey-semi-dark);
}
.image {
height: toRem(8);
margin-bottom: toRem(1);
}
</style>

View File

@@ -1,6 +1,9 @@
<template>
<div :class="['imported', { 'first-sequence': index === 0 }]">
<section :class="['information-section', { 'first-sequence': index === 0 }]">
<div class="uploaded-pictures">
<p v-if="index === 0" class="title-current-upload">
{{ $t('pages.upload.sequence_uploading_title') }}
</p>
<p v-if="sequence.pictures" class="uploaded-title">
<span
>{{ $t('pages.upload.import') }} {{ sequence.title }} -
@@ -39,7 +42,7 @@
/>
</ul>
</div>
</div>
</section>
</template>
<script setup lang="ts">
@@ -63,11 +66,16 @@ defineProps({
</script>
<style scoped lang="scss">
.imported {
.information-section {
display: flex;
margin-bottom: toRem(4);
padding-top: toRem(2);
padding-bottom: toRem(2);
width: 80%;
box-shadow: 0px 6.45694px 8.60925px rgba(0, 0, 0, 0.05);
}
.first-sequence {
background-color: var(--blue-pale);
border-radius: toRem(0.5);
}
.uploaded-pictures,
@@ -81,6 +89,7 @@ defineProps({
margin-left: toRem(2);
}
.uploaded-title {
margin-left: toRem(2);
margin-bottom: toRem(2);
width: 100%;
color: var(--grey-semi-dark);
@@ -92,11 +101,9 @@ defineProps({
}
.uploaded-picture-list,
.uploaded-error-list {
padding-top: 0rem;
padding-right: toRem(2);
padding-bottom: toRem(2);
padding: 0rem toRem(2) toRem(2);
overflow-y: auto;
max-height: toRem(50);
max-height: toRem(35);
}
.bi-check-circle {
color: var(--green);
@@ -121,13 +128,4 @@ defineProps({
padding: 0;
}
}
@media (max-width: toRem(50)) {
.imported {
flex-direction: column;
}
.uploaded-error-list {
padding-left: 0;
}
}
</style>

View File

@@ -34,9 +34,8 @@ const itemUploadedText = computed<string | null>(
margin-top: toRem(1);
border-radius: toRem(0.5);
width: 100%;
@include text(xs-r-regular);
@include text(s-regular);
}
.success {
background-color: var(--white);
border: toRem(0.1) solid var(--grey);

View File

@@ -1,46 +0,0 @@
<template>
<div v-if="isDisplayed" class="tooltip-banner">
<div class="top-banner">
<span
><i class="bi bi-info-circle"></i>
{{ $t('pages.upload.tooltip_banner_title') }}</span
>
<Button look="no-text" icon="bi bi-x-lg" @trigger="isDisplayed = false" />
</div>
<p class="text-banner">{{ $t('pages.upload.tooltip_banner_text') }}</p>
</div>
</template>
<script lang="ts" setup>
import { ref } from 'vue'
import Button from '@/components/Button.vue'
let isDisplayed = ref<boolean>(true)
</script>
<style lang="scss" scoped>
.tooltip-banner {
border-left: toRem(1) solid var(--blue);
border-radius: toRem(1);
padding: toRem(2);
background-color: var(--white);
}
.top-banner {
@include text(h2);
display: flex;
justify-content: space-between;
margin-bottom: toRem(0.5);
color: var(--blue-dark);
}
.text-banner {
@include text(s-regular);
margin-bottom: 0;
margin-right: 3rem;
color: var(--grey-semi-dark);
}
@media (max-width: toRem(76.8)) {
.text-banner {
margin-right: 0;
}
}
</style>

View File

@@ -57,6 +57,7 @@ const uploadPendingTitle = computed<string>(() => {
flex-direction: column;
justify-content: center;
align-items: center;
margin-top: toRem(4);
}
.loader {
position: relative;

View File

@@ -3,14 +3,15 @@ import { onMounted, ref } from 'vue'
import type { AuthConfigInterface } from './interfaces/Auth'
export default function authConfig() {
const authConf = ref<AuthConfigInterface>({})
const authConf = ref<AuthConfigInterface>()
async function getConfig(): Promise<object> {
async function getConfig(): Promise<AuthConfigInterface> {
const { data } = await axios.get('api/configuration', {
withCredentials: false
})
return data.auth
return data
}
onMounted(async () => (authConf.value = await getConfig()))
return { authConf }
}

View File

@@ -1,7 +1,7 @@
interface UserProfileInterface {
url?: string
}
export interface AuthConfigInterface {
enabled?: boolean
user_profile?: UserProfileInterface
license?: { id: string; url: string }
auth?: {
enabled?: boolean
user_profile?: { url?: string }
}
}

View File

@@ -6,11 +6,11 @@
"description": "Panoramax, lalternative libre pour photo-cartographier les territoires"
},
"header": {
"contribute_text": "Comment contribuer ?",
"upload_text": "Partager des photos",
"contribute_text": "À propos",
"upload_text": "Contribuer",
"sequences_text": "Mes photos",
"alt_logo": "Logo de l'instance",
"title": "Panoramax",
"title": "Instance\nPanoramax",
"beta_text": "Version beta",
"logout_text": "Déconnexion",
"my_information_text": "Mes informations",
@@ -24,8 +24,7 @@
},
"pages": {
"home": {
"report_mail_subject": "⚠️ Signalement sur l'image {id}",
"report_mail_body": "Bonjour, %0D%0A%0D%0A Problème sur l'image (garder le type de problème signalé) : %0D%0A%0D%0A contenu inapproprié / absence de floutage sur un élément à anonymiser ou flouter pour des raisons de sécurité /surfloutage (floutage en trop) %0D%0A%0D%0A Lien vers la photo concernée : {link} %0D%0A%0D%0A Précision sur les éléments concernés (en particulier pour les problèmes de floutage - que faut-il flouter ou déflouter?) : ",
"report_mail": "?subject=⚠️ Signalement sur l`image {picId}&body=Bonjour, %0D%0A%0D%0A Problème sur l`image (garder le type de problème signalé) : %0D%0A%0D%0A contenu inapproprié / absence de floutage sur un élément à anonymiser ou flouter pour des raisons de sécurité /surfloutage (floutage en trop) %0D%0A%0D%0A Lien vers la photo concernée : {link} %0D%0A%0D%0A Précision sur les éléments concernés (en particulier pour les problèmes de floutage - que faut-il flouter ou déflouter?) :",
"report_button_text": "Signaler la photo"
},
"settings": {
@@ -33,9 +32,9 @@
"setting_tooltip": "Afficher ou masquer le token"
},
"sequence": {
"sequence_published": "Séquence publiée",
"sequence_published": "Publiée",
"sequence_waiting": "En cours de publication",
"sequence_hidden": "Séquence masquée",
"sequence_hidden": "Masquée",
"hide_sequence_tooltip": "Masque la séquence sur la carte",
"delete_sequence_tooltip": "Supprime définitivement la séquence",
"hide_photo_tooltip": "Masque les photos sur la carte",
@@ -59,6 +58,7 @@
"unselect_text": "Tout désélectionner",
"select_shift_text": "Sélectionnez plusieurs photos avec shift",
"waiting_process": "Photo en cours de traitement",
"broken": "Traitement de la photo en erreur",
"no_image": "Aucune photo dans cette séquence"
},
"sequences": {
@@ -75,25 +75,18 @@
"sequence_deleted": "La séquence a bien été supprimée"
},
"share_pictures": {
"title": "Pourquoi partager ses photos sur Panoramax ?",
"description": "Avec Panoramax, vos photos sont accessibles à tous. Elles seront automatiquement floutées grâce à notre algorithme de floutage dans le respect de <a href='https://panoramax.fr/foire-aux-questions' target='_blank' style='color:black'>la législation</a> et libres de droit.\nLa mise en ligne se fera <a href='https://www.etalab.gouv.fr/licence-ouverte-open-licence/' target='_blank' style='color:black'>sous licence ouverte</a> sous forme «brute» pour des réutilisations variées (ex: préparation des chantiers, gestion de la voirie etc).",
"arg_title1": "Des lieux visibles depuis la voie publique",
"arg_title2": "Une gestion des photos au format 360° ou non",
"arg_title3": "Une fonctionnalité de visualisation de vue du sol",
"arg_title4": "Des photos automatiquement géolocalisées",
"arg_text1": "Le lorem ipsum est, en imprimerie, une suite de mots sans signification utilisée à titre provisoire pour calibrer une mise en page, le texte définitif venant remplacer le faux-texte dès qu'il est prêt ou que la",
"arg_text2": "Le lorem ipsum est, en imprimerie, une suite de mots sans signification utilisée à titre provisoire pour calibrer une mise en page, le texte définitif venant remplacer le faux-texte dès qu'il est prêt ou que la",
"arg_text3": "Le lorem ipsum est, en imprimerie, une suite de mots sans signification utilisée à titre provisoire pour calibrer une mise en page, le texte définitif venant remplacer le faux-texte dès qu'il est prêt ou que la",
"arg_text4": "Le lorem ipsum est, en imprimerie, une suite de mots sans signification utilisée à titre provisoire pour calibrer une mise en page, le texte définitif venant remplacer le faux-texte dès qu'il est prêt ou que la",
"arg_alt1": "Image d'un immeuble",
"arg_alt2": "Image d'un signe 360",
"arg_alt3": "Image d'un pointer de localisation sur une carte",
"arg_alt4": "Image d'un pointer de localisation",
"footer_block": "⚠️️️ Aujourd'hui, le versement de grands volumes d'images est possible via une ligne de commande. Bientôt, d'autres moyens de versement seront disponibles, notamment via une interface web.",
"title": "Partagez vos photos",
"sub_title": "Un compte utilisateur est obligatoire pour partager des photos",
"photo_type1": "Des lieux visibles depuis la voie publique",
"photo_type2": "360° ou non",
"photo_type3": "Vues du sol",
"photo_type4": "Géolocalisées",
"description": "Ici, vos photos sont accessibles à tous :\n\n{check} automatiquement floutées dans le respect de <a href='https://panoramax.fr/foire-aux-questions' target='_blank' style='color:black'>la législation</a>\n{check} libres de droit, <a href={licenseUrl} target='_blank' style='color:black'>sous licence {licenseName}</a>\n{check} sous forme «brute» pour des réutilisations variées (ex: préparation des chantiers)\n\n",
"footer_block": "⚠️️️ Aujourd'hui, le versement de grands volumes d'images est possible via une ligne de commande et via une interface web.",
"button": "Accéder à l'outil",
"alt_img_upload": "Image qui représente plusieurs photos en cours de téléchargement",
"title_terminal": "Utilisez la ligne de commande",
"footer_description_terminal": "Les données déposées seront sous <a href='https://www.etalab.gouv.fr/licence-ouverte-open-licence/' target='_blank' style='color:#808080'>la licence ouverte</a>.",
"footer_description_terminal": "Les données déposées seront sous {word}",
"cli_title": "Vous voulez accéder à plus de documentation ?",
"user_account_button": "Créer un compte",
"description_terminal": "<a href='https://gitlab.com/geovisio/cli' target='_blank' style='color:black'>L'outil en ligne de commande</a> vous permet de partager de grands volumes de photos. La procédure est simple et vous devez disposer <a target='_blank' href='https://www.python.org/downloads/' style='color:black'>de python (au moins la version 3.8)</a>.\n\n1. Installer loutil en ligne de commande geovisio\n2. Lancez la commande de versement dimages sur le dossier choisi. Après '--api-url', renseignez l'url de l'api de l'instance où partager les photos et le chemin vers votre dossier de photos sur votre machine. Loutil demandera vos informations de connexion avant l'import. Une fois les données chargées, un temps de traitement est nécessaire pour les rendre disponibles.",
@@ -103,16 +96,11 @@
},
"upload": {
"title": "Déposez vos photos",
"title_uploading": "Traitement des images",
"text": "Déposez vos fichiers dans cet espace. Chaque photo ou groupe de photos envoyé constituera une 'séquence'. Vous pourrez retrouver ensuite toutes vos séquences dans la section 'Mes images'. Déposez vos fichiers dans cet espace. Chaque photo ou groupe de photos envoyé",
"no_picture_uploading_text": "Aucune photo en cours de téléchargement actuellement",
"tooltip_banner_title": "Contribuez à la base de photos de Panoramax",
"tooltip_banner_text": "Déposez vos fichiers dans cet espace. Chaque photo ou groupe de photos envoyé constituera une 'séquence'. Vous pourrez retrouver ensuite toutes vos séquences dans la section 'Mes images'.",
"input_label": "Glissez vos images ici ou cliquez sur ",
"import_word": "importer",
"input_label": "Déposez des photos dans la zone ou ",
"import_word": "importez",
"import_type": "Format JPEG uniquement",
"sequence_title": "Séquence du ",
"button_text": "Télécharger",
"button_text": "Envoyer",
"uploaded_files": "{count} fichier| {count} fichiers",
"no_uploaded_files": "Aucun fichier sélectionné",
"uploaded_word": " Image téléchargée",
@@ -123,7 +111,7 @@
"upload_done": "Transfert terminé !",
"upload_pending_pictures": "Envoi de {count} photo en cours |Envoi de {count} photos en cours",
"sequence_link": "Voir la sequence",
"button_new_upload": "Nouveau téléchargement",
"button_new_upload": "Nouvel envoi",
"leave_message": "⚠️ Attention, le téléchargement sera interrompu si vous quittez la page avant la fin."
}
}

View File

@@ -43,7 +43,7 @@ const routes: Array<RouteRecordRaw> = [
component: SharePicturesView
},
{
path: '/telecharger',
path: '/envoyer',
name: 'upload-pictures',
component: UploadPicturesView
}

View File

@@ -1,22 +0,0 @@
describe('In the utils page', () => {
it('go to the login page', () => {
cy.visit('partager-des-photos')
cy.fixture('upload').then((uploadData) => {
cy.contains(uploadData.textButtonUpload).click()
})
})
it('go to the login page', () => {
cy.visit('partager-des-photos')
cy.fixture('upload').then((uploadData) => {
cy.contains(uploadData.textButtonUpload).click()
})
})
it('go to the cli page', () => {
cy.visit('partager-des-photos')
cy.fixture('upload').then((uploadData) => {
cy.contains(uploadData.textButtonCli).click()
})
})
})
export {}

View File

@@ -1,4 +0,0 @@
{
"textButtonUpload": "Créer un compte",
"textButtonCli": "Accéder à l'outil"
}

View File

@@ -1,5 +1,5 @@
import { test, describe, vi, expect } from 'vitest'
import { mount, shallowMount } from '@vue/test-utils'
import { shallowMount } from '@vue/test-utils'
import Link from '../../../components/Link.vue'
import { createI18n } from 'vue-i18n'
import { createRouter, createWebHistory } from 'vue-router'

View File

@@ -29,7 +29,7 @@ describe('Template', () => {
}
})
expect(wrapper.html()).contains('button-stub')
expect(wrapper.html()).contains('text="Nouveau téléchargement"')
expect(wrapper.html()).contains('text="Nouvel envoi"')
})
})
describe('When the component have an uploaded sequence', () => {
@@ -59,7 +59,7 @@ describe('Template', () => {
loadPercentage: '100%'
}
})
expect(wrapper.html()).contains('text="Nouveau téléchargement"')
expect(wrapper.html()).contains('text="Nouvel envoi"')
expect(wrapper.html()).contains('Transfert terminé !')
expect(wrapper.html()).contains('2345 Mo/2345 Mo')
})

View File

@@ -6,6 +6,7 @@ const i18n = createI18n({
fallbackLocale: 'fr',
globalInjection: true,
legacy: false,
warnHtmlMessage: false,
messages: {
fr
}

View File

@@ -7,11 +7,12 @@ import {
} from '../../views/utils/sequence/index'
import {
formatPictureSize,
formatTextSize
formatTextSize,
sortByName
} from '../../views/utils/upload/index'
import { getAuthRoute } from '../../utils/auth'
import { img, getPicId } from '../../utils/image'
import title from '../../utils/index'
import { img } from '../../utils/image'
import { title } from '../../utils/index'
describe('imageStatus', () => {
it('should render the "status" value', () => {
@@ -156,19 +157,6 @@ describe('img', () => {
})
})
describe('getPicId', () => {
it('should return the Id of the picture in the url', () => {
const url = 'http://dummy.com?pic=3205340583&test'
Object.defineProperty(window, 'location', {
value: {
href: url
},
writable: true // possibility to override
})
expect(getPicId()).toEqual('3205340583')
})
})
describe('title', () => {
it('should return the formated title with instance name', () => {
import.meta.env.VITE_INSTANCE_NAME = 'my instance'
@@ -181,3 +169,36 @@ describe('title', () => {
expect(title(myTitle)).toEqual('my title')
})
})
describe('sortByName', () => {
it('should return the the list sorted by name', () => {
const list1 = [
{ name: 'd_1_ct.jpg' },
{ name: 'd_11_ct.jpg' },
{ name: 'd_2_ct.jpg' }
]
expect(sortByName(list1)).toEqual([
{ name: 'd_1_ct.jpg' },
{ name: 'd_2_ct.jpg' },
{ name: 'd_11_ct.jpg' }
])
const list2 = [{ name: 'A.jpg' }, { name: 'Z.jpg' }, { name: 'B.jpg' }]
expect(sortByName(list2)).toEqual([
{ name: 'A.jpg' },
{ name: 'B.jpg' },
{ name: 'Z.jpg' }
])
const list3 = [
{ name: 'CAM1_001.jpg' },
{ name: 'CAM2_002.jpg' },
{ name: 'CAM1_011.jpg' }
]
expect(sortByName(list3)).toEqual([
{ name: 'CAM1_001.jpg' },
{ name: 'CAM1_011.jpg' },
{ name: 'CAM2_002.jpg' }
])
})
})

View File

@@ -1,19 +1,9 @@
import { it, describe, expect } from 'vitest'
import { shallowMount } from '@vue/test-utils'
import SharePicturesView from '../../../views/SharePicturesView.vue'
import { createI18n } from 'vue-i18n'
import fr from '../../../locales/fr.json'
import i18n from '../config'
import { createRouter, createWebHistory } from 'vue-router'
const i18n = createI18n({
locale: 'fr',
fallbackLocale: 'fr',
globalInjection: true,
legacy: false,
messages: {
fr
}
})
const router = createRouter({
history: createWebHistory(),
routes: []
@@ -32,8 +22,7 @@ describe('Template', () => {
}
})
expect(wrapper.html()).contains('<link')
expect(wrapper.html()).contains('path="')
expect(wrapper.html()).contains('/auth/login')
expect(wrapper.html()).contains('pathexternal=')
expect(wrapper.html()).contains('look="button button--blue"')
expect(wrapper.html()).contains('type="external"')
})

View File

@@ -1,10 +1,13 @@
import { it, describe, expect, vi } from 'vitest'
import { shallowMount } from '@vue/test-utils'
import { shallowMount, mount, flushPromises } from '@vue/test-utils'
import UploadPicturesView from '../../../views/UploadPicturesView.vue'
import { createRouter, createWebHistory } from 'vue-router'
import i18n from '../config'
import InputUpload from '../../../components/InputUpload.vue'
import Button from '../../../components/Button.vue'
import * as createAPictureToASequence from '@/views/utils/upload/request'
import * as createASequence from '@/views/utils/upload/request'
import { formatDate } from '../../../utils/dates'
import * as sortByName from '../../../views/utils/upload/index'
const router = createRouter({
history: createWebHistory(),
routes: []
@@ -39,12 +42,91 @@ describe('Template', () => {
}
})
const spy = vi.spyOn(wrapper.vm, 'addPictures')
const wrapperInputUpload = wrapper.findComponent(InputUpload)
const sortByNameMock = vi.spyOn(sortByName, 'sortByName')
sortByNameMock.mockReturnValue([{}, {}])
const wrapperInputUpload = await wrapper.findComponent(InputUpload)
await wrapperInputUpload.trigger('trigger')
await wrapperInputUpload.vm.$emit('trigger', [{}, {}])
expect(spy).toHaveBeenCalledTimes(1)
expect(wrapper.html()).contains('2 fichiers')
})
describe('submit uploadPicture', () => {
it('should trigger to uploadPictures', async () => {
const wrapper = mount(UploadPicturesView, {
global: {
plugins: [i18n, router],
mocks: {
$t: (msg) => msg
},
components: {
InputUpload
}
}
})
const spy = vi.spyOn(wrapper.vm, 'uploadPicture')
const wrapperInputUpload = wrapper.findComponent(InputUpload)
await wrapperInputUpload.trigger('trigger')
await wrapperInputUpload.vm.$emit('trigger', [{}, {}])
const buttonWrapper = await wrapper.find('[data-test="button-upload"]')
await buttonWrapper.trigger('submit.prevent')
expect(spy).toHaveBeenCalledTimes(1)
expect(wrapper.html()).contains('class="loader-percentage"')
expect(wrapper.html()).contains('class="lds-ring lg"')
})
})
describe('one sequence has been imported', () => {
it('should render a sequence with a list of uploaded pictures', async () => {
const wrapper = mount(UploadPicturesView, {
global: {
plugins: [i18n, router],
mocks: {
$t: (msg) => msg
}
}
})
const picture = {
lastModified: 1599133968750,
name: '100MSDCF_DSC02790.JPG',
size: 2345,
type: 'image/jpeg'
}
const spyASequence = vi.spyOn(createASequence, 'createASequence')
const spyPicture = vi.spyOn(
createAPictureToASequence,
'createAPictureToASequence'
)
const sortByNameMock = vi.spyOn(sortByName, 'sortByName')
sortByNameMock.mockReturnValue([picture])
const sequenceId = 'my-id'
spyASequence.mockReturnValue({ data: { id: sequenceId } })
spyPicture.mockReturnValue({ data: {} })
const sequenceTitle = `Séquence du ${formatDate(
new Date(),
'Do MMMM YYYY, hh:mm:ss'
)}`
const body = new FormData()
body.append('position', '1')
body.append('picture', picture)
const wrapperInputUpload = wrapper.findComponent(InputUpload)
await wrapperInputUpload.trigger('trigger')
await wrapperInputUpload.vm.$emit('trigger', [picture])
const buttonWrapper = await wrapper.find('[data-test="button-upload"]')
await buttonWrapper.trigger('submit.prevent')
expect(spyASequence).toHaveBeenCalledWith(sequenceTitle)
expect(spyPicture).toHaveBeenCalledWith(sequenceId, body)
expect(wrapper.html()).contains('class="uploaded-picture-list"')
expect(wrapper.html()).contains('class="uploaded-picture-item success"')
expect(wrapper.html()).contains(
'100MSDCF_DSC02790.JPG - pages.upload.uploaded_word'
)
expect(wrapper.html()).contains('<router-link')
expect(wrapper.html()).contains(
'class="default button button--white" title="pages.upload.sequence_link"'
)
})
})
})
})

View File

@@ -1,10 +1,3 @@
export function img(name: string): string {
return new URL(`../assets/images/${name}`, import.meta.url).toString()
}
export function getPicId(): string {
return window.location.href.substring(
window.location.href.indexOf('pic=') + 4,
window.location.href.lastIndexOf('&')
)
}

View File

@@ -1,5 +1,9 @@
export default function title(title: string): string {
export function title(title: string): string {
const instanceName = import.meta.env.VITE_INSTANCE_NAME
if (instanceName) return `${title} ${instanceName}`
return title
}
export function createUrlLink(picId: string): string {
return encodeURIComponent(`${window.location.origin}/#focus=pic&pic=${picId}`)
}

View File

@@ -1,44 +1,4 @@
import axios from 'axios'
import type {
ViewerMapInterface,
OptionalViewerMapInterface
} from '@/views/interfaces/common'
import GeoVisio from 'geovisio'
async function fetchMapAndViewer(params?: OptionalViewerMapInterface) {
const tiles = import.meta.env.VITE_TILES
let paramsGeovisio: ViewerMapInterface = {
map: {
startWide: true,
maxZoom: 19
}
}
if (tiles) {
const style = tiles.includes('wxs.ign.fr') ? await getIgnTiles() : tiles
paramsGeovisio = {
map: {
...paramsGeovisio.map,
style
}
}
}
if (params) {
const { picId } = params
const { fetchOptions } = params
if (picId) paramsGeovisio = { ...paramsGeovisio, picId: picId }
if (fetchOptions) {
paramsGeovisio = {
...paramsGeovisio,
fetchOptions: fetchOptions
}
}
}
return new GeoVisio(
'viewer', // Div ID
`${import.meta.env.VITE_API_URL}api/search`,
paramsGeovisio
)
}
async function getIgnTiles(): Promise<object> {
const { data } = await axios.get(
@@ -58,4 +18,4 @@ async function getIgnTiles(): Promise<object> {
return data
}
export { fetchMapAndViewer }
export { getIgnTiles }

View File

@@ -1,45 +1,74 @@
<template>
<main class="entry-page">
<section id="viewer" class="entry-viewer">
<div v-if="mapIsLoaded" class="entry-report-button">
<Button
:text="$t('pages.home.report_button_text')"
look="button--black"
@trigger="triggerMailto"
/>
</div>
</section>
<section id="viewer" class="entry-viewer"></section>
</main>
</template>
<script lang="ts" setup>
import { useI18n } from 'vue-i18n'
import { onMounted, computed, ref } from 'vue'
import Button from '@/components/Button.vue'
import { fetchMapAndViewer } from '@/utils/mapAndViewer'
import { getPicId } from '@/utils/image'
import { onMounted, ref } from 'vue'
import 'geovisio/build/index.css'
import GeoVisio from 'geovisio'
import { createLink } from '@/components-viewer/reportLink'
import { createUrlLink } from '@/utils/index'
import type { ViewerMapInterface } from '@/views/interfaces/common'
import { getIgnTiles } from '@/utils/mapAndViewer'
const { t } = useI18n()
let mapIsLoaded = ref<boolean>(false)
const mailtoPath = computed<string>(() => {
return `mailto:signalement.ign@panoramax.fr?subject=${t(
'pages.home.report_mail_subject',
{
id: getPicId()
}
)}&body=${t('pages.home.report_mail_body', {
id: getPicId(),
link: encodeURIComponent(window.location.href)
})}`
})
function triggerMailto(): void {
window.location.href = mailtoPath.value
}
let viewer = ref()
onMounted(async () => {
const reportLink = document.createElement('div')
reportLink.className = 'gvs-group gvs-group-large gvs-group-btnpanel'
const tiles = import.meta.env.VITE_TILES
const maxZoom = import.meta.env.VITE_MAX_ZOOM
let paramsGeovisio: ViewerMapInterface = {
map: {
startWide: true
}
}
if (maxZoom && maxZoom !== '') {
paramsGeovisio = {
map: {
...paramsGeovisio.map,
maxZoom: parseInt(maxZoom)
}
}
}
if (tiles) {
const style = tiles.includes('wxs.ign.fr') ? await getIgnTiles() : tiles
paramsGeovisio = {
map: {
...paramsGeovisio.map,
style
}
}
}
try {
await fetchMapAndViewer()
viewer.value = await new GeoVisio(
'viewer', // Div ID
`${import.meta.env.VITE_API_URL}/api/search`,
{
...paramsGeovisio,
widgets: { customWidget: reportLink }
}
)
if (viewer.value && viewer.value.addEventListener) {
viewer.value.addEventListener(
'picture-loaded',
async (e: { detail: { picId: string } }): Promise<void> => {
const href = t('pages.home.report_mail', {
picId: e.detail.picId,
link: createUrlLink(e.detail.picId)
})
reportLink.innerHTML = createLink(
href,
t('pages.home.report_button_text')
)
}
)
}
mapIsLoaded.value = true
} catch (err) {
console.log(err)
@@ -56,13 +85,6 @@ onMounted(async () => {
position: relative;
min-height: calc(100vh - #{toRem(8)});
}
.gvs-has-map .entry-report-button {
display: block;
position: absolute;
right: toRem(1);
top: toRem(2);
z-index: 1;
}
.gvs-focus-map .entry-report-button {
display: none;
}

View File

@@ -17,38 +17,42 @@
class="button-collapse"
@click="toggleMenu"
>
<span :class="[sequence.status, 'sequence-status']">{{
sequenceStatus
}}</span>
<h1 class="title">
{{ sequence.title }}
</h1>
<div class="wrapper-title">
<span :class="[sequence.status, 'sequence-status']">{{
sequenceStatus
}}</span>
<h1 class="title desktop">
{{ sequence.title }}
</h1>
</div>
<i :class="headerPanelIsOpen ? 'bi bi-dash' : 'bi bi-plus'"></i>
</button>
<div class="wrapper-button">
<div class="disable-button">
<Button
:tooltip="$t('pages.sequence.hide_sequence_tooltip')"
:text="
sequence.status === 'ready'
? $t('pages.sequence.button_disable')
: $t('pages.sequence.button_enable')
"
look="button--white"
:icon="
sequence.status === 'ready' ? 'bi bi-eye-slash' : 'bi bi-eye'
"
@trigger="patchCollection"
/>
</div>
<Button
:tooltip="$t('pages.sequence.delete_sequence_tooltip')"
:text="$t('pages.sequence.button_delete')"
look="button--red"
icon="bi bi-trash"
@trigger="deleteCollection"
/>
</div>
<h1 class="title responsive">
{{ sequence.title }}
</h1>
</div>
<div class="wrapper-button">
<Button
:tooltip="$t('pages.sequence.hide_sequence_tooltip')"
:text="
sequence.status === 'ready'
? $t('pages.sequence.button_disable')
: $t('pages.sequence.button_enable')
"
look="button--white"
:icon="
sequence.status === 'ready' ? 'bi bi-eye-slash' : 'bi bi-eye'
"
class="disable-button"
@trigger="patchCollection"
/>
<Button
:tooltip="$t('pages.sequence.delete_sequence_tooltip')"
:text="$t('pages.sequence.button_delete')"
look="button--red"
icon="bi bi-trash"
@trigger="deleteCollection"
/>
</div>
<div
:class="[sequence.status, 'collapse py-2 show']"
@@ -188,6 +192,7 @@ import Pagination from '@/components/Pagination.vue'
import InputCheckbox from '@/components/InputCheckbox.vue'
import Loader from '@/components/Loader.vue'
import ImageItem from '@/components/ImageItem.vue'
import 'geovisio/build/index.css'
import { durationCalc, formatDate } from '@/utils/dates'
import {
deleteACollectionItem,
@@ -206,7 +211,7 @@ import {
spliceIntoChunks,
formatPaginationItems
} from '@/views/utils/sequence/index'
import { fetchMapAndViewer } from '@/utils/mapAndViewer'
import { getIgnTiles } from '@/utils/mapAndViewer'
import type {
ResponseUserPhotoInterface,
ResponseUserPhotoLinksInterface,
@@ -214,6 +219,10 @@ import type {
UserSequenceInterface,
ResponseUserSequenceInterface
} from './interfaces/MySequenceView'
import type { ViewerMapInterface } from '@/views/interfaces/common'
import GeoVisio from 'geovisio'
import { createUrlLink } from '@/utils'
import { createLink } from '@/components-viewer/reportLink'
const { t } = useI18n()
const route = useRoute()
@@ -253,11 +262,58 @@ onMounted(async () => {
(el) => el.properties['geovisio:status'] === 'ready'
)
pictures.value = collectionItems
viewer.value = await fetchMapAndViewer({
fetchOptions: {
credentials: 'include'
const tiles = import.meta.env.VITE_TILES
const maxZoom = import.meta.env.VITE_MAX_ZOOM
let paramsGeovisio: ViewerMapInterface = {
map: {
startWide: true
}
})
}
if (maxZoom) {
paramsGeovisio = {
map: {
...paramsGeovisio.map,
maxZoom: parseInt(maxZoom)
}
}
}
if (tiles) {
const style = tiles.includes('wxs.ign.fr') ? await getIgnTiles() : tiles
paramsGeovisio = {
map: {
...paramsGeovisio.map,
style
}
}
}
const reportLink = document.createElement('div')
reportLink.className = 'gvs-group gvs-group-large gvs-group-btnpanel'
viewer.value = await new GeoVisio(
'viewer', // Div ID
`${import.meta.env.VITE_API_URL}/api/search`,
{
...paramsGeovisio,
fetchOptions: {
credentials: 'include'
},
widgets: { customWidget: reportLink }
}
)
if (viewer.value && viewer.value.addEventListener) {
viewer.value.addEventListener(
'picture-loaded',
async (e: { detail: { picId: string } }): Promise<void> => {
const href = t('pages.home.report_mail', {
picId: e.detail.picId,
link: createUrlLink(e.detail.picId)
})
reportLink.innerHTML = createLink(
href,
t('pages.home.report_button_text')
)
}
)
}
setHeightValue()
if (itemSelected.value.length || !collectionItemsReady[0]) return
viewer.value._api.onceReady().then(() => {
@@ -543,6 +599,9 @@ async function patchOrDeleteCollectionItems(
</script>
<style lang="scss" scoped>
.responsive {
display: none;
}
.entry-page {
display: flex;
}
@@ -550,7 +609,6 @@ async function patchOrDeleteCollectionItems(
width: 50vw;
position: relative;
height: calc(100vh - #{toRem(8)});
font-size: 137.5%;
}
.menu-right {
width: 50vw;
@@ -563,6 +621,10 @@ async function patchOrDeleteCollectionItems(
justify-content: center;
align-items: center;
}
.wrapper-title {
display: flex;
align-items: center;
}
.wrapper-button {
display: flex;
align-items: center;
@@ -600,9 +662,10 @@ async function patchOrDeleteCollectionItems(
}
}
.title {
@include text(h2);
@include text(h4);
color: var(--grey-dark);
margin-right: toRem(1);
margin-bottom: 0;
}
.button-close {
display: none;
@@ -619,9 +682,10 @@ async function patchOrDeleteCollectionItems(
justify-content: space-between;
}
.sequence-status {
@include text(xs-r-regular);
border-radius: toRem(3);
padding: toRem(0.5) toRem(1);
margin-right: toRem(2);
margin-right: toRem(1);
color: var(--white);
&.ready {
background-color: var(--orange);
@@ -642,6 +706,9 @@ async function patchOrDeleteCollectionItems(
padding: 0;
display: flex;
align-items: center;
justify-content: space-between;
width: 100%;
margin-bottom: toRem(1);
}
.bi-plus,
.bi-dash {
@@ -650,7 +717,7 @@ async function patchOrDeleteCollectionItems(
}
.photos-wrapper {
padding: toRem(1) 0rem toRem(2) toRem(1);
padding: toRem(1) 0 toRem(2) toRem(1);
height: calc(100vh - v-bind(menuHeight));
}
.delete-all {
@@ -660,6 +727,7 @@ async function patchOrDeleteCollectionItems(
margin-left: toRem(1);
margin-right: toRem(2);
margin-bottom: toRem(1);
@include text(xs-r-regular);
}
.wrapper-select {
display: flex;
@@ -676,9 +744,6 @@ async function patchOrDeleteCollectionItems(
margin-right: toRem(1);
margin-left: toRem(1);
}
.delete-all-text {
@include text(xs-r-regular);
}
.action-buttons {
display: flex;
align-items: center;
@@ -714,7 +779,6 @@ async function patchOrDeleteCollectionItems(
.header-menu {
flex-direction: column;
align-items: flex-start;
margin-bottom: toRem(1);
}
.block-collapse {
flex-direction: column;
@@ -735,6 +799,29 @@ async function patchOrDeleteCollectionItems(
}
}
@media (max-width: toRem(76.8)) {
.desktop {
display: none;
}
.responsive {
display: initial;
}
.title {
margin-right: 0;
margin-bottom: toRem(1);
}
.wrapper-button {
flex-direction: column;
align-items: initial;
}
.disable-button {
margin-right: 0;
margin-bottom: toRem(1);
}
.button-collapse {
width: 100%;
justify-content: space-between;
margin-bottom: 0;
}
.photo-item {
width: 100%;
margin-right: 0;
@@ -792,13 +879,17 @@ async function patchOrDeleteCollectionItems(
right: 0;
top: toRem(22);
z-index: 3;
background-color: var(--black);
height: toRem(5);
display: flex;
align-items: center;
justify-content: center;
border-top-left-radius: toRem(0.5);
border-bottom-left-radius: toRem(0.5);
background-color: var(--white);
border: toRem(0.1) solid var(--black);
}
.menu-top {
padding: toRem(1);
}
.menu-is-open {
.menu-right {

View File

@@ -1,63 +1,5 @@
<template>
<main class="entry-page">
<section>
<div class="information-wrapper">
<div class="information">
<h1 class="page-title">{{ $t('pages.share_pictures.title') }}</h1>
<p
class="page-description"
v-html="$t('pages.share_pictures.description')"
></p>
</div>
<img
src="@/assets/images/person-map.png"
alt=""
class="information-image"
/>
</div>
<ul class="card-list">
<li class="card-list-item">
<ShareCard
:image="{
url: 'building.png',
alt: $t('pages.share_pictures.arg_alt1')
}"
:title="$t('pages.share_pictures.arg_title1')"
:text="$t('pages.share_pictures.arg_text1')"
/>
</li>
<li class="card-list-item">
<ShareCard
:image="{
url: '360.png',
alt: $t('pages.share_pictures.arg_alt2')
}"
:title="$t('pages.share_pictures.arg_title2')"
:text="$t('pages.share_pictures.arg_text2')"
/>
</li>
<li class="card-list-item">
<ShareCard
:image="{
url: 'pointer-map.png',
alt: $t('pages.share_pictures.arg_alt3')
}"
:title="$t('pages.share_pictures.arg_title3')"
:text="$t('pages.share_pictures.arg_text3')"
/>
</li>
<li class="card-list-item">
<ShareCard
:image="{
url: 'pointer.png',
alt: $t('pages.share_pictures.arg_alt4')
}"
:title="$t('pages.share_pictures.arg_title4')"
:text="$t('pages.share_pictures.arg_text4')"
/>
</li>
</ul>
</section>
<section id="sec-1" class="section-upload">
<div class="wrapper-upload">
<div class="wrapper-upload-text">
@@ -86,17 +28,25 @@
</div>
</div>
<p
v-if="authConf && authConf.license && authConf.license.url"
class="upload-text"
v-html="
$t('pages.share_pictures.description', {
check: checkImg
check: checkImg,
licenseUrl: authConf.license.url,
licenseName: authConf.license.id
})
"
/>
<p class="upload-text">
{{ $t('pages.share_pictures.footer_block') }}
</p>
<div v-if="!isLogged && authConf.enabled" class="wrapper-account">
<div
v-if="
!isLogged && authConf && authConf.auth && authConf.auth.enabled
"
class="wrapper-account"
>
<h4 class="account-subtitle">
{{ $t('pages.share_pictures.sub_title') }}
</h4>
@@ -149,10 +99,26 @@
class="upload-text"
v-html="$t('pages.share_pictures.description_terminal')"
></p>
<p
<div
v-if="authConf && authConf.license && authConf.license.url"
class="upload-text grey"
v-html="$t('pages.share_pictures.footer_description_terminal')"
></p>
>
<p>
{{
$t('pages.share_pictures.footer_description_terminal', {
word: 'la license'
})
}}
</p>
<Link
:href="authConf.license.url"
:text="authConf.license.id"
target="_blank"
type="external"
look="link--grey"
@trigger="triggerHref"
/>
</div>
<div class="wrapper-account">
<h4 class="account-subtitle">
{{ $t('pages.share_pictures.cli_title') }}
@@ -162,7 +128,7 @@
:text="$t('pages.share_pictures.button')"
type="external"
look="button button--blue"
path="https://gitlab.com/geovisio/cli"
path-external="https://gitlab.com/geovisio/cli"
/>
</div>
</div>
@@ -175,7 +141,6 @@
<script lang="ts" setup>
import Link from '@/components/Link.vue'
import Terminal from '@/components/Terminal.vue'
import ShareCard from '@/components/share-pictures/ShareCard.vue'
import { useCookies } from 'vue3-cookies'
import { computed, ref } from 'vue'
import { useI18n } from 'vue-i18n'
@@ -210,46 +175,6 @@ function triggerHref(): void {
</script>
<style lang="scss" scoped>
.entry-page {
margin: auto;
background-color: var(--beige-pale);
padding: toRem(2) 5%;
}
.information-wrapper {
display: flex;
align-items: center;
padding: toRem(4);
background-color: var(--white);
border-radius: toRem(1);
}
.information {
width: 50%;
}
.information-image {
margin: auto;
}
.page-title {
@include text(h1);
color: var(--blue-dark);
}
.page-description {
color: var(--grey-semi-dark);
white-space: pre-wrap;
}
.card-list {
display: flex;
padding: toRem(4);
background-color: var(--white);
border-radius: toRem(1);
margin-top: toRem(2);
}
.card-list-item {
margin-right: toRem(3);
&:last-child {
margin-right: toRem(0);
}
}
.upload-title {
@include text(h1);
}
@@ -323,7 +248,7 @@ function triggerHref(): void {
min-height: 80vh;
}
.section-upload:first-child {
background-color: var(--beige-pale);
background-color: rgba(236, 236, 236, 0.5);
}
.wrapper-upload {
display: flex;
@@ -335,7 +260,14 @@ function triggerHref(): void {
white-space: pre-wrap;
}
.grey {
display: flex;
align-items: center;
flex-wrap: wrap;
color: var(--grey-semi-dark);
p {
margin-bottom: 0;
margin-right: toRem(0.5);
}
}
.wrapper-account {
display: flex;
@@ -380,23 +312,6 @@ function triggerHref(): void {
bottom: calc(20vh - #{toRem(10.5)});
}
@media (max-width: toRem(102.4)) {
.information-wrapper {
flex-direction: column;
}
.information {
width: 100%;
}
.card-list {
flex-wrap: wrap;
}
.card-list-item {
width: calc(50% - #{toRem(2)});
margin-right: 0;
&:nth-child(1),
&:nth-child(3) {
margin-right: toRem(2);
}
}
.section-upload {
height: initial;
}
@@ -421,17 +336,6 @@ function triggerHref(): void {
}
}
@media (max-width: toRem(76.8)) {
.card-list-item {
width: 100%;
margin-bottom: toRem(2);
&:nth-child(1),
&:nth-child(3) {
margin-right: 0;
}
&:last-child {
margin-bottom: 0;
}
}
.entry-page {
padding-top: toRem(11);
}

View File

@@ -1,77 +1,50 @@
<template>
<main class="entry-page">
<div class="entry-banner">
<TooltipBanner />
</div>
<section class="upload-section">
<div class="form-upload">
<h1 class="settings-title">{{ $t('pages.upload.title') }}</h1>
<p class="settings-text">{{ $t('pages.upload.text') }}</p>
<form v-if="!isLoading" @submit.prevent="uploadPicture">
<InputUpload
:text="$t('pages.upload.input_label')"
:text-second-part="$t('pages.upload.import_word')"
:text-picture-type="$t('pages.upload.import_type')"
accept="image/jpeg"
data-test="input-add-pictures"
@trigger="addPictures"
/>
<div class="footer-form">
<span v-if="fileUploaded" class="number-file-text">{{
fileUploaded
}}</span>
<span v-else class="number-file-text">{{
t('pages.upload.no_uploaded_files')
}}</span>
<Button
:text="$t('pages.upload.button_text')"
:disabled="!pictures || pictures.length < 1"
type="submit"
look="button button--blue"
/>
</div>
</form>
<UploadLoader
v-else
:load-percentage="loadPercentage"
:load-text-size="loadTextSize"
:is-loaded="isLoaded"
:uploaded-sequences="uploadedSequences"
:pictures-count="picturesCount"
@triggerNewUpload="triggerNewUpload"
<h1 class="settings-title">{{ $t('pages.upload.title') }}</h1>
<form v-if="!isLoading" @submit.prevent="uploadPicture">
<InputUpload
:text="$t('pages.upload.input_label')"
:text-second-part="$t('pages.upload.import_word')"
:text-picture-type="$t('pages.upload.import_type')"
accept="image/jpeg"
data-test="input-add-pictures"
@trigger="addPictures"
/>
</div>
<div class="form-upload">
<h1 class="settings-title">{{ $t('pages.upload.title_uploading') }}</h1>
<div v-if="!uploadedSequences.length" class="uploading-information">
<img src="@/assets/images/uploading.png" class="uploading-img" />
<p class="settings-text">
{{ $t('pages.upload.no_picture_uploading_text') }}
</p>
<div class="footer-form">
<span v-if="fileUploaded" class="number-file-text">{{
fileUploaded
}}</span>
<span v-else class="number-file-text">{{
t('pages.upload.no_uploaded_files')
}}</span>
<Button
:text="$t('pages.upload.button_text')"
:disabled="!pictures || pictures.length < 1"
type="submit"
data-test="button-upload"
look="button button--blue"
/>
</div>
<template v-for="(sequence, i) in uploadedSequences">
<ImportedSection
v-if="i === 0"
:index="i"
:sequence="sequence"
:pictures-count="picturesCount"
:upload-errors="sequence.picturesOnError"
:upload-pictures="sequence.pictures"
/>
</template>
</div>
</form>
<UploadLoader
v-else
:load-percentage="loadPercentage"
:load-text-size="loadTextSize"
:is-loaded="isLoaded"
:uploaded-sequences="uploadedSequences"
:pictures-count="picturesCount"
@triggerNewUpload="triggerNewUpload"
/>
</section>
<template v-for="(sequence, i) in uploadedSequences">
<section v-if="i !== 0" class="information-section">
<ImportedSection
:index="i"
:sequence="sequence"
:pictures-count="picturesCount"
:upload-errors="sequence.picturesOnError"
:upload-pictures="sequence.pictures"
/>
</section>
</template>
<ImportedSection
v-for="(sequence, i) in uploadedSequences"
:index="i"
:sequence="sequence"
:pictures-count="picturesCount"
:upload-errors="sequence.picturesOnError"
:upload-pictures="sequence.pictures"
/>
</main>
</template>
@@ -82,7 +55,6 @@ import { useI18n } from 'vue-i18n'
import InputUpload from '@/components/InputUpload.vue'
import Button from '@/components/Button.vue'
import ImportedSection from '@/components/upload/ImportedSection.vue'
import TooltipBanner from '@/components/upload/TooltipBanner.vue'
import UploadLoader from '@/components/upload/UploadLoader.vue'
import type { sequenceInterface } from './interfaces/UploadPicturesView'
import { formatDate } from '@/utils/dates'
@@ -90,9 +62,13 @@ import {
createAPictureToASequence,
createASequence
} from '@/views/utils/upload/request'
import { formatPictureSize, formatTextSize } from '@/views/utils/upload/index'
import {
formatPictureSize,
formatTextSize,
sortByName
} from '@/views/utils/upload/index'
const { t } = useI18n()
let pictures = ref<FileList | []>([])
let pictures = ref<File[] | []>([])
let picturesCount = ref<number>(0)
let isLoading = ref<boolean>(false)
let isLoaded = ref<boolean>(false)
@@ -152,7 +128,8 @@ function triggerNewUpload(): void {
picturesCount.value = 0
}
function addPictures(value: FileList): void {
pictures.value = value
const files = sortByName([].slice.call(value))
pictures.value = files
picturesCount.value = pictures.value.length
picturesUploadingSize.value = 0
picturesToUploadSize.value = 0
@@ -177,7 +154,7 @@ async function uploadPicture(): Promise<void> {
const sequenceTitle = `${t('pages.upload.sequence_title')}${formatDate(
new Date(),
'Do MMMM YY, hh:mm:ss'
'Do MMMM YYYY, hh:mm:ss'
)}`
const { data } = await createASequence(sequenceTitle)
const sequence: sequenceInterface = {
@@ -228,63 +205,22 @@ ul {
margin-bottom: 0;
}
.entry-page {
display: flex;
flex-direction: column;
align-items: center;
background-color: var(--beige-pale);
min-height: calc(100vh - #{toRem(8)});
padding-top: toRem(2);
padding-bottom: toRem(2);
overflow: hidden;
}
.entry-banner {
width: 90%;
margin-bottom: toRem(2);
}
.upload-section {
display: flex;
width: 90%;
}
.information-section {
width: 90%;
background-color: var(--white);
border-radius: toRem(1);
margin-top: toRem(2);
padding: toRem(2);
}
.uploading-information {
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
height: 100%;
}
.uploading-img {
height: toRem(20);
margin-bottom: toRem(2);
}
.form-upload {
height: toRem(70);
padding: toRem(2);
width: 50%;
background-color: var(--white);
border-radius: toRem(1);
&:first-child {
margin-right: toRem(1);
}
&:last-child {
margin-left: toRem(1);
}
.upload-section {
width: 80%;
padding: toRem(4);
margin-top: toRem(6);
margin-bottom: toRem(4);
box-shadow: 0px 6.45694px 8.60925px rgba(0, 0, 0, 0.05);
}
.settings-title {
color: var(--blue-dark);
@include text(h3);
margin-bottom: toRem(2);
}
.settings-text {
color: var(--grey-semi-dark);
margin-bottom: toRem(2);
@include text(s-regular);
text-align: center;
@include text(h1);
margin-bottom: toRem(4);
}
.footer-form {
display: flex;
@@ -298,6 +234,9 @@ ul {
color: var(--blue-dark);
@include text(s-regular);
}
::-webkit-scrollbar {
width: toRem(2);
}
::-webkit-scrollbar-track {
background: var(--grey-pale);
}
@@ -306,36 +245,22 @@ ul {
border-radius: toRem(5);
}
@media (max-width: toRem(76.8)) {
.upload-section {
flex-direction: column;
}
.form-upload {
width: 100%;
&:first-child,
&:last-child {
margin: toRem(0);
}
&:first-child {
margin-bottom: toRem(2);
}
.entry-page {
overflow-x: hidden;
}
.information-section {
flex-direction: column;
}
}
@media (max-width: toRem(50)) {
.entry-page {
margin-top: toRem(11);
overflow-y: hidden;
}
.upload-section,
.information-section {
.upload-section {
margin-top: toRem(15);
width: 100%;
padding-right: toRem(2);
padding-left: toRem(2);
}
.information-section {
width: 90%;
width: 100%;
}
}
</style>

View File

@@ -3,12 +3,15 @@ export interface OptionalViewerMapInterface {
credentials: string
}
picId?: string
widgets?: {
customWidget: HTMLAnchorElement
}
}
export interface ViewerMapInterface extends OptionalViewerMapInterface {
map: {
startWide: boolean
style?: object | string
maxZoom: number
maxZoom?: number
}
}

View File

@@ -5,4 +5,12 @@ function formatTextSize(size: number, i: number): string {
const sizes = ['0', 'Ko', 'Mo', 'Go', 'To', 'Po', 'Eo', 'Zo', 'o']
return `${parseFloat((size / Math.pow(1024, i)).toFixed(2))} ${sizes[i]}`
}
export { formatPictureSize, formatTextSize }
function sortByName(fileList: File[]): File[] {
return fileList.sort((a: File, b: File) =>
a.name.localeCompare(b.name, navigator.languages[0] || navigator.language, {
numeric: true,
ignorePunctuation: true
})
)
}
export { formatPictureSize, formatTextSize, sortByName }

3533
yarn.lock

File diff suppressed because it is too large Load Diff