Compare commits
48 Commits
group
...
vehicleLis
| Author | SHA1 | Date | |
|---|---|---|---|
| f09d9d0a16 | |||
|
|
11e2375456 | ||
|
|
e11514b6d5 | ||
|
|
3bfce72512 | ||
|
|
335656ec75 | ||
| f698891865 | |||
|
|
5d33ba3dcb | ||
| 193f5dfb6f | |||
|
|
af9626e2c4 | ||
| 49de2f22c4 | |||
|
|
d3df781e33 | ||
| ab8cbf6f73 | |||
| 10bde53c5e | |||
| 28c8fa8a90 | |||
| 11851fec61 | |||
| 696cd1d87f | |||
|
|
372647b3b9 | ||
|
|
56c1b45f45 | ||
|
|
62f64747c3 | ||
| 56d0914568 | |||
|
|
57accbddbb | ||
|
|
40cf6012a6 | ||
|
|
774a3c2301 | ||
| 1b847ea216 | |||
| 0c0a9d0496 | |||
|
|
e03f31d3f4 | ||
|
|
a2b644d548 | ||
|
|
d531fe63ff | ||
|
|
4c42d3efd3 | ||
|
|
47a2b1a3cb | ||
|
|
d9f5721a15 | ||
|
|
12a8a359ab | ||
| 707d9c63aa | |||
| a3ee210547 | |||
|
|
814a20c7e2 | ||
|
|
941a7ccc5e | ||
|
|
6accf3e4b9 | ||
|
|
6c29e2f2ae | ||
| 203aadaee1 | |||
| 58f5a23d1d | |||
| dc53589e7d | |||
| 34b46655bd | |||
| 72c9ca9635 | |||
| cb36a7c8dd | |||
| 0f9fe384a0 | |||
| d028256893 | |||
| 0dd4a723be | |||
| f4c2d61dc3 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -1,3 +1,4 @@
|
|||||||
/config.yaml
|
/config.yaml
|
||||||
|
themes/*
|
||||||
.vscode
|
.vscode
|
||||||
__debug_bin
|
__debug_bin
|
||||||
33
Dockerfile
Normal file
33
Dockerfile
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
FROM golang:alpine as builder
|
||||||
|
|
||||||
|
ARG ACCESS_TOKEN_USR="nothing"
|
||||||
|
ARG ACCESS_TOKEN_PWD="nothing"
|
||||||
|
|
||||||
|
RUN apk add --no-cache ca-certificates tzdata git
|
||||||
|
|
||||||
|
WORKDIR /
|
||||||
|
|
||||||
|
# Create a netrc file using the credentials specified using --build-arg
|
||||||
|
RUN printf "machine git.coopgo.io\n\
|
||||||
|
login ${ACCESS_TOKEN_USR}\n\
|
||||||
|
password ${ACCESS_TOKEN_PWD}\n\
|
||||||
|
\n"\
|
||||||
|
>> ~/.netrc
|
||||||
|
RUN chmod 600 ~/.netrc
|
||||||
|
|
||||||
|
COPY . .
|
||||||
|
|
||||||
|
RUN go mod download && CGO_ENABLED=0 GOOS=linux GOARCH=amd64 go build -o /server
|
||||||
|
|
||||||
|
RUN rm -r themes/*
|
||||||
|
RUN git clone --depth 1 https://git.coopgo.io/coopgo-apps/parcoursmob-default-theme themes/default
|
||||||
|
RUN git clone -b spie06 --depth 1 https://git.coopgo.io/coopgo-apps/parcoursmob-default-theme themes/spie06
|
||||||
|
|
||||||
|
FROM scratch
|
||||||
|
COPY --from=builder /etc/ssl/certs/ca-certificates.crt /etc/ssl/certs/
|
||||||
|
COPY --from=builder /usr/share/zoneinfo /usr/share/zoneinfo
|
||||||
|
COPY --from=builder /themes/ /themes/
|
||||||
|
COPY --from=builder /server /
|
||||||
|
|
||||||
|
EXPOSE 8080
|
||||||
|
ENTRYPOINT ["/server"]
|
||||||
34
go.mod
34
go.mod
@@ -2,22 +2,17 @@ module git.coopgo.io/coopgo-apps/parcoursmob
|
|||||||
|
|
||||||
go 1.18
|
go 1.18
|
||||||
|
|
||||||
replace git.coopgo.io/coopgo-platform/mobility-accounts => ../../coopgo-platform/mobility-accounts/
|
// replace git.coopgo.io/coopgo-platform/mobility-accounts => ../../coopgo-platform/mobility-accounts/
|
||||||
|
|
||||||
replace git.coopgo.io/coopgo-platform/groups-management => ../../coopgo-platform/groups-management/
|
replace git.coopgo.io/coopgo-platform/groups-management => ../../coopgo-platform/groups-management/
|
||||||
|
|
||||||
replace git.coopgo.io/coopgo-platform/fleets => ../../coopgo-platform/fleets/
|
// replace git.coopgo.io/coopgo-platform/fleets => ../../coopgo-platform/fleets/
|
||||||
|
|
||||||
replace git.coopgo.io/coopgo-platform/agenda => ../../coopgo-platform/agenda/
|
replace git.coopgo.io/coopgo-platform/agenda => ../../coopgo-platform/agenda/
|
||||||
|
|
||||||
replace git.coopgo.io/coopgo-platform/emailing => ../../coopgo-platform/emailing/
|
// replace git.coopgo.io/coopgo-platform/emailing => ../../coopgo-platform/emailing/
|
||||||
|
|
||||||
require (
|
require (
|
||||||
git.coopgo.io/coopgo-platform/agenda v0.0.0-00010101000000-000000000000
|
|
||||||
git.coopgo.io/coopgo-platform/emailing v0.0.0-00010101000000-000000000000
|
|
||||||
git.coopgo.io/coopgo-platform/fleets v0.0.0-00010101000000-000000000000
|
|
||||||
git.coopgo.io/coopgo-platform/groups-management v0.0.0-00010101000000-000000000000
|
|
||||||
git.coopgo.io/coopgo-platform/mobility-accounts v0.0.0-00010101000000-000000000000
|
|
||||||
github.com/coreos/go-oidc v2.2.1+incompatible
|
github.com/coreos/go-oidc v2.2.1+incompatible
|
||||||
github.com/fogleman/gg v1.3.0
|
github.com/fogleman/gg v1.3.0
|
||||||
github.com/go-playground/validator/v10 v10.11.0
|
github.com/go-playground/validator/v10 v10.11.0
|
||||||
@@ -34,30 +29,47 @@ require (
|
|||||||
google.golang.org/protobuf v1.28.1
|
google.golang.org/protobuf v1.28.1
|
||||||
)
|
)
|
||||||
|
|
||||||
|
require (
|
||||||
|
git.coopgo.io/coopgo-platform/agenda v0.0.0-20221205162112-5feb1b720ef9
|
||||||
|
git.coopgo.io/coopgo-platform/emailing v0.0.0-20221017030337-c71888d90c15
|
||||||
|
git.coopgo.io/coopgo-platform/fleets v0.0.0-20221205181518-91991789db9b
|
||||||
|
git.coopgo.io/coopgo-platform/groups-management v0.0.0-20221205161801-9705c8d898f0
|
||||||
|
git.coopgo.io/coopgo-platform/mobility-accounts v0.0.0-20220906130339-b9a32e41bffe
|
||||||
|
github.com/gorilla/securecookie v1.1.1
|
||||||
|
github.com/minio/minio-go/v7 v7.0.43
|
||||||
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/coreos/go-semver v0.3.0 // indirect
|
github.com/coreos/go-semver v0.3.0 // indirect
|
||||||
github.com/coreos/go-systemd/v22 v22.3.2 // indirect
|
github.com/coreos/go-systemd/v22 v22.3.2 // indirect
|
||||||
|
github.com/dustin/go-humanize v1.0.0 // indirect
|
||||||
github.com/fsnotify/fsnotify v1.5.4 // indirect
|
github.com/fsnotify/fsnotify v1.5.4 // indirect
|
||||||
github.com/go-mail/mail v2.3.1+incompatible // indirect
|
|
||||||
github.com/go-playground/locales v0.14.0 // indirect
|
github.com/go-playground/locales v0.14.0 // indirect
|
||||||
github.com/go-playground/universal-translator v0.18.0 // indirect
|
github.com/go-playground/universal-translator v0.18.0 // indirect
|
||||||
github.com/gogo/protobuf v1.3.2 // indirect
|
github.com/gogo/protobuf v1.3.2 // indirect
|
||||||
github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0 // indirect
|
github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0 // indirect
|
||||||
github.com/golang/protobuf v1.5.2 // indirect
|
github.com/golang/protobuf v1.5.2 // indirect
|
||||||
github.com/golang/snappy v0.0.1 // indirect
|
github.com/golang/snappy v0.0.1 // indirect
|
||||||
github.com/gorilla/securecookie v1.1.1 // indirect
|
|
||||||
github.com/hashicorp/hcl v1.0.0 // indirect
|
github.com/hashicorp/hcl v1.0.0 // indirect
|
||||||
github.com/klauspost/compress v1.13.6 // indirect
|
github.com/json-iterator/go v1.1.12 // indirect
|
||||||
|
github.com/klauspost/compress v1.15.9 // indirect
|
||||||
|
github.com/klauspost/cpuid/v2 v2.1.0 // indirect
|
||||||
github.com/leodido/go-urn v1.2.1 // indirect
|
github.com/leodido/go-urn v1.2.1 // indirect
|
||||||
github.com/magiconair/properties v1.8.6 // indirect
|
github.com/magiconair/properties v1.8.6 // indirect
|
||||||
github.com/mb0/wkt v0.0.0-20170420051526-a30afd545ee1 // indirect
|
github.com/mb0/wkt v0.0.0-20170420051526-a30afd545ee1 // indirect
|
||||||
|
github.com/minio/md5-simd v1.1.2 // indirect
|
||||||
|
github.com/minio/sha256-simd v1.0.0 // indirect
|
||||||
github.com/mitchellh/mapstructure v1.5.0 // indirect
|
github.com/mitchellh/mapstructure v1.5.0 // indirect
|
||||||
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
||||||
|
github.com/modern-go/reflect2 v1.0.2 // indirect
|
||||||
github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe // indirect
|
github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe // indirect
|
||||||
github.com/pelletier/go-toml v1.9.5 // indirect
|
github.com/pelletier/go-toml v1.9.5 // indirect
|
||||||
github.com/pelletier/go-toml/v2 v2.0.5 // indirect
|
github.com/pelletier/go-toml/v2 v2.0.5 // indirect
|
||||||
github.com/pkg/errors v0.9.1 // indirect
|
github.com/pkg/errors v0.9.1 // indirect
|
||||||
github.com/pquerna/cachecontrol v0.1.0 // indirect
|
github.com/pquerna/cachecontrol v0.1.0 // indirect
|
||||||
|
github.com/rs/xid v1.4.0 // indirect
|
||||||
github.com/santhosh-tekuri/jsonschema/v5 v5.0.0 // indirect
|
github.com/santhosh-tekuri/jsonschema/v5 v5.0.0 // indirect
|
||||||
|
github.com/sirupsen/logrus v1.9.0 // indirect
|
||||||
github.com/spf13/afero v1.8.2 // indirect
|
github.com/spf13/afero v1.8.2 // indirect
|
||||||
github.com/spf13/cast v1.5.0 // indirect
|
github.com/spf13/cast v1.5.0 // indirect
|
||||||
github.com/spf13/jwalterweatherman v1.1.0 // indirect
|
github.com/spf13/jwalterweatherman v1.1.0 // indirect
|
||||||
|
|||||||
45
go.sum
45
go.sum
@@ -36,6 +36,24 @@ cloud.google.com/go/storage v1.8.0/go.mod h1:Wv1Oy7z6Yz3DshWRJFhqM/UCfaWIRTdp0RX
|
|||||||
cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9ullr3+Kg0=
|
cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9ullr3+Kg0=
|
||||||
cloud.google.com/go/storage v1.14.0/go.mod h1:GrKmX003DSIwi9o29oFT7YDnHYwZoctc3fOKtUw0Xmo=
|
cloud.google.com/go/storage v1.14.0/go.mod h1:GrKmX003DSIwi9o29oFT7YDnHYwZoctc3fOKtUw0Xmo=
|
||||||
dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU=
|
dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU=
|
||||||
|
git.coopgo.io/coopgo-platform/agenda v0.0.0-20221017030035-4a26fc791c5b h1:7kLW1khfGguZ2aL+QpWFwZmAdEcY1MsUjLdiRufjr2s=
|
||||||
|
git.coopgo.io/coopgo-platform/agenda v0.0.0-20221017030035-4a26fc791c5b/go.mod h1:wqPvfYmzGF2cfXbs8XE1P2j5UYqZwp/La0llkl7dUkc=
|
||||||
|
git.coopgo.io/coopgo-platform/agenda v0.0.0-20221205162112-5feb1b720ef9 h1:aP/OxSxOUM7D5PcIs+VSU90Pyy+SSUOOM54FZvPmZ8w=
|
||||||
|
git.coopgo.io/coopgo-platform/agenda v0.0.0-20221205162112-5feb1b720ef9/go.mod h1:wqPvfYmzGF2cfXbs8XE1P2j5UYqZwp/La0llkl7dUkc=
|
||||||
|
git.coopgo.io/coopgo-platform/emailing v0.0.0-20221017030337-c71888d90c15 h1:+ZI4nGE6mqZ6pc7N/BizheEPRXn6Z84Sj7ikwfP2ZcU=
|
||||||
|
git.coopgo.io/coopgo-platform/emailing v0.0.0-20221017030337-c71888d90c15/go.mod h1:rmbqiHVkONcECOoPlsXlxZnD315Tiz2oRnn1M7646Kg=
|
||||||
|
git.coopgo.io/coopgo-platform/fleets v0.0.0-20221101232521-da16c90fc3ba h1:clfkgihzYa3xWKSY/Sn6VUSydOReh7cpuihvYNkWjNk=
|
||||||
|
git.coopgo.io/coopgo-platform/fleets v0.0.0-20221101232521-da16c90fc3ba/go.mod h1:s9OIFCNcjBAbBzRNHwoCTYV6kAntPG9CpT3GVweGdTY=
|
||||||
|
git.coopgo.io/coopgo-platform/fleets v0.0.0-20221205162030-cecdcb20e1d5 h1:HL/M681G9R1ZN8XPp4LvG9hcF20//R9yQmr5cdXwQaM=
|
||||||
|
git.coopgo.io/coopgo-platform/fleets v0.0.0-20221205162030-cecdcb20e1d5/go.mod h1:s9OIFCNcjBAbBzRNHwoCTYV6kAntPG9CpT3GVweGdTY=
|
||||||
|
git.coopgo.io/coopgo-platform/fleets v0.0.0-20221205181518-91991789db9b h1:wvK1V8fFmHgnzVyleEF6Jq0Jcp5cVpPfW7WjMYiTTrE=
|
||||||
|
git.coopgo.io/coopgo-platform/fleets v0.0.0-20221205181518-91991789db9b/go.mod h1:s9OIFCNcjBAbBzRNHwoCTYV6kAntPG9CpT3GVweGdTY=
|
||||||
|
git.coopgo.io/coopgo-platform/groups-management v0.0.0-20221017025751-671dc9a2c544 h1:rMLP77uIEequVXXZ0X9G1iK2k+xvW/+58ggwxxI6gqY=
|
||||||
|
git.coopgo.io/coopgo-platform/groups-management v0.0.0-20221017025751-671dc9a2c544/go.mod h1:lozSy6qlIIYhvKKXscZzz28HAtS0qBDUTv5nofLRmYA=
|
||||||
|
git.coopgo.io/coopgo-platform/groups-management v0.0.0-20221205161801-9705c8d898f0 h1:CnLKO1kzoGtaqPhDqfOX3WPRFRcJVJZdGzPdBE4X//w=
|
||||||
|
git.coopgo.io/coopgo-platform/groups-management v0.0.0-20221205161801-9705c8d898f0/go.mod h1:lozSy6qlIIYhvKKXscZzz28HAtS0qBDUTv5nofLRmYA=
|
||||||
|
git.coopgo.io/coopgo-platform/mobility-accounts v0.0.0-20220906130339-b9a32e41bffe h1:4OKwfKybR0VsIw2dSM9RtqGWveWPt+JjtiiMIBrg/w0=
|
||||||
|
git.coopgo.io/coopgo-platform/mobility-accounts v0.0.0-20220906130339-b9a32e41bffe/go.mod h1:1typNYtO+PQT6KG77vs/PUv0fO60/nbeSGZL2tt1LLg=
|
||||||
github.com/Azure/go-ansiterm v0.0.0-20170929234023-d6e3b3328b78/go.mod h1:LmzpDX56iTiv29bbRTIsUNlaFfuhWRQBWjQdVyAevI8=
|
github.com/Azure/go-ansiterm v0.0.0-20170929234023-d6e3b3328b78/go.mod h1:LmzpDX56iTiv29bbRTIsUNlaFfuhWRQBWjQdVyAevI8=
|
||||||
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
||||||
github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo=
|
github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo=
|
||||||
@@ -85,6 +103,7 @@ github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSs
|
|||||||
github.com/docker/go-connections v0.4.0/go.mod h1:Gbd7IOopHjR8Iph03tsViu4nIes5XhDvyHbTtUxmeec=
|
github.com/docker/go-connections v0.4.0/go.mod h1:Gbd7IOopHjR8Iph03tsViu4nIes5XhDvyHbTtUxmeec=
|
||||||
github.com/docker/go-units v0.3.3/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
|
github.com/docker/go-units v0.3.3/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
|
||||||
github.com/docker/go-units v0.4.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
|
github.com/docker/go-units v0.4.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
|
||||||
|
github.com/dustin/go-humanize v1.0.0 h1:VSnTsYCnlFHaM2/igO1h6X3HA71jcobQuxemgkq4zYo=
|
||||||
github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk=
|
github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk=
|
||||||
github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
|
github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
|
||||||
github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
|
github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
|
||||||
@@ -110,8 +129,6 @@ github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vb
|
|||||||
github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE=
|
github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE=
|
||||||
github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
|
github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
|
||||||
github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A=
|
github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A=
|
||||||
github.com/go-mail/mail v2.3.1+incompatible h1:UzNOn0k5lpfVtO31cK3hn6I4VEVGhe3lX8AJBAxXExM=
|
|
||||||
github.com/go-mail/mail v2.3.1+incompatible/go.mod h1:VPWjmmNyRsWXQZHVHT3g0YbIINUkSmuKOiLIDkWbL6M=
|
|
||||||
github.com/go-playground/assert/v2 v2.0.1 h1:MsBgLAaY856+nPRTKrp3/OZK38U/wa0CcBYNjji3q3A=
|
github.com/go-playground/assert/v2 v2.0.1 h1:MsBgLAaY856+nPRTKrp3/OZK38U/wa0CcBYNjji3q3A=
|
||||||
github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
|
github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
|
||||||
github.com/go-playground/locales v0.14.0 h1:u50s323jtVGugKlcYeyzC0etD1HifMjqmJqb8WugfUU=
|
github.com/go-playground/locales v0.14.0 h1:u50s323jtVGugKlcYeyzC0etD1HifMjqmJqb8WugfUU=
|
||||||
@@ -214,14 +231,21 @@ github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX
|
|||||||
github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU=
|
github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU=
|
||||||
github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
||||||
github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
||||||
|
github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
|
||||||
|
github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo=
|
||||||
github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU=
|
github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU=
|
||||||
github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk=
|
github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk=
|
||||||
github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w=
|
github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w=
|
||||||
github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM=
|
github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM=
|
||||||
github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8=
|
github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8=
|
||||||
github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
|
github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
|
||||||
github.com/klauspost/compress v1.13.6 h1:P76CopJELS0TiO2mebmnzgWaajssP/EszplttgQxcgc=
|
|
||||||
github.com/klauspost/compress v1.13.6/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk=
|
github.com/klauspost/compress v1.13.6/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk=
|
||||||
|
github.com/klauspost/compress v1.15.9 h1:wKRjX6JRtDdrE9qwa4b/Cip7ACOshUI4smpCQanqjSY=
|
||||||
|
github.com/klauspost/compress v1.15.9/go.mod h1:PhcZ0MbTNciWF3rruxRgKxI5NkcHHrHUDtV4Yw2GlzU=
|
||||||
|
github.com/klauspost/cpuid/v2 v2.0.1/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg=
|
||||||
|
github.com/klauspost/cpuid/v2 v2.0.4/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg=
|
||||||
|
github.com/klauspost/cpuid/v2 v2.1.0 h1:eyi1Ad2aNJMW95zcSbmGg7Cg6cq3ADwLpMAP96d8rF0=
|
||||||
|
github.com/klauspost/cpuid/v2 v2.1.0/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY=
|
||||||
github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
||||||
github.com/konsorten/go-windows-terminal-sequences v1.0.2/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
github.com/konsorten/go-windows-terminal-sequences v1.0.2/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
||||||
github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
||||||
@@ -248,14 +272,23 @@ github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOA
|
|||||||
github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
|
github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
|
||||||
github.com/mb0/wkt v0.0.0-20170420051526-a30afd545ee1 h1:VCgV+ng800r1/AChRHzHbWCtQI06cPxoZQUljQHTyXc=
|
github.com/mb0/wkt v0.0.0-20170420051526-a30afd545ee1 h1:VCgV+ng800r1/AChRHzHbWCtQI06cPxoZQUljQHTyXc=
|
||||||
github.com/mb0/wkt v0.0.0-20170420051526-a30afd545ee1/go.mod h1:IhobDa5AIyiMAsnH/qkytD0NbG0JMOJ2ihQqe1NdXyg=
|
github.com/mb0/wkt v0.0.0-20170420051526-a30afd545ee1/go.mod h1:IhobDa5AIyiMAsnH/qkytD0NbG0JMOJ2ihQqe1NdXyg=
|
||||||
|
github.com/minio/md5-simd v1.1.2 h1:Gdi1DZK69+ZVMoNHRXJyNcxrMA4dSxoYHZSQbirFg34=
|
||||||
|
github.com/minio/md5-simd v1.1.2/go.mod h1:MzdKDxYpY2BT9XQFocsiZf/NKVtR7nkE4RoEpN+20RM=
|
||||||
|
github.com/minio/minio-go/v7 v7.0.43 h1:14Q4lwblqTdlAmba05oq5xL0VBLHi06zS4yLnIkz6hI=
|
||||||
|
github.com/minio/minio-go/v7 v7.0.43/go.mod h1:nCrRzjoSUQh8hgKKtu3Y708OLvRLtuASMg2/nvmbarw=
|
||||||
|
github.com/minio/sha256-simd v1.0.0 h1:v1ta+49hkWZyvaKwrQB8elexRqm6Y0aMLjCNsrYxo6g=
|
||||||
|
github.com/minio/sha256-simd v1.0.0/go.mod h1:OuYzVNI5vcoYIAmbIvHPl3N3jUzVedXbKy5RFepssQM=
|
||||||
github.com/mitchellh/copystructure v1.0.0/go.mod h1:SNtv71yrdKgLRyLFxmLdkAbkKEFWgYaq1OVrnRcwhnw=
|
github.com/mitchellh/copystructure v1.0.0/go.mod h1:SNtv71yrdKgLRyLFxmLdkAbkKEFWgYaq1OVrnRcwhnw=
|
||||||
github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY=
|
github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY=
|
||||||
github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
|
github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
|
||||||
github.com/mitchellh/reflectwalk v1.0.0/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw=
|
github.com/mitchellh/reflectwalk v1.0.0/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw=
|
||||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
||||||
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
|
||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
||||||
github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
||||||
github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
||||||
|
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
|
||||||
|
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
|
||||||
github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe h1:iruDEfMl2E6fbMZ9s0scYfZQ84/6SPL6zC8ACM2oIL0=
|
github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe h1:iruDEfMl2E6fbMZ9s0scYfZQ84/6SPL6zC8ACM2oIL0=
|
||||||
github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe/go.mod h1:wL8QJuTMNUDYhXwkmfOly8iTdp5TEcJFWZD2D7SIkUc=
|
github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe/go.mod h1:wL8QJuTMNUDYhXwkmfOly8iTdp5TEcJFWZD2D7SIkUc=
|
||||||
github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U=
|
github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U=
|
||||||
@@ -302,12 +335,16 @@ github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFR
|
|||||||
github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc=
|
github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc=
|
||||||
github.com/rogpeppe/go-internal v1.8.0 h1:FCbCCtXNOY3UtUuHUYaghJg4y7Fd14rXifAYUAtL9R8=
|
github.com/rogpeppe/go-internal v1.8.0 h1:FCbCCtXNOY3UtUuHUYaghJg4y7Fd14rXifAYUAtL9R8=
|
||||||
github.com/rogpeppe/go-internal v1.8.0/go.mod h1:WmiCO8CzOY8rg0OYDC4/i/2WRWAB6poM+XZ2dLUbcbE=
|
github.com/rogpeppe/go-internal v1.8.0/go.mod h1:WmiCO8CzOY8rg0OYDC4/i/2WRWAB6poM+XZ2dLUbcbE=
|
||||||
|
github.com/rs/xid v1.4.0 h1:qd7wPTDkN6KQx2VmMBLrpHkiyQwgFXRnkOLacUiaSNY=
|
||||||
|
github.com/rs/xid v1.4.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg=
|
||||||
github.com/santhosh-tekuri/jsonschema/v5 v5.0.0 h1:TToq11gyfNlrMFZiYujSekIsPd9AmsA2Bj/iv+s4JHE=
|
github.com/santhosh-tekuri/jsonschema/v5 v5.0.0 h1:TToq11gyfNlrMFZiYujSekIsPd9AmsA2Bj/iv+s4JHE=
|
||||||
github.com/santhosh-tekuri/jsonschema/v5 v5.0.0/go.mod h1:FKdcjfQW6rpZSnxxUvEA5H/cDPdvJ/SZJQLWWXWGrZ0=
|
github.com/santhosh-tekuri/jsonschema/v5 v5.0.0/go.mod h1:FKdcjfQW6rpZSnxxUvEA5H/cDPdvJ/SZJQLWWXWGrZ0=
|
||||||
github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
|
github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
|
||||||
github.com/sirupsen/logrus v1.4.1/go.mod h1:ni0Sbl8bgC9z8RoU9G6nDWqqs/fq4eDPysMBDgk/93Q=
|
github.com/sirupsen/logrus v1.4.1/go.mod h1:ni0Sbl8bgC9z8RoU9G6nDWqqs/fq4eDPysMBDgk/93Q=
|
||||||
github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE=
|
github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE=
|
||||||
github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88=
|
github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88=
|
||||||
|
github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0=
|
||||||
|
github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
|
||||||
github.com/spf13/afero v1.8.2 h1:xehSyVa0YnHWsJ49JFljMpg1HX19V6NDZ1fkm1Xznbo=
|
github.com/spf13/afero v1.8.2 h1:xehSyVa0YnHWsJ49JFljMpg1HX19V6NDZ1fkm1Xznbo=
|
||||||
github.com/spf13/afero v1.8.2/go.mod h1:CtAatgMJh6bJEIs48Ay/FOnkljP3WeGUG0MC1RfAqwo=
|
github.com/spf13/afero v1.8.2/go.mod h1:CtAatgMJh6bJEIs48Ay/FOnkljP3WeGUG0MC1RfAqwo=
|
||||||
github.com/spf13/cast v1.5.0 h1:rj3WzYc11XZaIZMPKmwP96zkFEnnAmV8s6XbB2aY32w=
|
github.com/spf13/cast v1.5.0 h1:rj3WzYc11XZaIZMPKmwP96zkFEnnAmV8s6XbB2aY32w=
|
||||||
@@ -542,6 +579,8 @@ golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBc
|
|||||||
golang.org/x/sys v0.0.0-20210806184541-e5e7981a1069/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.0.0-20210806184541-e5e7981a1069/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.0.0-20220412211240-33da011f77ad/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.0.0-20220412211240-33da011f77ad/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/sys v0.0.0-20220704084225-05e143d24a9e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.0.0-20220728004956-3c1f35247d10 h1:WIoqL4EROvwiPdUtaip4VcDdpZ4kha7wBWZrbVKCIZg=
|
golang.org/x/sys v0.0.0-20220728004956-3c1f35247d10 h1:WIoqL4EROvwiPdUtaip4VcDdpZ4kha7wBWZrbVKCIZg=
|
||||||
golang.org/x/sys v0.0.0-20220728004956-3c1f35247d10/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.0.0-20220728004956-3c1f35247d10/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
||||||
|
|||||||
@@ -4,8 +4,8 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/services"
|
"git.coopgo.io/coopgo-apps/parcoursmob/services"
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/cache"
|
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
||||||
|
cache "git.coopgo.io/coopgo-apps/parcoursmob/utils/storage"
|
||||||
"github.com/spf13/viper"
|
"github.com/spf13/viper"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -13,10 +13,10 @@ type APIHandler struct {
|
|||||||
idp *identification.IdentificationProvider
|
idp *identification.IdentificationProvider
|
||||||
config *viper.Viper
|
config *viper.Viper
|
||||||
services *services.ServicesHandler
|
services *services.ServicesHandler
|
||||||
cache *cache.CacheHandler
|
cache cache.CacheHandler
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewAPIHandler(cfg *viper.Viper, idp *identification.IdentificationProvider, svc *services.ServicesHandler, cache *cache.CacheHandler) (*APIHandler, error) {
|
func NewAPIHandler(cfg *viper.Viper, idp *identification.IdentificationProvider, svc *services.ServicesHandler, cache cache.CacheHandler) (*APIHandler, error) {
|
||||||
return &APIHandler{
|
return &APIHandler{
|
||||||
idp: idp,
|
idp: idp,
|
||||||
config: cfg,
|
config: cfg,
|
||||||
|
|||||||
97
handlers/api/export.go
Normal file
97
handlers/api/export.go
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/csv"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"sort"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/gorilla/mux"
|
||||||
|
)
|
||||||
|
|
||||||
|
type FlatMaps []map[string]any
|
||||||
|
|
||||||
|
func (maps FlatMaps) GetHeaders() (res []string) {
|
||||||
|
keys := map[string]bool{}
|
||||||
|
for _, m := range maps {
|
||||||
|
for k, _ := range m {
|
||||||
|
if _, ok := keys[k]; !ok {
|
||||||
|
keys[k] = true
|
||||||
|
res = append(res, k)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sort.Strings(res)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (maps FlatMaps) GetValues() (res [][]string) {
|
||||||
|
headers := maps.GetHeaders()
|
||||||
|
for _, m := range maps {
|
||||||
|
line := []string{}
|
||||||
|
for _, k := range headers {
|
||||||
|
if v, ok := m[k]; ok && v != nil {
|
||||||
|
line = append(line, fmt.Sprint(v))
|
||||||
|
} else {
|
||||||
|
line = append(line, "")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
res = append(res, line)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h APIHandler) CacheExport(w http.ResponseWriter, r *http.Request) {
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
cacheid := vars["cacheid"]
|
||||||
|
|
||||||
|
d, err := h.cache.Get(cacheid)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusNotFound)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if data, ok := d.([]any); ok {
|
||||||
|
|
||||||
|
flatmaps := FlatMaps{}
|
||||||
|
//fmt.Println(data)
|
||||||
|
|
||||||
|
for _, v := range data {
|
||||||
|
fm := map[string]any{}
|
||||||
|
flatten("", v.(map[string]any), fm)
|
||||||
|
flatmaps = append(flatmaps, fm)
|
||||||
|
}
|
||||||
|
|
||||||
|
w.Header().Set("Content-Type", "text/csv")
|
||||||
|
w.Header().Set("Content-Disposition", fmt.Sprintf("attachment; filename=export-%s.csv", cacheid))
|
||||||
|
c := csv.NewWriter(w)
|
||||||
|
c.Write(flatmaps.GetHeaders())
|
||||||
|
c.WriteAll(flatmaps.GetValues())
|
||||||
|
return
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
w.WriteHeader(http.StatusNotFound)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func flatten(prefix string, src map[string]any, dest map[string]any) {
|
||||||
|
if len(prefix) > 0 {
|
||||||
|
prefix += "."
|
||||||
|
}
|
||||||
|
for k, v := range src {
|
||||||
|
switch child := v.(type) {
|
||||||
|
case map[string]any:
|
||||||
|
flatten(prefix+k, child, dest)
|
||||||
|
case []any:
|
||||||
|
for i := 0; i < len(child); i++ {
|
||||||
|
dest[prefix+k+"."+strconv.Itoa(i)] = child[i]
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
fmt.Println(prefix+k, " : ", v)
|
||||||
|
dest[prefix+k] = v
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -18,6 +18,7 @@ func (h APIHandler) OAuth2Callback(w http.ResponseWriter, r *http.Request) {
|
|||||||
// Extract the ID Token from OAuth2 token.
|
// Extract the ID Token from OAuth2 token.
|
||||||
rawIDToken, ok := oauth2Token.Extra("id_token").(string)
|
rawIDToken, ok := oauth2Token.Extra("id_token").(string)
|
||||||
if !ok {
|
if !ok {
|
||||||
|
fmt.Println("issue retrieving token")
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -40,7 +41,11 @@ func (h APIHandler) OAuth2Callback(w http.ResponseWriter, r *http.Request) {
|
|||||||
delete(session.Values, "redirect")
|
delete(session.Values, "redirect")
|
||||||
}
|
}
|
||||||
|
|
||||||
session.Save(r, w)
|
if err = session.Save(r, w); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
http.Redirect(w, r, redirect, http.StatusFound)
|
http.Redirect(w, r, redirect, http.StatusFound)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,27 +8,43 @@ import (
|
|||||||
"io"
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/sorting"
|
||||||
|
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
||||||
|
"git.coopgo.io/coopgo-platform/fleets/storage"
|
||||||
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
||||||
groupstorage "git.coopgo.io/coopgo-platform/groups-management/storage"
|
groupstorage "git.coopgo.io/coopgo-platform/groups-management/storage"
|
||||||
accounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
accounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
||||||
|
mobilityaccountsstorage "git.coopgo.io/coopgo-platform/mobility-accounts/storage"
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
"google.golang.org/protobuf/types/known/structpb"
|
"google.golang.org/protobuf/types/known/structpb"
|
||||||
)
|
)
|
||||||
|
|
||||||
type GroupsByName []groupstorage.Group
|
|
||||||
|
|
||||||
func (a GroupsByName) Len() int { return len(a) }
|
|
||||||
func (a GroupsByName) Less(i, j int) bool {
|
|
||||||
return strings.Compare(a[i].Data["name"].(string), a[j].Data["name"].(string)) < 0
|
|
||||||
}
|
|
||||||
func (a GroupsByName) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
|
||||||
|
|
||||||
func (h *ApplicationHandler) Administration(w http.ResponseWriter, r *http.Request) {
|
func (h *ApplicationHandler) Administration(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|
||||||
|
accounts, err := h.services.GetAccounts()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
beneficiaries, err := h.services.GetBeneficiaries()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
bookings, err := h.services.GetBookings()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
request := &groupsmanagement.GetGroupsRequest{
|
request := &groupsmanagement.GetGroupsRequest{
|
||||||
Namespaces: []string{"parcoursmob_organizations"},
|
Namespaces: []string{"parcoursmob_organizations"},
|
||||||
}
|
}
|
||||||
@@ -47,9 +63,9 @@ func (h *ApplicationHandler) Administration(w http.ResponseWriter, r *http.Reque
|
|||||||
groups = append(groups, g)
|
groups = append(groups, g)
|
||||||
}
|
}
|
||||||
|
|
||||||
sort.Sort(GroupsByName(groups))
|
sort.Sort(sorting.GroupsByName(groups))
|
||||||
|
|
||||||
h.Renderer.Administration(w, r, groups)
|
h.Renderer.Administration(w, r, accounts, beneficiaries, groups, bookings)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *ApplicationHandler) AdministrationCreateGroup(w http.ResponseWriter, r *http.Request) {
|
func (h *ApplicationHandler) AdministrationCreateGroup(w http.ResponseWriter, r *http.Request) {
|
||||||
@@ -69,7 +85,11 @@ func (h *ApplicationHandler) AdministrationCreateGroup(w http.ResponseWriter, r
|
|||||||
"vehicles": r.FormValue("modules.vehicles") == "on",
|
"vehicles": r.FormValue("modules.vehicles") == "on",
|
||||||
"vehicles_management": r.FormValue("modules.vehicles_management") == "on",
|
"vehicles_management": r.FormValue("modules.vehicles_management") == "on",
|
||||||
"events": r.FormValue("modules.events") == "on",
|
"events": r.FormValue("modules.events") == "on",
|
||||||
|
"agenda": r.FormValue("modules.agenda") == "on",
|
||||||
|
"groups": r.FormValue("modules.groups") == "on",
|
||||||
"administration": r.FormValue("modules.administration") == "on",
|
"administration": r.FormValue("modules.administration") == "on",
|
||||||
|
"support": r.FormValue("modules.support") == "on",
|
||||||
|
"group_module": r.FormValue("modules.group_module") == "on",
|
||||||
}
|
}
|
||||||
|
|
||||||
groupid := uuid.NewString()
|
groupid := uuid.NewString()
|
||||||
@@ -137,29 +157,12 @@ func (h *ApplicationHandler) AdministrationGroupDisplay(w http.ResponseWriter, r
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
members, err := h.members()
|
groupmembers, admins, err := h.groupmembers(groupid)
|
||||||
if err != nil {
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
groupmembers := []any{}
|
|
||||||
admins := []any{}
|
|
||||||
|
|
||||||
for _, m := range members {
|
|
||||||
mm := m.ToStorageType()
|
|
||||||
for _, g := range mm.Data["groups"].([]any) {
|
|
||||||
if g.(string) == groupid {
|
|
||||||
groupmembers = append(groupmembers, mm)
|
|
||||||
}
|
|
||||||
if g.(string) == groupid+":admin" {
|
|
||||||
admins = append(admins, mm)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
h.Renderer.AdministrationGroupDisplay(w, r, resp.Group.ToStorageType(), groupmembers, admins)
|
h.Renderer.AdministrationGroupDisplay(w, r, resp.Group.ToStorageType(), groupmembers, admins)
|
||||||
}
|
}
|
||||||
@@ -229,7 +232,7 @@ func (h *ApplicationHandler) AdministrationGroupInviteAdmin(w http.ResponseWrite
|
|||||||
}
|
}
|
||||||
key := base64.RawURLEncoding.EncodeToString(b)
|
key := base64.RawURLEncoding.EncodeToString(b)
|
||||||
|
|
||||||
h.cache.PutWithTTL("onboarding/"+key, onboarding, 72*time.Hour)
|
h.cache.PutWithTTL("onboarding/"+key, onboarding, 168*time.Hour) // 1 week TTL
|
||||||
|
|
||||||
data := map[string]any{
|
data := map[string]any{
|
||||||
"group": groupresp.Group.ToStorageType().Data["name"],
|
"group": groupresp.Group.ToStorageType().Data["name"],
|
||||||
@@ -247,6 +250,114 @@ func (h *ApplicationHandler) AdministrationGroupInviteAdmin(w http.ResponseWrite
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) AdministrationGroupInviteMember(w http.ResponseWriter, r *http.Request) {
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
groupid := vars["groupid"]
|
||||||
|
|
||||||
|
groupresp, err := h.services.GRPC.GroupsManagement.GetGroup(context.TODO(), &groupsmanagement.GetGroupRequest{
|
||||||
|
Id: groupid,
|
||||||
|
Namespace: "parcoursmob_organizations",
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
group := groupresp.Group.ToStorageType()
|
||||||
|
|
||||||
|
r.ParseForm()
|
||||||
|
|
||||||
|
accountresp, err := h.services.GRPC.MobilityAccounts.GetAccountUsername(context.TODO(), &accounts.GetAccountUsernameRequest{
|
||||||
|
Username: r.FormValue("username"),
|
||||||
|
Namespace: "parcoursmob",
|
||||||
|
})
|
||||||
|
|
||||||
|
if err == nil {
|
||||||
|
account := accountresp.Account.ToStorageType()
|
||||||
|
account.Data["groups"] = append(account.Data["groups"].([]any), group.ID)
|
||||||
|
|
||||||
|
as, _ := accounts.AccountFromStorageType(&account)
|
||||||
|
|
||||||
|
_, err = h.services.GRPC.MobilityAccounts.UpdateData(
|
||||||
|
context.TODO(),
|
||||||
|
&accounts.UpdateDataRequest{
|
||||||
|
Account: as,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
fmt.Println(err)
|
||||||
|
|
||||||
|
data := map[string]any{
|
||||||
|
"group": group.Data["name"],
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := h.emailing.Send("onboarding.existing_member", r.FormValue("username"), data); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, "/app/group/settings", http.StatusFound)
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
// Onboard now administrator
|
||||||
|
onboarding := map[string]any{
|
||||||
|
"username": r.FormValue("username"),
|
||||||
|
"group": group.ID,
|
||||||
|
"admin": false,
|
||||||
|
}
|
||||||
|
|
||||||
|
b := make([]byte, 16)
|
||||||
|
if _, err := io.ReadFull(rand.Reader, b); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
key := base64.RawURLEncoding.EncodeToString(b)
|
||||||
|
|
||||||
|
h.cache.PutWithTTL("onboarding/"+key, onboarding, 168*time.Hour) // 1 week TTL
|
||||||
|
|
||||||
|
data := map[string]any{
|
||||||
|
"group": group.Data["name"],
|
||||||
|
"key": key,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := h.emailing.Send("onboarding.new_member", r.FormValue("username"), data); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, "/app/administration/groups/"+group.ID, http.StatusFound)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h ApplicationHandler) AdminStatVehicles(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|
||||||
|
request := &fleets.GetBookingsRequest{}
|
||||||
|
resp, err := h.services.GRPC.Fleets.GetBookings(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusNotFound)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
bookings := []storage.Booking{}
|
||||||
|
|
||||||
|
for _, b := range resp.Bookings {
|
||||||
|
booking := b.ToStorageType()
|
||||||
|
bookings = append(bookings, booking)
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Sort(sorting.BookingsByStartdate(bookings))
|
||||||
|
|
||||||
|
vehicles, _ := h.services.GetVehiclesMap()
|
||||||
|
groups, _ := h.services.GetGroupsMap()
|
||||||
|
|
||||||
|
h.Renderer.VehicleBookingsList(w, r, bookings, vehicles, groups)
|
||||||
|
}
|
||||||
|
|
||||||
func (h *ApplicationHandler) members() ([]*accounts.Account, error) {
|
func (h *ApplicationHandler) members() ([]*accounts.Account, error) {
|
||||||
resp, err := h.services.GRPC.MobilityAccounts.GetAccounts(context.TODO(), &accounts.GetAccountsRequest{
|
resp, err := h.services.GRPC.MobilityAccounts.GetAccounts(context.TODO(), &accounts.GetAccountsRequest{
|
||||||
Namespaces: []string{"parcoursmob"},
|
Namespaces: []string{"parcoursmob"},
|
||||||
@@ -257,3 +368,30 @@ func (h *ApplicationHandler) members() ([]*accounts.Account, error) {
|
|||||||
|
|
||||||
return resp.Accounts, nil
|
return resp.Accounts, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) groupmembers(groupid string) (groupmembers []mobilityaccountsstorage.Account, admins []mobilityaccountsstorage.Account, err error) {
|
||||||
|
members, err := h.members()
|
||||||
|
if err != nil {
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
groupmembers = []mobilityaccountsstorage.Account{}
|
||||||
|
admins = []mobilityaccountsstorage.Account{}
|
||||||
|
|
||||||
|
for _, m := range members {
|
||||||
|
mm := m.ToStorageType()
|
||||||
|
for _, g := range mm.Data["groups"].([]any) {
|
||||||
|
if g.(string) == groupid {
|
||||||
|
groupmembers = append(groupmembers, mm)
|
||||||
|
}
|
||||||
|
if g.(string) == groupid+":admin" {
|
||||||
|
admins = append(admins, mm)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return groupmembers, admins, err
|
||||||
|
}
|
||||||
|
|||||||
@@ -7,10 +7,12 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
"sort"
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
formvalidators "git.coopgo.io/coopgo-apps/parcoursmob/utils/form-validators"
|
formvalidators "git.coopgo.io/coopgo-apps/parcoursmob/utils/form-validators"
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
||||||
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/sorting"
|
||||||
agenda "git.coopgo.io/coopgo-platform/agenda/grpcapi"
|
agenda "git.coopgo.io/coopgo-platform/agenda/grpcapi"
|
||||||
agendastorage "git.coopgo.io/coopgo-platform/agenda/storage"
|
agendastorage "git.coopgo.io/coopgo-platform/agenda/storage"
|
||||||
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
||||||
@@ -21,12 +23,6 @@ import (
|
|||||||
"google.golang.org/protobuf/types/known/timestamppb"
|
"google.golang.org/protobuf/types/known/timestamppb"
|
||||||
)
|
)
|
||||||
|
|
||||||
type EventsByStartdate []agendastorage.Event
|
|
||||||
|
|
||||||
func (e EventsByStartdate) Len() int { return len(e) }
|
|
||||||
func (e EventsByStartdate) Less(i, j int) bool { return e[i].Startdate.Before(e[j].Startdate) }
|
|
||||||
func (e EventsByStartdate) Swap(i, j int) { e[i], e[j] = e[j], e[i] }
|
|
||||||
|
|
||||||
type EventsForm struct {
|
type EventsForm struct {
|
||||||
Name string `json:"name" validate:"required"`
|
Name string `json:"name" validate:"required"`
|
||||||
Type string `json:"type" validate:"required"`
|
Type string `json:"type" validate:"required"`
|
||||||
@@ -60,7 +56,7 @@ func (h *ApplicationHandler) AgendaHome(w http.ResponseWriter, r *http.Request)
|
|||||||
responses = append(responses, e.ToStorageType())
|
responses = append(responses, e.ToStorageType())
|
||||||
}
|
}
|
||||||
|
|
||||||
sort.Sort(EventsByStartdate(responses))
|
sort.Sort(sorting.EventsByStartdate(responses))
|
||||||
|
|
||||||
groupsresp, err := h.services.GRPC.GroupsManagement.GetGroupsBatch(context.TODO(), &groupsmanagement.GetGroupsBatchRequest{
|
groupsresp, err := h.services.GRPC.GroupsManagement.GetGroupsBatch(context.TODO(), &groupsmanagement.GetGroupsBatchRequest{
|
||||||
Groupids: groupids,
|
Groupids: groupids,
|
||||||
@@ -94,8 +90,6 @@ func (h *ApplicationHandler) AgendaCreateEvent(w http.ResponseWriter, r *http.Re
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Println(eventForm)
|
|
||||||
|
|
||||||
data, _ := structpb.NewStruct(map[string]any{
|
data, _ := structpb.NewStruct(map[string]any{
|
||||||
"address": eventForm.Address,
|
"address": eventForm.Address,
|
||||||
})
|
})
|
||||||
@@ -114,6 +108,7 @@ func (h *ApplicationHandler) AgendaCreateEvent(w http.ResponseWriter, r *http.Re
|
|||||||
Allday: eventForm.Allday,
|
Allday: eventForm.Allday,
|
||||||
MaxSubscribers: int64(eventForm.MaxSubscribers),
|
MaxSubscribers: int64(eventForm.MaxSubscribers),
|
||||||
Data: data,
|
Data: data,
|
||||||
|
Deleted: false,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -126,7 +121,7 @@ func (h *ApplicationHandler) AgendaCreateEvent(w http.ResponseWriter, r *http.Re
|
|||||||
}
|
}
|
||||||
|
|
||||||
http.Redirect(w, r, fmt.Sprintf("/app/agenda/%s", resp.Event.Id), http.StatusFound)
|
http.Redirect(w, r, fmt.Sprintf("/app/agenda/%s", resp.Event.Id), http.StatusFound)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
h.Renderer.AgendaCreateEvent(w, r)
|
h.Renderer.AgendaCreateEvent(w, r)
|
||||||
}
|
}
|
||||||
@@ -212,6 +207,18 @@ func (h *ApplicationHandler) AgendaDisplayEvent(w http.ResponseWriter, r *http.R
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (h *ApplicationHandler) AgendaSubscribeEvent(w http.ResponseWriter, r *http.Request) {
|
func (h *ApplicationHandler) AgendaSubscribeEvent(w http.ResponseWriter, r *http.Request) {
|
||||||
|
current_group, err := h.currentGroup(r)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
current_user_token, current_user_claims, err := h.currentUser(r)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
vars := mux.Vars(r)
|
vars := mux.Vars(r)
|
||||||
eventid := vars["eventid"]
|
eventid := vars["eventid"]
|
||||||
|
|
||||||
@@ -220,15 +227,34 @@ func (h *ApplicationHandler) AgendaSubscribeEvent(w http.ResponseWriter, r *http
|
|||||||
w.WriteHeader(http.StatusBadRequest)
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
subscriber := r.FormValue("subscriber")
|
subscriber := r.FormValue("subscriber")
|
||||||
|
data := map[string]any{
|
||||||
|
"subscribed_by": map[string]any{
|
||||||
|
"user": map[string]any{
|
||||||
|
"id": current_user_token.Subject,
|
||||||
|
"display_name": current_user_claims["first_name"].(string) + " " + current_user_claims["last_name"].(string),
|
||||||
|
"email": current_user_claims["email"].(string),
|
||||||
|
},
|
||||||
|
"group": map[string]any{
|
||||||
|
"id": current_group.ID,
|
||||||
|
"name": current_group.Data["name"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
datapb, err := structpb.NewStruct(data)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
request := &agenda.SubscribeEventRequest{
|
request := &agenda.SubscribeEventRequest{
|
||||||
Eventid: eventid,
|
Eventid: eventid,
|
||||||
Subscriber: subscriber,
|
Subscriber: subscriber,
|
||||||
|
Data: datapb,
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err := h.services.GRPC.Agenda.SubscribeEvent(context.TODO(), request)
|
_, err = h.services.GRPC.Agenda.SubscribeEvent(context.TODO(), request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
@@ -306,11 +332,322 @@ func contains(s []*agenda.Subscription, e string) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// func contains[V string](s []V, e V) bool {
|
///////////////////////////////Update Event/////////////////////////////////////////
|
||||||
// for _, a := range s {
|
func (h *ApplicationHandler) AgendaUpdateEvent(w http.ResponseWriter, r *http.Request) {
|
||||||
// if a == e {
|
adm := strings.Split(r.URL.Path, "/")
|
||||||
// return true
|
eventID := adm[3]
|
||||||
// }
|
request := &agenda.GetEventRequest{
|
||||||
// }
|
Id: eventID,
|
||||||
// return false
|
}
|
||||||
// }
|
|
||||||
|
resp, err := h.services.GRPC.Agenda.GetEvent(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if r.Method == "POST" {
|
||||||
|
g := r.Context().Value(identification.GroupKey)
|
||||||
|
if g == nil {
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
group := g.(storage.Group)
|
||||||
|
|
||||||
|
eventForm, err := parseEventsForm(r)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
data, _ := structpb.NewStruct(map[string]any{
|
||||||
|
"address": eventForm.Address,
|
||||||
|
})
|
||||||
|
|
||||||
|
request := &agenda.UpdateEventRequest{
|
||||||
|
Event: &agenda.Event{
|
||||||
|
Namespace: "parcoursmob_dispositifs",
|
||||||
|
Id: eventID,
|
||||||
|
Owners: []string{group.ID},
|
||||||
|
Type: eventForm.Type,
|
||||||
|
Name: eventForm.Name,
|
||||||
|
Description: eventForm.Description,
|
||||||
|
Startdate: timestamppb.New(*eventForm.Startdate),
|
||||||
|
Enddate: timestamppb.New(*eventForm.Enddate),
|
||||||
|
Starttime: eventForm.Starttime,
|
||||||
|
Endtime: eventForm.Endtime,
|
||||||
|
Allday: eventForm.Allday,
|
||||||
|
MaxSubscribers: int64(eventForm.MaxSubscribers),
|
||||||
|
Data: data,
|
||||||
|
Subscriptions: resp.Event.Subscriptions,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := h.services.GRPC.Agenda.UpdateEvent(context.TODO(), request)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, fmt.Sprintf("/app/agenda/%s", resp.Event.Id), http.StatusFound)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
h.Renderer.AgendaUpdateEvent(w, r, resp.Event.ToStorageType())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) AgendaDeleteEvent(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
eventID := vars["eventid"]
|
||||||
|
request := &agenda.GetEventRequest{
|
||||||
|
Id: eventID,
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := h.services.GRPC.Agenda.GetEvent(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if r.Method == "POST" {
|
||||||
|
|
||||||
|
request := &agenda.UpdateEventRequest{
|
||||||
|
Event: &agenda.Event{
|
||||||
|
Namespace: resp.Event.Namespace,
|
||||||
|
Id: resp.Event.Id,
|
||||||
|
Owners: resp.Event.Owners,
|
||||||
|
Type: resp.Event.Type,
|
||||||
|
Name: resp.Event.Name,
|
||||||
|
Description: resp.Event.Description,
|
||||||
|
Startdate: resp.Event.Startdate,
|
||||||
|
Enddate: resp.Event.Enddate,
|
||||||
|
Starttime: resp.Event.Starttime,
|
||||||
|
Endtime: resp.Event.Endtime,
|
||||||
|
Allday: resp.Event.Allday,
|
||||||
|
MaxSubscribers: int64(resp.Event.MaxSubscribers),
|
||||||
|
Data: resp.Event.Data,
|
||||||
|
Subscriptions: resp.Event.Subscriptions,
|
||||||
|
Deleted: true,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := h.services.GRPC.Agenda.UpdateEvent(context.TODO(), request)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, "/app/agenda/", http.StatusFound)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
h.Renderer.AgendaDeleteEvent(w, r, resp.Event.ToStorageType())
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////Delete subscriber///////////////////////////////
|
||||||
|
func (h *ApplicationHandler) AgendaDeleteSubscribeEvent(w http.ResponseWriter, r *http.Request) {
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
eventId := vars["eventid"]
|
||||||
|
subscribeid := vars["subscribeid"]
|
||||||
|
s_b_id := ""
|
||||||
|
s_b_name := ""
|
||||||
|
s_b_email := ""
|
||||||
|
s_b_group_id := ""
|
||||||
|
s_b_group_name := ""
|
||||||
|
request := &agenda.GetEventRequest{
|
||||||
|
Id: eventId,
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := h.services.GRPC.Agenda.GetEvent(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := range resp.Event.Subscriptions {
|
||||||
|
if resp.Event.Subscriptions[i].Subscriber == subscribeid {
|
||||||
|
subscribed_by_id := resp.Event.Subscriptions[i].Data.Fields["subscribed_by"].GetStructValue().Fields["user"].GetStructValue().Fields["id"].GetStringValue()
|
||||||
|
subscribed_by_name := resp.Event.Subscriptions[i].Data.Fields["subscribed_by"].GetStructValue().Fields["user"].GetStructValue().Fields["display_name"].GetStringValue()
|
||||||
|
subscribed_by_email := resp.Event.Subscriptions[i].Data.Fields["subscribed_by"].GetStructValue().Fields["user"].GetStructValue().Fields["email"].GetStringValue()
|
||||||
|
subscribed_by_group_id := resp.Event.Subscriptions[i].Data.Fields["subscribed_by"].GetStructValue().Fields["group"].GetStructValue().Fields["id"].GetStringValue()
|
||||||
|
subscribed_by_group_name := resp.Event.Subscriptions[i].Data.Fields["subscribed_by"].GetStructValue().Fields["group"].GetStructValue().Fields["name"].GetStringValue()
|
||||||
|
s_b_id = subscribed_by_id
|
||||||
|
s_b_name = subscribed_by_name
|
||||||
|
s_b_email = subscribed_by_email
|
||||||
|
s_b_group_id = subscribed_by_group_id
|
||||||
|
s_b_group_name = subscribed_by_group_name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
current_group, err := h.currentGroup(r)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
current_user_token, current_user_claims, err := h.currentUser(r)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
data := map[string]any{
|
||||||
|
"subscribed_by": map[string]any{
|
||||||
|
"user": map[string]any{
|
||||||
|
"id": s_b_id,
|
||||||
|
"display_name": s_b_name,
|
||||||
|
"email": s_b_email,
|
||||||
|
},
|
||||||
|
"group": map[string]any{
|
||||||
|
"id": s_b_group_id,
|
||||||
|
"name": s_b_group_name,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"unsubscribed_by": map[string]any{
|
||||||
|
"user": map[string]any{
|
||||||
|
"id": current_user_token.Subject,
|
||||||
|
"display_name": current_user_claims["first_name"].(string) + " " + current_user_claims["last_name"].(string),
|
||||||
|
"email": current_user_claims["email"],
|
||||||
|
},
|
||||||
|
"group": map[string]any{
|
||||||
|
"id": current_group.ID,
|
||||||
|
"name": current_group.Data["name"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"motif": r.FormValue("motif"),
|
||||||
|
}
|
||||||
|
|
||||||
|
datapb, err := structpb.NewStruct(data)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if r.Method == "POST" {
|
||||||
|
request := &agenda.DeleteSubscriptionRequest{
|
||||||
|
Subscriber: subscribeid,
|
||||||
|
Eventid: eventId,
|
||||||
|
Data: datapb,
|
||||||
|
}
|
||||||
|
|
||||||
|
data := map[string]any{
|
||||||
|
"motif": r.FormValue("motif"),
|
||||||
|
"user": current_user_claims["first_name"].(string) + " " + current_user_claims["last_name"].(string),
|
||||||
|
"subscriber": fmt.Sprintf("http://localhost:9000/app/beneficiaries/%s", subscribeid),
|
||||||
|
"link": fmt.Sprintf("http://localhost:9000/app/agenda/%s", eventId),
|
||||||
|
}
|
||||||
|
|
||||||
|
// récupérer l'adresse mail de l'utilisateur qui a créé l'événement
|
||||||
|
mail := s_b_email
|
||||||
|
fmt.Println(mail)
|
||||||
|
|
||||||
|
_, err := h.services.GRPC.Agenda.DeleteSubscription(context.TODO(), request)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := h.emailing.Send("delete_subscriber.request", mail, data); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, fmt.Sprintf("/app/agenda/%s", eventId), http.StatusFound)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
h.Renderer.AgendaDeleteSubscribeEvent(w, r, eventId)
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////
|
||||||
|
// /////////////////////History Event////////////////////////
|
||||||
|
func (h *ApplicationHandler) AgendaHistoryEvent(w http.ResponseWriter, r *http.Request) {
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
eventId := vars["eventid"]
|
||||||
|
request := &agenda.GetEventRequest{
|
||||||
|
Id: eventId,
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := h.services.GRPC.Agenda.GetEvent(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
grouprequest := &groupsmanagement.GetGroupRequest{
|
||||||
|
Id: resp.Event.Owners[0],
|
||||||
|
}
|
||||||
|
|
||||||
|
groupresp, err := h.services.GRPC.GroupsManagement.GetGroup(context.TODO(), grouprequest)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
subscribers := map[string]any{}
|
||||||
|
|
||||||
|
accids := []string{}
|
||||||
|
for _, v := range resp.Event.DeletedSubscription {
|
||||||
|
accids = append(accids, v.Subscriber)
|
||||||
|
}
|
||||||
|
|
||||||
|
subscriberresp, err := h.services.GRPC.MobilityAccounts.GetAccountsBatch(
|
||||||
|
context.TODO(),
|
||||||
|
&mobilityaccounts.GetAccountsBatchRequest{
|
||||||
|
Accountids: accids,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if err == nil {
|
||||||
|
for _, sub := range subscriberresp.Accounts {
|
||||||
|
subscribers[sub.Id] = sub.ToStorageType()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
g := r.Context().Value(identification.GroupKey)
|
||||||
|
if g == nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
group := g.(storage.Group)
|
||||||
|
|
||||||
|
accountids := []string{}
|
||||||
|
for _, m := range group.Members {
|
||||||
|
if !contains(resp.Event.DeletedSubscription, m) {
|
||||||
|
accountids = append(accountids, m)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
accountresp, err := h.services.GRPC.MobilityAccounts.GetAccountsBatch(
|
||||||
|
context.TODO(),
|
||||||
|
&mobilityaccounts.GetAccountsBatchRequest{
|
||||||
|
Accountids: accountids,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
accounts := []any{}
|
||||||
|
|
||||||
|
if err == nil {
|
||||||
|
for _, acc := range accountresp.Accounts {
|
||||||
|
accounts = append(accounts, acc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
h.Renderer.AgendaHistoryEvent(w, r, resp.Event.ToStorageType(), groupresp.Group.ToStorageType(), subscribers, accounts)
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,12 +1,16 @@
|
|||||||
package application
|
package application
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/renderer"
|
"git.coopgo.io/coopgo-apps/parcoursmob/renderer"
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/services"
|
"git.coopgo.io/coopgo-apps/parcoursmob/services"
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/cache"
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
||||||
|
cache "git.coopgo.io/coopgo-apps/parcoursmob/utils/storage"
|
||||||
"git.coopgo.io/coopgo-platform/emailing"
|
"git.coopgo.io/coopgo-platform/emailing"
|
||||||
|
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
||||||
|
"github.com/coreos/go-oidc"
|
||||||
"github.com/spf13/viper"
|
"github.com/spf13/viper"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -14,11 +18,12 @@ type ApplicationHandler struct {
|
|||||||
config *viper.Viper
|
config *viper.Viper
|
||||||
Renderer *renderer.Renderer
|
Renderer *renderer.Renderer
|
||||||
services *services.ServicesHandler
|
services *services.ServicesHandler
|
||||||
cache *cache.CacheHandler
|
cache cache.CacheHandler
|
||||||
|
filestorage cache.FileStorage
|
||||||
emailing *emailing.Mailer
|
emailing *emailing.Mailer
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewApplicationHandler(cfg *viper.Viper, svc *services.ServicesHandler, cache *cache.CacheHandler, emailing *emailing.Mailer) (*ApplicationHandler, error) {
|
func NewApplicationHandler(cfg *viper.Viper, svc *services.ServicesHandler, cache cache.CacheHandler, filestorage cache.FileStorage, emailing *emailing.Mailer) (*ApplicationHandler, error) {
|
||||||
templates_root := cfg.GetString("templates.root")
|
templates_root := cfg.GetString("templates.root")
|
||||||
renderer := renderer.NewRenderer(cfg, templates_root)
|
renderer := renderer.NewRenderer(cfg, templates_root)
|
||||||
return &ApplicationHandler{
|
return &ApplicationHandler{
|
||||||
@@ -26,6 +31,7 @@ func NewApplicationHandler(cfg *viper.Viper, svc *services.ServicesHandler, cach
|
|||||||
Renderer: renderer,
|
Renderer: renderer,
|
||||||
services: svc,
|
services: svc,
|
||||||
cache: cache,
|
cache: cache,
|
||||||
|
filestorage: filestorage,
|
||||||
emailing: emailing,
|
emailing: emailing,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
@@ -37,3 +43,31 @@ func (h *ApplicationHandler) NotFound(w http.ResponseWriter, r *http.Request) {
|
|||||||
func (h *ApplicationHandler) templateFile(file string) string {
|
func (h *ApplicationHandler) templateFile(file string) string {
|
||||||
return h.config.GetString("templates.root") + file
|
return h.config.GetString("templates.root") + file
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) currentGroup(r *http.Request) (current_group storage.Group, err error) {
|
||||||
|
g := r.Context().Value(identification.GroupKey)
|
||||||
|
if g == nil {
|
||||||
|
return storage.Group{}, errors.New("current group not found")
|
||||||
|
}
|
||||||
|
current_group = g.(storage.Group)
|
||||||
|
|
||||||
|
return current_group, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) currentUser(r *http.Request) (current_user_token *oidc.IDToken, current_user_claims map[string]any, err error) {
|
||||||
|
// Get current user ID
|
||||||
|
u := r.Context().Value(identification.IdtokenKey)
|
||||||
|
if u == nil {
|
||||||
|
return nil, nil, errors.New("current user not found")
|
||||||
|
}
|
||||||
|
current_user_token = u.(*oidc.IDToken)
|
||||||
|
|
||||||
|
// Get current user claims
|
||||||
|
c := r.Context().Value(identification.ClaimsKey)
|
||||||
|
if c == nil {
|
||||||
|
return current_user_token, nil, errors.New("current user claims not found")
|
||||||
|
}
|
||||||
|
current_user_claims = c.(map[string]any)
|
||||||
|
|
||||||
|
return current_user_token, current_user_claims, nil
|
||||||
|
}
|
||||||
|
|||||||
@@ -7,8 +7,10 @@ import (
|
|||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"image/png"
|
"image/png"
|
||||||
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
@@ -16,10 +18,13 @@ import (
|
|||||||
formvalidators "git.coopgo.io/coopgo-apps/parcoursmob/utils/form-validators"
|
formvalidators "git.coopgo.io/coopgo-apps/parcoursmob/utils/form-validators"
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
||||||
profilepictures "git.coopgo.io/coopgo-apps/parcoursmob/utils/profile-pictures"
|
profilepictures "git.coopgo.io/coopgo-apps/parcoursmob/utils/profile-pictures"
|
||||||
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/sorting"
|
||||||
|
filestorage "git.coopgo.io/coopgo-apps/parcoursmob/utils/storage"
|
||||||
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
||||||
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
||||||
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
||||||
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
||||||
|
mobilityaccountsstorage "git.coopgo.io/coopgo-platform/mobility-accounts/storage"
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
"google.golang.org/protobuf/types/known/structpb"
|
"google.golang.org/protobuf/types/known/structpb"
|
||||||
@@ -29,8 +34,9 @@ type BeneficiariesForm struct {
|
|||||||
FirstName string `json:"first_name" validate:"required"`
|
FirstName string `json:"first_name" validate:"required"`
|
||||||
LastName string `json:"last_name" validate:"required"`
|
LastName string `json:"last_name" validate:"required"`
|
||||||
Email string `json:"email" validate:"required,email"`
|
Email string `json:"email" validate:"required,email"`
|
||||||
Birthdate *time.Time `json:"birthdate"`
|
Birthdate *time.Time `json:"birthdate" validate:"required"`
|
||||||
PhoneNumber string `json:"phone_number" validate:"required,phoneNumber"`
|
PhoneNumber string `json:"phone_number" validate:"required,phoneNumber"`
|
||||||
|
FileNumber string `json:"file_number"`
|
||||||
Address any `json:"address,omitempty"`
|
Address any `json:"address,omitempty"`
|
||||||
Gender string `json:"gender"`
|
Gender string `json:"gender"`
|
||||||
}
|
}
|
||||||
@@ -44,6 +50,8 @@ func (h *ApplicationHandler) BeneficiariesList(w http.ResponseWriter, r *http.Re
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sort.Sort(sorting.BeneficiariesByName(accounts))
|
||||||
|
|
||||||
cacheid := uuid.NewString()
|
cacheid := uuid.NewString()
|
||||||
h.cache.PutWithTTL(cacheid, accounts, 1*time.Hour)
|
h.cache.PutWithTTL(cacheid, accounts, 1*time.Hour)
|
||||||
|
|
||||||
@@ -53,12 +61,12 @@ func (h *ApplicationHandler) BeneficiariesList(w http.ResponseWriter, r *http.Re
|
|||||||
func (h *ApplicationHandler) BeneficiaryCreate(w http.ResponseWriter, r *http.Request) {
|
func (h *ApplicationHandler) BeneficiaryCreate(w http.ResponseWriter, r *http.Request) {
|
||||||
g := r.Context().Value(identification.GroupKey)
|
g := r.Context().Value(identification.GroupKey)
|
||||||
if g == nil {
|
if g == nil {
|
||||||
|
fmt.Println("Create beneficiary : could not find group")
|
||||||
w.WriteHeader(http.StatusBadRequest)
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
group := g.(storage.Group)
|
group := g.(storage.Group)
|
||||||
fmt.Println(group)
|
|
||||||
|
|
||||||
if r.Method == "POST" {
|
if r.Method == "POST" {
|
||||||
|
|
||||||
@@ -115,6 +123,8 @@ func (h *ApplicationHandler) BeneficiaryDisplay(w http.ResponseWriter, r *http.R
|
|||||||
vars := mux.Vars(r)
|
vars := mux.Vars(r)
|
||||||
beneficiaryID := vars["beneficiaryid"]
|
beneficiaryID := vars["beneficiaryid"]
|
||||||
|
|
||||||
|
documents := h.filestorage.List(filestorage.PREFIX_BENEFICIARIES + "/" + beneficiaryID)
|
||||||
|
|
||||||
request := &mobilityaccounts.GetAccountRequest{
|
request := &mobilityaccounts.GetAccountRequest{
|
||||||
Id: beneficiaryID,
|
Id: beneficiaryID,
|
||||||
}
|
}
|
||||||
@@ -126,9 +136,6 @@ func (h *ApplicationHandler) BeneficiaryDisplay(w http.ResponseWriter, r *http.R
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
//TODO filter namespaces
|
|
||||||
//TODO filter groups
|
|
||||||
|
|
||||||
bookingsrequest := &fleets.GetDriverBookingsRequest{
|
bookingsrequest := &fleets.GetDriverBookingsRequest{
|
||||||
Driver: beneficiaryID,
|
Driver: beneficiaryID,
|
||||||
}
|
}
|
||||||
@@ -145,7 +152,25 @@ func (h *ApplicationHandler) BeneficiaryDisplay(w http.ResponseWriter, r *http.R
|
|||||||
bookings = append(bookings, b.ToStorageType())
|
bookings = append(bookings, b.ToStorageType())
|
||||||
}
|
}
|
||||||
|
|
||||||
h.Renderer.BeneficiaryDisplay(w, r, resp.Account.ToStorageType(), bookings)
|
groupsrequest := &groupsmanagement.GetGroupsRequest{
|
||||||
|
Namespaces: []string{"parcoursmob_organizations"},
|
||||||
|
Member: beneficiaryID,
|
||||||
|
}
|
||||||
|
|
||||||
|
groupsresp, err := h.services.GRPC.GroupsManagement.GetGroups(context.TODO(), groupsrequest)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
organizations := []any{}
|
||||||
|
for _, o := range groupsresp.Groups {
|
||||||
|
organizations = append(organizations, o.ToStorageType())
|
||||||
|
}
|
||||||
|
|
||||||
|
beneficiaries_file_types := h.config.GetStringSlice("modules.beneficiaries.documents_types")
|
||||||
|
file_types_map := h.config.GetStringMapString("storage.files.file_types")
|
||||||
|
|
||||||
|
h.Renderer.BeneficiaryDisplay(w, r, resp.Account.ToStorageType(), bookings, organizations, beneficiaries_file_types, file_types_map, documents)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *ApplicationHandler) BeneficiaryUpdate(w http.ResponseWriter, r *http.Request) {
|
func (h *ApplicationHandler) BeneficiaryUpdate(w http.ResponseWriter, r *http.Request) {
|
||||||
@@ -239,6 +264,63 @@ func (h *ApplicationHandler) BeneficiaryPicture(w http.ResponseWriter, r *http.R
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) BeneficiaryDocuments(w http.ResponseWriter, r *http.Request) {
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
beneficiaryID := vars["beneficiaryid"]
|
||||||
|
|
||||||
|
//r.ParseForm()
|
||||||
|
r.ParseMultipartForm(10 * 1024 * 1024)
|
||||||
|
|
||||||
|
document_type := r.FormValue("type")
|
||||||
|
document_name := r.FormValue("name")
|
||||||
|
|
||||||
|
file, header, err := r.FormFile("file-upload")
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
fileid := uuid.NewString()
|
||||||
|
|
||||||
|
metadata := map[string]string{
|
||||||
|
"type": document_type,
|
||||||
|
"name": document_name,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := h.filestorage.Put(file, filestorage.PREFIX_BENEFICIARIES, fmt.Sprintf("%s/%s_%s", beneficiaryID, fileid, header.Filename), header.Size, metadata); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, fmt.Sprintf("/app/beneficiaries/%s", beneficiaryID), http.StatusFound)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) BeneficiaryDocumentDownload(w http.ResponseWriter, r *http.Request) {
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
beneficiaryID := vars["beneficiaryid"]
|
||||||
|
document := vars["document"]
|
||||||
|
|
||||||
|
file, info, err := h.filestorage.Get(filestorage.PREFIX_BENEFICIARIES, fmt.Sprintf("%s/%s", beneficiaryID, document))
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
w.Header().Set("Content-Type", info.ContentType)
|
||||||
|
if _, err = io.Copy(w, file); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, fmt.Sprintf("/app/beneficiaries/%s", beneficiaryID), http.StatusFound)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
func filterAccount(r *http.Request, a *mobilityaccounts.Account) bool {
|
func filterAccount(r *http.Request, a *mobilityaccounts.Account) bool {
|
||||||
searchFilter, ok := r.URL.Query()["search"]
|
searchFilter, ok := r.URL.Query()["search"]
|
||||||
|
|
||||||
@@ -252,8 +334,8 @@ func filterAccount(r *http.Request, a *mobilityaccounts.Account) bool {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *ApplicationHandler) beneficiaries(r *http.Request) ([]any, error) {
|
func (h *ApplicationHandler) beneficiaries(r *http.Request) ([]mobilityaccountsstorage.Account, error) {
|
||||||
var accounts = []any{}
|
var accounts = []mobilityaccountsstorage.Account{}
|
||||||
g := r.Context().Value(identification.GroupKey)
|
g := r.Context().Value(identification.GroupKey)
|
||||||
if g == nil {
|
if g == nil {
|
||||||
return accounts, errors.New("no group provided")
|
return accounts, errors.New("no group provided")
|
||||||
@@ -301,6 +383,7 @@ func parseBeneficiariesForm(r *http.Request) (map[string]any, error) {
|
|||||||
Email: r.PostFormValue("email"),
|
Email: r.PostFormValue("email"),
|
||||||
Birthdate: date,
|
Birthdate: date,
|
||||||
PhoneNumber: r.PostFormValue("phone_number"),
|
PhoneNumber: r.PostFormValue("phone_number"),
|
||||||
|
FileNumber: r.PostFormValue("file_number"),
|
||||||
Gender: r.PostFormValue("gender"),
|
Gender: r.PostFormValue("gender"),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -7,10 +7,12 @@ import (
|
|||||||
"sort"
|
"sort"
|
||||||
|
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
||||||
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/sorting"
|
||||||
agenda "git.coopgo.io/coopgo-platform/agenda/grpcapi"
|
agenda "git.coopgo.io/coopgo-platform/agenda/grpcapi"
|
||||||
agendastorage "git.coopgo.io/coopgo-platform/agenda/storage"
|
agendastorage "git.coopgo.io/coopgo-platform/agenda/storage"
|
||||||
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
||||||
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
||||||
|
"google.golang.org/protobuf/types/known/timestamppb"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (h *ApplicationHandler) Dashboard(w http.ResponseWriter, r *http.Request) {
|
func (h *ApplicationHandler) Dashboard(w http.ResponseWriter, r *http.Request) {
|
||||||
@@ -50,7 +52,7 @@ func (h *ApplicationHandler) Dashboard(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
members, err := h.members()
|
members, _, err := h.groupmembers(group.ID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
@@ -63,13 +65,14 @@ func (h *ApplicationHandler) Dashboard(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
eventsresp, err := h.services.GRPC.Agenda.GetEvents(context.TODO(), &agenda.GetEventsRequest{
|
eventsresp, err := h.services.GRPC.Agenda.GetEvents(context.TODO(), &agenda.GetEventsRequest{
|
||||||
Namespaces: []string{"parcoursmob_dispositifs"},
|
Namespaces: []string{"parcoursmob_dispositifs"},
|
||||||
|
Mindate: timestamppb.Now(),
|
||||||
})
|
})
|
||||||
|
|
||||||
for _, e := range eventsresp.Events {
|
for _, e := range eventsresp.Events {
|
||||||
events = append(events, e.ToStorageType())
|
events = append(events, e.ToStorageType())
|
||||||
}
|
}
|
||||||
|
|
||||||
sort.Sort(EventsByStartdate(events))
|
sort.Sort(sorting.EventsByStartdate(events))
|
||||||
|
|
||||||
h.Renderer.Dashboard(w, r, accounts, count, count_members, events)
|
h.Renderer.Dashboard(w, r, accounts, count, count_members, events)
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package application
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
"sort"
|
"sort"
|
||||||
@@ -16,6 +17,8 @@ import (
|
|||||||
"google.golang.org/protobuf/types/known/structpb"
|
"google.golang.org/protobuf/types/known/structpb"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var Addres any
|
||||||
|
|
||||||
type BeneficiariesGroupForm struct {
|
type BeneficiariesGroupForm struct {
|
||||||
FirstName string `json:"first_name" validate:"required"`
|
FirstName string `json:"first_name" validate:"required"`
|
||||||
LastName string `json:"last_name" validate:"required"`
|
LastName string `json:"last_name" validate:"required"`
|
||||||
@@ -61,6 +64,12 @@ func (h *ApplicationHandler) Groups(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
func (h *ApplicationHandler) CreateGroupModule(w http.ResponseWriter, r *http.Request) {
|
func (h *ApplicationHandler) CreateGroupModule(w http.ResponseWriter, r *http.Request) {
|
||||||
if r.Method == "POST" {
|
if r.Method == "POST" {
|
||||||
|
if r.PostFormValue("address") != "" {
|
||||||
|
var a any
|
||||||
|
json.Unmarshal([]byte(r.PostFormValue("address")), &a)
|
||||||
|
|
||||||
|
Addres = a
|
||||||
|
}
|
||||||
r.ParseForm()
|
r.ParseForm()
|
||||||
|
|
||||||
if r.FormValue("name") == "" {
|
if r.FormValue("name") == "" {
|
||||||
@@ -81,6 +90,8 @@ func (h *ApplicationHandler) CreateGroupModule(w http.ResponseWriter, r *http.Re
|
|||||||
dataMap := map[string]any{
|
dataMap := map[string]any{
|
||||||
"name": r.FormValue("name"),
|
"name": r.FormValue("name"),
|
||||||
"type": r.FormValue("type"),
|
"type": r.FormValue("type"),
|
||||||
|
"description": r.FormValue("description"),
|
||||||
|
"address": Addres,
|
||||||
}
|
}
|
||||||
|
|
||||||
data, err := structpb.NewValue(dataMap)
|
data, err := structpb.NewValue(dataMap)
|
||||||
@@ -138,7 +149,6 @@ func (h *ApplicationHandler) DisplayGroupModule(w http.ResponseWriter, r *http.R
|
|||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
fmt.Println(resp.Group.Members)
|
|
||||||
|
|
||||||
var accounts = []any{}
|
var accounts = []any{}
|
||||||
|
|
||||||
@@ -150,14 +160,12 @@ func (h *ApplicationHandler) DisplayGroupModule(w http.ResponseWriter, r *http.R
|
|||||||
// if err != nil {
|
// if err != nil {
|
||||||
// return err
|
// return err
|
||||||
// }
|
// }
|
||||||
|
|
||||||
for _, account := range ressp.Accounts {
|
for _, account := range ressp.Accounts {
|
||||||
if filterAcccount(r, account) {
|
if filterAcccount(r, account) {
|
||||||
a := account.ToStorageType()
|
a := account.ToStorageType()
|
||||||
accounts = append(accounts, a)
|
accounts = append(accounts, a)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fmt.Println(accounts)
|
|
||||||
|
|
||||||
cacheid := uuid.NewString()
|
cacheid := uuid.NewString()
|
||||||
h.cache.PutWithTTL(cacheid, accounts, 1*time.Hour)
|
h.cache.PutWithTTL(cacheid, accounts, 1*time.Hour)
|
||||||
@@ -198,22 +206,16 @@ func (h *ApplicationHandler) DisplayGroupModule(w http.ResponseWriter, r *http.R
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Println("yeesssssssssssssssssssssssss")
|
|
||||||
fmt.Println(respbeneficiary.Account.Id)
|
|
||||||
fmt.Println("===================yeesssssssssssssssssssssssss")
|
|
||||||
fmt.Println(r.FormValue("beneficiaryid"))
|
|
||||||
fmt.Println("=====================yeesssssssssssssssssssssssss")
|
|
||||||
// }
|
|
||||||
http.Redirect(w, r, fmt.Sprintf("/app/group_module/groups/%s", resp.Group.ToStorageType().ID), http.StatusFound)
|
http.Redirect(w, r, fmt.Sprintf("/app/group_module/groups/%s", resp.Group.ToStorageType().ID), http.StatusFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
accountsB, err := h.beneficiaries(r)
|
accountsBeneficaire, err := h.beneficiaries(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
w.WriteHeader(http.StatusBadRequest)
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
//h.Renderer.BeneficaireSearch(w, r, accounts, searched, beneficiary, resp.Group.ToStorageType())
|
//h.Renderer.BeneficaireSearch(w, r, accounts, searched, beneficiary, resp.Group.ToStorageType())
|
||||||
h.Renderer.DisplayGroupModule(w, r, resp.Group.ToStorageType().ID, accounts, cacheid, searched, beneficiary, resp.Group.ToStorageType(), accountsB)
|
h.Renderer.DisplayGroupModule(w, r, resp.Group.ToStorageType().ID, accounts, cacheid, searched, beneficiary, resp.Group.ToStorageType(), accountsBeneficaire)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,14 +5,26 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"sort"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
||||||
|
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
||||||
|
groupstorage "git.coopgo.io/coopgo-platform/groups-management/storage"
|
||||||
|
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"github.com/gorilla/mux"
|
||||||
geojson "github.com/paulmach/go.geojson"
|
geojson "github.com/paulmach/go.geojson"
|
||||||
"gitlab.scity.coop/maas/navitia-golang"
|
"gitlab.scity.coop/maas/navitia-golang"
|
||||||
"gitlab.scity.coop/maas/navitia-golang/types"
|
"gitlab.scity.coop/maas/navitia-golang/types"
|
||||||
|
"google.golang.org/protobuf/types/known/structpb"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var Depart any
|
||||||
|
var Arrive any
|
||||||
|
|
||||||
func (h *ApplicationHandler) JourneysSearch(w http.ResponseWriter, r *http.Request) {
|
func (h *ApplicationHandler) JourneysSearch(w http.ResponseWriter, r *http.Request) {
|
||||||
r.ParseForm()
|
r.ParseForm()
|
||||||
|
|
||||||
@@ -26,7 +38,6 @@ func (h *ApplicationHandler) JourneysSearch(w http.ResponseWriter, r *http.Reque
|
|||||||
departuredate := r.FormValue("departuredate")
|
departuredate := r.FormValue("departuredate")
|
||||||
departuretime := r.FormValue("departuretime")
|
departuretime := r.FormValue("departuretime")
|
||||||
departuredatetime, _ := time.ParseInLocation("2006-01-02 15:04", fmt.Sprintf("%s %s", departuredate, departuretime), locTime)
|
departuredatetime, _ := time.ParseInLocation("2006-01-02 15:04", fmt.Sprintf("%s %s", departuredate, departuretime), locTime)
|
||||||
fmt.Println(departuredatetime)
|
|
||||||
|
|
||||||
departure := r.FormValue("departure")
|
departure := r.FormValue("departure")
|
||||||
destination := r.FormValue("destination")
|
destination := r.FormValue("destination")
|
||||||
@@ -70,7 +81,7 @@ func (h *ApplicationHandler) JourneysSearch(w http.ResponseWriter, r *http.Reque
|
|||||||
w.WriteHeader(http.StatusBadRequest)
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
fmt.Println(session)
|
|
||||||
request := navitia.JourneyRequest{
|
request := navitia.JourneyRequest{
|
||||||
From: types.ID(fmt.Sprintf("%f", departuregeo.Geometry.Point[0]) + ";" + fmt.Sprintf("%f", departuregeo.Geometry.Point[1])),
|
From: types.ID(fmt.Sprintf("%f", departuregeo.Geometry.Point[0]) + ";" + fmt.Sprintf("%f", departuregeo.Geometry.Point[1])),
|
||||||
To: types.ID(fmt.Sprintf("%f", destinationgeo.Geometry.Point[0]) + ";" + fmt.Sprintf("%f", destinationgeo.Geometry.Point[1])),
|
To: types.ID(fmt.Sprintf("%f", destinationgeo.Geometry.Point[0]) + ";" + fmt.Sprintf("%f", destinationgeo.Geometry.Point[1])),
|
||||||
@@ -81,8 +92,8 @@ func (h *ApplicationHandler) JourneysSearch(w http.ResponseWriter, r *http.Reque
|
|||||||
journeys, err = session.Journeys(context.Background(), request)
|
journeys, err = session.Journeys(context.Background(), request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
w.WriteHeader(http.StatusBadRequest)
|
// w.WriteHeader(http.StatusBadRequest)
|
||||||
return
|
// return
|
||||||
}
|
}
|
||||||
|
|
||||||
//CARPOOL
|
//CARPOOL
|
||||||
@@ -93,8 +104,6 @@ func (h *ApplicationHandler) JourneysSearch(w http.ResponseWriter, r *http.Reque
|
|||||||
// departuredatetime.Format("2006-01-02"), departuredatetime.Add(24*time.Hour).Format("2006-01-02"))
|
// departuredatetime.Format("2006-01-02"), departuredatetime.Add(24*time.Hour).Format("2006-01-02"))
|
||||||
carpoolrequest := "https://api.rdex.ridygo.fr/journeys.json"
|
carpoolrequest := "https://api.rdex.ridygo.fr/journeys.json"
|
||||||
|
|
||||||
fmt.Println(carpoolrequest)
|
|
||||||
|
|
||||||
client := &http.Client{}
|
client := &http.Client{}
|
||||||
req, err := http.NewRequest("GET", carpoolrequest, nil)
|
req, err := http.NewRequest("GET", carpoolrequest, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -113,11 +122,19 @@ func (h *ApplicationHandler) JourneysSearch(w http.ResponseWriter, r *http.Reque
|
|||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if err == nil && resp.StatusCode == http.StatusOK {
|
||||||
err = json.NewDecoder(resp.Body).Decode(&carpoolresults)
|
err = json.NewDecoder(resp.Body).Decode(&carpoolresults)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if carpoolresults == nil {
|
||||||
|
carpoolresults = []any{}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
carpoolresults = []any{}
|
||||||
|
}
|
||||||
|
|
||||||
// Vehicles
|
// Vehicles
|
||||||
|
|
||||||
vehiclerequest := &fleets.GetVehiclesRequest{
|
vehiclerequest := &fleets.GetVehiclesRequest{
|
||||||
@@ -139,3 +156,413 @@ func (h *ApplicationHandler) JourneysSearch(w http.ResponseWriter, r *http.Reque
|
|||||||
|
|
||||||
h.Renderer.JourneysSearch(w, r, carpoolresults, journeys, vehicles, searched, departuregeo, destinationgeo, departuredate, departuretime)
|
h.Renderer.JourneysSearch(w, r, carpoolresults, journeys, vehicles, searched, departuregeo, destinationgeo, departuredate, departuretime)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type GroupsModule []groupstorage.Group
|
||||||
|
|
||||||
|
func (a GroupsModule) Len() int { return len(a) }
|
||||||
|
func (a GroupsModule) Less(i, j int) bool {
|
||||||
|
return strings.Compare(a[i].Data["name"].(string), a[j].Data["name"].(string)) < 0
|
||||||
|
}
|
||||||
|
func (a GroupsModule) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) GroupsGestion(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|
||||||
|
request := &groupsmanagement.GetGroupsRequest{
|
||||||
|
Namespaces: []string{"parcoursmob_groups_covoiturage"},
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := h.services.GRPC.GroupsManagement.GetGroups(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var groups = []groupstorage.Group{}
|
||||||
|
|
||||||
|
for _, group := range resp.Groups {
|
||||||
|
g := group.ToStorageType()
|
||||||
|
groups = append(groups, g)
|
||||||
|
}
|
||||||
|
|
||||||
|
cacheid := uuid.NewString()
|
||||||
|
|
||||||
|
sort.Sort(GroupsModule(groups))
|
||||||
|
h.cache.PutWithTTL(cacheid, groups, 1*time.Hour)
|
||||||
|
|
||||||
|
h.Renderer.GroupsGestion(w, r, groups, cacheid)
|
||||||
|
}
|
||||||
|
func filterAcc(r *http.Request, a *mobilityaccounts.Account) bool {
|
||||||
|
searchFilter, ok := r.URL.Query()["search"]
|
||||||
|
|
||||||
|
if ok && len(searchFilter[0]) > 0 {
|
||||||
|
name := a.Data.AsMap()["first_name"].(string) + " " + a.Data.AsMap()["last_name"].(string)
|
||||||
|
if !strings.Contains(strings.ToLower(name), strings.ToLower(searchFilter[0])) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) CreateGroup(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|
||||||
|
var beneficiary any
|
||||||
|
var (
|
||||||
|
departurgeo *geojson.Feature
|
||||||
|
dstinationgeo *geojson.Feature
|
||||||
|
)
|
||||||
|
searched := false
|
||||||
|
|
||||||
|
if r.FormValue("beneficiaryid") != "" {
|
||||||
|
|
||||||
|
searched = true
|
||||||
|
|
||||||
|
requestbeneficiary := &mobilityaccounts.GetAccountRequest{
|
||||||
|
Id: r.FormValue("beneficiaryid"),
|
||||||
|
}
|
||||||
|
|
||||||
|
respbeneficiary, err := h.services.GRPC.MobilityAccounts.GetAccount(context.TODO(), requestbeneficiary)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
beneficiary = respbeneficiary.Account.ToStorageType()
|
||||||
|
|
||||||
|
if r.Method == "POST" {
|
||||||
|
departure := r.FormValue("departure")
|
||||||
|
destination := r.FormValue("destination")
|
||||||
|
|
||||||
|
if departure != "" && destination != "" {
|
||||||
|
|
||||||
|
var err error
|
||||||
|
|
||||||
|
departurgeo, err = geojson.UnmarshalFeature([]byte(departure))
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
dstinationgeo, err = geojson.UnmarshalFeature([]byte(destination))
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if r.FormValue("departure") != "" {
|
||||||
|
var a any
|
||||||
|
json.Unmarshal([]byte(r.FormValue("departure")), &a)
|
||||||
|
|
||||||
|
Depart = a
|
||||||
|
}
|
||||||
|
if r.FormValue("destination") != "" {
|
||||||
|
var a any
|
||||||
|
json.Unmarshal([]byte(r.FormValue("destination")), &a)
|
||||||
|
|
||||||
|
Arrive = a
|
||||||
|
}
|
||||||
|
r.ParseForm()
|
||||||
|
|
||||||
|
if r.FormValue("name") == "" {
|
||||||
|
|
||||||
|
fmt.Println("invalid name")
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if r.FormValue("number") == "" {
|
||||||
|
|
||||||
|
fmt.Println("invalid number of personne")
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
planDays := map[string]any{
|
||||||
|
"lundi": r.FormValue("lundi") == "on",
|
||||||
|
"mardi": r.FormValue("mardi") == "on",
|
||||||
|
"mercredi": r.FormValue("mercredi") == "on",
|
||||||
|
"jeudi": r.FormValue("jeudi") == "on",
|
||||||
|
"vendredi": r.FormValue("vendredi") == "on",
|
||||||
|
"samedi": r.FormValue("samedi") == "on",
|
||||||
|
"dimanche": r.FormValue("dimanche") == "on",
|
||||||
|
}
|
||||||
|
|
||||||
|
groupidd := uuid.NewString()
|
||||||
|
|
||||||
|
dataMap := map[string]any{
|
||||||
|
"name": r.FormValue("name"),
|
||||||
|
"number": r.FormValue("number"),
|
||||||
|
"driver_first_name": respbeneficiary.Account.ToStorageType().Data["first_name"],
|
||||||
|
"driver_last_name": respbeneficiary.Account.ToStorageType().Data["last_name"],
|
||||||
|
"depart": Depart,
|
||||||
|
"arrive": Arrive,
|
||||||
|
"departdate": r.FormValue("departdate"),
|
||||||
|
"date": r.FormValue("date"),
|
||||||
|
"enddate": r.FormValue("enddate"),
|
||||||
|
"departtime": r.FormValue("departtime"),
|
||||||
|
"time": r.FormValue("time"),
|
||||||
|
|
||||||
|
"planDays": planDays,
|
||||||
|
"recurrent": r.FormValue("recurrent"),
|
||||||
|
"pontuelle": r.FormValue("ponctuelle"),
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := structpb.NewValue(dataMap)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
request_organization := &groupsmanagement.AddGroupRequest{
|
||||||
|
Group: &groupsmanagement.Group{
|
||||||
|
Id: groupidd,
|
||||||
|
Namespace: "parcoursmob_groups_covoiturage",
|
||||||
|
Data: data.GetStructValue(),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = h.services.GRPC.GroupsManagement.AddGroup(context.TODO(), request_organization)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, fmt.Sprintf("/app/journeys/groups_covoiturage/create/%s", request_organization.Group.ToStorageType().ID), http.StatusFound)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
accountsBeneficaire, err := h.beneficiaries(r)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
h.Renderer.CreateGroup(w, r, Depart, Arrive, searched, beneficiary, accountsBeneficaire, departurgeo, dstinationgeo)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) DisplayGroupCovoiturage(w http.ResponseWriter, r *http.Request) {
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
groupid := vars["groupid"]
|
||||||
|
|
||||||
|
request := &groupsmanagement.GetGroupRequest{
|
||||||
|
Id: groupid,
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := h.services.GRPC.GroupsManagement.GetGroup(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var accounts = []any{}
|
||||||
|
|
||||||
|
requesst := &mobilityaccounts.GetAccountsBatchRequest{
|
||||||
|
Accountids: resp.Group.Members,
|
||||||
|
}
|
||||||
|
|
||||||
|
ressp, _ := h.services.GRPC.MobilityAccounts.GetAccountsBatch(context.TODO(), requesst)
|
||||||
|
|
||||||
|
for _, account := range ressp.Accounts {
|
||||||
|
if filterAcc(r, account) {
|
||||||
|
a := account.ToStorageType()
|
||||||
|
accounts = append(accounts, a)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cacheid := uuid.NewString()
|
||||||
|
h.cache.PutWithTTL(cacheid, accounts, 1*time.Hour)
|
||||||
|
r.ParseForm()
|
||||||
|
|
||||||
|
var beneficiary any
|
||||||
|
searched := false
|
||||||
|
|
||||||
|
if r.FormValue("beneficiaryid") != "" {
|
||||||
|
|
||||||
|
searched = true
|
||||||
|
|
||||||
|
requestbeneficiary := &mobilityaccounts.GetAccountRequest{
|
||||||
|
Id: r.FormValue("beneficiaryid"),
|
||||||
|
}
|
||||||
|
|
||||||
|
respbeneficiary, err := h.services.GRPC.MobilityAccounts.GetAccount(context.TODO(), requestbeneficiary)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
beneficiary = respbeneficiary.Account.ToStorageType()
|
||||||
|
|
||||||
|
subscribe := &groupsmanagement.SubscribeRequest{
|
||||||
|
Groupid: resp.Group.ToStorageType().ID,
|
||||||
|
Memberid: respbeneficiary.Account.Id,
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = h.services.GRPC.GroupsManagement.Subscribe(context.TODO(), subscribe)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
/*******************Code to store more information about mermbers groupscovoiturage**************/
|
||||||
|
if r.FormValue("departure") != "" {
|
||||||
|
var a any
|
||||||
|
json.Unmarshal([]byte(r.FormValue("departure")), &a)
|
||||||
|
|
||||||
|
Depart = a
|
||||||
|
}
|
||||||
|
if r.FormValue("destination") != "" {
|
||||||
|
var a any
|
||||||
|
json.Unmarshal([]byte(r.FormValue("destination")), &a)
|
||||||
|
|
||||||
|
Arrive = a
|
||||||
|
}
|
||||||
|
r.ParseForm()
|
||||||
|
dataMap := map[string]any{
|
||||||
|
|
||||||
|
"depart": Depart,
|
||||||
|
"arrive": Arrive,
|
||||||
|
}
|
||||||
|
id := uuid.NewString()
|
||||||
|
data, err := structpb.NewValue(dataMap)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
request_organizatio := &groupsmanagement.AddGroupMemberRequest{
|
||||||
|
Group: &groupsmanagement.GroupMember{
|
||||||
|
Id: id,
|
||||||
|
Memberid: respbeneficiary.Account.Id,
|
||||||
|
Groupid: resp.Group.ToStorageType().ID,
|
||||||
|
Data: data.GetStructValue(),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = h.services.GRPC.GroupsManagement.AddGroupMember(context.TODO(), request_organizatio)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, fmt.Sprintf("/app/journeys/groups_covoiturage/create/%s", resp.Group.ToStorageType().ID), http.StatusFound)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
//////////find all groups to store the adresse passenger///////
|
||||||
|
// grp := &groupsmanagement.GetGroupsBatchMemberRequest{
|
||||||
|
|
||||||
|
// Groupids: []string{resp.Group.ToStorageType().ID},
|
||||||
|
// }
|
||||||
|
// s, err := h.services.GRPC.GroupsManagement.GetGroupsBatchMember(context.TODO(), grp)
|
||||||
|
// if err != nil {
|
||||||
|
// fmt.Println(err)
|
||||||
|
// w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
// return
|
||||||
|
// }
|
||||||
|
// groups := map[string]any{}
|
||||||
|
|
||||||
|
// if err == nil {
|
||||||
|
// for _, g := range s.Groups {
|
||||||
|
// groups[g.Memberid] = g.ToStorageType()
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//////////find all groups to store the adresse passenger///////
|
||||||
|
///////////try to optimise the code ////////////////////////////
|
||||||
|
groups, _ := h.services.GetGroupsMemberMap(resp.Group.ToStorageType().ID)
|
||||||
|
//fmt.Println(groups)
|
||||||
|
var number string = strconv.Itoa(len(resp.Group.Members))
|
||||||
|
/////////////////////
|
||||||
|
accountsBeneficaire, err := h.beneficiaries(r)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
h.Renderer.DisplayGroupCovoiturage(w, r, number, resp.Group.ToStorageType().ID, Depart, Arrive, accounts, cacheid, searched, beneficiary, resp.Group.ToStorageType(), accountsBeneficaire, groups)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) UpdateGroupCovoiturage(w http.ResponseWriter, r *http.Request) {
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
id := vars["id"]
|
||||||
|
groupid := vars["groupid"]
|
||||||
|
memberid := vars["memberid"]
|
||||||
|
|
||||||
|
if r.Method == "POST" {
|
||||||
|
|
||||||
|
//////////get groupid covoiturage//////////
|
||||||
|
request := &groupsmanagement.GetGroupRequest{
|
||||||
|
Id: groupid,
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := h.services.GRPC.GroupsManagement.GetGroup(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////get group member////////////////////////////////
|
||||||
|
|
||||||
|
reequest := &groupsmanagement.GetGroupMemberRequest{
|
||||||
|
Id: id,
|
||||||
|
}
|
||||||
|
|
||||||
|
ressp, err := h.services.GRPC.GroupsManagement.GetGroupMember(context.TODO(), reequest)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
req := &groupsmanagement.UnsubscribeMemberRequest{
|
||||||
|
Id: ressp.Group.Id,
|
||||||
|
}
|
||||||
|
|
||||||
|
_, errr := h.services.GRPC.GroupsManagement.UnsubscribeMember(context.TODO(), req)
|
||||||
|
if errr != nil {
|
||||||
|
fmt.Println(errr)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
members := resp.Group.Members
|
||||||
|
for i := 0; i < len(members); i++ {
|
||||||
|
if members[i] == memberid {
|
||||||
|
members = append(members[:i], members[(i+1):]...)
|
||||||
|
resp.Group.Members = members
|
||||||
|
reequest := &groupsmanagement.UnsubscribeRequest{
|
||||||
|
Groupid: resp.Group.ToStorageType().ID,
|
||||||
|
Memberid: memberid,
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := h.services.GRPC.GroupsManagement.Unsubscribe(context.TODO(), reequest)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
http.Redirect(w, r, fmt.Sprintf("/app/journeys/groups_covoiturage/create/%s", groupid), http.StatusFound)
|
||||||
|
/*
|
||||||
|
I must add "return" to resolve the err
|
||||||
|
http: superfluous response.WriteHeader call from git.coopgo.io/coopgo-apps/parcoursmob/renderer.(*Renderer).Render (renderer.go:50)
|
||||||
|
*/
|
||||||
|
return
|
||||||
|
}
|
||||||
|
h.Renderer.UpdateGroupCovoiturage(w, r, groupid, memberid)
|
||||||
|
}
|
||||||
|
|||||||
128
handlers/application/members.go
Normal file
128
handlers/application/members.go
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
package application
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
formvalidators "git.coopgo.io/coopgo-apps/parcoursmob/utils/form-validators"
|
||||||
|
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
||||||
|
"google.golang.org/protobuf/types/known/structpb"
|
||||||
|
)
|
||||||
|
|
||||||
|
type UserForm struct {
|
||||||
|
FirstName string `json:"first_name" validate:"required"`
|
||||||
|
LastName string `json:"last_name" validate:"required"`
|
||||||
|
Email string `json:"email" validate:"required,email"`
|
||||||
|
PhoneNumber string `json:"phone_number" `
|
||||||
|
Address any `json:"address,omitempty"`
|
||||||
|
Gender string `json:"gender"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) MemberDisplay(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|
||||||
|
adm := strings.Split(r.URL.Path, "/")
|
||||||
|
adminid := adm[3]
|
||||||
|
|
||||||
|
request := &mobilityaccounts.GetAccountRequest{
|
||||||
|
Id: adminid,
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := h.services.GRPC.MobilityAccounts.GetAccount(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
h.Renderer.MemberDisplay(w, r, resp.Account.ToStorageType())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) MemberUpdate(w http.ResponseWriter, r *http.Request) {
|
||||||
|
adm := strings.Split(r.URL.Path, "/")
|
||||||
|
userID := adm[3]
|
||||||
|
|
||||||
|
if r.Method == "POST" {
|
||||||
|
|
||||||
|
dataMap, err := parseUserForm(r)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := structpb.NewValue(dataMap)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
request := &mobilityaccounts.UpdateDataRequest{
|
||||||
|
Account: &mobilityaccounts.Account{
|
||||||
|
Id: userID,
|
||||||
|
Namespace: "parcoursmob",
|
||||||
|
Data: data.GetStructValue(),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := h.services.GRPC.MobilityAccounts.UpdateData(context.TODO(), request)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, fmt.Sprintf("/app/members/%s", resp.Account.Id), http.StatusFound)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
request := &mobilityaccounts.GetAccountRequest{
|
||||||
|
Id: userID,
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := h.services.GRPC.MobilityAccounts.GetAccount(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
h.Renderer.MemberUpdate(w, r, resp.Account.ToStorageType())
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseUserForm(r *http.Request) (map[string]any, error) {
|
||||||
|
if err := r.ParseForm(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
formData := UserForm{
|
||||||
|
FirstName: r.PostFormValue("first_name"),
|
||||||
|
LastName: r.PostFormValue("last_name"),
|
||||||
|
Email: r.PostFormValue("email"),
|
||||||
|
PhoneNumber: r.PostFormValue("phone_number"),
|
||||||
|
Gender: r.PostFormValue("gender"),
|
||||||
|
}
|
||||||
|
|
||||||
|
validate := formvalidators.New()
|
||||||
|
if err := validate.Struct(formData); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
d, err := json.Marshal(formData)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var dataMap map[string]any
|
||||||
|
err = json.Unmarshal(d, &dataMap)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return dataMap, nil
|
||||||
|
}
|
||||||
@@ -29,17 +29,12 @@ func (h *ApplicationHandler) SupportSend(w http.ResponseWriter, r *http.Request)
|
|||||||
"user": current_user_claims["email"],
|
"user": current_user_claims["email"],
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := h.emailing.Send("onboarding.Support_email", "support@parcoursmob.fr", data); err != nil {
|
if err := h.emailing.Send("support.request", "support@parcoursmob.fr", data); err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
fmt.Println("error")
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
}
|
}
|
||||||
fmt.Println("success")
|
|
||||||
http.Redirect(w, r, "/app/", http.StatusFound)
|
http.Redirect(w, r, "/app/", http.StatusFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Println("comment page!")
|
|
||||||
fmt.Println(comment)
|
|
||||||
fmt.Print(current_user_claims["email"])
|
|
||||||
h.Renderer.SupportSend(w, r, comment, current_user_claims)
|
h.Renderer.SupportSend(w, r, comment, current_user_claims)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,13 +5,19 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"sort"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
||||||
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/sorting"
|
||||||
|
filestorage "git.coopgo.io/coopgo-apps/parcoursmob/utils/storage"
|
||||||
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
||||||
|
fleetsstorage "git.coopgo.io/coopgo-platform/fleets/storage"
|
||||||
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
||||||
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
||||||
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
||||||
|
mobilityaccountsstorage "git.coopgo.io/coopgo-platform/mobility-accounts/storage"
|
||||||
|
"github.com/coreos/go-oidc"
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
"google.golang.org/protobuf/types/known/structpb"
|
"google.golang.org/protobuf/types/known/structpb"
|
||||||
@@ -29,15 +35,63 @@ func (h *ApplicationHandler) VehiclesManagementOverview(w http.ResponseWriter, r
|
|||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
}
|
}
|
||||||
|
|
||||||
var vehicles = []any{}
|
vehicles := []fleetsstorage.Vehicle{}
|
||||||
|
bookings := []fleetsstorage.Booking{}
|
||||||
|
vehicles_map := map[string]fleetsstorage.Vehicle{}
|
||||||
|
|
||||||
for _, vehicle := range resp.Vehicles {
|
for _, vehicle := range resp.Vehicles {
|
||||||
if filterVehicle(r, vehicle) {
|
if filterVehicle(r, vehicle) {
|
||||||
v := vehicle.ToStorageType()
|
v := vehicle.ToStorageType()
|
||||||
vehicles = append(vehicles, v)
|
vehicles = append(vehicles, v)
|
||||||
|
vehicles_map[v.ID] = v
|
||||||
|
for _, b := range v.Bookings {
|
||||||
|
if b.Status() != fleetsstorage.StatusOld {
|
||||||
|
bookings = append(bookings, b)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
h.Renderer.VehiclesManagementOverview(w, r, vehicles)
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Sort(sorting.VehiclesByLicencePlate(vehicles))
|
||||||
|
sort.Sort(sorting.BookingsByStartdate(bookings))
|
||||||
|
|
||||||
|
h.Renderer.VehiclesManagementOverview(w, r, vehicles, vehicles_map, bookings)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) VehiclesManagementBookingsList(w http.ResponseWriter, r *http.Request) {
|
||||||
|
//Get Vehicles
|
||||||
|
request := &fleets.GetVehiclesRequest{
|
||||||
|
Namespaces: []string{"parcoursmob"},
|
||||||
|
}
|
||||||
|
resp, err := h.services.GRPC.Fleets.GetVehicles(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
}
|
||||||
|
|
||||||
|
bookings := []fleetsstorage.Booking{}
|
||||||
|
vehicles_map := map[string]fleetsstorage.Vehicle{}
|
||||||
|
|
||||||
|
for _, vehicle := range resp.Vehicles {
|
||||||
|
if filterVehicle(r, vehicle) {
|
||||||
|
v := vehicle.ToStorageType()
|
||||||
|
vehicles_map[v.ID] = v
|
||||||
|
// bookings = append(bookings, v.Bookings...)
|
||||||
|
for _, b := range v.Bookings {
|
||||||
|
if v, ok := b.Data["administrator_unavailability"].(bool); !ok || !v {
|
||||||
|
bookings = append(bookings, b)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Sort(sorting.BookingsByStartdate(bookings))
|
||||||
|
|
||||||
|
cacheid := uuid.NewString()
|
||||||
|
h.cache.PutWithTTL(cacheid, bookings, 1*time.Hour)
|
||||||
|
|
||||||
|
h.Renderer.VehiclesManagementBookingsList(w, r, vehicles_map, bookings, cacheid)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *ApplicationHandler) VehiclesFleetAdd(w http.ResponseWriter, r *http.Request) {
|
func (h *ApplicationHandler) VehiclesFleetAdd(w http.ResponseWriter, r *http.Request) {
|
||||||
@@ -76,6 +130,10 @@ func (h *ApplicationHandler) VehiclesFleetAdd(w http.ResponseWriter, r *http.Req
|
|||||||
if v := r.FormValue("licence_plate"); v != "" {
|
if v := r.FormValue("licence_plate"); v != "" {
|
||||||
dataMap["licence_plate"] = v
|
dataMap["licence_plate"] = v
|
||||||
}
|
}
|
||||||
|
if v := r.FormValue("automatic"); v != "" {
|
||||||
|
fmt.Println(v)
|
||||||
|
dataMap["automatic"] = (v == "on")
|
||||||
|
}
|
||||||
|
|
||||||
data, err := structpb.NewValue(dataMap)
|
data, err := structpb.NewValue(dataMap)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -107,7 +165,9 @@ func (h *ApplicationHandler) VehiclesFleetAdd(w http.ResponseWriter, r *http.Req
|
|||||||
http.Redirect(w, r, fmt.Sprintf("/app/vehicles-management/fleet/%s", vehicle.Id), http.StatusFound)
|
http.Redirect(w, r, fmt.Sprintf("/app/vehicles-management/fleet/%s", vehicle.Id), http.StatusFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
h.Renderer.VehiclesFleetAdd(w, r)
|
|
||||||
|
vehicles_types := h.config.GetStringSlice("modules.fleets.vehicle_types")
|
||||||
|
h.Renderer.VehiclesFleetAdd(w, r, vehicles_types)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *ApplicationHandler) VehiclesFleetDisplay(w http.ResponseWriter, r *http.Request) {
|
func (h *ApplicationHandler) VehiclesFleetDisplay(w http.ResponseWriter, r *http.Request) {
|
||||||
@@ -172,22 +232,17 @@ func (h ApplicationHandler) VehicleManagementBookingDisplay(w http.ResponseWrite
|
|||||||
vars := mux.Vars(r)
|
vars := mux.Vars(r)
|
||||||
bookingid := vars["bookingid"]
|
bookingid := vars["bookingid"]
|
||||||
|
|
||||||
request := &fleets.GetBookingRequest{
|
booking, err := h.services.GetBooking(bookingid)
|
||||||
Bookingid: bookingid,
|
|
||||||
}
|
|
||||||
resp, err := h.services.GRPC.Fleets.GetBooking(context.TODO(), request)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
booking := resp.Booking.ToStorageType()
|
|
||||||
|
|
||||||
if r.Method == "POST" {
|
if r.Method == "POST" {
|
||||||
r.ParseForm()
|
r.ParseForm()
|
||||||
|
|
||||||
newbooking := resp.Booking
|
newbooking, _ := fleets.BookingFromStorageType(&booking)
|
||||||
|
|
||||||
startdate := r.FormValue("startdate")
|
startdate := r.FormValue("startdate")
|
||||||
if startdate != "" {
|
if startdate != "" {
|
||||||
@@ -235,6 +290,9 @@ func (h ApplicationHandler) VehicleManagementBookingDisplay(w http.ResponseWrite
|
|||||||
booking = newbooking.ToStorageType()
|
booking = newbooking.ToStorageType()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
beneficiary := mobilityaccountsstorage.Account{}
|
||||||
|
|
||||||
|
if booking.Driver != "" {
|
||||||
beneficiaryrequest := &mobilityaccounts.GetAccountRequest{
|
beneficiaryrequest := &mobilityaccounts.GetAccountRequest{
|
||||||
Id: booking.Driver,
|
Id: booking.Driver,
|
||||||
}
|
}
|
||||||
@@ -246,6 +304,9 @@ func (h ApplicationHandler) VehicleManagementBookingDisplay(w http.ResponseWrite
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
beneficiary = beneficiaryresp.Account.ToStorageType()
|
||||||
|
}
|
||||||
|
|
||||||
grouprequest := &groupsmanagement.GetGroupRequest{
|
grouprequest := &groupsmanagement.GetGroupRequest{
|
||||||
Id: booking.Vehicle.Administrators[0],
|
Id: booking.Vehicle.Administrators[0],
|
||||||
}
|
}
|
||||||
@@ -257,5 +318,142 @@ func (h ApplicationHandler) VehicleManagementBookingDisplay(w http.ResponseWrite
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
h.Renderer.VehicleManagementBookingDisplay(w, r, booking, booking.Vehicle, beneficiaryresp.Account.ToStorageType(), groupresp.Group.ToStorageType())
|
alternativerequest := &fleets.GetVehiclesRequest{
|
||||||
|
Namespaces: []string{"parcoursmob"},
|
||||||
|
AvailabilityFrom: timestamppb.New(booking.Startdate),
|
||||||
|
AvailabilityTo: timestamppb.New(booking.Enddate.Add(24 * time.Hour)),
|
||||||
|
}
|
||||||
|
|
||||||
|
alternativeresp, err := h.services.GRPC.Fleets.GetVehicles(context.TODO(), alternativerequest)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
alternatives := []any{}
|
||||||
|
|
||||||
|
for _, a := range alternativeresp.Vehicles {
|
||||||
|
alternatives = append(alternatives, a.ToStorageType())
|
||||||
|
}
|
||||||
|
|
||||||
|
documents := h.filestorage.List(filestorage.PREFIX_BOOKINGS + "/" + bookingid)
|
||||||
|
file_types_map := h.config.GetStringMapString("storage.files.file_types")
|
||||||
|
|
||||||
|
h.Renderer.VehicleManagementBookingDisplay(w, r, booking, booking.Vehicle, beneficiary, groupresp.Group.ToStorageType(), documents, file_types_map, alternatives)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h ApplicationHandler) VehicleManagementBookingChangeVehicle(w http.ResponseWriter, r *http.Request) {
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
bookingid := vars["bookingid"]
|
||||||
|
|
||||||
|
r.ParseForm()
|
||||||
|
|
||||||
|
newvehicle := r.FormValue("vehicle")
|
||||||
|
|
||||||
|
booking, err := h.services.GetBooking(bookingid)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
booking.Vehicleid = newvehicle
|
||||||
|
|
||||||
|
b, _ := fleets.BookingFromStorageType(&booking)
|
||||||
|
|
||||||
|
request := &fleets.UpdateBookingRequest{
|
||||||
|
Booking: b,
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = h.services.GRPC.Fleets.UpdateBooking(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, fmt.Sprintf("/app/vehicles-management/bookings/%s", bookingid), http.StatusFound)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h ApplicationHandler) VehiclesFleetMakeUnavailable(w http.ResponseWriter, r *http.Request) { // Get Group
|
||||||
|
g := r.Context().Value(identification.GroupKey)
|
||||||
|
if g == nil {
|
||||||
|
fmt.Println("no current group")
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
current_group := g.(storage.Group)
|
||||||
|
|
||||||
|
// Get current user ID
|
||||||
|
u := r.Context().Value(identification.IdtokenKey)
|
||||||
|
if u == nil {
|
||||||
|
fmt.Println("no current user")
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
current_user_token := u.(*oidc.IDToken)
|
||||||
|
|
||||||
|
// Get current user claims
|
||||||
|
c := r.Context().Value(identification.ClaimsKey)
|
||||||
|
if c == nil {
|
||||||
|
fmt.Println("no current user claims")
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
current_user_claims := c.(map[string]any)
|
||||||
|
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
vehicleid := vars["vehicleid"]
|
||||||
|
|
||||||
|
r.ParseForm()
|
||||||
|
|
||||||
|
start := r.FormValue("unavailablefrom")
|
||||||
|
end := r.FormValue("unavailableto")
|
||||||
|
comment := r.FormValue("comment")
|
||||||
|
|
||||||
|
unavailablefrom, _ := time.Parse("2006-01-02", start)
|
||||||
|
unavailableto, _ := time.Parse("2006-01-02", end)
|
||||||
|
|
||||||
|
data := map[string]any{
|
||||||
|
"comment": comment,
|
||||||
|
"administrator_unavailability": true,
|
||||||
|
"booked_by": map[string]any{
|
||||||
|
"user": map[string]any{
|
||||||
|
"id": current_user_token.Subject,
|
||||||
|
"display_name": current_user_claims["display_name"],
|
||||||
|
},
|
||||||
|
"group": map[string]any{
|
||||||
|
"id": current_group.ID,
|
||||||
|
"name": current_group.Data["name"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
datapb, err := structpb.NewStruct(data)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
booking := &fleets.Booking{
|
||||||
|
Id: uuid.NewString(),
|
||||||
|
Vehicleid: vehicleid,
|
||||||
|
Unavailablefrom: timestamppb.New(unavailablefrom),
|
||||||
|
Unavailableto: timestamppb.New(unavailableto),
|
||||||
|
Data: datapb,
|
||||||
|
}
|
||||||
|
|
||||||
|
request := &fleets.CreateBookingRequest{
|
||||||
|
Booking: booking,
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = h.services.GRPC.Fleets.CreateBooking(context.TODO(), request)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, fmt.Sprintf("/app/vehicles-management/fleet/%s", vehicleid), http.StatusFound)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,15 +3,21 @@ package application
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
||||||
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/sorting"
|
||||||
|
filestorage "git.coopgo.io/coopgo-apps/parcoursmob/utils/storage"
|
||||||
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
||||||
|
"git.coopgo.io/coopgo-platform/fleets/storage"
|
||||||
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
||||||
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
groupsmanagementstorage "git.coopgo.io/coopgo-platform/groups-management/storage"
|
||||||
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
||||||
"github.com/coreos/go-oidc"
|
mobilityaccountsstorage "git.coopgo.io/coopgo-platform/mobility-accounts/storage"
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
"google.golang.org/protobuf/types/known/structpb"
|
"google.golang.org/protobuf/types/known/structpb"
|
||||||
@@ -21,7 +27,9 @@ import (
|
|||||||
func (h ApplicationHandler) VehiclesSearch(w http.ResponseWriter, r *http.Request) {
|
func (h ApplicationHandler) VehiclesSearch(w http.ResponseWriter, r *http.Request) {
|
||||||
r.ParseForm()
|
r.ParseForm()
|
||||||
|
|
||||||
var beneficiary any
|
var beneficiary mobilityaccountsstorage.Account
|
||||||
|
|
||||||
|
beneficiarydocuments := []filestorage.FileInfo{}
|
||||||
|
|
||||||
vehicles := []any{}
|
vehicles := []any{}
|
||||||
searched := false
|
searched := false
|
||||||
@@ -30,6 +38,9 @@ func (h ApplicationHandler) VehiclesSearch(w http.ResponseWriter, r *http.Reques
|
|||||||
|
|
||||||
startdate, _ := time.Parse("2006-01-02", start)
|
startdate, _ := time.Parse("2006-01-02", start)
|
||||||
enddate, _ := time.Parse("2006-01-02", end)
|
enddate, _ := time.Parse("2006-01-02", end)
|
||||||
|
automatic := (r.FormValue("automatic") == "on")
|
||||||
|
|
||||||
|
administrators := []string{}
|
||||||
|
|
||||||
if r.FormValue("beneficiaryid") != "" {
|
if r.FormValue("beneficiaryid") != "" {
|
||||||
// Handler form
|
// Handler form
|
||||||
@@ -50,7 +61,14 @@ func (h ApplicationHandler) VehiclesSearch(w http.ResponseWriter, r *http.Reques
|
|||||||
|
|
||||||
request := &fleets.GetVehiclesRequest{
|
request := &fleets.GetVehiclesRequest{
|
||||||
Namespaces: []string{"parcoursmob"},
|
Namespaces: []string{"parcoursmob"},
|
||||||
|
AvailabilityFrom: timestamppb.New(startdate),
|
||||||
|
AvailabilityTo: timestamppb.New(enddate),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if r.FormValue("type") != "" {
|
||||||
|
request.Types = []string{r.FormValue("type")}
|
||||||
|
}
|
||||||
|
|
||||||
resp, err := h.services.GRPC.Fleets.GetVehicles(context.TODO(), request)
|
resp, err := h.services.GRPC.Fleets.GetVehicles(context.TODO(), request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
@@ -59,10 +77,30 @@ func (h ApplicationHandler) VehiclesSearch(w http.ResponseWriter, r *http.Reques
|
|||||||
|
|
||||||
for _, vehicle := range resp.Vehicles {
|
for _, vehicle := range resp.Vehicles {
|
||||||
v := vehicle.ToStorageType()
|
v := vehicle.ToStorageType()
|
||||||
if v.Free(startdate, enddate) {
|
|
||||||
|
if r.FormValue("type") == "Voiture" && automatic {
|
||||||
|
fmt.Println(v.Data["automatic"])
|
||||||
|
if auto, ok := v.Data["automatic"].(bool); !ok || !auto {
|
||||||
|
fmt.Println(v.Data["automatic"])
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
adminfound := false
|
||||||
|
for _, a := range administrators {
|
||||||
|
if a == v.Administrators[0] {
|
||||||
|
adminfound = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !adminfound {
|
||||||
|
administrators = append(administrators, v.Administrators[0])
|
||||||
|
}
|
||||||
|
|
||||||
vehicles = append(vehicles, v)
|
vehicles = append(vehicles, v)
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
beneficiarydocuments = h.filestorage.List(filestorage.PREFIX_BENEFICIARIES + "/" + beneficiary.ID)
|
||||||
}
|
}
|
||||||
|
|
||||||
accounts, err := h.beneficiaries(r)
|
accounts, err := h.beneficiaries(r)
|
||||||
@@ -71,43 +109,64 @@ func (h ApplicationHandler) VehiclesSearch(w http.ResponseWriter, r *http.Reques
|
|||||||
w.WriteHeader(http.StatusBadRequest)
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
groups := map[string]any{}
|
||||||
|
|
||||||
h.Renderer.VehiclesSearch(w, r, accounts, searched, vehicles, beneficiary, r.FormValue("startdate"), r.FormValue("enddate"))
|
if len(administrators) > 0 {
|
||||||
|
admingroups, err := h.services.GRPC.GroupsManagement.GetGroupsBatch(context.TODO(), &groupsmanagement.GetGroupsBatchRequest{
|
||||||
|
Groupids: administrators,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, g := range admingroups.Groups {
|
||||||
|
groups[g.Id] = g.ToStorageType()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
sort.Sort(sorting.BeneficiariesByName(accounts))
|
||||||
|
|
||||||
|
mandatory_documents := h.config.GetStringSlice("modules.fleets.booking_documents.mandatory")
|
||||||
|
file_types_map := h.config.GetStringMapString("storage.files.file_types")
|
||||||
|
vehicles_types := h.config.GetStringSlice("modules.fleets.vehicle_types")
|
||||||
|
|
||||||
|
h.Renderer.VehiclesSearch(w, r, accounts, searched, vehicles, beneficiary, r.FormValue("startdate"), r.FormValue("enddate"), mandatory_documents, file_types_map, beneficiarydocuments, r.FormValue("type"), automatic, vehicles_types, groups)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h ApplicationHandler) Book(w http.ResponseWriter, r *http.Request) {
|
func (h ApplicationHandler) Book(w http.ResponseWriter, r *http.Request) {
|
||||||
// Get Group
|
fmt.Println("Book")
|
||||||
g := r.Context().Value(identification.GroupKey)
|
current_group, err := h.currentGroup(r)
|
||||||
if g == nil {
|
if err != nil {
|
||||||
fmt.Println("no current group")
|
fmt.Println(err)
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
current_group := g.(storage.Group)
|
|
||||||
|
|
||||||
// Get current user ID
|
current_user_token, current_user_claims, err := h.currentUser(r)
|
||||||
u := r.Context().Value(identification.IdtokenKey)
|
if err != nil {
|
||||||
if u == nil {
|
fmt.Println(err)
|
||||||
fmt.Println("no current user")
|
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
current_user_token := u.(*oidc.IDToken)
|
|
||||||
|
|
||||||
// Get current user claims
|
|
||||||
c := r.Context().Value(identification.ClaimsKey)
|
|
||||||
if c == nil {
|
|
||||||
fmt.Println("no current user claims")
|
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
current_user_claims := c.(map[string]any)
|
|
||||||
|
|
||||||
vars := mux.Vars(r)
|
vars := mux.Vars(r)
|
||||||
vehicleid := vars["vehicleid"]
|
vehicleid := vars["vehicleid"]
|
||||||
beneficiaryid := vars["beneficiaryid"]
|
beneficiaryid := vars["beneficiaryid"]
|
||||||
|
|
||||||
r.ParseForm()
|
vehicle, err := h.services.GRPC.Fleets.GetVehicle(context.TODO(), &fleets.GetVehicleRequest{
|
||||||
|
Vehicleid: vehicleid,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
w.Write([]byte("Vehicle not found"))
|
||||||
|
w.Write([]byte(err.Error()))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
r.ParseMultipartForm(100 * 1024 * 1024)
|
||||||
|
|
||||||
start := r.FormValue("startdate")
|
start := r.FormValue("startdate")
|
||||||
end := r.FormValue("enddate")
|
end := r.FormValue("enddate")
|
||||||
@@ -119,7 +178,7 @@ func (h ApplicationHandler) Book(w http.ResponseWriter, r *http.Request) {
|
|||||||
"booked_by": map[string]any{
|
"booked_by": map[string]any{
|
||||||
"user": map[string]any{
|
"user": map[string]any{
|
||||||
"id": current_user_token.Subject,
|
"id": current_user_token.Subject,
|
||||||
"display_name": current_user_claims["display_name"],
|
"display_name": fmt.Sprintf("%s %s", current_user_claims["first_name"], current_user_claims["last_name"]),
|
||||||
},
|
},
|
||||||
"group": map[string]any{
|
"group": map[string]any{
|
||||||
"id": current_group.ID,
|
"id": current_group.ID,
|
||||||
@@ -149,6 +208,41 @@ func (h ApplicationHandler) Book(w http.ResponseWriter, r *http.Request) {
|
|||||||
Booking: booking,
|
Booking: booking,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for _, v := range h.config.GetStringSlice("modules.fleets.booking_documents.mandatory") {
|
||||||
|
existing_file := r.FormValue("type-" + v)
|
||||||
|
if existing_file == "" {
|
||||||
|
file, header, err := r.FormFile("doc-" + v)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
w.Write([]byte("Document manquant : " + v))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
fileid := uuid.NewString()
|
||||||
|
|
||||||
|
metadata := map[string]string{
|
||||||
|
"type": v,
|
||||||
|
"name": header.Filename,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := h.filestorage.Put(file, filestorage.PREFIX_BOOKINGS, fmt.Sprintf("%s/%s_%s", booking.Id, fileid, header.Filename), header.Size, metadata); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
path := strings.Split(existing_file, "/")
|
||||||
|
|
||||||
|
if err := h.filestorage.Copy(existing_file, fmt.Sprintf("%s/%s/%s", filestorage.PREFIX_BOOKINGS, booking.Id, path[len(path)-1])); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
_, err = h.services.GRPC.Fleets.CreateBooking(context.TODO(), request)
|
_, err = h.services.GRPC.Fleets.CreateBooking(context.TODO(), request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
@@ -156,6 +250,20 @@ func (h ApplicationHandler) Book(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//NOTIFY GROUP MEMBERS
|
||||||
|
members, _, err := h.groupmembers(vehicle.Vehicle.Administrators[0])
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
} else {
|
||||||
|
for _, m := range members {
|
||||||
|
if email, ok := m.Data["email"].(string); ok {
|
||||||
|
h.emailing.Send("fleets.bookings.creation_admin_alert", email, map[string]string{
|
||||||
|
"bookingid": booking.Id,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
http.Redirect(w, r, fmt.Sprintf("/app/vehicles/bookings/%s", booking.Id), http.StatusFound)
|
http.Redirect(w, r, fmt.Sprintf("/app/vehicles/bookings/%s", booking.Id), http.StatusFound)
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -182,9 +290,9 @@ func (h ApplicationHandler) VehicleBookingDisplay(w http.ResponseWriter, r *http
|
|||||||
|
|
||||||
beneficiaryresp, err := h.services.GRPC.MobilityAccounts.GetAccount(context.TODO(), beneficiaryrequest)
|
beneficiaryresp, err := h.services.GRPC.MobilityAccounts.GetAccount(context.TODO(), beneficiaryrequest)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
beneficiaryresp = &mobilityaccounts.GetAccountResponse{
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
Account: &mobilityaccounts.Account{},
|
||||||
return
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
grouprequest := &groupsmanagement.GetGroupRequest{
|
grouprequest := &groupsmanagement.GetGroupRequest{
|
||||||
@@ -198,10 +306,22 @@ func (h ApplicationHandler) VehicleBookingDisplay(w http.ResponseWriter, r *http
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
h.Renderer.VehicleBookingDisplay(w, r, booking, booking.Vehicle, beneficiaryresp.Account.ToStorageType(), groupresp.Group.ToStorageType())
|
documents := h.filestorage.List(filestorage.PREFIX_BOOKINGS + "/" + bookingid)
|
||||||
|
file_types_map := h.config.GetStringMapString("storage.files.file_types")
|
||||||
|
|
||||||
|
h.Renderer.VehicleBookingDisplay(w, r, booking, booking.Vehicle, beneficiaryresp.Account.ToStorageType(), groupresp.Group.ToStorageType(), documents, file_types_map)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h ApplicationHandler) VehiclesBookingsList(w http.ResponseWriter, r *http.Request) {
|
func (h ApplicationHandler) VehiclesBookingsList(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|
||||||
|
g := r.Context().Value(identification.GroupKey)
|
||||||
|
if g == nil {
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
group := g.(groupsmanagementstorage.Group)
|
||||||
|
|
||||||
request := &fleets.GetBookingsRequest{}
|
request := &fleets.GetBookingsRequest{}
|
||||||
resp, err := h.services.GRPC.Fleets.GetBookings(context.TODO(), request)
|
resp, err := h.services.GRPC.Fleets.GetBookings(context.TODO(), request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -210,11 +330,49 @@ func (h ApplicationHandler) VehiclesBookingsList(w http.ResponseWriter, r *http.
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
bookings := []any{}
|
bookings := []storage.Booking{}
|
||||||
|
|
||||||
for _, b := range resp.Bookings {
|
for _, b := range resp.Bookings {
|
||||||
bookings = append(bookings, b.ToStorageType())
|
booking := b.ToStorageType()
|
||||||
|
if b1, ok := booking.Data["booked_by"].(map[string]any); ok {
|
||||||
|
if b2, ok := b1["group"].(map[string]any); ok {
|
||||||
|
if b2["id"] == group.ID {
|
||||||
|
bookings = append(bookings, booking)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
h.Renderer.VehicleBookingsList(w, r, bookings)
|
}
|
||||||
|
|
||||||
|
sort.Sort(sorting.BookingsByStartdate(bookings))
|
||||||
|
|
||||||
|
vehicles, _ := h.services.GetVehiclesMap()
|
||||||
|
groups, _ := h.services.GetGroupsMap()
|
||||||
|
|
||||||
|
h.Renderer.VehicleBookingsList(w, r, bookings, vehicles, groups)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ApplicationHandler) BookingDocumentDownload(w http.ResponseWriter, r *http.Request) {
|
||||||
|
vars := mux.Vars(r)
|
||||||
|
bookingid := vars["bookingid"]
|
||||||
|
document := vars["document"]
|
||||||
|
|
||||||
|
fmt.Println(fmt.Sprintf("%s/%s", bookingid, document))
|
||||||
|
|
||||||
|
file, info, err := h.filestorage.Get(filestorage.PREFIX_BOOKINGS, fmt.Sprintf("%s/%s", bookingid, document))
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
w.Header().Set("Content-Type", info.ContentType)
|
||||||
|
if _, err = io.Copy(w, file); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, fmt.Sprintf("/app/vehicles/bookings/%s", bookingid), http.StatusFound)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,8 +3,9 @@ package auth
|
|||||||
import (
|
import (
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/renderer"
|
"git.coopgo.io/coopgo-apps/parcoursmob/renderer"
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/services"
|
"git.coopgo.io/coopgo-apps/parcoursmob/services"
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/cache"
|
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
||||||
|
cache "git.coopgo.io/coopgo-apps/parcoursmob/utils/storage"
|
||||||
|
"git.coopgo.io/coopgo-platform/emailing"
|
||||||
"github.com/spf13/viper"
|
"github.com/spf13/viper"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -13,10 +14,11 @@ type AuthHandler struct {
|
|||||||
config *viper.Viper
|
config *viper.Viper
|
||||||
services *services.ServicesHandler
|
services *services.ServicesHandler
|
||||||
Renderer *renderer.Renderer
|
Renderer *renderer.Renderer
|
||||||
cache *cache.CacheHandler
|
cache cache.CacheHandler
|
||||||
|
emailing *emailing.Mailer
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewAuthHandler(cfg *viper.Viper, idp *identification.IdentificationProvider, svc *services.ServicesHandler, cache *cache.CacheHandler) (*AuthHandler, error) {
|
func NewAuthHandler(cfg *viper.Viper, idp *identification.IdentificationProvider, svc *services.ServicesHandler, cache cache.CacheHandler, emailing *emailing.Mailer) (*AuthHandler, error) {
|
||||||
templates_root := cfg.GetString("templates.root")
|
templates_root := cfg.GetString("templates.root")
|
||||||
renderer := renderer.NewRenderer(cfg, templates_root)
|
renderer := renderer.NewRenderer(cfg, templates_root)
|
||||||
return &AuthHandler{
|
return &AuthHandler{
|
||||||
@@ -25,5 +27,6 @@ func NewAuthHandler(cfg *viper.Viper, idp *identification.IdentificationProvider
|
|||||||
services: svc,
|
services: svc,
|
||||||
Renderer: renderer,
|
Renderer: renderer,
|
||||||
cache: cache,
|
cache: cache,
|
||||||
|
emailing: emailing,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|||||||
13
handlers/auth/disconnect.go
Normal file
13
handlers/auth/disconnect.go
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
package auth
|
||||||
|
|
||||||
|
import "net/http"
|
||||||
|
|
||||||
|
func (h *AuthHandler) Disconnect(w http.ResponseWriter, r *http.Request) {
|
||||||
|
session, err := h.idp.SessionsStore.Get(r, "parcoursmob_session")
|
||||||
|
if err == nil {
|
||||||
|
session.Options.MaxAge = -1
|
||||||
|
session.Save(r, w)
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, "/", http.StatusOK)
|
||||||
|
}
|
||||||
97
handlers/auth/lost_password.go
Normal file
97
handlers/auth/lost_password.go
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
package auth
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"crypto/rand"
|
||||||
|
"encoding/base64"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (h *AuthHandler) LostPasswordInit(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.Method == "POST" {
|
||||||
|
r.ParseForm()
|
||||||
|
email := r.FormValue("email")
|
||||||
|
if email != "" {
|
||||||
|
account, err := h.services.GRPC.MobilityAccounts.GetAccountUsername(context.TODO(), &grpcapi.GetAccountUsernameRequest{
|
||||||
|
Username: email,
|
||||||
|
Namespace: "parcoursmob",
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
http.Redirect(w, r, "/app/", http.StatusFound)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
b := make([]byte, 16)
|
||||||
|
if _, err := io.ReadFull(rand.Reader, b); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
key := base64.RawURLEncoding.EncodeToString(b)
|
||||||
|
|
||||||
|
passwordretrieval := map[string]any{
|
||||||
|
"username": email,
|
||||||
|
"account_id": account.Account.Id,
|
||||||
|
"key": key,
|
||||||
|
}
|
||||||
|
|
||||||
|
h.cache.PutWithTTL("retrieve-password/"+key, passwordretrieval, 72*time.Hour)
|
||||||
|
|
||||||
|
if err := h.emailing.Send("auth.retrieve_password", email, passwordretrieval); err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, "/app/", http.StatusFound)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
h.Renderer.LostPasswordInit(w, r)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *AuthHandler) LostPasswordRecover(w http.ResponseWriter, r *http.Request) {
|
||||||
|
r.ParseForm()
|
||||||
|
|
||||||
|
key := r.FormValue("key")
|
||||||
|
recover, err := h.cache.Get("retrieve-password/" + key)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
h.Renderer.LostPasswordRecoverKO(w, r, key)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if r.Method == "POST" {
|
||||||
|
newpassword := r.FormValue("password")
|
||||||
|
if newpassword == "" {
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
w.Write([]byte("Password is empty"))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := h.services.GRPC.MobilityAccounts.ChangePassword(context.TODO(), &grpcapi.ChangePasswordRequest{
|
||||||
|
Id: recover.(map[string]any)["account_id"].(string),
|
||||||
|
Password: newpassword,
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = h.cache.Delete("retrieve-password/" + key)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
http.Redirect(w, r, "/app/", http.StatusFound)
|
||||||
|
|
||||||
|
}
|
||||||
|
h.Renderer.LostPasswordRecover(w, r, recover)
|
||||||
|
}
|
||||||
@@ -16,7 +16,7 @@ func (h *AuthHandler) Onboarding(w http.ResponseWriter, r *http.Request) {
|
|||||||
onboarding, err := h.cache.Get("onboarding/" + key)
|
onboarding, err := h.cache.Get("onboarding/" + key)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
h.Renderer.AuthOnboardingKO(w, r, key)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -37,7 +37,7 @@ func (h *AuthHandler) Onboarding(w http.ResponseWriter, r *http.Request) {
|
|||||||
if onboardingmap["admin"].(bool) {
|
if onboardingmap["admin"].(bool) {
|
||||||
groups = append(groups, onboardingmap["group"].(string)+":admin")
|
groups = append(groups, onboardingmap["group"].(string)+":admin")
|
||||||
}
|
}
|
||||||
|
display_name := fmt.Sprint(r.FormValue("first_name")) + " " + fmt.Sprint(r.FormValue("last_name"))
|
||||||
account := &ma.Account{
|
account := &ma.Account{
|
||||||
Authentication: ma.AccountAuth{
|
Authentication: ma.AccountAuth{
|
||||||
Local: ma.LocalAuth{
|
Local: ma.LocalAuth{
|
||||||
@@ -46,7 +46,9 @@ func (h *AuthHandler) Onboarding(w http.ResponseWriter, r *http.Request) {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
Namespace: "parcoursmob",
|
Namespace: "parcoursmob",
|
||||||
|
|
||||||
Data: map[string]any{
|
Data: map[string]any{
|
||||||
|
"display_name": display_name,
|
||||||
"first_name": r.FormValue("first_name"),
|
"first_name": r.FormValue("first_name"),
|
||||||
"last_name": r.FormValue("last_name"),
|
"last_name": r.FormValue("last_name"),
|
||||||
"email": onboardingmap["username"],
|
"email": onboardingmap["username"],
|
||||||
@@ -72,6 +74,12 @@ func (h *AuthHandler) Onboarding(w http.ResponseWriter, r *http.Request) {
|
|||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
err = h.cache.Delete("onboarding/" + key)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
}
|
||||||
|
|
||||||
http.Redirect(w, r, "/app/", http.StatusFound)
|
http.Redirect(w, r, "/app/", http.StatusFound)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
53
main.go
53
main.go
@@ -11,8 +11,8 @@ import (
|
|||||||
"git.coopgo.io/coopgo-apps/parcoursmob/handlers/auth"
|
"git.coopgo.io/coopgo-apps/parcoursmob/handlers/auth"
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/renderer"
|
"git.coopgo.io/coopgo-apps/parcoursmob/renderer"
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/services"
|
"git.coopgo.io/coopgo-apps/parcoursmob/services"
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/cache"
|
|
||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
||||||
|
cache "git.coopgo.io/coopgo-apps/parcoursmob/utils/storage"
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -33,12 +33,14 @@ func main() {
|
|||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
idp, err := identification.NewIdentificationProvider(cfg, svc)
|
kv, err := cache.NewKVHandler(cfg)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
cache, err := cache.NewCacheHandler(cfg)
|
filestorage, err := cache.NewFileStorage(cfg)
|
||||||
|
|
||||||
|
idp, err := identification.NewIdentificationProvider(cfg, svc, kv)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
@@ -48,9 +50,9 @@ func main() {
|
|||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
apiHandler, _ := api.NewAPIHandler(cfg, idp, svc, cache)
|
apiHandler, _ := api.NewAPIHandler(cfg, idp, svc, kv)
|
||||||
applicationHandler, _ := application.NewApplicationHandler(cfg, svc, cache, emailing)
|
applicationHandler, _ := application.NewApplicationHandler(cfg, svc, kv, filestorage, emailing)
|
||||||
authHandler, _ := auth.NewAuthHandler(cfg, idp, svc, cache)
|
authHandler, _ := auth.NewAuthHandler(cfg, idp, svc, kv, emailing)
|
||||||
|
|
||||||
fmt.Println("Running", service_name, ":")
|
fmt.Println("Running", service_name, ":")
|
||||||
|
|
||||||
@@ -59,6 +61,9 @@ func main() {
|
|||||||
r.PathPrefix("/public/").Handler(http.StripPrefix("/public/", http.FileServer(http.Dir(templates_public_dir))))
|
r.PathPrefix("/public/").Handler(http.StripPrefix("/public/", http.FileServer(http.Dir(templates_public_dir))))
|
||||||
|
|
||||||
r.HandleFunc("/auth/onboarding", authHandler.Onboarding)
|
r.HandleFunc("/auth/onboarding", authHandler.Onboarding)
|
||||||
|
r.HandleFunc("/auth/disconnect", authHandler.Disconnect)
|
||||||
|
r.HandleFunc("/auth/lost-password", authHandler.LostPasswordInit)
|
||||||
|
r.HandleFunc("/auth/lost-password/recover", authHandler.LostPasswordRecover)
|
||||||
r.HandleFunc("/auth/groups/", authHandler.Groups)
|
r.HandleFunc("/auth/groups/", authHandler.Groups)
|
||||||
r.HandleFunc("/auth/groups/switch", authHandler.GroupSwitch)
|
r.HandleFunc("/auth/groups/switch", authHandler.GroupSwitch)
|
||||||
r.HandleFunc("/", redirectApp)
|
r.HandleFunc("/", redirectApp)
|
||||||
@@ -67,6 +72,7 @@ func main() {
|
|||||||
api_router.HandleFunc("/", apiHandler.NotFound)
|
api_router.HandleFunc("/", apiHandler.NotFound)
|
||||||
api_router.HandleFunc("/geo/autocomplete", apiHandler.GeoAutocomplete)
|
api_router.HandleFunc("/geo/autocomplete", apiHandler.GeoAutocomplete)
|
||||||
api_router.HandleFunc("/cache/{cacheid}", apiHandler.GetCache)
|
api_router.HandleFunc("/cache/{cacheid}", apiHandler.GetCache)
|
||||||
|
api_router.HandleFunc("/cache/{cacheid}/export", apiHandler.CacheExport)
|
||||||
api_router.HandleFunc("/oauth2/callback", apiHandler.OAuth2Callback)
|
api_router.HandleFunc("/oauth2/callback", apiHandler.OAuth2Callback)
|
||||||
|
|
||||||
application := r.PathPrefix("/app").Subrouter()
|
application := r.PathPrefix("/app").Subrouter()
|
||||||
@@ -75,41 +81,55 @@ func main() {
|
|||||||
application.HandleFunc("/beneficiaries/create", applicationHandler.BeneficiaryCreate)
|
application.HandleFunc("/beneficiaries/create", applicationHandler.BeneficiaryCreate)
|
||||||
application.HandleFunc("/beneficiaries/{beneficiaryid}", applicationHandler.BeneficiaryDisplay)
|
application.HandleFunc("/beneficiaries/{beneficiaryid}", applicationHandler.BeneficiaryDisplay)
|
||||||
application.HandleFunc("/beneficiaries/{beneficiaryid}/update", applicationHandler.BeneficiaryUpdate)
|
application.HandleFunc("/beneficiaries/{beneficiaryid}/update", applicationHandler.BeneficiaryUpdate)
|
||||||
|
application.HandleFunc("/beneficiaries/{beneficiaryid}/documents", applicationHandler.BeneficiaryDocuments)
|
||||||
|
application.HandleFunc("/beneficiaries/{beneficiaryid}/documents/{document}", applicationHandler.BeneficiaryDocumentDownload)
|
||||||
application.HandleFunc("/beneficiaries/{beneficiaryid}/picture", applicationHandler.BeneficiaryPicture)
|
application.HandleFunc("/beneficiaries/{beneficiaryid}/picture", applicationHandler.BeneficiaryPicture)
|
||||||
application.HandleFunc("/members/{beneficiaryid}/picture", applicationHandler.BeneficiaryPicture)
|
application.HandleFunc("/members/{beneficiaryid}/picture", applicationHandler.BeneficiaryPicture)
|
||||||
|
application.HandleFunc("/members/{adminid}", applicationHandler.MemberDisplay)
|
||||||
|
application.HandleFunc("/members/{adminid}/update", applicationHandler.MemberUpdate)
|
||||||
application.HandleFunc("/journeys/", applicationHandler.JourneysSearch)
|
application.HandleFunc("/journeys/", applicationHandler.JourneysSearch)
|
||||||
application.HandleFunc("/vehicles/", applicationHandler.VehiclesSearch)
|
application.HandleFunc("/vehicles/", applicationHandler.VehiclesSearch)
|
||||||
application.HandleFunc("/vehicles/bookings/", applicationHandler.VehiclesBookingsList)
|
application.HandleFunc("/vehicles/bookings/", applicationHandler.VehiclesBookingsList)
|
||||||
application.HandleFunc("/vehicles/bookings/{bookingid}", applicationHandler.VehicleBookingDisplay)
|
application.HandleFunc("/vehicles/bookings/{bookingid}", applicationHandler.VehicleBookingDisplay)
|
||||||
application.HandleFunc("/vehicles/v/{vehicleid}/b/{beneficiaryid}", applicationHandler.Book)
|
application.HandleFunc("/vehicles/v/{vehicleid}/b/{beneficiaryid}", applicationHandler.Book)
|
||||||
|
application.HandleFunc("/vehicles/bookings/{bookingid}/documents/{document}", applicationHandler.BookingDocumentDownload)
|
||||||
application.HandleFunc("/vehicles-management/", applicationHandler.VehiclesManagementOverview)
|
application.HandleFunc("/vehicles-management/", applicationHandler.VehiclesManagementOverview)
|
||||||
application.HandleFunc("/vehicles-management/fleet/add", applicationHandler.VehiclesFleetAdd)
|
application.HandleFunc("/vehicles-management/fleet/add", applicationHandler.VehiclesFleetAdd)
|
||||||
application.HandleFunc("/vehicles-management/fleet/{vehicleid}", applicationHandler.VehiclesFleetDisplay)
|
application.HandleFunc("/vehicles-management/fleet/{vehicleid}", applicationHandler.VehiclesFleetDisplay)
|
||||||
|
application.HandleFunc("/vehicles-management/fleet/{vehicleid}/unavailability", applicationHandler.VehiclesFleetMakeUnavailable)
|
||||||
application.HandleFunc("/vehicles-management/fleet/{vehicleid}/update", applicationHandler.VehiclesFleetUpdate)
|
application.HandleFunc("/vehicles-management/fleet/{vehicleid}/update", applicationHandler.VehiclesFleetUpdate)
|
||||||
|
application.HandleFunc("/vehicles-management/bookings/", applicationHandler.VehiclesManagementBookingsList)
|
||||||
application.HandleFunc("/vehicles-management/bookings/{bookingid}", applicationHandler.VehicleManagementBookingDisplay)
|
application.HandleFunc("/vehicles-management/bookings/{bookingid}", applicationHandler.VehicleManagementBookingDisplay)
|
||||||
|
application.HandleFunc("/vehicles-management/bookings/{bookingid}/documents/{document}", applicationHandler.BookingDocumentDownload)
|
||||||
application.HandleFunc("/agenda/", applicationHandler.AgendaHome)
|
application.HandleFunc("/agenda/", applicationHandler.AgendaHome)
|
||||||
application.HandleFunc("/agenda/create-event", applicationHandler.AgendaCreateEvent)
|
application.HandleFunc("/agenda/create-event", applicationHandler.AgendaCreateEvent)
|
||||||
application.HandleFunc("/agenda/{eventid}", applicationHandler.AgendaDisplayEvent)
|
application.HandleFunc("/agenda/{eventid}", applicationHandler.AgendaDisplayEvent)
|
||||||
|
///////////////////////////////Code to modify event///////////////////////
|
||||||
|
application.HandleFunc("/agenda/{eventid}/update", applicationHandler.AgendaUpdateEvent)
|
||||||
|
application.HandleFunc("/agenda/{eventid}/delete", applicationHandler.AgendaDeleteEvent)
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////
|
||||||
application.HandleFunc("/agenda/{eventid}/subscribe", applicationHandler.AgendaSubscribeEvent)
|
application.HandleFunc("/agenda/{eventid}/subscribe", applicationHandler.AgendaSubscribeEvent)
|
||||||
application.HandleFunc("/directory/", applicationHandler.DirectoryHome)
|
application.HandleFunc("/directory/", applicationHandler.DirectoryHome)
|
||||||
|
|
||||||
application.HandleFunc("/group/settings", applicationHandler.GroupSettingsDisplay)
|
application.HandleFunc("/group/settings", applicationHandler.GroupSettingsDisplay)
|
||||||
application.HandleFunc("/group/settings/invite-member", applicationHandler.GroupSettingsInviteMember)
|
application.HandleFunc("/group/settings/invite-member", applicationHandler.GroupSettingsInviteMember)
|
||||||
|
|
||||||
|
/****************************Groupe Déplacement ************************************/
|
||||||
|
application.HandleFunc("/journeys/groups_covoiturage", applicationHandler.GroupsGestion)
|
||||||
|
application.HandleFunc("/journeys/groups_covoiturage/create", applicationHandler.CreateGroup)
|
||||||
|
application.HandleFunc("/journeys/groups_covoiturage/create/{groupid}", applicationHandler.DisplayGroupCovoiturage)
|
||||||
|
application.HandleFunc("/journeys/groups_covoiturage/create/{id}/{groupid}/{memberid}", applicationHandler.UpdateGroupCovoiturage)
|
||||||
|
/****************************************************************/
|
||||||
|
|
||||||
/********************Code Supprt Emailing************************/
|
/********************Code Supprt Emailing************************/
|
||||||
application.HandleFunc("/support/", applicationHandler.SupportSend)
|
application.HandleFunc("/support/", applicationHandler.SupportSend)
|
||||||
// application.HandleFunc("/group_module/", applicationHandler.GroupModule)
|
|
||||||
// application.HandleFunc("/group_module/create", applicationHandler.GroupModuleCreate)
|
|
||||||
/*********************** CODE GROUP **************************/
|
/*********************** CODE GROUP **************************/
|
||||||
|
|
||||||
appGroup := application.PathPrefix("/group_module").Subrouter()
|
appGroup := application.PathPrefix("/group_module").Subrouter()
|
||||||
appGroup.HandleFunc("/", applicationHandler.Groups)
|
appGroup.HandleFunc("/", applicationHandler.Groups)
|
||||||
appGroup.HandleFunc("/groups", applicationHandler.CreateGroupModule)
|
appGroup.HandleFunc("/groups", applicationHandler.CreateGroupModule)
|
||||||
//appGroup.HandleFunc("/createBeneficaire", applicationHandler.CreateGroupBeneficaire)
|
|
||||||
//appGroup.HandleFunc("/groups/{groupid}/createBeneficaire", applicationHandler.BeneficaireSearch)
|
|
||||||
//appGroup.HandleFunc("/groups/{groupid}/{beneficiaryid}", applicationHandler.DisplayGroupBeneficaire)
|
|
||||||
appGroup.HandleFunc("/groups/{groupid}", applicationHandler.DisplayGroupModule)
|
appGroup.HandleFunc("/groups/{groupid}", applicationHandler.DisplayGroupModule)
|
||||||
//appGroup.HandleFunc("/groups/{groupid}/invite-admin", applicationHandler.AdministrationGroupInviteAdmin)
|
|
||||||
/****************************************************************/
|
|
||||||
|
|
||||||
//TODO Subrouters with middlewares checking security for each module ?
|
//TODO Subrouters with middlewares checking security for each module ?
|
||||||
application.Use(idp.Middleware)
|
application.Use(idp.Middleware)
|
||||||
@@ -120,8 +140,13 @@ func main() {
|
|||||||
appAdmin.HandleFunc("/groups/", applicationHandler.AdministrationCreateGroup)
|
appAdmin.HandleFunc("/groups/", applicationHandler.AdministrationCreateGroup)
|
||||||
appAdmin.HandleFunc("/groups/{groupid}", applicationHandler.AdministrationGroupDisplay)
|
appAdmin.HandleFunc("/groups/{groupid}", applicationHandler.AdministrationGroupDisplay)
|
||||||
appAdmin.HandleFunc("/groups/{groupid}/invite-admin", applicationHandler.AdministrationGroupInviteAdmin)
|
appAdmin.HandleFunc("/groups/{groupid}/invite-admin", applicationHandler.AdministrationGroupInviteAdmin)
|
||||||
//TODO Secure with Middleware checking for modules
|
appAdmin.HandleFunc("/groups/{groupid}/invite-member", applicationHandler.AdministrationGroupInviteMember)
|
||||||
|
appAdmin.HandleFunc("/stats/vehicles", applicationHandler.AdminStatVehicles)
|
||||||
|
|
||||||
|
/////////////////////////////////////Delete subscriber///////////////////////////////////////////////
|
||||||
|
application.HandleFunc("/agenda/{eventid}/{subscribeid}/delete", applicationHandler.AgendaDeleteSubscribeEvent)
|
||||||
|
application.HandleFunc("/agenda/{eventid}/history", applicationHandler.AgendaHistoryEvent)
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
fmt.Println("-> HTTP server listening on", address)
|
fmt.Println("-> HTTP server listening on", address)
|
||||||
|
|
||||||
srv := &http.Server{
|
srv := &http.Server{
|
||||||
|
|||||||
@@ -1,13 +1,20 @@
|
|||||||
package renderer
|
package renderer
|
||||||
|
|
||||||
import "net/http"
|
import (
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
mobilityaccountsstorage "git.coopgo.io/coopgo-platform/mobility-accounts/storage"
|
||||||
|
)
|
||||||
|
|
||||||
const administrationMenu = "administration"
|
const administrationMenu = "administration"
|
||||||
|
|
||||||
func (renderer *Renderer) Administration(w http.ResponseWriter, r *http.Request, groups any) {
|
func (renderer *Renderer) Administration(w http.ResponseWriter, r *http.Request, accounts any, beneficiaries any, groups any, bookings any) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.administration.home.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.administration.home.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, administrationMenu)
|
state := NewState(r, renderer.ThemeConfig, administrationMenu)
|
||||||
state.ViewState = map[string]any{
|
state.ViewState = map[string]any{
|
||||||
|
"accounts": accounts,
|
||||||
|
"beneficiaries": beneficiaries,
|
||||||
|
"bookings": bookings,
|
||||||
"groups": groups,
|
"groups": groups,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -21,7 +28,7 @@ func (renderer *Renderer) AdministrationCreateGroup(w http.ResponseWriter, r *ht
|
|||||||
renderer.Render("administration", w, r, files, state)
|
renderer.Render("administration", w, r, files, state)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) AdministrationGroupDisplay(w http.ResponseWriter, r *http.Request, group any, groupmembers []any, admins []any) {
|
func (renderer *Renderer) AdministrationGroupDisplay(w http.ResponseWriter, r *http.Request, group any, groupmembers []mobilityaccountsstorage.Account, admins []mobilityaccountsstorage.Account) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.administration.display_group.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.administration.display_group.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, administrationMenu)
|
state := NewState(r, renderer.ThemeConfig, administrationMenu)
|
||||||
state.ViewState = map[string]any{
|
state.ViewState = map[string]any{
|
||||||
|
|||||||
@@ -29,6 +29,7 @@ func (renderer *Renderer) AgendaCreateEvent(w http.ResponseWriter, r *http.Reque
|
|||||||
func (renderer *Renderer) AgendaDisplayEvent(w http.ResponseWriter, r *http.Request, event any, group any, subscribers map[string]any, beneficiaries any) {
|
func (renderer *Renderer) AgendaDisplayEvent(w http.ResponseWriter, r *http.Request, event any, group any, subscribers map[string]any, beneficiaries any) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.agenda.display_event.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.agenda.display_event.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, agendaMenu)
|
state := NewState(r, renderer.ThemeConfig, agendaMenu)
|
||||||
|
|
||||||
state.ViewState = map[string]any{
|
state.ViewState = map[string]any{
|
||||||
"event": event,
|
"event": event,
|
||||||
"group": group,
|
"group": group,
|
||||||
@@ -38,3 +39,53 @@ func (renderer *Renderer) AgendaDisplayEvent(w http.ResponseWriter, r *http.Requ
|
|||||||
|
|
||||||
renderer.Render("agenda create event", w, r, files, state)
|
renderer.Render("agenda create event", w, r, files, state)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//////////////////////////DElete subscriber//////////////////////////////////
|
||||||
|
func (renderer *Renderer) AgendaDeleteSubscribeEvent(w http.ResponseWriter, r *http.Request, eventid string) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.agenda.delete_subscriber.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, agendaMenu)
|
||||||
|
state.ViewState = map[string]any{
|
||||||
|
"eventid": eventid,
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer.Render("agenda delete subscriber", w, r, files, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
//////////////////////////History Event//////////////////////////////////
|
||||||
|
func (renderer *Renderer) AgendaHistoryEvent(w http.ResponseWriter, r *http.Request, event any, group any, subscribers map[string]any, beneficiaries any) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.agenda.history_event.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, agendaMenu)
|
||||||
|
state.ViewState = map[string]any{
|
||||||
|
"event": event,
|
||||||
|
"group": group,
|
||||||
|
"subscribers": subscribers,
|
||||||
|
"beneficiaries": beneficiaries,
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer.Render("agenda history event", w, r, files, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////
|
||||||
|
func (renderer *Renderer) AgendaUpdateEvent(w http.ResponseWriter, r *http.Request, event any) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.agenda.update.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, agendaMenu)
|
||||||
|
|
||||||
|
state.ViewState = map[string]any{
|
||||||
|
"event": event,
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer.Render("event_update", w, r, files, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (renderer *Renderer) AgendaDeleteEvent(w http.ResponseWriter, r *http.Request, event any) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.agenda.delete.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, agendaMenu)
|
||||||
|
|
||||||
|
state.ViewState = map[string]any{
|
||||||
|
"event": event,
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer.Render("event_deleteEvent", w, r, files, state)
|
||||||
|
}
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ func (renderer *Renderer) AuthGroups(w http.ResponseWriter, r *http.Request, gro
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) AuthOnboarding(w http.ResponseWriter, r *http.Request, key string, onboarding any) {
|
func (renderer *Renderer) AuthOnboarding(w http.ResponseWriter, r *http.Request, key string, onboarding any) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.auth.onboarding.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.auth.onboarding.form.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, "")
|
state := NewState(r, renderer.ThemeConfig, "")
|
||||||
state.ViewState = map[string]any{
|
state.ViewState = map[string]any{
|
||||||
"key": key,
|
"key": key,
|
||||||
@@ -22,3 +22,41 @@ func (renderer *Renderer) AuthOnboarding(w http.ResponseWriter, r *http.Request,
|
|||||||
|
|
||||||
renderer.RenderNoLayout("onboarding", w, r, files, state)
|
renderer.RenderNoLayout("onboarding", w, r, files, state)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (renderer *Renderer) AuthOnboardingKO(w http.ResponseWriter, r *http.Request, key string) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.auth.onboarding.ko.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, "")
|
||||||
|
state.ViewState = map[string]any{
|
||||||
|
"key": key,
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer.RenderNoLayout("onboarding", w, r, files, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (renderer *Renderer) LostPasswordInit(w http.ResponseWriter, r *http.Request) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.auth.lost_password.init.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, "")
|
||||||
|
state.ViewState = map[string]any{}
|
||||||
|
|
||||||
|
renderer.RenderNoLayout("lost_password_init", w, r, files, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (renderer *Renderer) LostPasswordRecover(w http.ResponseWriter, r *http.Request, recover any) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.auth.lost_password.recover.form.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, "")
|
||||||
|
state.ViewState = map[string]any{
|
||||||
|
"recover": recover,
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer.RenderNoLayout("lost_password_recover", w, r, files, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (renderer *Renderer) LostPasswordRecoverKO(w http.ResponseWriter, r *http.Request, key string) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.auth.lost_password.recover.ko.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, "")
|
||||||
|
state.ViewState = map[string]any{
|
||||||
|
"key": key,
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer.RenderNoLayout("lost_password_recover_ko", w, r, files, state)
|
||||||
|
}
|
||||||
|
|||||||
@@ -4,6 +4,8 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"html/template"
|
"html/template"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
|
mobilityaccountsstorage "git.coopgo.io/coopgo-platform/mobility-accounts/storage"
|
||||||
)
|
)
|
||||||
|
|
||||||
const beneficiariesMenu = "beneficiaries"
|
const beneficiariesMenu = "beneficiaries"
|
||||||
@@ -11,7 +13,7 @@ const beneficiariesMenu = "beneficiaries"
|
|||||||
type BeneficiariesListState struct {
|
type BeneficiariesListState struct {
|
||||||
Count int `json:"count"`
|
Count int `json:"count"`
|
||||||
CacheId string `json:"cache_id"`
|
CacheId string `json:"cache_id"`
|
||||||
Beneficiaries []any `json:"beneficiaries"`
|
Beneficiaries []mobilityaccountsstorage.Account `json:"beneficiaries"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s BeneficiariesListState) JSON() template.JS {
|
func (s BeneficiariesListState) JSON() template.JS {
|
||||||
@@ -26,7 +28,7 @@ func (s BeneficiariesListState) JSONWithLimits(a int, b int) template.JS {
|
|||||||
return s.JSON()
|
return s.JSON()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) BeneficiariesList(w http.ResponseWriter, r *http.Request, accounts []any, cacheid string) {
|
func (renderer *Renderer) BeneficiariesList(w http.ResponseWriter, r *http.Request, accounts []mobilityaccountsstorage.Account, cacheid string) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.beneficiaries.list.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.beneficiaries.list.files")
|
||||||
|
|
||||||
state := NewState(r, renderer.ThemeConfig, beneficiariesMenu)
|
state := NewState(r, renderer.ThemeConfig, beneficiariesMenu)
|
||||||
@@ -50,12 +52,16 @@ type BeneficiariesDisplayState struct {
|
|||||||
Beneficiary any
|
Beneficiary any
|
||||||
}
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) BeneficiaryDisplay(w http.ResponseWriter, r *http.Request, beneficiary any, bookings []any) {
|
func (renderer *Renderer) BeneficiaryDisplay(w http.ResponseWriter, r *http.Request, beneficiary any, bookings []any, organizations []any, beneficiaries_file_types []string, file_types_map map[string]string, documents any) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.beneficiaries.display.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.beneficiaries.display.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, beneficiariesMenu)
|
state := NewState(r, renderer.ThemeConfig, beneficiariesMenu)
|
||||||
state.ViewState = map[string]any{
|
state.ViewState = map[string]any{
|
||||||
"beneficiary": beneficiary,
|
"beneficiary": beneficiary,
|
||||||
"bookings": bookings,
|
"bookings": bookings,
|
||||||
|
"beneficiaries_file_types": beneficiaries_file_types,
|
||||||
|
"file_types_map": file_types_map,
|
||||||
|
"documents": documents,
|
||||||
|
"organizations": organizations,
|
||||||
}
|
}
|
||||||
|
|
||||||
renderer.Render("beneficiaries_display", w, r, files, state)
|
renderer.Render("beneficiaries_display", w, r, files, state)
|
||||||
|
|||||||
@@ -23,6 +23,17 @@ func TimeFrom(d any) *time.Time {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TimeFormat(d any, f string) string {
|
||||||
|
date := TimeFrom(d)
|
||||||
|
if date == nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
if date.Before(time.Now().Add(-24 * 365 * 30 * time.Hour)) {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
return date.Format(f)
|
||||||
|
}
|
||||||
|
|
||||||
func GenderISO5218(d any) string {
|
func GenderISO5218(d any) string {
|
||||||
if date, ok := d.(string); ok {
|
if date, ok := d.(string); ok {
|
||||||
switch date {
|
switch date {
|
||||||
|
|||||||
@@ -2,9 +2,10 @@ package renderer
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
|
||||||
"html/template"
|
"html/template"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
|
mobilityaccountsstorage "git.coopgo.io/coopgo-platform/mobility-accounts/storage"
|
||||||
)
|
)
|
||||||
|
|
||||||
const groupMenu = "group_module"
|
const groupMenu = "group_module"
|
||||||
@@ -47,7 +48,7 @@ func (renderer *Renderer) CreateGroupModule(w http.ResponseWriter, r *http.Reque
|
|||||||
renderer.Render("group_module", w, r, files, state)
|
renderer.Render("group_module", w, r, files, state)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) DisplayGroupModule(w http.ResponseWriter, r *http.Request, groupid string, accounts []any, cacheid string, searched bool, beneficiary any, group any, beneficiaries []any) {
|
func (renderer *Renderer) DisplayGroupModule(w http.ResponseWriter, r *http.Request, groupid string, accounts []any, cacheid string, searched bool, beneficiary any, group any, beneficiaries []mobilityaccountsstorage.Account) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.group_module.display_group.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.group_module.display_group.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, groupMenu)
|
state := NewState(r, renderer.ThemeConfig, groupMenu)
|
||||||
|
|
||||||
@@ -70,12 +71,7 @@ func (renderer *Renderer) DisplayGroupModule(w http.ResponseWriter, r *http.Requ
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Println(beneficiary)
|
|
||||||
|
|
||||||
state.ViewState = viewstate
|
state.ViewState = viewstate
|
||||||
fmt.Println("èèèèèèèèèèèèèèèèèèèèèèèèèè")
|
|
||||||
fmt.Println(state.ViewState)
|
|
||||||
fmt.Println(group)
|
|
||||||
|
|
||||||
renderer.Render("beneficiaries_list", w, r, files, state)
|
renderer.Render("beneficiaries_list", w, r, files, state)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,40 @@
|
|||||||
package renderer
|
package renderer
|
||||||
|
|
||||||
import "net/http"
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"html/template"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
groupstorage "git.coopgo.io/coopgo-platform/groups-management/storage"
|
||||||
|
mobilityaccountsstorage "git.coopgo.io/coopgo-platform/mobility-accounts/storage"
|
||||||
|
)
|
||||||
|
|
||||||
const journeysMenu = "journeys"
|
const journeysMenu = "journeys"
|
||||||
|
|
||||||
|
type BeneficiariesCovoiturage struct {
|
||||||
|
Group string `json:"group"`
|
||||||
|
Count int `json:"count"`
|
||||||
|
CacheId string `json:"cache_id"`
|
||||||
|
Beneficiaries []any `json:"beneficiaries"`
|
||||||
|
}
|
||||||
|
type BeneficiariesCovoiturageA struct {
|
||||||
|
Count int `json:"count"`
|
||||||
|
CacheId string `json:"cache_id"`
|
||||||
|
Beneficiaries []any `json:"beneficiaries"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s BeneficiariesCovoiturage) JSON() template.JS {
|
||||||
|
result, _ := json.Marshal(s)
|
||||||
|
return template.JS(result)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s BeneficiariesCovoiturage) JSONWithLimits(a int, b int) template.JS {
|
||||||
|
if b < len(s.Beneficiaries) {
|
||||||
|
s.Beneficiaries = s.Beneficiaries[a:b]
|
||||||
|
}
|
||||||
|
return s.JSON()
|
||||||
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) JourneysSearch(w http.ResponseWriter, r *http.Request, carpools any, transitjourneys any, vehicles []any, searched bool, departure any, destination any, departuredate string, departuretime string) {
|
func (renderer *Renderer) JourneysSearch(w http.ResponseWriter, r *http.Request, carpools any, transitjourneys any, vehicles []any, searched bool, departure any, destination any, departuredate string, departuretime string) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.journeys.search.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.journeys.search.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, journeysMenu)
|
state := NewState(r, renderer.ThemeConfig, journeysMenu)
|
||||||
@@ -20,3 +51,100 @@ func (renderer *Renderer) JourneysSearch(w http.ResponseWriter, r *http.Request,
|
|||||||
|
|
||||||
renderer.Render("journeys", w, r, files, state)
|
renderer.Render("journeys", w, r, files, state)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type BeneficiariesListstate struct {
|
||||||
|
Count int `json:"count"`
|
||||||
|
CacheId string `json:"cache_id"`
|
||||||
|
Beneficiaries []groupstorage.Group `json:"beneficiaries"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s BeneficiariesListstate) JSON() template.JS {
|
||||||
|
result, _ := json.Marshal(s)
|
||||||
|
return template.JS(result)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s BeneficiariesListstate) JSONWithLimits(a int, b int) template.JS {
|
||||||
|
if b < len(s.Beneficiaries) {
|
||||||
|
s.Beneficiaries = s.Beneficiaries[a:b]
|
||||||
|
}
|
||||||
|
return s.JSON()
|
||||||
|
}
|
||||||
|
func (renderer *Renderer) GroupsGestion(w http.ResponseWriter, r *http.Request, groups []groupstorage.Group, cacheid string) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.journeys.list.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, journeysMenu)
|
||||||
|
|
||||||
|
state.ViewState = BeneficiariesListstate{
|
||||||
|
Count: len(groups),
|
||||||
|
CacheId: cacheid,
|
||||||
|
Beneficiaries: groups,
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer.Render("journeys", w, r, files, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (renderer *Renderer) CreateGroup(w http.ResponseWriter, r *http.Request, depart any, arrive any, searched bool, beneficiary any, beneficiaries []mobilityaccountsstorage.Account, departure any, destination any) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.journeys.create.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, journeysMenu)
|
||||||
|
viewstate := map[string]any{
|
||||||
|
"deeparture": depart,
|
||||||
|
"deestination": arrive,
|
||||||
|
"beneficiaries": beneficiaries,
|
||||||
|
"searched": searched,
|
||||||
|
"departure": departure,
|
||||||
|
"destination": destination,
|
||||||
|
}
|
||||||
|
|
||||||
|
if searched {
|
||||||
|
viewstate["search"] = map[string]any{
|
||||||
|
"beneficiary": beneficiary,
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
state.ViewState = viewstate
|
||||||
|
renderer.Render("journeys", w, r, files, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (renderer *Renderer) DisplayGroupCovoiturage(w http.ResponseWriter, r *http.Request, number string, groupid string, depart any, arrive any, accounts []any, cacheid string, searched bool, beneficiary any, group any, beneficiaries []mobilityaccountsstorage.Account, groups map[string]any) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.journeys.display.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, journeysMenu)
|
||||||
|
|
||||||
|
viewstate := map[string]any{
|
||||||
|
"beneficiaries": beneficiaries,
|
||||||
|
"searched": searched,
|
||||||
|
"group": group,
|
||||||
|
"deeparture": depart,
|
||||||
|
"deestination": arrive,
|
||||||
|
"groups": groups,
|
||||||
|
"ben": accounts,
|
||||||
|
"number": number,
|
||||||
|
"list": BeneficiariesCovoiturage{
|
||||||
|
Group: groupid,
|
||||||
|
Count: len(accounts),
|
||||||
|
CacheId: cacheid,
|
||||||
|
Beneficiaries: accounts,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
if searched {
|
||||||
|
viewstate["search"] = map[string]any{
|
||||||
|
"beneficiary": beneficiary,
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
state.ViewState = viewstate
|
||||||
|
|
||||||
|
renderer.Render("journeys", w, r, files, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (renderer *Renderer) UpdateGroupCovoiturage(w http.ResponseWriter, r *http.Request, groupid string, memberid string) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.journeys.update.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, journeysMenu)
|
||||||
|
|
||||||
|
state.ViewState = map[string]any{
|
||||||
|
"groupid": groupid,
|
||||||
|
"memberid": memberid,
|
||||||
|
}
|
||||||
|
renderer.Render("journeys", w, r, files, state)
|
||||||
|
|
||||||
|
}
|
||||||
|
|||||||
24
renderer/members.go
Normal file
24
renderer/members.go
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
package renderer
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
const membersMenu = "members"
|
||||||
|
|
||||||
|
func (renderer *Renderer) MemberDisplay(w http.ResponseWriter, r *http.Request, admins any) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.members.display.files")
|
||||||
|
|
||||||
|
state := NewState(r, renderer.ThemeConfig, membersMenu)
|
||||||
|
state.ViewState = map[string]any{
|
||||||
|
"admins": admins,
|
||||||
|
}
|
||||||
|
renderer.Render("members_list", w, r, files, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (renderer *Renderer) MemberUpdate(w http.ResponseWriter, r *http.Request, user any) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.members.update.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, membersMenu)
|
||||||
|
state.ViewState = user
|
||||||
|
renderer.Render("members_update", w, r, files, state)
|
||||||
|
}
|
||||||
@@ -9,6 +9,7 @@ import (
|
|||||||
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
||||||
"git.coopgo.io/coopgo-platform/emailing"
|
"git.coopgo.io/coopgo-platform/emailing"
|
||||||
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
||||||
|
"github.com/coreos/go-oidc"
|
||||||
"github.com/spf13/viper"
|
"github.com/spf13/viper"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -35,6 +36,7 @@ func NewRenderer(global *viper.Viper, templates_dir string) *Renderer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) Render(name string, w http.ResponseWriter, r *http.Request, files []string, state RenderState) {
|
func (renderer *Renderer) Render(name string, w http.ResponseWriter, r *http.Request, files []string, state RenderState) {
|
||||||
|
|
||||||
genericFiles := renderer.ThemeConfig.GetStringSlice("views.generic.files")
|
genericFiles := renderer.ThemeConfig.GetStringSlice("views.generic.files")
|
||||||
|
|
||||||
prefixed_files := []string{}
|
prefixed_files := []string{}
|
||||||
@@ -49,6 +51,7 @@ func (renderer *Renderer) Render(name string, w http.ResponseWriter, r *http.Req
|
|||||||
t := template.New(name).Funcs(
|
t := template.New(name).Funcs(
|
||||||
template.FuncMap{
|
template.FuncMap{
|
||||||
"timeFrom": TimeFrom,
|
"timeFrom": TimeFrom,
|
||||||
|
"timeFormat": TimeFormat,
|
||||||
"genderISO5218": GenderISO5218,
|
"genderISO5218": GenderISO5218,
|
||||||
"dict": Dict,
|
"dict": Dict,
|
||||||
"json": JSON,
|
"json": JSON,
|
||||||
@@ -66,6 +69,7 @@ func (renderer *Renderer) Render(name string, w http.ResponseWriter, r *http.Req
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) RenderNoLayout(name string, w http.ResponseWriter, r *http.Request, files []string, state RenderState) {
|
func (renderer *Renderer) RenderNoLayout(name string, w http.ResponseWriter, r *http.Request, files []string, state RenderState) {
|
||||||
|
|
||||||
prefixed_files := []string{}
|
prefixed_files := []string{}
|
||||||
for _, f := range files {
|
for _, f := range files {
|
||||||
prefixed_files = append(prefixed_files, renderer.templateFile(f))
|
prefixed_files = append(prefixed_files, renderer.templateFile(f))
|
||||||
@@ -75,6 +79,7 @@ func (renderer *Renderer) RenderNoLayout(name string, w http.ResponseWriter, r *
|
|||||||
t := template.New(name).Funcs(
|
t := template.New(name).Funcs(
|
||||||
template.FuncMap{
|
template.FuncMap{
|
||||||
"timeFrom": TimeFrom,
|
"timeFrom": TimeFrom,
|
||||||
|
"timeFormat": TimeFormat,
|
||||||
"genderISO5218": GenderISO5218,
|
"genderISO5218": GenderISO5218,
|
||||||
"dict": Dict,
|
"dict": Dict,
|
||||||
"json": JSON,
|
"json": JSON,
|
||||||
@@ -97,14 +102,34 @@ func (r *Renderer) templateFile(file string) string {
|
|||||||
type RenderState struct {
|
type RenderState struct {
|
||||||
icons.IconSet
|
icons.IconSet
|
||||||
LayoutState
|
LayoutState
|
||||||
Group any
|
UserID string
|
||||||
|
UserClaims map[string]any
|
||||||
|
Group storage.Group
|
||||||
Roles any
|
Roles any
|
||||||
ViewState any // This is a state specific to a given view
|
ViewState any // This is a state specific to a given view
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewState(r *http.Request, themeConfig *viper.Viper, menuState string) RenderState {
|
func NewState(r *http.Request, themeConfig *viper.Viper, menuState string) RenderState {
|
||||||
iconset := themeConfig.GetStringMapString("icons.svg")
|
iconset := themeConfig.GetStringMapString("icons.svg")
|
||||||
|
|
||||||
// Get State elements from Request
|
// Get State elements from Request
|
||||||
|
var userid string
|
||||||
|
var claims map[string]any
|
||||||
|
|
||||||
|
u := r.Context().Value(identification.IdtokenKey)
|
||||||
|
if u != nil {
|
||||||
|
if current_user_token, ok := u.(*oidc.IDToken); ok {
|
||||||
|
userid = current_user_token.Subject
|
||||||
|
}
|
||||||
|
|
||||||
|
c := r.Context().Value(identification.ClaimsKey)
|
||||||
|
if c != nil {
|
||||||
|
if current_user_claims, ok := c.(map[string]any); ok {
|
||||||
|
claims = current_user_claims
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
g := r.Context().Value(identification.GroupKey)
|
g := r.Context().Value(identification.GroupKey)
|
||||||
if g == nil {
|
if g == nil {
|
||||||
@@ -187,7 +212,6 @@ func NewState(r *http.Request, themeConfig *viper.Viper, menuState string) Rende
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/*************************** my code ******************************/
|
|
||||||
if modules["support"] != nil && modules["support"].(bool) {
|
if modules["support"] != nil && modules["support"].(bool) {
|
||||||
ls.MenuItems = append(ls.MenuItems, MenuItem{
|
ls.MenuItems = append(ls.MenuItems, MenuItem{
|
||||||
Title: "Support",
|
Title: "Support",
|
||||||
@@ -197,7 +221,7 @@ func NewState(r *http.Request, themeConfig *viper.Viper, menuState string) Rende
|
|||||||
})
|
})
|
||||||
|
|
||||||
}
|
}
|
||||||
/******************************************************************/
|
|
||||||
if modules["group_module"] != nil && modules["group_module"].(bool) {
|
if modules["group_module"] != nil && modules["group_module"].(bool) {
|
||||||
ls.MenuItems = append(ls.MenuItems, MenuItem{
|
ls.MenuItems = append(ls.MenuItems, MenuItem{
|
||||||
Title: "Groupes / Communautés",
|
Title: "Groupes / Communautés",
|
||||||
@@ -207,7 +231,6 @@ func NewState(r *http.Request, themeConfig *viper.Viper, menuState string) Rende
|
|||||||
})
|
})
|
||||||
|
|
||||||
}
|
}
|
||||||
/*************************** my code ******************************/
|
|
||||||
|
|
||||||
if modules["directory"] != nil && modules["directory"].(bool) {
|
if modules["directory"] != nil && modules["directory"].(bool) {
|
||||||
ls.MenuItems = append(ls.MenuItems, MenuItem{
|
ls.MenuItems = append(ls.MenuItems, MenuItem{
|
||||||
@@ -217,11 +240,20 @@ func NewState(r *http.Request, themeConfig *viper.Viper, menuState string) Rende
|
|||||||
Icon: "hero:outline/document-text",
|
Icon: "hero:outline/document-text",
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
if modules["conseillers"] != nil && modules["conseillers"].(bool) {
|
||||||
|
ls.MenuItems = append(ls.MenuItems, MenuItem{
|
||||||
|
Title: "Conseillers",
|
||||||
|
Link: "/app/conseillers/",
|
||||||
|
Active: menuState == membersMenu,
|
||||||
|
Icon: "hero:outline/user-group",
|
||||||
|
})
|
||||||
|
}
|
||||||
return RenderState{
|
return RenderState{
|
||||||
IconSet: icons.NewIconSet(iconset),
|
IconSet: icons.NewIconSet(iconset),
|
||||||
Group: group,
|
Group: group,
|
||||||
Roles: roles,
|
Roles: roles,
|
||||||
|
UserID: userid,
|
||||||
|
UserClaims: claims,
|
||||||
LayoutState: ls,
|
LayoutState: ls,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,11 +5,12 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const commentMenu = "comment"
|
const commentMenu = "comment"
|
||||||
const commentsend = "sendComment"
|
|
||||||
|
//const commentsend = "sendComment"
|
||||||
|
|
||||||
func (renderer *Renderer) SupportSend(w http.ResponseWriter, r *http.Request, comment any, admins any) {
|
func (renderer *Renderer) SupportSend(w http.ResponseWriter, r *http.Request, comment any, admins any) {
|
||||||
|
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.support.search.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.support.request.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, commentMenu)
|
state := NewState(r, renderer.ThemeConfig, commentMenu)
|
||||||
state.ViewState = map[string]any{
|
state.ViewState = map[string]any{
|
||||||
"comment": comment,
|
"comment": comment,
|
||||||
|
|||||||
@@ -1,22 +1,44 @@
|
|||||||
package renderer
|
package renderer
|
||||||
|
|
||||||
import "net/http"
|
import (
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
filestorage "git.coopgo.io/coopgo-apps/parcoursmob/utils/storage"
|
||||||
|
fleetsstorage "git.coopgo.io/coopgo-platform/fleets/storage"
|
||||||
|
)
|
||||||
|
|
||||||
const vehiclesmanagementMenu = "vehicles_management"
|
const vehiclesmanagementMenu = "vehicles_management"
|
||||||
|
|
||||||
func (renderer *Renderer) VehiclesManagementOverview(w http.ResponseWriter, r *http.Request, vehicles []any) {
|
func (renderer *Renderer) VehiclesManagementOverview(w http.ResponseWriter, r *http.Request, vehicles []fleetsstorage.Vehicle, vehicles_map map[string]fleetsstorage.Vehicle, bookings []fleetsstorage.Booking) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.vehicles_management.overview.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.vehicles_management.overview.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, vehiclesmanagementMenu)
|
state := NewState(r, renderer.ThemeConfig, vehiclesmanagementMenu)
|
||||||
state.ViewState = map[string]any{
|
state.ViewState = map[string]any{
|
||||||
"vehicles": vehicles,
|
"vehicles": vehicles,
|
||||||
|
"bookings": bookings,
|
||||||
|
"vehicles_map": vehicles_map,
|
||||||
}
|
}
|
||||||
|
|
||||||
renderer.Render("fleet overview", w, r, files, state)
|
renderer.Render("fleet overview", w, r, files, state)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) VehiclesFleetAdd(w http.ResponseWriter, r *http.Request) {
|
func (renderer *Renderer) VehiclesManagementBookingsList(w http.ResponseWriter, r *http.Request, vehicles_map map[string]fleetsstorage.Vehicle, bookings []fleetsstorage.Booking, cacheid string) {
|
||||||
|
files := renderer.ThemeConfig.GetStringSlice("views.vehicles_management.bookings_list.files")
|
||||||
|
state := NewState(r, renderer.ThemeConfig, vehiclesmanagementMenu)
|
||||||
|
state.ViewState = map[string]any{
|
||||||
|
"bookings": bookings,
|
||||||
|
"vehicles_map": vehicles_map,
|
||||||
|
"cacheid": cacheid,
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer.Render("fleet overview", w, r, files, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (renderer *Renderer) VehiclesFleetAdd(w http.ResponseWriter, r *http.Request, vehicle_types []string) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.vehicles_management.fleet_add.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.vehicles_management.fleet_add.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, vehiclesmanagementMenu)
|
state := NewState(r, renderer.ThemeConfig, vehiclesmanagementMenu)
|
||||||
|
state.ViewState = map[string]any{
|
||||||
|
"vehicle_types": vehicle_types,
|
||||||
|
}
|
||||||
|
|
||||||
renderer.Render("fleet add vehicle", w, r, files, state)
|
renderer.Render("fleet add vehicle", w, r, files, state)
|
||||||
}
|
}
|
||||||
@@ -41,7 +63,7 @@ func (renderer *Renderer) VehiclesFleetUpdate(w http.ResponseWriter, r *http.Req
|
|||||||
renderer.Render("fleet display vehicle", w, r, files, state)
|
renderer.Render("fleet display vehicle", w, r, files, state)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) VehicleManagementBookingDisplay(w http.ResponseWriter, r *http.Request, booking any, vehicle any, beneficiary any, group any) {
|
func (renderer *Renderer) VehicleManagementBookingDisplay(w http.ResponseWriter, r *http.Request, booking any, vehicle any, beneficiary any, group any, documents []filestorage.FileInfo, file_types_map map[string]string, alternative_vehicles []any) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.vehicles_management.booking_display.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.vehicles_management.booking_display.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, vehiclesmanagementMenu)
|
state := NewState(r, renderer.ThemeConfig, vehiclesmanagementMenu)
|
||||||
state.ViewState = map[string]any{
|
state.ViewState = map[string]any{
|
||||||
@@ -49,6 +71,9 @@ func (renderer *Renderer) VehicleManagementBookingDisplay(w http.ResponseWriter,
|
|||||||
"vehicle": vehicle,
|
"vehicle": vehicle,
|
||||||
"beneficiary": beneficiary,
|
"beneficiary": beneficiary,
|
||||||
"group": group,
|
"group": group,
|
||||||
|
"documents": documents,
|
||||||
|
"file_types_map": file_types_map,
|
||||||
|
"alternative_vehicles": alternative_vehicles,
|
||||||
}
|
}
|
||||||
|
|
||||||
renderer.Render("vehicles search", w, r, files, state)
|
renderer.Render("vehicles search", w, r, files, state)
|
||||||
|
|||||||
@@ -1,15 +1,34 @@
|
|||||||
package renderer
|
package renderer
|
||||||
|
|
||||||
import "net/http"
|
import (
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
filestorage "git.coopgo.io/coopgo-apps/parcoursmob/utils/storage"
|
||||||
|
"git.coopgo.io/coopgo-platform/fleets/storage"
|
||||||
|
mobilityaccountsstorage "git.coopgo.io/coopgo-platform/mobility-accounts/storage"
|
||||||
|
)
|
||||||
|
|
||||||
const vehiclesMenu = "vehicles"
|
const vehiclesMenu = "vehicles"
|
||||||
|
|
||||||
func (renderer *Renderer) VehiclesSearch(w http.ResponseWriter, r *http.Request, beneficiaries []any, searched bool, vehicles []any, beneficiary any, startdate any, enddate any) {
|
func selectDocumentsDefaults(beneficiarydocuments []filestorage.FileInfo, mandatory_documents []string) map[string]string {
|
||||||
|
res := map[string]string{}
|
||||||
|
for _, v := range mandatory_documents {
|
||||||
|
for _, d := range beneficiarydocuments {
|
||||||
|
if d.Metadata["Type"] == v {
|
||||||
|
res[v] = d.Key
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
func (renderer *Renderer) VehiclesSearch(w http.ResponseWriter, r *http.Request, beneficiaries []mobilityaccountsstorage.Account, searched bool, vehicles []any, beneficiary any, startdate any, enddate any, mandatory_documents []string, file_types_map map[string]string, beneficiarydocuments []filestorage.FileInfo, selected_type string, automatic bool, vehicles_types []string, admingroups map[string]any) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.vehicles.search.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.vehicles.search.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, vehiclesMenu)
|
state := NewState(r, renderer.ThemeConfig, vehiclesMenu)
|
||||||
viewstate := map[string]any{
|
viewstate := map[string]any{
|
||||||
"beneficiaries": beneficiaries,
|
"beneficiaries": beneficiaries,
|
||||||
"searched": searched,
|
"searched": searched,
|
||||||
|
"vehicles_types": vehicles_types,
|
||||||
}
|
}
|
||||||
|
|
||||||
if searched {
|
if searched {
|
||||||
@@ -18,6 +37,13 @@ func (renderer *Renderer) VehiclesSearch(w http.ResponseWriter, r *http.Request,
|
|||||||
"enddate": enddate,
|
"enddate": enddate,
|
||||||
"vehicles": vehicles,
|
"vehicles": vehicles,
|
||||||
"beneficiary": beneficiary,
|
"beneficiary": beneficiary,
|
||||||
|
"mandatory_documents": mandatory_documents,
|
||||||
|
"file_types_map": file_types_map,
|
||||||
|
"beneficiary_documents": beneficiarydocuments,
|
||||||
|
"selected_type": selected_type,
|
||||||
|
"automatic": automatic,
|
||||||
|
"admingroups": admingroups,
|
||||||
|
"documents_defaults": selectDocumentsDefaults(beneficiarydocuments, mandatory_documents),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -26,7 +52,7 @@ func (renderer *Renderer) VehiclesSearch(w http.ResponseWriter, r *http.Request,
|
|||||||
renderer.Render("vehicles search", w, r, files, state)
|
renderer.Render("vehicles search", w, r, files, state)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) VehicleBookingDisplay(w http.ResponseWriter, r *http.Request, booking any, vehicle any, beneficiary any, group any) {
|
func (renderer *Renderer) VehicleBookingDisplay(w http.ResponseWriter, r *http.Request, booking any, vehicle any, beneficiary any, group any, documents []filestorage.FileInfo, file_types_map map[string]string) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.vehicles.booking_display.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.vehicles.booking_display.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, vehiclesMenu)
|
state := NewState(r, renderer.ThemeConfig, vehiclesMenu)
|
||||||
state.ViewState = map[string]any{
|
state.ViewState = map[string]any{
|
||||||
@@ -34,16 +60,20 @@ func (renderer *Renderer) VehicleBookingDisplay(w http.ResponseWriter, r *http.R
|
|||||||
"vehicle": vehicle,
|
"vehicle": vehicle,
|
||||||
"beneficiary": beneficiary,
|
"beneficiary": beneficiary,
|
||||||
"group": group,
|
"group": group,
|
||||||
|
"documents": documents,
|
||||||
|
"file_types_map": file_types_map,
|
||||||
}
|
}
|
||||||
|
|
||||||
renderer.Render("vehicles search", w, r, files, state)
|
renderer.Render("vehicles search", w, r, files, state)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (renderer *Renderer) VehicleBookingsList(w http.ResponseWriter, r *http.Request, bookings []any) {
|
func (renderer *Renderer) VehicleBookingsList(w http.ResponseWriter, r *http.Request, bookings []storage.Booking, vehiclesMap any, groupsMap any) {
|
||||||
files := renderer.ThemeConfig.GetStringSlice("views.vehicles.bookings_list.files")
|
files := renderer.ThemeConfig.GetStringSlice("views.vehicles.bookings_list.files")
|
||||||
state := NewState(r, renderer.ThemeConfig, vehiclesMenu)
|
state := NewState(r, renderer.ThemeConfig, vehiclesMenu)
|
||||||
state.ViewState = map[string]any{
|
state.ViewState = map[string]any{
|
||||||
"bookings": bookings,
|
"bookings": bookings,
|
||||||
|
"vehicles_map": vehiclesMap,
|
||||||
|
"groups_map": groupsMap,
|
||||||
}
|
}
|
||||||
|
|
||||||
renderer.Render("vehicles search", w, r, files, state)
|
renderer.Render("vehicles search", w, r, files, state)
|
||||||
|
|||||||
@@ -1,7 +1,12 @@
|
|||||||
package services
|
package services
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
|
"sort"
|
||||||
|
|
||||||
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/sorting"
|
||||||
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
||||||
|
"git.coopgo.io/coopgo-platform/fleets/storage"
|
||||||
"google.golang.org/grpc"
|
"google.golang.org/grpc"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -21,3 +26,43 @@ func NewFleetsService(dial string) (*FleetsService, error) {
|
|||||||
FleetsClient: client,
|
FleetsClient: client,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *ServicesHandler) GetBooking(bookingid string) (booking storage.Booking, err error) {
|
||||||
|
request := &fleets.GetBookingRequest{
|
||||||
|
Bookingid: bookingid,
|
||||||
|
}
|
||||||
|
resp, err := s.GRPC.Fleets.GetBooking(context.TODO(), request)
|
||||||
|
if err == nil {
|
||||||
|
booking = resp.Booking.ToStorageType()
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ServicesHandler) GetBookings() (bookings []storage.Booking, err error) {
|
||||||
|
bookings = []storage.Booking{}
|
||||||
|
|
||||||
|
request := &fleets.GetBookingsRequest{}
|
||||||
|
resp, err := s.GRPC.Fleets.GetBookings(context.TODO(), request)
|
||||||
|
if err == nil {
|
||||||
|
for _, booking := range resp.Bookings {
|
||||||
|
bookings = append(bookings, booking.ToStorageType())
|
||||||
|
}
|
||||||
|
sort.Sort(sorting.BookingsByStartdate(bookings))
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ServicesHandler) GetVehiclesMap() (vehicles map[string]storage.Vehicle, err error) {
|
||||||
|
vehicles = map[string]storage.Vehicle{}
|
||||||
|
|
||||||
|
request := &fleets.GetVehiclesRequest{}
|
||||||
|
resp, err := s.GRPC.Fleets.GetVehicles(context.TODO(), request)
|
||||||
|
if err == nil {
|
||||||
|
for _, vehicle := range resp.Vehicles {
|
||||||
|
vehicles[vehicle.Id] = vehicle.ToStorageType()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,7 +1,10 @@
|
|||||||
package services
|
package services
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
||||||
|
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
||||||
"google.golang.org/grpc"
|
"google.golang.org/grpc"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -21,3 +24,35 @@ func NewGroupsManagementService(groupsManagementDial string) (*GroupsManagementS
|
|||||||
GroupsManagementClient: client,
|
GroupsManagementClient: client,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *ServicesHandler) GetGroupsMap() (groups map[string]storage.Group, err error) {
|
||||||
|
groups = map[string]storage.Group{}
|
||||||
|
|
||||||
|
request := &groupsmanagement.GetGroupsRequest{
|
||||||
|
Namespaces: []string{"parcoursmob_organizations"},
|
||||||
|
}
|
||||||
|
resp, err := s.GRPC.GroupsManagement.GetGroups(context.TODO(), request)
|
||||||
|
if err == nil {
|
||||||
|
for _, group := range resp.Groups {
|
||||||
|
groups[group.Id] = group.ToStorageType()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////optimize the code//////////////////////////////////////
|
||||||
|
func (s *ServicesHandler) GetGroupsMemberMap(id string) (groups map[string]any, err error) {
|
||||||
|
groups = map[string]any{}
|
||||||
|
|
||||||
|
request := &groupsmanagement.GetGroupsBatchMemberRequest{
|
||||||
|
Groupids: []string{id},
|
||||||
|
}
|
||||||
|
resp, err := s.GRPC.GroupsManagement.GetGroupsBatchMember(context.TODO(), request)
|
||||||
|
if err == nil {
|
||||||
|
for _, group := range resp.Groups {
|
||||||
|
groups[group.Memberid] = group.ToStorageType()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,7 +1,10 @@
|
|||||||
package services
|
package services
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
mobilityaccounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
||||||
|
"git.coopgo.io/coopgo-platform/mobility-accounts/storage"
|
||||||
"google.golang.org/grpc"
|
"google.golang.org/grpc"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -21,3 +24,37 @@ func NewMobilityAccountService(mobilityAccountsDial string) (*MobilityAccountSer
|
|||||||
MobilityAccountsClient: client,
|
MobilityAccountsClient: client,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *ServicesHandler) GetBeneficiaries() (accounts []storage.Account, err error) {
|
||||||
|
accounts = []storage.Account{}
|
||||||
|
request := &mobilityaccounts.GetAccountsRequest{
|
||||||
|
Namespaces: []string{"parcoursmob_beneficiaries"},
|
||||||
|
}
|
||||||
|
resp, err := s.GRPC.MobilityAccounts.GetAccounts(context.TODO(), request)
|
||||||
|
|
||||||
|
if err == nil {
|
||||||
|
for _, v := range resp.Accounts {
|
||||||
|
a := v.ToStorageType()
|
||||||
|
accounts = append(accounts, a)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ServicesHandler) GetAccounts() (accounts []storage.Account, err error) {
|
||||||
|
accounts = []storage.Account{}
|
||||||
|
request := &mobilityaccounts.GetAccountsRequest{
|
||||||
|
Namespaces: []string{"parcoursmob"},
|
||||||
|
}
|
||||||
|
resp, err := s.GRPC.MobilityAccounts.GetAccounts(context.TODO(), request)
|
||||||
|
|
||||||
|
if err == nil {
|
||||||
|
for _, v := range resp.Accounts {
|
||||||
|
a := v.ToStorageType()
|
||||||
|
accounts = append(accounts, a)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,24 +0,0 @@
|
|||||||
# PARCOURSMOB default template
|
|
||||||
|
|
||||||
This theme uses :
|
|
||||||
|
|
||||||
- [TailwindCSS](https://tailwindcss.com/) as CSS framework and [Tailwind UI components](https://tailwindui.com/)
|
|
||||||
- [AlpineJS](https://alpinejs.dev/) lightweight Javascript framework
|
|
||||||
|
|
||||||
## TailwindCSS
|
|
||||||
|
|
||||||
Look at the [Tailwind CSS docs](https://tailwindcss.com/docs/installation) to know how to install and use Tailwind.
|
|
||||||
|
|
||||||
If you installed the Tailwind CLI, run this command from this repository while developing from the web/ directory.
|
|
||||||
|
|
||||||
```
|
|
||||||
npx tailwind -i ./assets/css/main.css -o public/css/main.css --watch
|
|
||||||
```
|
|
||||||
|
|
||||||
## Esbuild
|
|
||||||
|
|
||||||
To bundle Javascript with esbuild :
|
|
||||||
|
|
||||||
```
|
|
||||||
npx esbuild assets/js/main.js --bundle --outfile=public/js/main.js
|
|
||||||
```
|
|
||||||
@@ -1,191 +0,0 @@
|
|||||||
name: PARCOURSMOB
|
|
||||||
|
|
||||||
views:
|
|
||||||
generic:
|
|
||||||
files:
|
|
||||||
- web/layouts/layout.html
|
|
||||||
- web/layouts/_partials/mainmenu.html
|
|
||||||
dashboard:
|
|
||||||
files:
|
|
||||||
- web/layouts/dashboard/_partials/agenda-widget.html
|
|
||||||
- web/layouts/dashboard/_partials/beneficiaries-widget.html
|
|
||||||
- web/layouts/dashboard/dashboard.html
|
|
||||||
beneficiaries:
|
|
||||||
list:
|
|
||||||
files:
|
|
||||||
- web/layouts/beneficiaries/list.html
|
|
||||||
create:
|
|
||||||
files:
|
|
||||||
- web/layouts/_partials/address_autocomplete.html
|
|
||||||
- web/layouts/beneficiaries/create.html
|
|
||||||
display:
|
|
||||||
files:
|
|
||||||
- web/layouts/_partials/address_autocomplete.html
|
|
||||||
- web/layouts/vehicles_management/_partials/vehicle-type-select.html
|
|
||||||
- web/layouts/beneficiaries/_partials/beneficiary-vehicles.html
|
|
||||||
- web/layouts/beneficiaries/_partials/beneficiary-notes.html
|
|
||||||
- web/layouts/beneficiaries/_partials/beneficiary-journeys.html
|
|
||||||
- web/layouts/beneficiaries/_partials/beneficiary-events.html
|
|
||||||
- web/layouts/beneficiaries/_partials/beneficiary-files.html
|
|
||||||
- web/layouts/beneficiaries/display.html
|
|
||||||
update:
|
|
||||||
files:
|
|
||||||
- web/layouts/_partials/address_autocomplete.html
|
|
||||||
- web/layouts/beneficiaries/update.html
|
|
||||||
|
|
||||||
vehicles:
|
|
||||||
search:
|
|
||||||
files:
|
|
||||||
- web/layouts/_partials/address_autocomplete.html
|
|
||||||
- web/layouts/vehicles_management/_partials/vehicle-type-select.html
|
|
||||||
- web/layouts/vehicles/search.html
|
|
||||||
booking_display:
|
|
||||||
files:
|
|
||||||
- web/layouts/vehicles/booking-display.html
|
|
||||||
bookings_list:
|
|
||||||
files:
|
|
||||||
- web/layouts/vehicles_management/_partials/bookings-list.html
|
|
||||||
- web/layouts/vehicles/bookings-list.html
|
|
||||||
vehicles_management:
|
|
||||||
overview:
|
|
||||||
files:
|
|
||||||
- web/layouts/vehicles_management/_partials/bookings-list.html
|
|
||||||
- web/layouts/vehicles_management/_partials/vehicles-list.html
|
|
||||||
- web/layouts/vehicles_management/overview.html
|
|
||||||
fleet_add:
|
|
||||||
files:
|
|
||||||
- web/layouts/_partials/address_autocomplete.html
|
|
||||||
- web/layouts/vehicles_management/_partials/vehicle-type-select.html
|
|
||||||
- web/layouts/vehicles_management/fleet-add.html
|
|
||||||
fleet_display:
|
|
||||||
files:
|
|
||||||
- web/layouts/vehicles_management/_partials/calendar.html
|
|
||||||
- web/layouts/vehicles_management/fleet-display.html
|
|
||||||
fleet_update:
|
|
||||||
files:
|
|
||||||
- web/layouts/_partials/address_autocomplete.html
|
|
||||||
- web/layouts/vehicles_management/_partials/vehicle-type-select.html
|
|
||||||
- web/layouts/vehicles_management/fleet-update.html
|
|
||||||
booking_display:
|
|
||||||
files:
|
|
||||||
- web/layouts/vehicles_management/booking-display.html
|
|
||||||
agenda:
|
|
||||||
list:
|
|
||||||
files:
|
|
||||||
- web/layouts/agenda/home.html
|
|
||||||
display_event:
|
|
||||||
files:
|
|
||||||
- web/layouts/agenda/display-event.html
|
|
||||||
create_event:
|
|
||||||
files:
|
|
||||||
- web/layouts/_partials/address_autocomplete.html
|
|
||||||
- web/layouts/agenda/create-event.html
|
|
||||||
directory:
|
|
||||||
home:
|
|
||||||
files:
|
|
||||||
- web/layouts/directory/home.html
|
|
||||||
journeys:
|
|
||||||
search:
|
|
||||||
files:
|
|
||||||
- web/layouts/_partials/address_autocomplete.html
|
|
||||||
- web/layouts/journeys/_partials/journeys-all.html
|
|
||||||
- web/layouts/journeys/_partials/journeys-others.html
|
|
||||||
- web/layouts/journeys/_partials/journeys-carpool.html
|
|
||||||
- web/layouts/journeys/_partials/journeys-public-transit.html
|
|
||||||
- web/layouts/journeys/search.html
|
|
||||||
support:
|
|
||||||
search:
|
|
||||||
files:
|
|
||||||
- web/layouts/support/support.html
|
|
||||||
group_module:
|
|
||||||
home:
|
|
||||||
files:
|
|
||||||
- web/layouts/group_module/home.html
|
|
||||||
create_group:
|
|
||||||
files:
|
|
||||||
- web/layouts/group_module/create_group.html
|
|
||||||
|
|
||||||
display_group:
|
|
||||||
files:
|
|
||||||
- web/layouts/group_module/display_group.html
|
|
||||||
|
|
||||||
administration:
|
|
||||||
home:
|
|
||||||
files:
|
|
||||||
- web/layouts/administration/home.html
|
|
||||||
create_group:
|
|
||||||
files:
|
|
||||||
- web/layouts/administration/create_group.html
|
|
||||||
display_group:
|
|
||||||
files:
|
|
||||||
- web/layouts/administration/_partials/groups_admins.html
|
|
||||||
- web/layouts/administration/_partials/group_members.html
|
|
||||||
- web/layouts/administration/display_group.html
|
|
||||||
group:
|
|
||||||
settings:
|
|
||||||
files:
|
|
||||||
- web/layouts/administration/_partials/groups_admins.html
|
|
||||||
- web/layouts/administration/_partials/group_members.html
|
|
||||||
- web/layouts/group/settings.html
|
|
||||||
auth:
|
|
||||||
groups:
|
|
||||||
files:
|
|
||||||
- web/layouts/auth/groups.html
|
|
||||||
onboarding:
|
|
||||||
files:
|
|
||||||
- web/layouts/auth/onboarding.html
|
|
||||||
|
|
||||||
icons:
|
|
||||||
svg:
|
|
||||||
coopgo:parcoursmob/monogram: <svg xmlns="http://www.w3.org/2000/svg" class="%s" viewBox="0 0 61.85 33.58"><defs><style>.cls-1{fill:#ff1300;}.cls-2{fill:#243887;}</style></defs><g id="Calque_2" data-name="Calque 2"><g id="Calque_1-2" data-name="Calque 1"><path class="cls-1" d="M44.978,0C31.337,0,28.1,6.824,27.875,15.505H39.536V9.434a.727.727,0,0,1,1.123-.607L52.6,16.453,40.659,24.08a.729.729,0,0,1-1.123-.608v-6.1H27.865c.075,8.427,1.527,16.213,17.113,16.213,14.867,0,16.872-7.764,16.872-17.032C61.85,7.91,59.894,0,44.978,0Z"/><polygon class="cls-1" points="41.412 21.385 49.133 16.453 41.412 11.521 41.412 21.385"/><path class="cls-2" d="M14.175,11.4l-.019,4.151H26.311a14.781,14.781,0,0,0,.819-5.141C27.046,3.767,22.545,0,14.764,0H1.052A1.147,1.147,0,0,0,0,1.24V31.87a1.149,1.149,0,0,0,1.094,1.239H11.525a1.145,1.145,0,0,0,1.051-1.239V10.41h.758C13.88,10.41,14.175,10.756,14.175,11.4Z"/><path class="cls-2" d="M14.148,17.3l-.015,3.514H18.97A7.521,7.521,0,0,0,25.458,17.3Z"/></g></g></svg>
|
|
||||||
hero:outline/briefcase: <svg xmlns="http://www.w3.org/2000/svg" class="%s" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M21 13.255A23.931 23.931 0 0112 15c-3.183 0-6.22-.62-9-1.745M16 6V4a2 2 0 00-2-2h-4a2 2 0 00-2 2v2m4 6h.01M5 20h14a2 2 0 002-2V8a2 2 0 00-2-2H5a2 2 0 00-2 2v10a2 2 0 002 2z" /></svg>
|
|
||||||
hero:outline/support: <svg xmlns="http://www.w3.org/2000/svg" class="%s" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-6 h-6"><path stroke-linecap="round" stroke-linejoin="round" d="M7.5 8.25h9m-9 3H12m-9.75 1.51c0 1.6 1.123 2.994 2.707 3.227 1.129.166 2.27.293 3.423.379.35.026.67.21.865.501L12 21l2.755-4.133a1.14 1.14 0 01.865-.501 48.172 48.172 0 003.423-.379c1.584-.233 2.707-1.626 2.707-3.228V6.741c0-1.602-1.123-2.995-2.707-3.228A48.394 48.394 0 0012 3c-2.392 0-4.744.175-7.043.513C3.373 3.746 2.25 5.14 2.25 6.741v6.018z" /></svg>
|
|
||||||
hero:outline/group_module: <svg xmlns="http://www.w3.org/2000/svg" class="%s" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-6 h-6"><path stroke-linecap="round" stroke-linejoin="round" d="M18 18.72a9.094 9.094 0 003.741-.479 3 3 0 00-4.682-2.72m.94 3.198l.001.031c0 .225-.012.447-.037.666A11.944 11.944 0 0112 21c-2.17 0-4.207-.576-5.963-1.584A6.062 6.062 0 016 18.719m12 0a5.971 5.971 0 00-.941-3.197m0 0A5.995 5.995 0 0012 12.75a5.995 5.995 0 00-5.058 2.772m0 0a3 3 0 00-4.681 2.72 8.986 8.986 0 003.74.477m.94-3.197a5.971 5.971 0 00-.94 3.197M15 6.75a3 3 0 11-6 0 3 3 0 016 0zm6 3a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0zm-13.5 0a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0z" /></svg>
|
|
||||||
hero:outline/calendar: <svg xmlns="http://www.w3.org/2000/svg" class="%s" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z" /></svg>
|
|
||||||
hero:outline/chevron-right: <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="%s"><path stroke-linecap="round" stroke-linejoin="round" d="M8.25 4.5l7.5 7.5-7.5 7.5" /></svg>
|
|
||||||
hero:outline/cog: <svg xmlns="http://www.w3.org/2000/svg" class="%s" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.065 2.572c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.572 1.065c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.065-2.572c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z" /><path stroke-linecap="round" stroke-linejoin="round" d="M15 12a3 3 0 11-6 0 3 3 0 016 0z" /></svg>
|
|
||||||
hero:outline/document-text: <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="%s"><path stroke-linecap="round" stroke-linejoin="round" d="M19.5 14.25v-2.625a3.375 3.375 0 00-3.375-3.375h-1.5A1.125 1.125 0 0113.5 7.125v-1.5a3.375 3.375 0 00-3.375-3.375H8.25m0 12.75h7.5m-7.5 3H12M10.5 2.25H5.625c-.621 0-1.125.504-1.125 1.125v17.25c0 .621.504 1.125 1.125 1.125h12.75c.621 0 1.125-.504 1.125-1.125V11.25a9 9 0 00-9-9z" /></svg>
|
|
||||||
hero:outline/home: <svg xmlns="http://www.w3.org/2000/svg" class="%s" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M3 12l2-2m0 0l7-7 7 7M5 10v10a1 1 0 001 1h3m10-11l2 2m-2-2v10a1 1 0 01-1 1h-3m-6 0a1 1 0 001-1v-4a1 1 0 011-1h2a1 1 0 011 1v4a1 1 0 001 1m-6 0h6" /></svg>
|
|
||||||
hero:outline/map: <svg xmlns="http://www.w3.org/2000/svg" class="%s" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M9 20l-5.447-2.724A1 1 0 013 16.382V5.618a1 1 0 011.447-.894L9 7m0 13l6-3m-6 3V7m6 10l4.553 2.276A1 1 0 0021 18.382V7.618a1 1 0 00-.553-.894L15 4m0 13V4m0 0L9 7" /></svg>
|
|
||||||
hero:outline/office-building: <svg xmlns="http://www.w3.org/2000/svg" class="%s" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M19 21V5a2 2 0 00-2-2H7a2 2 0 00-2 2v16m14 0h2m-2 0h-5m-9 0H3m2 0h5M9 7h1m-1 4h1m4-4h1m-1 4h1m-5 10v-5a1 1 0 011-1h2a1 1 0 011 1v5m-4 0h4" /></svg>
|
|
||||||
hero:outline/plus-circle: <svg xmlns="http://www.w3.org/2000/svg" class="%s" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M12 9v3m0 0v3m0-3h3m-3 0H9m12 0a9 9 0 11-18 0 9 9 0 0118 0z" /></svg>
|
|
||||||
hero:outline/shield-check: <svg xmlns="http://www.w3.org/2000/svg" class="%s" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M9 12l2 2 4-4m5.618-4.016A11.955 11.955 0 0112 2.944a11.955 11.955 0 01-8.618 3.04A12.02 12.02 0 003 9c0 5.591 3.824 10.29 9 11.622 5.176-1.332 9-6.03 9-11.622 0-1.042-.133-2.052-.382-3.016z" /></svg>
|
|
||||||
hero:outline/user-group: <svg xmlns="http://www.w3.org/2000/svg" class="%s" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M17 20h5v-2a3 3 0 00-5.356-1.857M17 20H7m10 0v-2c0-.656-.126-1.283-.356-1.857M7 20H2v-2a3 3 0 015.356-1.857M7 20v-2c0-.656.126-1.283.356-1.857m0 0a5.002 5.002 0 019.288 0M15 7a3 3 0 11-6 0 3 3 0 016 0zm6 3a2 2 0 11-4 0 2 2 0 014 0zM7 10a2 2 0 11-4 0 2 2 0 014 0z" /></svg>
|
|
||||||
hero:outline/x: <svg class="%s text-white" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" aria-hidden="true"><path stroke-linecap="round" stroke-linejoin="round" d="M6 18L18 6M6 6l12 12" /></svg>
|
|
||||||
hero:solid/chevron-right: <svg class="%s text-gray-400" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true"><path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd" /></svg>
|
|
||||||
hero:solid/question-mark-icon: <svg class="%s" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true"><path fill-rule="evenodd" d="M18 10a8 8 0 11-16 0 8 8 0 0116 0zm-8-3a1 1 0 00-.867.5 1 1 0 11-1.731-1A3 3 0 0113 8a3.001 3.001 0 01-2 2.83V11a1 1 0 11-2 0v-1a1 1 0 011-1 1 1 0 100-2zm0 8a1 1 0 100-2 1 1 0 000 2z" clip-rule="evenodd" /></svg>
|
|
||||||
hero:solid/search: <svg class="%s" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true"><path fill-rule="evenodd" d="M8 4a4 4 0 100 8 4 4 0 000-8zM2 8a6 6 0 1110.89 3.476l4.817 4.817a1 1 0 01-1.414 1.414l-4.816-4.816A6 6 0 012 8z" clip-rule="evenodd" /></svg>
|
|
||||||
hero:solid/selector: <svg class="%s" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true"><path fill-rule="evenodd" d="M10 3a1 1 0 01.707.293l3 3a1 1 0 01-1.414 1.414L10 5.414 7.707 7.707a1 1 0 01-1.414-1.414l3-3A1 1 0 0110 3zm-3.707 9.293a1 1 0 011.414 0L10 14.586l2.293-2.293a1 1 0 011.414 1.414l-3 3a1 1 0 01-1.414 0l-3-3a1 1 0 010-1.414z" clip-rule="evenodd" /></svg>
|
|
||||||
img:profile-picture-placeholder: <svg class="%s" fill="currentColor" viewBox="0 0 24 24"><path d="M24 20.993V24H0v-2.996A14.977 14.977 0 0112.004 15c4.904 0 9.26 2.354 11.996 5.993zM16.002 8.999a4 4 0 11-8 0 4 4 0 018 0z" /></svg>
|
|
||||||
tabler-icons:car: <svg xmlns="http://www.w3.org/2000/svg" class="icon icon-tabler icon-tabler-car %s" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round"><path stroke="none" d="M0 0h24v24H0z" fill="none"></path><circle cx="7" cy="17" r="2"></circle><circle cx="17" cy="17" r="2"></circle><path d="M5 17h-2v-6l2 -5h9l4 5h1a2 2 0 0 1 2 2v4h-2m-4 0h-6m-6 -6h15m-6 0v-5"></path></svg>
|
|
||||||
tabler-icons:walk: <svg xmlns="http://www.w3.org/2000/svg" class="icon icon-tabler icon-tabler-walk %s" width="44" height="44" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round"><path stroke="none" d="M0 0h24v24H0z" fill="none"/><circle cx="13" cy="4" r="1" /><line x1="7" y1="21" x2="10" y2="17" /><path d="M16 21l-2 -4l-3 -3l1 -6" /><path d="M6 12l2 -3l4 -1l3 3l3 1" /></svg>
|
|
||||||
tabler-icons:bus: <svg xmlns="http://www.w3.org/2000/svg" class="icon icon-tabler icon-tabler-bus %s" width="44" height="44" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round"><path stroke="none" d="M0 0h24v24H0z" fill="none"/><circle cx="6" cy="17" r="2" /><circle cx="18" cy="17" r="2" /><path d="M4 17h-2v-11a1 1 0 0 1 1 -1h14a5 7 0 0 1 5 7v5h-2m-4 0h-8" /><polyline points="16 5 17.5 12 22 12" /><line x1="2" y1="10" x2="17" y2="10" /><line x1="7" y1="5" x2="7" y2="10" /><line x1="12" y1="5" x2="12" y2="10" /></svg>
|
|
||||||
|
|
||||||
emails:
|
|
||||||
onboarding:
|
|
||||||
new_administrator:
|
|
||||||
subject: PARCOURSMOB - Vous avez été invité comme administrateur
|
|
||||||
files:
|
|
||||||
- emails/layout.html
|
|
||||||
- emails/onboarding/new-administrator.html
|
|
||||||
existing_administrator:
|
|
||||||
subject: PARCOURSMOB - Vous avez été invité comme administrateur
|
|
||||||
files:
|
|
||||||
- emails/layout.html
|
|
||||||
- emails/onboarding/existing-administrator.html
|
|
||||||
new_member:
|
|
||||||
subject: PARCOURSMOB - Vous avez été invité à rejoindre une organisation
|
|
||||||
files:
|
|
||||||
- emails/layout.html
|
|
||||||
- emails/onboarding/new-member.html
|
|
||||||
existing_member:
|
|
||||||
subject: PARCOURSMOB - Vous avez été invité à rejoindre une organisation
|
|
||||||
files:
|
|
||||||
- emails/layout.html
|
|
||||||
- emails/onboarding/existing-member.html
|
|
||||||
support_email:
|
|
||||||
subject: PARCOURMOB - Vous avez reçu un commentaire
|
|
||||||
files:
|
|
||||||
- emails/layout.html
|
|
||||||
- emails/onboarding/support_emailing.html
|
|
||||||
|
|
||||||
@@ -1,62 +0,0 @@
|
|||||||
{{define "main"}}
|
|
||||||
<html>
|
|
||||||
<head>
|
|
||||||
<style>
|
|
||||||
@font-face {
|
|
||||||
font-family: "Bitter";
|
|
||||||
font-style: normal;
|
|
||||||
src: url("https://coopgo.fr/fonts/Bitter-Regular.woff") format("woff"); }
|
|
||||||
|
|
||||||
html {
|
|
||||||
font-family: Bitter, serif;
|
|
||||||
}
|
|
||||||
|
|
||||||
.bg-co-blue {
|
|
||||||
--tw-bg-opacity: 1;
|
|
||||||
background-color: rgb(36 56 135 / var(--tw-bg-opacity));
|
|
||||||
}
|
|
||||||
|
|
||||||
.text-center {
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.w-96 {
|
|
||||||
width: 24rem/* 384px */;
|
|
||||||
}
|
|
||||||
|
|
||||||
.p-10 {
|
|
||||||
padding: 2.5rem/* 40px */;
|
|
||||||
}
|
|
||||||
|
|
||||||
.m-10 {
|
|
||||||
margin: 2.5rem/* 40px */;
|
|
||||||
}
|
|
||||||
|
|
||||||
.max-w-3xl {
|
|
||||||
max-width: 48rem/* 768px */;
|
|
||||||
}
|
|
||||||
|
|
||||||
.max-w-4xl {
|
|
||||||
max-width: 56rem/* 896px */;
|
|
||||||
}
|
|
||||||
|
|
||||||
.m-auto {
|
|
||||||
margin: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
.text-white {
|
|
||||||
--tw-text-opacity: 1;
|
|
||||||
color: rgb(255 255 255 / var(--tw-text-opacity));
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<div class="bg-co-blue text-center text-white p-10">
|
|
||||||
<img class="w-96" src="https://coopgo.fr/images/coopgo-logo-bluegreen.svg" alt="PARCOURSMOB" />
|
|
||||||
</div>
|
|
||||||
<div class="max-w-3xl m-auto">
|
|
||||||
{{template "content" .}}
|
|
||||||
</div>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<p>Vous avez été ajouté comme administrateur de l'organisation {{.group}} sur PARCOURSMOB.</p>
|
|
||||||
<p>Connectez vous sur <a href="http://localhost:9000">http://localhost:9000</a> pour y accéder</p>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<p>Vous avez été ajouté à l'organisation {{.group}} sur PARCOURSMOB.</p>
|
|
||||||
<p>Connectez vous sur <a href="http://localhost:9000">http://localhost:9000</a> pour y accéder</p>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<p>Vous avez été ajouté comme administrateur de l'organisation {{.group}} sur PARCOURSMOB.</p>
|
|
||||||
<p>Vous devez créer votre compte pour y accéder.</p>
|
|
||||||
<p>Pour créer votre compte PARCOURSMOB, cliquez sur : <a href="http://localhost:9000/auth/onboarding?key={{.key}}">http://localhost:9000/onboarding?key={{.key}}</a></p>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<p>Vous avez été ajouté à l'organisation {{.group}} sur PARCOURSMOB.</p>
|
|
||||||
<p>Vous devez créer votre compte pour y accéder.</p>
|
|
||||||
<p>Pour créer votre compte PARCOURSMOB, cliquez sur : <a href="http://localhost:9000/auth/onboarding?key={{.key}}">http://localhost:9000/onboarding?key={{.key}}</a></p>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<p>Vous avez reçu un commentaire sur PARCOURSMOB de la part de <b>{{.user}}</b></p>
|
|
||||||
<p>{{.key}}</p>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
@tailwind base;
|
|
||||||
@tailwind components;
|
|
||||||
@tailwind utilities;
|
|
||||||
|
|
||||||
@font-face {
|
|
||||||
font-family: "Manometer";
|
|
||||||
src: url("https://coopgo.fr/fonts/manometer.woff2") format("woff2"), url("/fonts/manometer.woff") format("woff"); }
|
|
||||||
|
|
||||||
@font-face {
|
|
||||||
font-family: "Bitter";
|
|
||||||
font-style: normal;
|
|
||||||
src: url("https://coopgo.fr/fonts/Bitter-Regular.woff") format("woff"); }
|
|
||||||
|
|
||||||
|
|
||||||
@layer base {
|
|
||||||
html {
|
|
||||||
font-family: Bitter, serif;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
import '@kingshott/iodine';
|
|
||||||
import Alpine from 'alpinejs'
|
|
||||||
|
|
||||||
window.Alpine = Alpine
|
|
||||||
|
|
||||||
Alpine.start()
|
|
||||||
Binary file not shown.
@@ -1,48 +0,0 @@
|
|||||||
{{ define "address_autocomplete" }}
|
|
||||||
|
|
||||||
|
|
||||||
<div class="col-span-6 relative" x-data="{
|
|
||||||
input: {{if .Address}}'{{.Address.Properties.label}}'{{else}}null{{end}},
|
|
||||||
address: null,
|
|
||||||
addressObject: {{if .Address}}{{printf "%s" .Address.MarshalJSON}}{{else}}null{{end}},
|
|
||||||
responselength: 0,
|
|
||||||
async autocomplete() {
|
|
||||||
if(this.input == null || this.input == '') {
|
|
||||||
this.responselength = 0
|
|
||||||
return []
|
|
||||||
}
|
|
||||||
if(this.addressObject != null && this.input == this.addressObject.properties.label) {
|
|
||||||
this.responselength = 0
|
|
||||||
return []
|
|
||||||
}
|
|
||||||
result = await fetch('https://geocode.ridygo.fr/v1/autocomplete/\?text=' + this.input)
|
|
||||||
json = await result.json()
|
|
||||||
|
|
||||||
this.responselength = json['features'].length
|
|
||||||
return json['features']
|
|
||||||
},
|
|
||||||
select(a) {
|
|
||||||
this.address = JSON.stringify(a)
|
|
||||||
this.addressObject = a
|
|
||||||
this.input = a.properties.label
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<input type="hidden" name="{{ .FieldName }}" x-model="address">
|
|
||||||
<label for="address" class="block text-sm font-medium text-gray-700">{{ if .FieldLabel }}{{.FieldLabel}}{{else}}Adresse{{end}}</label>
|
|
||||||
<input type="text"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm border-gray-300 rounded-2xl"
|
|
||||||
x-model="input">
|
|
||||||
|
|
||||||
<ul x-show="responselength > 0"
|
|
||||||
class="absolute z-10 mt-1 w-full bg-white shadow-lg max-h-60 rounded-xl py-1 text-base ring-1 ring-black ring-opacity-5 overflow-auto focus:outline-none sm:text-sm" tabindex="-1" role="listbox" aria-labelledby="listbox-label" aria-activedescendant="listbox-option-3">
|
|
||||||
<template x-for="item in autocomplete">
|
|
||||||
<a href="#">
|
|
||||||
<li class="text-gray-900 hover:bg-gray-200 cursor-default select-none relative py-2 pl-3 pr-9"
|
|
||||||
@click="select(item)">
|
|
||||||
<span class="font-normal block truncate" x-text="item.properties.label" ></span>
|
|
||||||
</li>
|
|
||||||
</a>
|
|
||||||
</template>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
{{ end }}
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
{{define "mainmenu"}}
|
|
||||||
|
|
||||||
{{range .LayoutState.MenuItems}}
|
|
||||||
<nav class="px-2 space-y-1">
|
|
||||||
<a href="{{.Link}}"
|
|
||||||
{{ if .Active }}
|
|
||||||
class="bg-white text-co-blue group flex items-center px-2 py-2 text-base font-medium rounded-2xl">
|
|
||||||
{{ else}}
|
|
||||||
class="text-white hover:bg-white hover:bg-opacity-5 group flex items-center px-2 py-2 text-base font-medium rounded-2xl">
|
|
||||||
{{end}}
|
|
||||||
|
|
||||||
{{$.IconSet.Icon .Icon "mr-4 flex-shrink-0 h-6 w-6"}}
|
|
||||||
|
|
||||||
{{.Title}}
|
|
||||||
</a>
|
|
||||||
</nav>
|
|
||||||
{{end}}
|
|
||||||
|
|
||||||
{{end}}
|
|
||||||
@@ -1,47 +0,0 @@
|
|||||||
{{define "groups_members"}}
|
|
||||||
<div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
|
|
||||||
<h2 class="text-md font-medium text-gray-900">Membres de l'organisation</h2>
|
|
||||||
|
|
||||||
<div class="grid grid-cols-1 gap-4 sm:grid-cols-2">
|
|
||||||
|
|
||||||
{{range .ViewState.members}}
|
|
||||||
<div class="relative rounded-lg bg-white px-6 py-5 flex items-center space-x-3 focus-within:ring-2 focus-within:ring-offset-2 focus-within:ring-co-blue">
|
|
||||||
<div class="flex-shrink-0">
|
|
||||||
<img class="h-10 w-10 rounded-co" src="/app/beneficiaries/{{.ID}}/picture" alt="">
|
|
||||||
</div>
|
|
||||||
<div class="flex-1 min-w-0">
|
|
||||||
<a href="#" class="focus:outline-none">
|
|
||||||
<span class="absolute inset-0" aria-hidden="true"></span>
|
|
||||||
<p class="text-sm font-medium text-gray-900">{{.Data.first_name}} {{.Data.last_name}}</p>
|
|
||||||
<p class="text-sm text-gray-500 truncate">{{.Data.email}}</p>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- <div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
|
|
||||||
<h2 class="text-md font-medium text-gray-900">Membres de l'organisation</h2>
|
|
||||||
|
|
||||||
<div class="grid grid-cols-1 gap-4 sm:grid-cols-2">
|
|
||||||
|
|
||||||
{{range .ViewState.members}}
|
|
||||||
<div class="relative rounded-lg bg-white px-6 py-5 flex items-center space-x-3 focus-within:ring-2 focus-within:ring-offset-2 focus-within:ring-co-blue">
|
|
||||||
<div class="flex-shrink-0">
|
|
||||||
<img class="h-10 w-10 rounded-co" src="/app/beneficiaries/{{.ID}}/picture" alt="">
|
|
||||||
</div>
|
|
||||||
<div class="flex-1 min-w-0">
|
|
||||||
<a href="#" class="focus:outline-none">
|
|
||||||
<span class="absolute inset-0" aria-hidden="true"></span>
|
|
||||||
<p class="text-sm font-medium text-gray-900">{{.Data.first_name}} {{.Data.last_name}}</p>
|
|
||||||
<p class="text-sm text-gray-500 truncate">{{.Data.email}}</p>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
</div>
|
|
||||||
</div> -->
|
|
||||||
{{end}}
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
{{define "groups_admins"}}
|
|
||||||
<div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
|
|
||||||
<h2 class="text-md font-medium text-gray-900">Administrateurs</h2>
|
|
||||||
|
|
||||||
<div class="grid grid-cols-1 gap-4 sm:grid-cols-2">
|
|
||||||
|
|
||||||
{{range .ViewState.admins}}
|
|
||||||
<div class="relative rounded-lg bg-white px-6 py-5 flex items-center space-x-3 focus-within:ring-2 focus-within:ring-offset-2 focus-within:ring-co-blue">
|
|
||||||
<div class="flex-shrink-0">
|
|
||||||
<img class="h-10 w-10 rounded-co" src="/app/beneficiaries/{{.ID}}/picture" alt="">
|
|
||||||
</div>
|
|
||||||
<div class="flex-1 min-w-0">
|
|
||||||
<a href="#" class="focus:outline-none">
|
|
||||||
<span class="absolute inset-0" aria-hidden="true"></span>
|
|
||||||
<p class="text-sm font-medium text-gray-900">{{.Data.first_name}} {{.Data.last_name}}</p>
|
|
||||||
<p class="text-sm text-gray-500 truncate">{{.Data.email}}</p>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<!-- <div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
|
|
||||||
<h2 class="text-md font-medium text-gray-900">Membres de l'organisation</h2>
|
|
||||||
|
|
||||||
<div class="grid grid-cols-1 gap-4 sm:grid-cols-2">
|
|
||||||
|
|
||||||
{{range .ViewState.members}}
|
|
||||||
<div class="relative rounded-lg bg-white px-6 py-5 flex items-center space-x-3 focus-within:ring-2 focus-within:ring-offset-2 focus-within:ring-co-blue">
|
|
||||||
<div class="flex-shrink-0">
|
|
||||||
<img class="h-10 w-10 rounded-co" src="/app/beneficiaries/{{.ID}}/picture" alt="">
|
|
||||||
</div>
|
|
||||||
<div class="flex-1 min-w-0">
|
|
||||||
<a href="#" class="focus:outline-none">
|
|
||||||
<span class="absolute inset-0" aria-hidden="true"></span>
|
|
||||||
<p class="text-sm font-medium text-gray-900">{{.Data.first_name}} {{.Data.last_name}}</p>
|
|
||||||
<p class="text-sm text-gray-500 truncate">{{.Data.email}}</p>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
</div>
|
|
||||||
</div> -->
|
|
||||||
{{end}}
|
|
||||||
@@ -1,142 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Administration > Créer une organisation</h1>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8 mt-8" x-data="{
|
|
||||||
fields: {
|
|
||||||
name: null,
|
|
||||||
},
|
|
||||||
rules: {
|
|
||||||
name: ['required'],
|
|
||||||
},
|
|
||||||
formValidation: {
|
|
||||||
valid: false,
|
|
||||||
fields: {
|
|
||||||
name: {valid: null},
|
|
||||||
}
|
|
||||||
},
|
|
||||||
isFormValid: true,
|
|
||||||
validate() {
|
|
||||||
this.formValidation = Iodine.assert(this.fields, this.rules)
|
|
||||||
},
|
|
||||||
validateField(field) {
|
|
||||||
this.formValidation.fields[field] = Iodine.assert(this.fields[field], this.rules[field])
|
|
||||||
},
|
|
||||||
submit(event) {
|
|
||||||
this.validate()
|
|
||||||
if(!this.formValidation.valid) {
|
|
||||||
this.isFormValid = false
|
|
||||||
event.preventDefault()
|
|
||||||
}
|
|
||||||
return this.formValidation.valid
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<form class="space-y-6" method="POST" @submit="submit">
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Nouvelle organisation</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Informations de base sur la nouvelle organisation à créer</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 md:mt-0 md:col-span-2">
|
|
||||||
<div class="grid grid-cols-6 gap-6">
|
|
||||||
<div class="col-span-6">
|
|
||||||
<label for="name" class="block text-sm font-medium text-gray-700">Nom de
|
|
||||||
l'organisation</label>
|
|
||||||
<input type="text" name="name" id="name"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.name" @blur="validateField('name')"
|
|
||||||
:class="formValidation.fields.name.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Paramètres</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Paramètres de configuration de l'organisation (modules
|
|
||||||
accessibles, ...)</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 md:mt-0 md:col-span-2">
|
|
||||||
|
|
||||||
<fieldset>
|
|
||||||
<legend class="sr-only">Droits d'accès aux modules</legend>
|
|
||||||
<div class="text-base font-medium text-gray-900" aria-hidden="true">Droits d'accès aux modules</div>
|
|
||||||
<div class="mt-4 space-y-4">
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="h-5 flex items-center">
|
|
||||||
<input id="beneficiaries" name="modules.beneficiaries" type="checkbox" checked
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.beneficiaries" class="font-medium text-gray-700">Bénéficiaires</label>
|
|
||||||
<p class="text-gray-500">Gestion des bénéficiaires assignés à sa propre organisation.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="journeys" name="modules.journeys" type="checkbox"
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.journeys" class="font-medium text-gray-700">Déplacements</label>
|
|
||||||
<p class="text-gray-500">Trouver des solutions et organiser les déplacements de ses bénéficiaires.</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="vehicles" name="modules.vehicles" type="checkbox"
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.vehicles" class="font-medium text-gray-700">Véhicules</label>
|
|
||||||
<p class="text-gray-500">Trouver et réserver des véhicules pour ses bénéficiaires.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="vehicles_management" name="modules.vehicles_management" type="checkbox"
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.vehicles_management" class="font-medium text-gray-700">Gestion des véhicules</label>
|
|
||||||
<p class="text-gray-500">Gérer les véhicules et réservations (pour les gestionnaires de flottes)
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="events" name="modules.events" type="checkbox"
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.events" class="font-medium text-gray-700">Agenda dispositifs</label>
|
|
||||||
<p class="text-gray-500">Agenda des dispositifs pour les bénéficiaires (sessions permis, événements, ...)
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</fieldset>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex justify-end">
|
|
||||||
<p x-show="! isFormValid" class="px-4 py-2 text-sm text-co-red">Certains champs de sont pas valides.</p>
|
|
||||||
<a href="/app/administration/">
|
|
||||||
<button type="button"
|
|
||||||
class="bg-white py-2 px-4 border border-gray-300 rounded-2xl shadow-sm text-sm font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Annuler</button>
|
|
||||||
</a>
|
|
||||||
<button type="submit"
|
|
||||||
class="ml-3 inline-flex justify-center py-2 px-4 border border-transparent shadow-sm text-sm font-medium rounded-2xl text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Créer l'organisation</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,126 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<main class="py-10">
|
|
||||||
<div class="max-w-3xl mx-auto px-4 sm:px-6 md:flex md:items-center md:justify-between md:space-x-5 lg:max-w-7xl lg:px-8">
|
|
||||||
<div class="flex items-center space-x-5">
|
|
||||||
<!-- <div class="flex-shrink-0">
|
|
||||||
<div class="relative">
|
|
||||||
<img class="h-16 w-16 rounded-co" src="/app/beneficiaries/{{.ViewState.ID}}/picture" alt="">
|
|
||||||
<span class="absolute inset-0 shadow-inner rounded-full" aria-hidden="true"></span>
|
|
||||||
</div>
|
|
||||||
</div> -->
|
|
||||||
<div>
|
|
||||||
<h1 class="text-2xl font-bold text-gray-900">{{.ViewState.group.Data.name}}</h1>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="mt-6 flex flex-col-reverse justify-stretch space-y-4 space-y-reverse sm:flex-row-reverse sm:justify-end sm:space-x-reverse sm:space-y-0 sm:space-x-3 md:mt-0 md:flex-row md:space-x-3">
|
|
||||||
<!-- <button type="button"
|
|
||||||
class="inline-flex items-center justify-center px-4 py-2 border border-gray-300 shadow-sm text-sm font-medium rounded-2xl text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">Supprimer</button>
|
|
||||||
<a href="/app/administration/groups/{{.ViewState.group.ID}}/update" class="inline-flex"><button type="button"
|
|
||||||
class="w-full px-4 py-2 border border-transparent text-sm font-medium rounded-2xl shadow-sm text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">Modifier</button></a> -->
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="mt-8 max-w-3xl mx-auto grid grid-cols-1 gap-6 sm:px-6 lg:max-w-7xl lg:grid-flow-col-dense lg:grid-cols-3">
|
|
||||||
<div class="space-y-6 lg:col-start-1 lg:col-span-2">
|
|
||||||
<section aria-labelledby="beneficiary-information-title">
|
|
||||||
<div class="bg-white shadow sm:rounded-lg">
|
|
||||||
<div class="px-4 py-5 sm:px-6">
|
|
||||||
<h2 id="beneficiary-information-title" class="text-lg leading-6 font-medium text-gray-900">
|
|
||||||
Paramètres de l'organisation</h2>
|
|
||||||
<p class="mt-1 max-w-2xl text-sm text-gray-500">Paramètres généraux de l'organisation</p>
|
|
||||||
</div>
|
|
||||||
{{template "groups_admins" .}}
|
|
||||||
<div class="px-2 py-4">
|
|
||||||
<form class="flex" method="POST" action="/app/administration/groups/{{.ViewState.group.ID}}/invite-admin">
|
|
||||||
<div class="pr-2 flex-1">
|
|
||||||
<input type="text" name="username" id="username"
|
|
||||||
class="mt-1 border-gray-300 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
placeholder="Email">
|
|
||||||
</div>
|
|
||||||
<button class="px-1 py-1 border border-transparent text-sm font-medium rounded-2xl shadow-sm text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">Ajouter administrateur</button>
|
|
||||||
|
|
||||||
</form>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
<section aria-labelledby="modules-title" class="lg:col-start-3 lg:col-span-1">
|
|
||||||
<div class="bg-white px-4 py-5 shadow sm:rounded-lg sm:px-6">
|
|
||||||
<h2 id="modules-title" class="text-lg font-medium text-gray-900">Modules activés</h2>
|
|
||||||
|
|
||||||
<fieldset>
|
|
||||||
<div class="mt-4 space-y-4">
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="h-5 flex items-center">
|
|
||||||
<input id="beneficiaries" name="modules.beneficiaries" type="checkbox" disabled {{if .ViewState.group.Data.modules.beneficiaries}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.beneficiaries" class="font-medium text-gray-700">Bénéficiaires</label>
|
|
||||||
<p class="text-gray-500">Gestion des bénéficiaires assignés à sa propre organisation.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="journeys" name="modules.journeys" type="checkbox" disabled {{if .ViewState.group.Data.modules.journeys}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.journeys" class="font-medium text-gray-700">Déplacements</label>
|
|
||||||
<p class="text-gray-500">Trouver des solutions et organiser les déplacements de ses bénéficiaires.</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="vehicles" name="modules.vehicles" type="checkbox" disabled {{if .ViewState.group.Data.modules.vehicles}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.vehicles" class="font-medium text-gray-700">Véhicules</label>
|
|
||||||
<p class="text-gray-500">Trouver et réserver des véhicules pour ses bénéficiaires.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="vehicles_management" name="modules.vehicles_management" type="checkbox" disabled {{if .ViewState.group.Data.modules.vehicles_management}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.vehicles_management" class="font-medium text-gray-700">Gestion des véhicules</label>
|
|
||||||
<p class="text-gray-500">Gérer les véhicules et réservations (pour les gestionnaires de flottes)
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="events" name="modules.events" type="checkbox" disabled {{if .ViewState.group.Data.modules.events}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.events" class="font-medium text-gray-700">Agenda dispositifs</label>
|
|
||||||
<p class="text-gray-500">Agenda des dispositifs pour les bénéficiaires (sessions permis, événements, ...)
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="events" name="modules.events" type="checkbox" disabled {{if .ViewState.group.Data.modules.administration}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.events" class="font-medium text-gray-700">Administration</label>
|
|
||||||
<p class="text-gray-500">Administration générale de la plateforme PARCOURSMOB. Créer, ajouter des organisations et administrateurs.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</fieldset>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
</main>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,124 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Administration</h1>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8 mt-8">
|
|
||||||
<h2 class="text-gray-500 text-xs font-medium uppercase tracking-wide">Statistiques</h2>
|
|
||||||
<ul role="list" class="mt-3 grid grid-cols-1 gap-5 sm:gap-6 sm:grid-cols-2 lg:grid-cols-4">
|
|
||||||
<li class="col-span-1 flex shadow-sm rounded-3xl">
|
|
||||||
<div
|
|
||||||
class="flex-shrink-0 flex items-center justify-center w-16 bg-co-blue text-white text-sm font-medium rounded-l-3xl">
|
|
||||||
{{.IconSet.Icon "hero:outline/user-group" "h-6 w-6"}}
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="flex-1 flex items-center justify-between border-t border-r border-b border-gray-200 bg-white rounded-r-3xl truncate">
|
|
||||||
<div class="flex-1 px-4 py-2 text-sm truncate">
|
|
||||||
<a href="#" class="text-gray-900 font-medium hover:text-gray-600">Bénéficiaires</a>
|
|
||||||
<p class="text-gray-500">{{len (index .ViewState.groups 0).Members}} bénéficiaires</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
<li class="col-span-1 flex shadow-sm rounded-3xl">
|
|
||||||
<div
|
|
||||||
class="flex-shrink-0 flex items-center justify-center w-16 bg-co-green text-white text-sm font-medium rounded-l-3xl">
|
|
||||||
{{.IconSet.Icon "hero:outline/office-building" "h-6 w-6"}}
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="flex-1 flex items-center justify-between border-t border-r border-b border-gray-200 bg-white rounded-r-3xl truncate">
|
|
||||||
<div class="flex-1 px-4 py-2 text-sm truncate">
|
|
||||||
<a href="#" class="text-gray-900 font-medium hover:text-gray-600">Organisations</a>
|
|
||||||
<p class="text-gray-500">{{len .ViewState.groups}} organisations</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
<li class="col-span-1 flex shadow-sm rounded-3xl">
|
|
||||||
<div
|
|
||||||
class="flex-shrink-0 flex items-center justify-center w-16 bg-co-red text-white text-sm font-medium rounded-l-3xl">
|
|
||||||
{{.IconSet.Icon "hero:outline/briefcase" "h-6 w-6"}}
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="flex-1 flex items-center justify-between border-t border-r border-b border-gray-200 bg-white rounded-r-3xl truncate">
|
|
||||||
<div class="flex-1 px-4 py-2 text-sm truncate">
|
|
||||||
<a href="#" class="text-gray-900 font-medium hover:text-gray-600">Référents</a>
|
|
||||||
<p class="text-gray-500">1 membres</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
<li class="col-span-1 flex shadow-sm rounded-3xl">
|
|
||||||
<div
|
|
||||||
class="flex-shrink-0 flex items-center justify-center w-16 bg-co-yellow text-white text-sm font-medium rounded-l-3xl">
|
|
||||||
{{.IconSet.Icon "hero:outline/shield-check" "h-6 w-6"}}
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="flex-1 flex items-center justify-between border-t border-r border-b border-gray-200 bg-white rounded-r-3xl truncate">
|
|
||||||
<div class="flex-1 px-4 py-2 text-sm truncate">
|
|
||||||
<a href="#" class="text-gray-900 font-medium hover:text-gray-600">Accompagnement</a>
|
|
||||||
<p class="text-gray-500">0 actions réalisées</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mt-10 mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h2 class="text-xl font-semibold text-gray-500">Gestion des organisations</h2>
|
|
||||||
|
|
||||||
<div class="sm:flex sm:items-center">
|
|
||||||
<div class="sm:flex-auto">
|
|
||||||
<p class="mt-2 text-sm text-gray-700"></p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-4 sm:mt-0 sm:ml-16 sm:flex-none">
|
|
||||||
<a href="/app/administration/groups/">
|
|
||||||
<button type="button"
|
|
||||||
class="inline-flex items-center justify-center rounded-2xl border border-transparent bg-co-blue px-4 py-2 text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
{{$.IconSet.Icon "hero:outline/plus-circle" "h-5 w-5 mr-3"}}
|
|
||||||
Ajouter une organisation
|
|
||||||
</button>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="bg-white shadow overflow-hidden sm:rounded-3xl mt-4">
|
|
||||||
<ul role="list" class="divide-y divide-gray-200">
|
|
||||||
{{range .ViewState.groups}}
|
|
||||||
<li>
|
|
||||||
<a href="/app/administration/groups/{{.ID}}" class="block hover:bg-gray-50">
|
|
||||||
<div class="px-4 py-4 flex items-center sm:px-6">
|
|
||||||
<div class="min-w-0 flex-1 sm:flex sm:items-center sm:justify-between">
|
|
||||||
<div class="truncate">
|
|
||||||
<div class="flex text-sm">
|
|
||||||
<p class="font-medium text-lg text-co-blue truncate">{{.Data.name}}</p>
|
|
||||||
<p class="ml-1 flex-shrink-0 font-normal text-gray-500"></p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-2 flex">
|
|
||||||
<div class="flex items-center text-sm text-gray-500">
|
|
||||||
{{$.IconSet.Icon "hero:outline/user-group" "flex-shrink-0 mr-1.5 h-5 w-5"}}
|
|
||||||
<p>
|
|
||||||
{{ len .Members }} bénéficiaires
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="mt-4 flex-shrink-0 sm:mt-0 sm:ml-5">
|
|
||||||
<div class="flex overflow-hidden -space-x-1">
|
|
||||||
<!-- <img class="inline-block h-6 w-6 rounded-full ring-2 ring-white"
|
|
||||||
src="http://localhost:9000/app/beneficiaries/e7616eac-4a87-4396-a505-23e0421b9c4c/picture"
|
|
||||||
alt="Dries Vincent"> -->
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="ml-5 flex-shrink-0">
|
|
||||||
{{$.IconSet.Icon "hero:solid/chevron-right" "h-5 w-5 text-gray-400"}}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
</li>
|
|
||||||
{{end}}
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,217 +0,0 @@
|
|||||||
{{ define "content" }}
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Ajouter à l'agenda</h1>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8 mt-8"
|
|
||||||
x-data="{
|
|
||||||
fields: {
|
|
||||||
name: null,
|
|
||||||
type: null,
|
|
||||||
description: null,
|
|
||||||
allday: false,
|
|
||||||
startdate: null,
|
|
||||||
enddate: null,
|
|
||||||
starttime: null,
|
|
||||||
endtime: null,
|
|
||||||
max_subscribers: 0,
|
|
||||||
},
|
|
||||||
rules: {
|
|
||||||
name: ['required'],
|
|
||||||
type: ['required'],
|
|
||||||
startdate: ['required'],
|
|
||||||
enddate: ['required'],
|
|
||||||
starttime: ['optional'],
|
|
||||||
endtime: ['optional'],
|
|
||||||
description: ['optional'],
|
|
||||||
max_subscribers: ['required', 'min:0']
|
|
||||||
},
|
|
||||||
formValidation: {
|
|
||||||
valid: false,
|
|
||||||
fields: {
|
|
||||||
name: {valid: null},
|
|
||||||
type: {valid: null},
|
|
||||||
description: {valid: null},
|
|
||||||
startdate: {valid: null},
|
|
||||||
enddate: {valid: null},
|
|
||||||
starttime: {valid: null},
|
|
||||||
endtime: {valid: null},
|
|
||||||
allday: {valid: null},
|
|
||||||
max_subscribers: {valid: null},
|
|
||||||
}
|
|
||||||
},
|
|
||||||
isFormValid: true,
|
|
||||||
validate() {
|
|
||||||
this.formValidation = Iodine.assert(this.fields, this.rules)
|
|
||||||
|
|
||||||
if(!Iodine.assertAfterOrEqual(new Date(this.fields.enddate), new Date(this.fields.startdate))) {
|
|
||||||
this.formValidation.fields.enddate.valid = false
|
|
||||||
this.formValidation.valid = false
|
|
||||||
}
|
|
||||||
},
|
|
||||||
validateField(field) {
|
|
||||||
this.formValidation.fields[field] = Iodine.assert(this.fields[field], this.rules[field])
|
|
||||||
},
|
|
||||||
submit(event) {
|
|
||||||
this.validate()
|
|
||||||
if(!this.formValidation.valid) {
|
|
||||||
this.isFormValid = false
|
|
||||||
event.preventDefault()
|
|
||||||
}
|
|
||||||
return this.formValidation.valid
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<form class="space-y-6" method="POST" @submit="submit">
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Informations sur le dispositif</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Informations générales sur le dispositif d'accompagnement à ajouter à l'agenda</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 md:mt-0 md:col-span-2">
|
|
||||||
<div class="grid grid-cols-6 gap-6">
|
|
||||||
<div class="col-span-6 sm:col-span-3">
|
|
||||||
<label for="name" class="block text-sm font-medium text-gray-700">Nom</label>
|
|
||||||
<input type="text" name="name" id="name"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.name" @blur="validateField('name')"
|
|
||||||
:class="formValidation.fields.name.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
<div class="sm:col-span-3">
|
|
||||||
<label for="type" class="block text-sm font-medium text-gray-700">Type de dispositif</label>
|
|
||||||
<select id="type" name="type"
|
|
||||||
x-model="fields.type" @blur="validateField('type')"
|
|
||||||
class="max-w-lg mt-1 block focus:ring-co-blue focus:border-co-blue w-full shadow-sm sm:max-w-xs sm:text-sm rounded-2xl"
|
|
||||||
:class="formValidation.fields.type.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
<option></option>
|
|
||||||
<option value="Permis accéléré">Permis accéléré</option>
|
|
||||||
<option value="Auto-école sociale (classique)">Auto-école sociale (classique)</option>
|
|
||||||
<option value="Information collective">Information collective</option>
|
|
||||||
<option value="Autres">Autres</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
<div class="col-span-6">
|
|
||||||
<label for="description" class="block text-sm font-medium text-gray-700">Description</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<textarea rows="4" name="description" id="descrpition"
|
|
||||||
x-model="fields.description" @blur="validateField('description')"
|
|
||||||
:class="formValidation.fields.description.valid == false ? 'border-co-red border-2' : 'border-gray-300'"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block w-full sm:text-sm border-gray-300 rounded-2xl"></textarea>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{{ $fieldName := "address" }}
|
|
||||||
{{ template "address_autocomplete" dict "FieldName" $fieldName }}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Planification</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Dates et horaires de l'événement</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 space-y-6 md:mt-0 md:col-span-2 align-middle">
|
|
||||||
<div class="grid grid-cols-6 gap-6">
|
|
||||||
|
|
||||||
<div class="col-span-6">
|
|
||||||
|
|
||||||
<button type="button" class="relative inline-flex h-6 w-11 flex-shrink-0 cursor-pointer rounded-full border-2 border-transparent transition-colors duration-200 ease-in-out focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2" role="switch" aria-checked="false"
|
|
||||||
:class="fields.allday ? 'bg-co-blue' : 'bg-gray-200'"
|
|
||||||
@click="fields.allday = ! fields.allday">
|
|
||||||
<span class="sr-only">Use setting</span>
|
|
||||||
<!-- Enabled: "translate-x-5", Not Enabled: "translate-x-0" -->
|
|
||||||
<span aria-hidden="true" class="pointer-events-none inline-block h-5 w-5 transform rounded-full bg-white shadow ring-0 transition duration-200 ease-in-out"
|
|
||||||
:class="fields.allday ? 'translate-x-5' : 'translate-x-0'"></span>
|
|
||||||
</button> <span class="text-md font-medium text-gray-700 ml-2">Toute la journée</span>
|
|
||||||
<input type="hidden" name="allday" x-model="fields.allday">
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="sm:col-span-6">
|
|
||||||
<div class="inline-flex w-full">
|
|
||||||
<div class="flex-1">
|
|
||||||
<label for="startdate" class="block text-sm font-medium text-gray-700">Date de début</label>
|
|
||||||
<input type="date" name="startdate" id="startdate" placeholder="JJ/MM/AAAA"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-l-2xl"
|
|
||||||
x-model="fields.startdate" @blur="validateField('startdate')"
|
|
||||||
:class="formValidation.fields.startdate.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
<div class="flex-1">
|
|
||||||
<label for="enddate" class="block text-sm font-medium text-gray-700">Date de fin</label>
|
|
||||||
<input type="date" name="enddate" id="enddate" placeholder="JJ/MM/AAAA"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-r-2xl"
|
|
||||||
x-model="fields.enddate" @blur="validateField('enddate')"
|
|
||||||
:class="formValidation.fields.enddate.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="sm:col-span-6" x-show="!fields.allday">
|
|
||||||
<div class="inline-flex w-full">
|
|
||||||
<div class="flex-1">
|
|
||||||
<label for="startdate" class="block text-sm font-medium text-gray-700">Horaire de début</label>
|
|
||||||
<input type="time" name="starttime" id="starttime" placeholder="00:00"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-l-2xl"
|
|
||||||
x-model="fields.starttime" @blur="validateField('starttime')"
|
|
||||||
:class="formValidation.fields.starttime.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
<div class="flex-1">
|
|
||||||
<label for="endtime" class="block text-sm font-medium text-gray-700">Horaire de fin</label>
|
|
||||||
<input type="time" name="endtime" id="endtime" placeholder="00:00"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-r-2xl"
|
|
||||||
x-model="fields.endtime" @blur="validateField('endtime')"
|
|
||||||
:class="formValidation.fields.endtime.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Paramètres</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Paramètres du dispositift (nombre de places disponibles, etc...)</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 space-y-6 md:mt-0 md:col-span-2 align-middle">
|
|
||||||
<div class="grid grid-cols-6 gap-6">
|
|
||||||
<div class="col-span-6 sm:col-span-3">
|
|
||||||
<label for="max_subscribers" class="block text-sm font-medium text-gray-700">Places disponibles (0 = illimité)</label>
|
|
||||||
<input type="number" name="max_subscribers" id="max_subscribers"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.max_subscribers" @blur="validateField('max_subscribers')"
|
|
||||||
:class="formValidation.fields.max_subscribers.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<div class="flex justify-end">
|
|
||||||
<p x-show="! isFormValid" class="px-4 py-2 text-sm text-co-red">Certains champs de sont pas valides.</p>
|
|
||||||
<a href="/app/agenda/">
|
|
||||||
<button type="button"
|
|
||||||
class="bg-white py-2 px-4 border border-gray-300 rounded-2xl shadow-sm text-sm font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Annuler</button>
|
|
||||||
</a>
|
|
||||||
<button type="submit"
|
|
||||||
class="ml-3 inline-flex justify-center py-2 px-4 border border-transparent shadow-sm text-sm font-medium rounded-2xl text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Ajouter</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{{end}}
|
|
||||||
@@ -1,140 +0,0 @@
|
|||||||
{{ define "content" }}
|
|
||||||
<main class="py-10">
|
|
||||||
<div class="max-w-3xl mx-auto px-4 sm:px-6 md:flex md:items-center md:justify-between md:space-x-5 lg:max-w-7xl lg:px-8">
|
|
||||||
<div class="flex items-center space-x-5">
|
|
||||||
<div>
|
|
||||||
<h1 class="text-2xl font-bold text-gray-900">{{.ViewState.event.Name}}</h1>
|
|
||||||
<p class="text-m font-medium text-gray-500">
|
|
||||||
{{if eq .ViewState.event.Startdate .ViewState.event.Enddate}}
|
|
||||||
Le {{(timeFrom .ViewState.event.Startdate).Format "02/01/2006"}}
|
|
||||||
{{else}}
|
|
||||||
Du {{(timeFrom .ViewState.event.Startdate).Format "02/01/2006"}} au {{(timeFrom .ViewState.event.Enddate).Format "02/01/2006"}}
|
|
||||||
{{end}}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="mt-8 max-w-3xl mx-auto grid grid-cols-1 gap-6 sm:px-6 lg:max-w-7xl lg:grid-flow-col-dense lg:grid-cols-3">
|
|
||||||
<div class="space-y-6 lg:col-start-1 lg:col-span-2">
|
|
||||||
<section aria-labelledby="event-information-title">
|
|
||||||
<div class="bg-white shadow sm:rounded-lg">
|
|
||||||
<div class="px-4 py-5 sm:px-6">
|
|
||||||
<h2 id="event-information-title" class="text-lg leading-6 font-medium text-gray-900">Informations</h2>
|
|
||||||
<p class="mt-1 max-w-2xl text-sm text-gray-500">Informations sur le dispositif.</p>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
<dl class="grid grid-cols-1 gap-x-4 gap-y-8 sm:grid-cols-2">
|
|
||||||
{{if .ViewState.event.Type}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Type</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.event.Type}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.group}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Structure gestionnaire</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.group.Data.name}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.event.MaxSubscribers}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Total places</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.event.MaxSubscribers}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if ne .ViewState.event.MaxSubscribers 0}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Places restantes</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.event.RemainingSubscriptions}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.event.Data.address}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Adresse</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.event.Data.address.properties.label}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.event.Description}}
|
|
||||||
<div class="sm:col-span-2">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Description</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.event.Description}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
<section aria-labelledby="subscribers-title" class="lg:col-start-3 lg:col-span-1">
|
|
||||||
<div class="bg-white px-4 py-5 shadow sm:rounded-lg sm:px-6">
|
|
||||||
<h2 id="subscribers-title" class="text-lg font-medium text-gray-900">Inscrire un bénéficiaire</h2>
|
|
||||||
{{if gt .ViewState.event.RemainingSubscriptions 0}}
|
|
||||||
<form class="mt-4" action="/app/agenda/{{.ViewState.event.ID}}/subscribe" method="POST">
|
|
||||||
<div class="relative mt-1 mb-4" x-data="{
|
|
||||||
text: '',
|
|
||||||
beneficiariesListOpen: false,
|
|
||||||
beneficiaries: {{json .ViewState.beneficiaries}},
|
|
||||||
filteredBeneficiaries: (text) => {
|
|
||||||
if(text=='') return beneficiaries
|
|
||||||
return this.beneficiaries.filter(b => b['data']['first_name'].includes(text) || b['data']['last_name'].includes(text))
|
|
||||||
},
|
|
||||||
fields: {
|
|
||||||
beneficiaryid: {{if .ViewState.search}}'{{.ViewState.search.beneficiary.ID}}'{{else}}null{{end}},
|
|
||||||
},
|
|
||||||
selectbeneficiary(beneficiary) {
|
|
||||||
console.log(beneficiary)
|
|
||||||
this.fields.beneficiaryid = beneficiary.id
|
|
||||||
this.text = beneficiary.data.first_name + ' ' + beneficiary.data.last_name
|
|
||||||
this.beneficiariesListOpen = false
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<input @focus="beneficiariesListOpen = true" x-model="text" id="combobox" type="text" class="w-full rounded-2xl border border-gray-300 bg-white py-2 pl-3 pr-12 shadow-sm focus:border-co-blue focus:outline-none focus:ring-1 focus:ring-co-blue sm:text-sm" role="combobox" aria-controls="options" aria-expanded="false">
|
|
||||||
|
|
||||||
<button @click="beneficiariesListOpen = ! beneficiariesListOpen" type="button" class="absolute inset-y-0 right-0 flex items-center rounded-r-2xl px-2 focus:outline-none">
|
|
||||||
<!-- Heroicon name: solid/selector -->
|
|
||||||
<svg class="h-5 w-5 text-gray-400" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd" d="M10 3a1 1 0 01.707.293l3 3a1 1 0 01-1.414 1.414L10 5.414 7.707 7.707a1 1 0 01-1.414-1.414l3-3A1 1 0 0110 3zm-3.707 9.293a1 1 0 011.414 0L10 14.586l2.293-2.293a1 1 0 011.414 1.414l-3 3a1 1 0 01-1.414 0l-3-3a1 1 0 010-1.414z" clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<ul x-show="beneficiariesListOpen" class="absolute z-10 mt-1 max-h-60 w-full overflow-auto rounded-xl bg-white py-1 text-base shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none sm:text-sm" id="options" role="listbox">
|
|
||||||
|
|
||||||
<template x-for="beneficiary in beneficiaries">
|
|
||||||
<li @click="selectbeneficiary(beneficiary)" class="relative cursor-default hover:bg-gray-100 select-none py-2 pl-3 pr-9 text-gray-900" id="option-0" role="option" tabindex="-1">
|
|
||||||
|
|
||||||
<span class="truncate" x-text="beneficiary.data.first_name"></span> <span class="truncate" x-text="beneficiary.data.last_name"></span>
|
|
||||||
|
|
||||||
<span class="absolute inset-y-0 right-0 flex items-center pr-4 text-co-blue">
|
|
||||||
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
</template>
|
|
||||||
</ul>
|
|
||||||
<input type="hidden" name="subscriber" x-model="fields.beneficiaryid">
|
|
||||||
</div>
|
|
||||||
<button type="submit"
|
|
||||||
class="rounded-2xl border border-transparent bg-co-blue px-4 py-2 w-full text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
Inscrire le bénéficiaire
|
|
||||||
</button>
|
|
||||||
</form>
|
|
||||||
{{else}}
|
|
||||||
<p class="p-12 text-gray-500 text-center text-md">Il n'y a plus de place disponible</p>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.subscribers}}
|
|
||||||
<h2 id="subscribers-title" class="text-lg font-medium text-gray-900 mt-10">Inscrits</h2>
|
|
||||||
<div class="mt-2">
|
|
||||||
{{range .ViewState.subscribers}}
|
|
||||||
<ul class="p-1">
|
|
||||||
<a href="/app/beneficiaries/{{.ID}}">
|
|
||||||
<li class="inline-flex text-sm p-2"><img class="h-6 w-6 rounded-co mr-2" src="/app/beneficiaries/{{.ID}}/picture"> {{.Data.first_name}} {{.Data.last_name}}</li>
|
|
||||||
</a>
|
|
||||||
</ul>
|
|
||||||
{{end}}
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
</main>
|
|
||||||
{{ end }}
|
|
||||||
@@ -1,114 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Agenda dispositifs</h1>
|
|
||||||
|
|
||||||
<div class="sm:flex sm:items-center">
|
|
||||||
<div class="sm:flex-auto">
|
|
||||||
<p class="mt-2 text-sm text-gray-700"></p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-4 sm:mt-0 sm:ml-16 sm:flex-none">
|
|
||||||
<a href="/app/agenda/create-event">
|
|
||||||
<button type="button"
|
|
||||||
class="inline-flex items-center justify-center rounded-2xl border border-transparent bg-co-blue px-4 py-2 text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-ci-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
{{$.IconSet.Icon "hero:outline/plus-circle" "h-5 w-5 mr-3"}}
|
|
||||||
Ajouter un dispositif
|
|
||||||
</button>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<div class="mt-8 flex flex-col">
|
|
||||||
<div class="-my-2 -mx-4 overflow-x-auto sm:-mx-6 lg:-mx-8">
|
|
||||||
<div class="inline-block min-w-full py-2 align-middle md:px-6 lg:px-8">
|
|
||||||
<div class="overflow-hidden shadow ring-1 ring-black ring-opacity-5 md:rounded-lg">
|
|
||||||
<table class="min-w-full divide-y divide-gray-300">
|
|
||||||
<thead class="bg-gray-50">
|
|
||||||
<tr>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Type de dispositif
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Structure
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Nom
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Lieu
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Dates
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Places disponibles
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Bénéficiaires positionnés
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="relative py-3.5 pl-3 pr-4 sm:pr-6">
|
|
||||||
<span class="sr-only">Actions</span>
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
<tbody class="divide-y divide-gray-200 bg-white">
|
|
||||||
{{range .ViewState.events}}
|
|
||||||
<a href="/app/agenda/{{.ID}}">
|
|
||||||
<tr>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >{{.Type}}</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
|
|
||||||
<div class="text-gray-900" >
|
|
||||||
{{range .Owners}}
|
|
||||||
{{if (index $.ViewState.groups .)}}
|
|
||||||
{{(index $.ViewState.groups .).Data.name}}
|
|
||||||
{{end}}
|
|
||||||
{{end}}
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >{{.Name}}</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >{{if .Data.address}}{{.Data.address.properties.label}}{{end}}</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >
|
|
||||||
{{if eq .Startdate .Enddate}}
|
|
||||||
Le {{(timeFrom .Startdate).Format "02/01/2006"}}
|
|
||||||
{{else}}
|
|
||||||
Du {{(timeFrom .Startdate).Format "02/01/2006"}} <br />Au {{(timeFrom .Enddate).Format "02/01/2006"}}
|
|
||||||
{{end}}
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
{{if ne .RemainingSubscriptions 999}}
|
|
||||||
<div class="text-gray-900" >{{.RemainingSubscriptions}}</div>
|
|
||||||
{{end}}
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="flex -space-x-1 overflow-hidden" >
|
|
||||||
{{range .Subscriptions}}
|
|
||||||
<img class="inline-block h-6 w-6 rounded-co ring-2 ring-white" src="/app/beneficiaries/{{.Subscriber}}/picture" >
|
|
||||||
{{end}}
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
|
|
||||||
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<a href="/app/agenda/{{.ID}}" class="text-co-blue hover:text-co-blue">Voir</a>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</a>
|
|
||||||
{{end}}
|
|
||||||
</tbody>
|
|
||||||
</thead>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,40 +0,0 @@
|
|||||||
{{define "main"}}
|
|
||||||
<html class="h-full bg-gray-50">
|
|
||||||
<head>
|
|
||||||
<title>PARCOURSMOB</title>
|
|
||||||
<link rel="stylesheet" href="/public/css/main.css" />
|
|
||||||
<!-- <script defer type="text/javascript" src="/public/js/main.js" defer></script> -->
|
|
||||||
<script src="https://cdn.jsdelivr.net/npm/@kingshott/iodine@8.1.0/dist/iodine.min.umd.js" defer></script>
|
|
||||||
<script defer src="https://unpkg.com/alpinejs@3.x.x/dist/cdn.min.js" defer></script>
|
|
||||||
</head>
|
|
||||||
<body class="h-full">
|
|
||||||
<div class="min-h-full flex flex-col justify-center py-12 sm:px-6 lg:px-8 h-">
|
|
||||||
<div class="sm:mx-auto sm:w-full sm:max-w-md">
|
|
||||||
{{.IconSet.Icon "coopgo:parcoursmob/monogram" "mx-auto h-16 w-auto"}}
|
|
||||||
<h2 class="mt-6 text-center text-3xl font-extrabold text-gray-900">Connectez vous à votre organisation</h2>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="mt-8 sm:mx-auto sm:w-full sm:max-w-md">
|
|
||||||
<div class="bg-white shadow sm:rounded-3xl">
|
|
||||||
<form id="groupform" class="space-y-6" action="" method="POST" x-data="{group: null}" x-init="$watch('group', value => document.getElementById('groupform').submit())">
|
|
||||||
<fieldset>
|
|
||||||
<legend class="sr-only">Organisations</legend>
|
|
||||||
{{range .ViewState.groups}}
|
|
||||||
<div class="relative bg-white rounded-md -space-y-px sm:rounded-3xl">
|
|
||||||
<label class="sm:rounded-3xl relative text-co-blue hover:text-white hover:bg-co-blue p-4 flex flex-row cursor-pointer md:pl-4 md:pr-6 focus:outline-none">
|
|
||||||
<span class="flex-1 items-center text-xl">
|
|
||||||
<input x-model="group" type="radio" name="group" value="{{.ID}}" class="h-4 w-4 sr-only" aria-labelledby="group-{{.ID}}-label" aria-describedby="group-{{.ID}}-description-0">
|
|
||||||
<span id="group-{{.ID}}-label" class="ml-3 font-medium">{{.Data.name}}</span>
|
|
||||||
</span>
|
|
||||||
<span id="group-{{.ID}}-description-0" class="ml-6 pl-1 text-sm md:ml-0 md:pl-0 md:text-right">{{$.IconSet.Icon "hero:solid/chevron-right" "w-6 h-6"}}</span>
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
</fieldset>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,54 +0,0 @@
|
|||||||
{{define "main"}}
|
|
||||||
<html class="h-full bg-gray-50">
|
|
||||||
<head>
|
|
||||||
<title>PARCOURSMOB - Identification</title>
|
|
||||||
<link rel="stylesheet" href="http://localhost:9000/public/css/main.css" />
|
|
||||||
</head>
|
|
||||||
<body class="h-full">
|
|
||||||
<form method="post">
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<div class="flex min-h-full flex-col justify-center py-12 sm:px-6 lg:px-8">
|
|
||||||
<div class="sm:mx-auto sm:w-full sm:max-w-md">
|
|
||||||
<!-- <img class="mx-auto h-12 w-auto" src="https://tailwindui.com/img/logos/mark.svg?color=indigo&shade=600" alt="Your Company"> -->
|
|
||||||
<h2 class="mt-6 text-center text-3xl font-bold tracking-tight text-gray-900">Créez votre compte PARCOURSMOB</h2>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="mt-8 sm:mx-auto sm:w-full sm:max-w-md">
|
|
||||||
<div class="bg-white py-8 px-4 shadow sm:rounded-lg sm:px-10">
|
|
||||||
<div>
|
|
||||||
<label for="last_name" class="block text-sm font-medium text-gray-700">Nom</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<input id="last_name" name="last_name" type="text" autocomplete="last_name" required class="block w-full appearance-none rounded-2xl border border-gray-300 px-3 py-2 placeholder-gray-400 shadow-sm focus:border-co-blue focus:outline-none focus:ring-co-blue sm:text-sm">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="first_name" class="block text-sm font-medium text-gray-700">Prénom</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<input id="first_name" name="first_name" type="text" autocomplete="first_name" required class="block w-full appearance-none rounded-2xl border border-gray-300 px-3 py-2 placeholder-gray-400 shadow-sm focus:border-co-blue focus:outline-none focus:ring-co-blue sm:text-sm">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="password" class="block text-sm font-medium text-gray-700">Mot de passe</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<input id="password" name="password" type="password" required class="block w-full appearance-none rounded-2xl border border-gray-300 px-3 py-2 placeholder-gray-400 shadow-sm focus:border-co-blue focus:outline-none focus:ring-co-blue sm:text-sm">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<button type="submit" class="mt-2 flex w-full justify-center rounded-2xl border border-transparent bg-co-blue py-2 px-4 text-sm font-medium text-white shadow-sm hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2">Créez votre compte</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
{{define "beneficiary_events"}}
|
|
||||||
<div class="px-4 py-6 sm:px-6">
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
{{define "beneficiary_files"}}
|
|
||||||
<div class="px-4 py-6 sm:px-6">
|
|
||||||
TODO Fichiers
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
{{define "beneficiary_journeys"}}
|
|
||||||
<div class="px-4 py-6 sm:px-6">
|
|
||||||
<form action="/app/journeys/" method="GET">
|
|
||||||
|
|
||||||
{{ $departureField := "departure" }}
|
|
||||||
{{ $departureLabel := "Départ" }}
|
|
||||||
{{ template "address_autocomplete" dict "FieldName" $departureField "FieldLabel" $departureLabel }}
|
|
||||||
|
|
||||||
{{ $destinationField := "destination" }}
|
|
||||||
{{ $destinationLabel := "Destination" }}
|
|
||||||
{{ template "address_autocomplete" dict "FieldName" $destinationField "FieldLabel" $destinationLabel }}
|
|
||||||
|
|
||||||
|
|
||||||
<div class="py-4 grid grid-cols-2">
|
|
||||||
<div class="lg:col-span-1">
|
|
||||||
<label for="departuredate" class="block text-sm font-medium text-gray-700">Le</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<input type="date" id="departuredate" name="departuredate" value="{{.ViewState.departuredate}}"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block w-full sm:text-sm border-gray-300 rounded-l-2xl border-r-1">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="lg:col-span-1">
|
|
||||||
<label for="departuretime" class="block text-sm font-medium text-gray-700">A</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<input type="time" id="departuretime" name="departuretime" value="{{.ViewState.departuretime}}"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block w-full sm:text-sm border-gray-300 rounded-r-2xl border-l-0">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<button type="submit"
|
|
||||||
class="rounded-2xl border border-transparent bg-co-blue px-4 py-2 my-4 mt-8 w-full text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
Chercher
|
|
||||||
</button>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,108 +0,0 @@
|
|||||||
{{define "beneficiary_notes"}}
|
|
||||||
<div class="px-4 py-6 sm:px-6">
|
|
||||||
<ul role="list" class="space-y-8">
|
|
||||||
<!-- <li>
|
|
||||||
<div class="flex space-x-3">
|
|
||||||
<div class="flex-shrink-0">
|
|
||||||
<img class="h-10 w-10 rounded-full"
|
|
||||||
src="https://images.unsplash.com/photo-1494790108377-be9c29b29330?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=facearea&facepad=2&w=256&h=256&q=80"
|
|
||||||
alt="">
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<div class="text-sm">
|
|
||||||
<a href="#" class="font-medium text-gray-900">Leslie Alexander</a>
|
|
||||||
</div>
|
|
||||||
<div class="mt-1 text-sm text-gray-700">
|
|
||||||
<p>Ducimus quas delectus ad maxime totam doloribus reiciendis ex.
|
|
||||||
Tempore dolorem maiores. Similique voluptatibus tempore non ut.</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-2 text-sm space-x-2">
|
|
||||||
<span class="text-gray-500 font-medium">4d ago</span>
|
|
||||||
<span class="text-gray-500 font-medium">·</span>
|
|
||||||
<button type="button" class="text-gray-900 font-medium">Reply</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
<li>
|
|
||||||
<div class="flex space-x-3">
|
|
||||||
<div class="flex-shrink-0">
|
|
||||||
<img class="h-10 w-10 rounded-full"
|
|
||||||
src="https://images.unsplash.com/photo-1519244703995-f4e0f30006d5?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=facearea&facepad=2&w=256&h=256&q=80"
|
|
||||||
alt="">
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<div class="text-sm">
|
|
||||||
<a href="#" class="font-medium text-gray-900">Michael Foster</a>
|
|
||||||
</div>
|
|
||||||
<div class="mt-1 text-sm text-gray-700">
|
|
||||||
<p>Et ut autem. Voluptatem eum dolores sint necessitatibus quos. Quis
|
|
||||||
eum qui dolorem accusantium voluptas voluptatem ipsum. Quo facere
|
|
||||||
iusto quia accusamus veniam id explicabo et aut.</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-2 text-sm space-x-2">
|
|
||||||
<span class="text-gray-500 font-medium">4d ago</span>
|
|
||||||
<span class="text-gray-500 font-medium">·</span>
|
|
||||||
<button type="button" class="text-gray-900 font-medium">Reply</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
<li>
|
|
||||||
<div class="flex space-x-3">
|
|
||||||
<div class="flex-shrink-0">
|
|
||||||
<img class="h-10 w-10 rounded-full"
|
|
||||||
src="https://images.unsplash.com/photo-1506794778202-cad84cf45f1d?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=facearea&facepad=2&w=256&h=256&q=80"
|
|
||||||
alt="">
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<div class="text-sm">
|
|
||||||
<a href="#" class="font-medium text-gray-900">Dries Vincent</a>
|
|
||||||
</div>
|
|
||||||
<div class="mt-1 text-sm text-gray-700">
|
|
||||||
<p>Expedita consequatur sit ea voluptas quo ipsam recusandae. Ab sint et
|
|
||||||
voluptatem repudiandae voluptatem et eveniet. Nihil quas consequatur
|
|
||||||
autem. Perferendis rerum et.</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-2 text-sm space-x-2">
|
|
||||||
<span class="text-gray-500 font-medium">4d ago</span>
|
|
||||||
<span class="text-gray-500 font-medium">·</span>
|
|
||||||
<button type="button" class="text-gray-900 font-medium">Reply</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li> -->
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
<div class="bg-gray-50 px-4 py-6 sm:px-6">
|
|
||||||
<div class="flex space-x-3">
|
|
||||||
<div class="flex-shrink-0">
|
|
||||||
<img class="h-10 w-10 rounded-co"
|
|
||||||
src="https://images.unsplash.com/photo-1517365830460-955ce3ccd263?ixlib=rb-=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=facearea&facepad=8&w=256&h=256&q=80"
|
|
||||||
alt="">
|
|
||||||
</div>
|
|
||||||
<div class="min-w-0 flex-1">
|
|
||||||
<form action="#">
|
|
||||||
<div>
|
|
||||||
<label for="comment" class="sr-only">Note</label>
|
|
||||||
<textarea id="comment" name="comment" rows="3"
|
|
||||||
class="shadow-sm block w-full focus:ring-blue-500 focus:border-blue-500 sm:text-sm border border-gray-300 rounded-md"
|
|
||||||
placeholder="Ajouter une note"></textarea>
|
|
||||||
</div>
|
|
||||||
<div class="mt-3 flex items-center justify-between">
|
|
||||||
<a href="#"
|
|
||||||
class="group inline-flex items-start text-sm space-x-2 text-gray-500 hover:text-gray-900">
|
|
||||||
<!-- Heroicon name: solid/question-mark-circle -->
|
|
||||||
{{.IconSet.Icon "hero:solid/question-mark-icon" "flex-shrink-0 h-5 w-5 text-gray-400 group-hover:text-gray-500" }}
|
|
||||||
<span> Accepte le "markdown". </span>
|
|
||||||
</a>
|
|
||||||
<button type="submit"
|
|
||||||
class="inline-flex items-center justify-center px-4 py-2 border border-transparent text-sm font-medium rounded-md shadow-sm text-white bg-co-blue hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Ajouter la note</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
{{define "beneficiary_vehicles"}}
|
|
||||||
<div class="px-4 py-6 sm:px-6">
|
|
||||||
{{if .ViewState.bookings}}
|
|
||||||
<h3 class="text-lg">Mises à disposition réalisées</h3>
|
|
||||||
<ul class="my-8">
|
|
||||||
{{range .ViewState.bookings}}
|
|
||||||
<li class="text-sm">Du {{(timeFrom .Startdate).Format "02/01/2006"}} au {{(timeFrom .Enddate).Format "02/01/2006"}}</li>
|
|
||||||
{{end}}
|
|
||||||
</ul>
|
|
||||||
{{end}}
|
|
||||||
<h3 class="text-lg">Réserver un véhicule</h3>
|
|
||||||
<form method="GET" action="/app/vehicles/">
|
|
||||||
<input type="hidden" name="beneficiaryid" value="{{.ViewState.beneficiary.ID}}">
|
|
||||||
<div class="py-4 grid grid-cols-2">
|
|
||||||
<div class="lg:col-span-1">
|
|
||||||
<label for="startdate" class="block text-sm font-medium text-gray-700">Du</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<input type="date" id="startdate" name="startdate"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block w-full sm:text-sm border-gray-300 rounded-l-2xl border-r-1">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="lg:col-span-1">
|
|
||||||
<label for="enddate" class="block text-sm font-medium text-gray-700">Au</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<input type="date" id="enddate" name="enddate"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block w-full sm:text-sm border-gray-300 rounded-r-2xl border-l-0">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
{{template "vehicle_type_select" .}}
|
|
||||||
|
|
||||||
<button type="submit"
|
|
||||||
class="rounded-2xl border border-transparent bg-co-blue px-4 py-2 my-4 mt-8 w-full text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
Chercher
|
|
||||||
</button>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,173 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Ajouter un bénéficiaire</h1>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8 mt-8"
|
|
||||||
x-data="{
|
|
||||||
fields: {
|
|
||||||
first_name: null,
|
|
||||||
last_name: null,
|
|
||||||
email: null,
|
|
||||||
phone_number: null,
|
|
||||||
birthdate: null
|
|
||||||
},
|
|
||||||
rules: {
|
|
||||||
first_name: ['required'],
|
|
||||||
last_name: ['required'],
|
|
||||||
email: ['required', 'email'],
|
|
||||||
phone_number: ['required', 'regexMatch:^((\\+)33|0)[1-9](\\d{2}){4}$'],
|
|
||||||
birthdate: ['optional'],
|
|
||||||
},
|
|
||||||
formValidation: {
|
|
||||||
valid: false,
|
|
||||||
fields: {
|
|
||||||
first_name: {valid: null},
|
|
||||||
last_name: {valid: null},
|
|
||||||
email: {valid: null},
|
|
||||||
phone_number: {valid: null},
|
|
||||||
birthdate: {valid: null},
|
|
||||||
}
|
|
||||||
},
|
|
||||||
isFormValid: true,
|
|
||||||
validate() {
|
|
||||||
this.formValidation = Iodine.assert(this.fields, this.rules)
|
|
||||||
},
|
|
||||||
validateField(field) {
|
|
||||||
this.formValidation.fields[field] = Iodine.assert(this.fields[field], this.rules[field])
|
|
||||||
},
|
|
||||||
submit(event) {
|
|
||||||
this.validate()
|
|
||||||
if(!this.formValidation.valid) {
|
|
||||||
this.isFormValid = false
|
|
||||||
event.preventDefault()
|
|
||||||
}
|
|
||||||
return this.formValidation.valid
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<form class="space-y-6" method="POST" @submit="submit">
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Informations obligatoires</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Informations personnelles sur le bénéficiaire obligatoires
|
|
||||||
pour créer son profil dans PARCOURSMOB</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 md:mt-0 md:col-span-2">
|
|
||||||
<div class="grid grid-cols-6 gap-6">
|
|
||||||
<div class="col-span-6 sm:col-span-3">
|
|
||||||
<label for="first_name" class="block text-sm font-medium text-gray-700">Prénom</label>
|
|
||||||
<input type="text" name="first_name" id="first_name" autocomplete="given-name"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.first_name" @blur="validateField('first_name')"
|
|
||||||
:class="formValidation.fields.first_name.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="col-span-6 sm:col-span-3">
|
|
||||||
<label for="last_name" class="block text-sm font-medium text-gray-700">Nom</label>
|
|
||||||
<input type="text" name="last_name" id="last_name" autocomplete="family-name"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.last_name" @blur="validateField('last_name')"
|
|
||||||
:class="formValidation.fields.last_name.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="col-span-6 sm:col-span-4">
|
|
||||||
<label for="email" class="block text-sm font-medium text-gray-700">Email</label>
|
|
||||||
<input type="text" name="email" id="email" autocomplete="email"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.email" @blur="validateField('email')"
|
|
||||||
:class="formValidation.fields.email.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="col-span-6 sm:col-span-4">
|
|
||||||
<label for="phone_number" class="block text-sm font-medium text-gray-700">Numéro de
|
|
||||||
téléphone</label>
|
|
||||||
<input type="text" name="phone_number" id="phone_number" autocomplete="phone" placeholder="+33612345678"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.phone_number" @blur="validateField('phone_number')"
|
|
||||||
:class="formValidation.fields.phone_number.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Informations optionnelles</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Autres informations de profil optionnelles</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 space-y-6 md:mt-0 md:col-span-2">
|
|
||||||
<div class="grid grid-cols-6 gap-6">
|
|
||||||
|
|
||||||
<div class="col-span-6 sm:col-span-3">
|
|
||||||
<label for="birthdate" class="block text-sm font-medium text-gray-700">Date de
|
|
||||||
naissance</label>
|
|
||||||
<input type="date" name="birthdate" id="birthdate" autocomplete="birthdate" placeholder="JJ/MM/AAAA"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.birthdate" @blur="validateField('birthdate')"
|
|
||||||
:class="formValidation.fields.birthdate.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="col-span-6 sm:col-span-3">
|
|
||||||
<label for="gender" class="block text-sm font-medium text-gray-700">Genre</label>
|
|
||||||
<div class="sm:mt-0 sm:col-span-2">
|
|
||||||
<select id="gender" name="gender" autocomplete="gender" x-model="gender"
|
|
||||||
class="max-w-lg mt-1 block focus:ring-co-blue focus:border-co-blue w-full shadow-sm sm:max-w-xs sm:text-sm border-gray-300 rounded-2xl">
|
|
||||||
<option value="0">Inconnu</option>
|
|
||||||
<option value="1">Masculin</option>
|
|
||||||
<option value="2">Féminin</option>
|
|
||||||
<option value="9">Sans objet</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- <div class="col-span-3 sm:col-span-3">
|
|
||||||
<label class="block text-sm font-medium text-gray-700"> Photo </label>
|
|
||||||
<div class="mt-1 flex items-center space-x-5">
|
|
||||||
<span class="inline-block h-12 w-12 rounded-co overflow-hidden bg-gray-100">
|
|
||||||
{{.IconSet.Icon "img:profile-picture-placeholder" "h-full w-full"}}
|
|
||||||
</span>
|
|
||||||
<button type="button"
|
|
||||||
class="bg-white py-2 px-3 border border-gray-300 rounded-2xl shadow-sm text-sm leading-4 font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">
|
|
||||||
Charger la photo
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div> -->
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Paramètres</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Paramètres liés au bénéficiaire, utiles pour exploiter les fonctionnalités de PARCOURSMOB</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 space-y-6 md:mt-0 md:col-span-2">
|
|
||||||
|
|
||||||
{{ $fieldName := "address" }}
|
|
||||||
{{ template "address_autocomplete" dict "FieldName" $fieldName }}
|
|
||||||
|
|
||||||
<!-- will dolater : tags, groups, ... -->
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex justify-end">
|
|
||||||
<p x-show="! isFormValid" class="px-4 py-2 text-sm text-co-red">Certains champs de sont pas valides.</p>
|
|
||||||
<a href="/app/beneficiaries/">
|
|
||||||
<button type="button"
|
|
||||||
class="bg-white py-2 px-4 border border-gray-300 rounded-2xl shadow-sm text-sm font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Annuler</button>
|
|
||||||
</a>
|
|
||||||
<button type="submit"
|
|
||||||
class="ml-3 inline-flex justify-center py-2 px-4 border border-transparent shadow-sm text-sm font-medium rounded-2xl text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Créer le bénéficiaire</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,150 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<main class="py-10">
|
|
||||||
<!-- Page header -->
|
|
||||||
<div class="max-w-3xl mx-auto px-4 sm:px-6 md:flex md:items-center md:justify-between md:space-x-5 lg:max-w-7xl lg:px-8">
|
|
||||||
<div class="flex items-center space-x-5">
|
|
||||||
<div class="flex-shrink-0">
|
|
||||||
<div class="relative">
|
|
||||||
<img class="h-16 w-16 rounded-co" src="/app/beneficiaries/{{.ViewState.beneficiary.ID}}/picture" alt="">
|
|
||||||
<span class="absolute inset-0 shadow-inner rounded-full" aria-hidden="true"></span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<h1 class="text-2xl font-bold text-gray-900">{{.ViewState.beneficiary.Data.first_name}}
|
|
||||||
{{.ViewState.beneficiary.Data.last_name}}</h1>
|
|
||||||
<p class="text-sm font-medium text-gray-500">{{if .ViewState.beneficiary.Metadata.created}}Ajouté le <time
|
|
||||||
datetime="2022-07-25">{{.ViewState.beneficiary.Metadata.created}}</time> par
|
|
||||||
<a href="#" class="text-gray-900">Conseiller 1</a>{{end}}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="mt-6 flex flex-col-reverse justify-stretch space-y-4 space-y-reverse sm:flex-row-reverse sm:justify-end sm:space-x-reverse sm:space-y-0 sm:space-x-3 md:mt-0 md:flex-row md:space-x-3">
|
|
||||||
<!-- <button type="button"
|
|
||||||
class="inline-flex items-center justify-center px-4 py-2 border border-gray-300 shadow-sm text-sm font-medium rounded-2xl text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">Supprimer</button> -->
|
|
||||||
<a href="/app/beneficiaries/{{.ViewState.beneficiary.ID}}/update" class="inline-flex"><button type="button"
|
|
||||||
class="w-full px-4 py-2 border border-transparent text-sm font-medium rounded-2xl shadow-sm text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">Modifier</button></a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="mt-8 max-w-3xl mx-auto grid grid-cols-1 gap-6 sm:px-6 lg:max-w-7xl lg:grid-flow-col-dense lg:grid-cols-3">
|
|
||||||
<div class="space-y-6 lg:col-start-1 lg:col-span-2">
|
|
||||||
<section aria-labelledby="beneficiary-information-title">
|
|
||||||
<div class="bg-white shadow sm:rounded-lg">
|
|
||||||
<div class="px-4 py-5 sm:px-6">
|
|
||||||
<h2 id="beneficiary-information-title" class="text-lg leading-6 font-medium text-gray-900">
|
|
||||||
Informations personnelles</h2>
|
|
||||||
<p class="mt-1 max-w-2xl text-sm text-gray-500">Informations générales sur le bénéficiaire.</p>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
<dl class="grid grid-cols-1 gap-x-4 gap-y-8 sm:grid-cols-2">
|
|
||||||
{{if .ViewState.beneficiary.Data.email}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Email</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.beneficiary.Data.email}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.beneficiary.Data.phone_number}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Téléphone</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.beneficiary.Data.phone_number}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.beneficiary.Data.birthdate}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Date de naissance</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{(timeFrom .ViewState.beneficiary.Data.birthdate).Format
|
|
||||||
"02/01/2006"}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if and .ViewState.beneficiary.Data.gender (ne .ViewState.beneficiary.Data.gender "0")}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Genre</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{genderISO5218 .ViewState.beneficiary.Data.gender}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.beneficiary.Data.address}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Adresse</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.beneficiary.Data.address.properties.label}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
<section aria-labelledby="functionalities-title" x-data="{
|
|
||||||
tab: 'vehicles',
|
|
||||||
to(event) {
|
|
||||||
this.tab = event.target.value
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<div class="bg-white shadow sm:rounded-lg sm:overflow-hidden">
|
|
||||||
<div class="divide-y divide-gray-200">
|
|
||||||
<div>
|
|
||||||
<div class="sm:hidden">
|
|
||||||
<label for="tabs" class="sr-only">Select a tab</label>
|
|
||||||
<select id="tabs" name="tabs" @change="to"
|
|
||||||
class="block w-full pl-3 pr-10 py-2 text-base border-gray-300 focus:outline-none focus:ring-indigo-500 focus:border-indigo-500 sm:text-sm rounded-md">
|
|
||||||
<!-- <option value="notes">Notes</option> -->
|
|
||||||
|
|
||||||
<option value="journeys">Déplacements</option>
|
|
||||||
|
|
||||||
<option value="events">Dispositifs</option>
|
|
||||||
|
|
||||||
<option value="files">Documents</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
<div class="hidden sm:block">
|
|
||||||
<div class="border-b border-gray-200 pl-4">
|
|
||||||
<nav class="-mb-px flex space-x-8" aria-label="Tabs">
|
|
||||||
<!-- Current: "border-indigo-500 text-indigo-600", Default: "border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300" -->
|
|
||||||
<!-- <a href="#" @click="tab = 'notes'"
|
|
||||||
class="whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"
|
|
||||||
:class="tab == 'notes' ? 'border-co-blue text-co-blue' : 'border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300'">
|
|
||||||
Notes </a> -->
|
|
||||||
|
|
||||||
<a href="#" @click="tab = 'journeys'"
|
|
||||||
class="whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"
|
|
||||||
:class="tab == 'journeys' ? 'border-co-blue text-co-blue' : 'border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300'">
|
|
||||||
Déplacements </a>
|
|
||||||
|
|
||||||
<a href="#" @click="tab = 'vehicles'"
|
|
||||||
class="whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"
|
|
||||||
:class="tab == 'vehicles' ? 'border-co-blue text-co-blue' : 'border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300'">
|
|
||||||
Véhicules </a>
|
|
||||||
|
|
||||||
<a href="#" @click="tab = 'events'"
|
|
||||||
class="whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"
|
|
||||||
:class="tab == 'events' ? 'border-co-blue text-co-blue' : 'border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300'">
|
|
||||||
Dispositifs </a>
|
|
||||||
|
|
||||||
<!-- <a href="#" @click="tab = 'files'"
|
|
||||||
class="whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"
|
|
||||||
:class="tab == 'files' ? 'border-co-blue text-co-blue' : 'border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300'">
|
|
||||||
Documents </a> -->
|
|
||||||
</nav>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div x-show="tab == 'journeys'">{{template "beneficiary_journeys" .}}</div>
|
|
||||||
<div x-show="tab == 'vehicles'">{{template "beneficiary_vehicles" .}}</div>
|
|
||||||
<div x-show="tab == 'events'">{{template "beneficiary_events" .}}</div>
|
|
||||||
<div x-show="tab == 'files'">{{template "beneficiary_files" .}}</div>
|
|
||||||
<div x-show="tab == 'notes'">{{template "beneficiary_notes" .}}</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<section aria-labelledby="timeline-title" class="lg:col-start-3 lg:col-span-1">
|
|
||||||
<div class="bg-white px-4 py-5 shadow sm:rounded-lg sm:px-6">
|
|
||||||
<h2 id="timeline-title" class="text-lg font-medium text-gray-900">Actions réalisées</h2>
|
|
||||||
<p class="p-12 text-gray-500 text-center text-md">Aucune action réalisée pour le moment</p>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
</main>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,159 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Bénéficiaires</h1>
|
|
||||||
|
|
||||||
<div class="sm:flex sm:items-center">
|
|
||||||
<div class="sm:flex-auto">
|
|
||||||
<p class="mt-2 text-sm text-gray-700"></p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-4 sm:mt-0 sm:ml-16 sm:flex-none">
|
|
||||||
<a href="/app/beneficiaries/create">
|
|
||||||
<button type="button"
|
|
||||||
class="inline-flex items-center justify-center rounded-2xl border border-transparent bg-co-blue px-4 py-2 text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-ci-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
{{$.IconSet.Icon "hero:outline/plus-circle" "h-5 w-5 mr-3"}}
|
|
||||||
Ajouter
|
|
||||||
</button>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8" x-data="{
|
|
||||||
state: {{.ViewState.JSONWithLimits 0 10}},
|
|
||||||
current: 0,
|
|
||||||
nb_pages() {
|
|
||||||
let nbEl = this.state.count
|
|
||||||
return Math.ceil(nbEl/10)
|
|
||||||
},
|
|
||||||
async paginate(page) {
|
|
||||||
let start = (page-1)*10
|
|
||||||
if(start < 0|| start > this.state.count) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
let resp = await fetch('/api/cache/' + this.state.cache_id + '?limits.min=' + start + '&limits.max=' + (start+10))
|
|
||||||
let data = await resp.json()
|
|
||||||
this.state.beneficiaries = data
|
|
||||||
this.current=page-1
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<div class="mt-8 flex flex-col">
|
|
||||||
<div class="-my-2 -mx-4 overflow-x-auto sm:-mx-6 lg:-mx-8">
|
|
||||||
<div class="inline-block min-w-full py-2 align-middle md:px-6 lg:px-8">
|
|
||||||
<div class="overflow-hidden shadow ring-1 ring-black ring-opacity-5 md:rounded-lg">
|
|
||||||
<table class="min-w-full divide-y divide-gray-300">
|
|
||||||
<thead class="bg-gray-50">
|
|
||||||
<tr>
|
|
||||||
<th scope="col"
|
|
||||||
class="py-3.5 pl-4 pr-3 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Nom
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900">
|
|
||||||
Téléphone
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900">
|
|
||||||
Adresse
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900">
|
|
||||||
Labels
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="relative py-3.5 pl-3 pr-4 sm:pr-6">
|
|
||||||
<span class="sr-only">Modifier</span>
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="divide-y divide-gray-200 bg-white">
|
|
||||||
<template x-for="beneficiary in state.beneficiaries">
|
|
||||||
<tr>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="flex items-center">
|
|
||||||
<div class="h-10 w-10 flex-shrink-0">
|
|
||||||
<img class="h-10 w-10 rounded-co"
|
|
||||||
:src="'/app/beneficiaries/' + beneficiary.id + '/picture'" alt="">
|
|
||||||
</div>
|
|
||||||
<div class="ml-4">
|
|
||||||
<div class="font-medium text-gray-900"><span
|
|
||||||
x-text="beneficiary.data.first_name"></span> <span
|
|
||||||
x-text="beneficiary.data.last_name"></span></div>
|
|
||||||
<div class="text-gray-500" x-text="beneficiary.data.email"></div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-gray-500">
|
|
||||||
<div class="text-gray-900" x-text="beneficiary.data.phone_number"></div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-gray-500"
|
|
||||||
x-text="beneficiary.data.address ? beneficiary.data.address.properties.label : ''">
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-gray-500">
|
|
||||||
<template x-for="tag in beneficiary.data.tags">
|
|
||||||
<span
|
|
||||||
class="inline-flex rounded-full bg-green-100 px-2 text-xs font-semibold leading-5 text-green-800"
|
|
||||||
x-text="tag"></span>
|
|
||||||
</template>
|
|
||||||
</td>
|
|
||||||
<td
|
|
||||||
class="relative whitespace-nowrap py-4 pl-3 pr-4 text-right text-sm font-medium sm:pr-6">
|
|
||||||
<a :href="'/app/beneficiaries/' + beneficiary.id"
|
|
||||||
class="text-co-blue hover:text-co-blue">Voir<span class="sr-only">, <span
|
|
||||||
x-text="beneficiary.data.first_name"></span> <span
|
|
||||||
x-text="beneficiary.data.last_name"></span></a>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<!-- More people... -->
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
<div class="bg-white px-4 py-3 flex items-center justify-between border-t border-gray-200 sm:px-6">
|
|
||||||
<div class="flex-1 flex justify-between sm:hidden">
|
|
||||||
<a href="#" class="relative inline-flex items-center px-4 py-2 border border-gray-300 text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50"
|
|
||||||
@click="paginate(current)"> Previous </a>
|
|
||||||
<a href="#" class="ml-3 relative inline-flex items-center px-4 py-2 border border-gray-300 text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50"
|
|
||||||
@click="paginate(current+2)"> Next </a>
|
|
||||||
</div>
|
|
||||||
<div class="hidden sm:flex-1 sm:flex sm:items-center sm:justify-between">
|
|
||||||
<div>
|
|
||||||
<p class="text-sm text-gray-700">
|
|
||||||
Résultats
|
|
||||||
<span class="font-medium" x-text="Math.min((current * 10)+1, state.count)"></span>
|
|
||||||
à
|
|
||||||
<span class="font-medium" x-text="Math.min((current * 10)+10, state.count)"></span>
|
|
||||||
sur
|
|
||||||
<span class="font-medium" x-text="state.count"></span>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<nav class="relative z-0 inline-flex rounded-md shadow-sm -space-x-px" aria-label="Pagination">
|
|
||||||
<a href="#" class="relative inline-flex items-center px-2 py-2 rounded-l-md border border-gray-300 bg-white text-sm font-medium text-gray-500 hover:bg-gray-50"
|
|
||||||
@click="paginate(current)">
|
|
||||||
<span class="sr-only">Previous</span>
|
|
||||||
<!-- Heroicon name: solid/chevron-left -->
|
|
||||||
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd" d="M12.707 5.293a1 1 0 010 1.414L9.414 10l3.293 3.293a1 1 0 01-1.414 1.414l-4-4a1 1 0 010-1.414l4-4a1 1 0 011.414 0z" clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</a>
|
|
||||||
<template x-for="i in nb_pages">
|
|
||||||
<a href="#" @click="paginate(i)"
|
|
||||||
class="relative inline-flex items-center px-4 py-2 border text-sm font-medium"
|
|
||||||
:class="i == current+1 ? 'z-10 bg-indigo-50 border-co-blue text-co-blue' : 'bg-white border-gray-300 text-gray-500 hover:bg-gray-50'"
|
|
||||||
x-text="i"></a>
|
|
||||||
</template>
|
|
||||||
<!-- Current: "z-10 bg-indigo-50 border-indigo-500 text-indigo-600", Default: "bg-white border-gray-300 text-gray-500 hover:bg-gray-50" -->
|
|
||||||
<a href="#" class="relative inline-flex items-center px-2 py-2 rounded-r-md border border-gray-300 bg-white text-sm font-medium text-gray-500 hover:bg-gray-50"
|
|
||||||
@click="paginate(current+2)">
|
|
||||||
<span class="sr-only">Next</span>
|
|
||||||
<!-- Heroicon name: solid/chevron-right -->
|
|
||||||
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</a>
|
|
||||||
</nav>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,173 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Modifier un bénéficiaire</h1>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8 mt-8"
|
|
||||||
x-data="{
|
|
||||||
fields: {
|
|
||||||
first_name: '{{ .ViewState.Data.first_name }}',
|
|
||||||
last_name: '{{ .ViewState.Data.last_name }}',
|
|
||||||
email: '{{ .ViewState.Data.email }}',
|
|
||||||
phone_number: '{{ .ViewState.Data.phone_number }}',
|
|
||||||
birthdate: {{if .ViewState.Data.birthdate}}'{{ (timeFrom .ViewState.Data.birthdate).Format "2006-01-02" }}'{{else}}null{{end}},
|
|
||||||
gender: {{.ViewState.Data.gender}}
|
|
||||||
},
|
|
||||||
rules: {
|
|
||||||
first_name: ['required'],
|
|
||||||
last_name: ['required'],
|
|
||||||
email: ['required', 'email'],
|
|
||||||
phone_number: ['required', 'regexMatch:^((\\+)33|0)[1-9](\\d{2}){4}$'],
|
|
||||||
birthdate: ['optional'],
|
|
||||||
},
|
|
||||||
formValidation: {
|
|
||||||
valid: false,
|
|
||||||
fields: {
|
|
||||||
first_name: {valid: null},
|
|
||||||
last_name: {valid: null},
|
|
||||||
email: {valid: null},
|
|
||||||
phone_number: {valid: null},
|
|
||||||
birthdate: {valid: null},
|
|
||||||
}
|
|
||||||
},
|
|
||||||
isFormValid: true,
|
|
||||||
validate() {
|
|
||||||
this.formValidation = Iodine.assert(this.fields, this.rules)
|
|
||||||
},
|
|
||||||
validateField(field) {
|
|
||||||
this.formValidation.fields[field] = Iodine.assert(this.fields[field], this.rules[field])
|
|
||||||
},
|
|
||||||
submit(event) {
|
|
||||||
this.validate()
|
|
||||||
if(!this.formValidation.valid) {
|
|
||||||
this.isFormValid = false
|
|
||||||
event.preventDefault()
|
|
||||||
}
|
|
||||||
return this.formValidation.valid
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<form class="space-y-6" method="POST" @submit="submit">
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Informations obligatoires</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Informations personnelles sur le bénéficiaire obligatoires
|
|
||||||
pour créer son profil dans PARCOURSMOB</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 md:mt-0 md:col-span-2">
|
|
||||||
<div class="grid grid-cols-6 gap-6">
|
|
||||||
<div class="col-span-6 sm:col-span-3">
|
|
||||||
<label for="first_name" class="block text-sm font-medium text-gray-700">Prénom</label>
|
|
||||||
<input type="text" name="first_name" id="first_name" autocomplete="given-name"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.first_name" @blur="validateField('first_name')"
|
|
||||||
:class="formValidation.fields.first_name.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="col-span-6 sm:col-span-3">
|
|
||||||
<label for="last_name" class="block text-sm font-medium text-gray-700">Nom</label>
|
|
||||||
<input type="text" name="last_name" id="last_name" autocomplete="family-name"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.last_name" @blur="validateField('last_name')"
|
|
||||||
:class="formValidation.fields.last_name.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="col-span-6 sm:col-span-4">
|
|
||||||
<label for="email" class="block text-sm font-medium text-gray-700">Email</label>
|
|
||||||
<input type="text" name="email" id="email" autocomplete="email"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.email" @blur="validateField('email')"
|
|
||||||
:class="formValidation.fields.email.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="col-span-6 sm:col-span-4">
|
|
||||||
<label for="phone_number" class="block text-sm font-medium text-gray-700">Numéro de
|
|
||||||
téléphone</label>
|
|
||||||
<input type="text" name="phone_number" id="phone_number" autocomplete="phone" placeholder="+33612345678"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.phone_number" @blur="validateField('phone_number')"
|
|
||||||
:class="formValidation.fields.phone_number.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Informations optionnelles</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Autres informations de profil optionnelles</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 space-y-6 md:mt-0 md:col-span-2">
|
|
||||||
<div class="grid grid-cols-6 gap-6">
|
|
||||||
|
|
||||||
<div class="col-span-6 sm:col-span-3">
|
|
||||||
<label for="birthdate" class="block text-sm font-medium text-gray-700">Date de
|
|
||||||
naissance</label>
|
|
||||||
<input type="date" name="birthdate" id="birthdate" autocomplete="birthdate" placeholder="JJ/MM/AAAA"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.birthdate" @blur="validateField('birthdate')"
|
|
||||||
:class="formValidation.fields.birthdate.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="col-span-6 sm:col-span-3">
|
|
||||||
<label for="gender" class="block text-sm font-medium text-gray-700">Genre</label>
|
|
||||||
<div class="sm:mt-0 sm:col-span-2">
|
|
||||||
<select id="gender" name="gender" autocomplete="gender" x-model="gender"
|
|
||||||
class="max-w-lg mt-1 block focus:ring-co-blue focus:border-co-blue w-full shadow-sm sm:max-w-xs sm:text-sm border-gray-300 rounded-2xl">
|
|
||||||
<option value="0">Inconnu</option>
|
|
||||||
<option value="1" :selected="fields.gender == '1'">Masculin</option>
|
|
||||||
<option value="2" :selected="fields.gender == '2'">Féminin</option>
|
|
||||||
<option value="9" :selected="fields.gender == '9'">Sans objet</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- <div class="col-span-3 sm:col-span-3">
|
|
||||||
<label class="block text-sm font-medium text-gray-700"> Photo </label>
|
|
||||||
<div class="mt-1 flex items-center space-x-5">
|
|
||||||
<span class="inline-block h-12 w-12 rounded-co overflow-hidden bg-gray-100">
|
|
||||||
{{.IconSet.Icon "img:profile-picture-placeholder" "h-full w-full"}}
|
|
||||||
</span>
|
|
||||||
<button type="button"
|
|
||||||
class="bg-white py-2 px-3 border border-gray-300 rounded-2xl shadow-sm text-sm leading-4 font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">
|
|
||||||
Charger la photo
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div> -->
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Paramètres</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Paramètres liés au bénéficiaire, utiles pour exploiter les fonctionnalités de PARCOURSMOB</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 space-y-6 md:mt-0 md:col-span-2">
|
|
||||||
|
|
||||||
{{ $fieldName := "address" }}
|
|
||||||
{{ template "address_autocomplete" (dict "FieldName" $fieldName "Address" .ViewState.Data.address) }}
|
|
||||||
<!-- will dolater : tags, groups, ... -->
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex justify-end">
|
|
||||||
<p x-show="! isFormValid" class="px-4 py-2 text-sm text-co-red">Certains champs de sont pas valides.</p>
|
|
||||||
<a href="/app/beneficiaries/{{.ViewState.ID}}">
|
|
||||||
<button type="button"
|
|
||||||
class="bg-white py-2 px-4 border border-gray-300 rounded-2xl shadow-sm text-sm font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Annuler</button>
|
|
||||||
</a>
|
|
||||||
<button type="submit"
|
|
||||||
class="ml-3 inline-flex justify-center py-2 px-4 border border-transparent shadow-sm text-sm font-medium rounded-2xl text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Modifier</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
{{define "agenda_widget"}}
|
|
||||||
<div class="col-span-1 bg-white rounded-2xl shadow divide-y divide-gray-200 flex flex-col">
|
|
||||||
<div class="-ml-4 -mt-2 px-4 py-2 flex items-center justify-between flex-wrap sm:flex-nowrap">
|
|
||||||
<div class="ml-4 mt-2">
|
|
||||||
<h3 class="text-lg leading-6 font-medium text-gray-900">Prochains dispositifs</h3>
|
|
||||||
</div>
|
|
||||||
<!-- <div class="ml-4 mt-2 flex-shrink-0">
|
|
||||||
<button type="button" class="relative inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-sm font-medium rounded-md text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Voir</button>
|
|
||||||
</div> -->
|
|
||||||
</div>
|
|
||||||
<ul role="list" class="divide-y divide-gray-200 flex-1">
|
|
||||||
{{range .}}
|
|
||||||
<li class="py-2 px-4 flex">
|
|
||||||
<a href="/app/agenda/{{.ID}}" class="flex w-full">
|
|
||||||
<div class="ml-3">
|
|
||||||
<p class="text-sm font-medium text-gray-900">{{(timeFrom .Startdate).Format "02/01"}} - {{.Type}} - {{.Name}}</p>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
</li>
|
|
||||||
{{end}}
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
<a href="/app/agenda/">
|
|
||||||
<button class="w-full p-2 text-center bg-co-blue text-white rounded-b-2xl text-sm">
|
|
||||||
Agenda des dispositifs
|
|
||||||
</button>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,31 +0,0 @@
|
|||||||
{{define "beneficiaries_widget"}}
|
|
||||||
<div class="col-span-1 bg-white rounded-2xl shadow divide-y divide-gray-200">
|
|
||||||
<div class="-ml-4 -mt-2 px-4 py-2 flex items-center justify-between flex-wrap sm:flex-nowrap">
|
|
||||||
<div class="ml-4 mt-2">
|
|
||||||
<h3 class="text-lg leading-6 font-medium text-gray-900">Bénéficiaires</h3>
|
|
||||||
<h2 class="text-sm leading-6 font-medium text-gray-600">Derniers bénéficiaires ajoutés</h2>
|
|
||||||
</div>
|
|
||||||
<!-- <div class="ml-4 mt-2 flex-shrink-0">
|
|
||||||
<button type="button" class="relative inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-sm font-medium rounded-md text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Voir</button>
|
|
||||||
</div> -->
|
|
||||||
</div>
|
|
||||||
<ul role="list" class="divide-y divide-gray-200">
|
|
||||||
{{range .latest}}
|
|
||||||
<li class="py-2 px-4 flex">
|
|
||||||
<a href="/app/beneficiaries/{{.ID}}" class="flex w-full">
|
|
||||||
<img class="h-6 w-6 rounded-co" src="/app/beneficiaries/{{.ID}}/picture" alt="">
|
|
||||||
<div class="ml-3">
|
|
||||||
<p class="text-sm font-medium text-gray-900">{{.Data.first_name}} {{.Data.last_name}}</p>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
</li>
|
|
||||||
{{end}}
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
<a href="/app/beneficiaries/">
|
|
||||||
<button class="w-full p-2 text-center bg-co-blue text-white rounded-b-2xl text-sm">
|
|
||||||
Gérer les bénéficiaires
|
|
||||||
</button>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,77 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Tableau de bord</h1>
|
|
||||||
</div>
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8 mt-8">
|
|
||||||
<h2 class="text-gray-500 text-xs font-medium uppercase tracking-wide">Statistiques de votre organisation</h2>
|
|
||||||
<ul role="list" class="mt-3 grid grid-cols-1 gap-5 sm:gap-6 sm:grid-cols-2 lg:grid-cols-4">
|
|
||||||
<li class="col-span-1 flex shadow-sm rounded-3xl">
|
|
||||||
<div
|
|
||||||
class="flex-shrink-0 flex items-center justify-center w-16 bg-co-blue text-white text-sm font-medium rounded-l-3xl">
|
|
||||||
{{.IconSet.Icon "hero:outline/user-group" "h-6 w-6"}}
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="flex-1 flex items-center justify-between border-t border-r border-b border-gray-200 bg-white rounded-r-3xl truncate">
|
|
||||||
<div class="flex-1 px-4 py-2 text-sm truncate">
|
|
||||||
<a href="#" class="text-gray-900 font-medium hover:text-gray-600">Bénéficiaires</a>
|
|
||||||
<p class="text-gray-500">{{.ViewState.beneficiaries.count}} bénéficiaires</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
<li class="col-span-1 flex shadow-sm rounded-3xl">
|
|
||||||
<div
|
|
||||||
class="flex-shrink-0 flex items-center justify-center w-16 bg-co-green text-white text-sm font-medium rounded-l-3xl">
|
|
||||||
{{.IconSet.Icon "hero:outline/shield-check" "h-6 w-6"}}
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="flex-1 flex items-center justify-between border-t border-r border-b border-gray-200 bg-white rounded-r-3xl truncate">
|
|
||||||
<div class="flex-1 px-4 py-2 text-sm truncate">
|
|
||||||
<a href="#" class="text-gray-900 font-medium hover:text-gray-600">Accompagnement</a>
|
|
||||||
<p class="text-gray-500">0 actions réalisées</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
<li class="col-span-1 flex shadow-sm rounded-3xl">
|
|
||||||
<div
|
|
||||||
class="flex-shrink-0 flex items-center justify-center w-16 bg-co-yellow text-white text-sm font-medium rounded-l-3xl">
|
|
||||||
{{.IconSet.Icon "hero:outline/office-building" "h-6 w-6"}}
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="flex-1 flex items-center justify-between border-t border-r border-b border-gray-200 bg-white rounded-r-3xl truncate">
|
|
||||||
<div class="flex-1 px-4 py-2 text-sm truncate">
|
|
||||||
<a href="#" class="text-gray-900 font-medium hover:text-gray-600">Groupes</a>
|
|
||||||
<p class="text-gray-500">0 groupes créés</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
<li class="col-span-1 flex shadow-sm rounded-3xl">
|
|
||||||
<div
|
|
||||||
class="flex-shrink-0 flex items-center justify-center w-16 bg-co-red text-white text-sm font-medium rounded-l-3xl">
|
|
||||||
{{.IconSet.Icon "hero:outline/briefcase" "h-6 w-6"}}
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="flex-1 flex items-center justify-between border-t border-r border-b border-gray-200 bg-white rounded-r-3xl truncate">
|
|
||||||
<div class="flex-1 px-4 py-2 text-sm truncate">
|
|
||||||
<a href="#" class="text-gray-900 font-medium hover:text-gray-600">Référents</a>
|
|
||||||
<p class="text-gray-500">{{.ViewState.count_members}} membres</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto py-8 px-4 sm:px-6 md:px-8">
|
|
||||||
<div class="py-4 grid grid-cols-1 gap-6 sm:grid-cols-2 lg:grid-cols-3">
|
|
||||||
|
|
||||||
|
|
||||||
{{template "beneficiaries_widget" .ViewState.beneficiaries}}
|
|
||||||
{{template "agenda_widget" .ViewState.events}}
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,23 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Répertoire des solutions</h1>
|
|
||||||
|
|
||||||
<div class="sm:flex sm:items-center">
|
|
||||||
<div class="sm:flex-auto">
|
|
||||||
<p class="mt-2 text-sm text-gray-700"></p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-4 sm:mt-0 sm:ml-16 sm:flex-none">
|
|
||||||
<select id="type" name="type"
|
|
||||||
x-model="fields.type" @blur="validateField('type')"
|
|
||||||
class="max-w-lg mt-1 block focus:ring-co-blue focus:border-co-blue w-full shadow-sm sm:max-w-xs sm:text-sm rounded-2xl border-gray-300">
|
|
||||||
<option>Types de solution</option>
|
|
||||||
<option value="Auto-écoles sociales">Auto-écoles sociales</option>
|
|
||||||
<option value="Conseil en Mobilité">Conseil en Mobilité</option>
|
|
||||||
<option value="Garages solidaires">Garages solidaires</option>
|
|
||||||
<option value="Information collective">Information collective</option>
|
|
||||||
<option value="Autres">Autres</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,129 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<main class="py-10">
|
|
||||||
<div class="max-w-3xl mx-auto px-4 sm:px-6 md:flex md:items-center md:justify-between md:space-x-5 lg:max-w-7xl lg:px-8">
|
|
||||||
<div class="flex items-center space-x-5">
|
|
||||||
<!-- <div class="flex-shrink-0">
|
|
||||||
<div class="relative">
|
|
||||||
<img class="h-16 w-16 rounded-co" src="/app/beneficiaries/{{.ViewState.ID}}/picture" alt="">
|
|
||||||
<span class="absolute inset-0 shadow-inner rounded-full" aria-hidden="true"></span>
|
|
||||||
</div>
|
|
||||||
</div> -->
|
|
||||||
<div>
|
|
||||||
<h1 class="text-2xl font-bold text-gray-900">{{.ViewState.group.Data.name}}</h1>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="mt-6 flex flex-col-reverse justify-stretch space-y-4 space-y-reverse sm:flex-row-reverse sm:justify-end sm:space-x-reverse sm:space-y-0 sm:space-x-3 md:mt-0 md:flex-row md:space-x-3">
|
|
||||||
<!-- <button type="button"
|
|
||||||
class="inline-flex items-center justify-center px-4 py-2 border border-gray-300 shadow-sm text-sm font-medium rounded-2xl text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">Supprimer</button>
|
|
||||||
<a href="/app/administration/groups/{{.ViewState.group.ID}}/update" class="inline-flex"><button type="button"
|
|
||||||
class="w-full px-4 py-2 border border-transparent text-sm font-medium rounded-2xl shadow-sm text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">Modifier</button></a> -->
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="mt-8 max-w-3xl mx-auto grid grid-cols-1 gap-6 sm:px-6 lg:max-w-7xl lg:grid-flow-col-dense lg:grid-cols-3">
|
|
||||||
<div class="space-y-6 lg:col-start-1 lg:col-span-2">
|
|
||||||
<section aria-labelledby="beneficiary-information-title">
|
|
||||||
<div class="bg-white shadow sm:rounded-lg">
|
|
||||||
<div class="px-4 py-5 sm:px-6">
|
|
||||||
<h2 id="beneficiary-information-title" class="text-lg leading-6 font-medium text-gray-900">
|
|
||||||
Paramètres de l'organisation</h2>
|
|
||||||
<p class="mt-1 max-w-2xl text-sm text-gray-500">Paramètres généraux de l'organisation</p>
|
|
||||||
</div>
|
|
||||||
{{template "groups_members" .}}
|
|
||||||
|
|
||||||
<div class="px-2 py-4">
|
|
||||||
<form class="flex" method="POST" action="/app/group/settings/invite-member">
|
|
||||||
<div class="pr-2 flex-1">
|
|
||||||
<input type="text" name="username" id="username"
|
|
||||||
class="mt-1 border-gray-300 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
placeholder="Email">
|
|
||||||
</div>
|
|
||||||
<button class="px-1 py-1 border border-transparent text-sm font-medium rounded-2xl shadow-sm text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">Ajouter un membre</button>
|
|
||||||
|
|
||||||
</form>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{{template "groups_admins" .}}
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
<section aria-labelledby="modules-title" class="lg:col-start-3 lg:col-span-1">
|
|
||||||
<div class="bg-white px-4 py-5 shadow sm:rounded-lg sm:px-6">
|
|
||||||
<h2 id="modules-title" class="text-lg font-medium text-gray-900">Modules activés</h2>
|
|
||||||
|
|
||||||
<fieldset>
|
|
||||||
<div class="mt-4 space-y-4">
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="h-5 flex items-center">
|
|
||||||
<input id="beneficiaries" name="modules.beneficiaries" type="checkbox" disabled {{if .ViewState.group.Data.modules.beneficiaries}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.beneficiaries" class="font-medium text-gray-700">Bénéficiaires</label>
|
|
||||||
<p class="text-gray-500">Gestion des bénéficiaires assignés à sa propre organisation.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="journeys" name="modules.journeys" type="checkbox" disabled {{if .ViewState.group.Data.modules.journeys}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.journeys" class="font-medium text-gray-700">Déplacements</label>
|
|
||||||
<p class="text-gray-500">Trouver des solutions et organiser les déplacements de ses bénéficiaires.</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="vehicles" name="modules.vehicles" type="checkbox" disabled {{if .ViewState.group.Data.modules.vehicles}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.vehicles" class="font-medium text-gray-700">Véhicules</label>
|
|
||||||
<p class="text-gray-500">Trouver et réserver des véhicules pour ses bénéficiaires.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="vehicles_management" name="modules.vehicles_management" type="checkbox" disabled {{if .ViewState.group.Data.modules.vehicles_management}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.vehicles_management" class="font-medium text-gray-700">Gestion des véhicules</label>
|
|
||||||
<p class="text-gray-500">Gérer les véhicules et réservations (pour les gestionnaires de flottes)
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="events" name="modules.events" type="checkbox" disabled {{if .ViewState.group.Data.modules.events}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.events" class="font-medium text-gray-700">Agenda dispositifs</label>
|
|
||||||
<p class="text-gray-500">Agenda des dispositifs pour les bénéficiaires (sessions permis, événements, ...)
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-start">
|
|
||||||
<div class="flex items-center h-5">
|
|
||||||
<input id="events" name="modules.events" type="checkbox" disabled {{if .ViewState.group.Data.modules.administration}} checked{{end}}
|
|
||||||
class="focus:ring-co-blue h-4 w-4 text-co-blue border-gray-300 rounded">
|
|
||||||
</div>
|
|
||||||
<div class="ml-3 text-sm">
|
|
||||||
<label for="modules.events" class="font-medium text-gray-700">Administration</label>
|
|
||||||
<p class="text-gray-500">Administration générale de la plateforme PARCOURSMOB. Créer, ajouter des organisations et administrateurs.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</fieldset>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
</main>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,86 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Groups > Créer un group</h1>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8 mt-8" x-data="{
|
|
||||||
fields: {
|
|
||||||
name: null,
|
|
||||||
type: null,
|
|
||||||
},
|
|
||||||
rules: {
|
|
||||||
name: ['required'],
|
|
||||||
type: ['required'],
|
|
||||||
},
|
|
||||||
formValidation: {
|
|
||||||
valid: false,
|
|
||||||
fields: {
|
|
||||||
name: {valid: null},
|
|
||||||
type: {valid: null},
|
|
||||||
}
|
|
||||||
},
|
|
||||||
isFormValid: true,
|
|
||||||
validate() {
|
|
||||||
this.formValidation = Iodine.assert(this.fields, this.rules)
|
|
||||||
},
|
|
||||||
validateField(field) {
|
|
||||||
this.formValidation.fields[field] = Iodine.assert(this.fields[field], this.rules[field])
|
|
||||||
},
|
|
||||||
submit(event) {
|
|
||||||
this.validate()
|
|
||||||
if(!this.formValidation.valid) {
|
|
||||||
this.isFormValid = false
|
|
||||||
event.preventDefault()
|
|
||||||
}
|
|
||||||
return this.formValidation.valid
|
|
||||||
}
|
|
||||||
|
|
||||||
}">
|
|
||||||
<form class="space-y-6" method="POST" @submit="submit">
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Nouveau groupe</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Informations de base sur le groupe à créer</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 md:mt-0 md:col-span-2">
|
|
||||||
<div class="grid grid-cols-6 gap-6">
|
|
||||||
<div class="col-span-5">
|
|
||||||
<label for="name" class="block text-sm font-medium text-gray-700">Nom de groupe</label>
|
|
||||||
<input type="text" name="name" id="name"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.name" @blur="validateField('name')"
|
|
||||||
:class="formValidation.fields.name.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
<div class="col-span-3">
|
|
||||||
<label for="type" class="block text-sm font-medium text-gray-700">Type de groupe</label>
|
|
||||||
<select id="type" name="type"
|
|
||||||
x-model="fields.type" @blur="validateField('type')"
|
|
||||||
class="max-w-lg mt-1 block focus:ring-co-blue focus:border-co-blue w-full shadow-sm sm:max-w-xs sm:text-sm rounded-2xl"
|
|
||||||
:class="formValidation.fields.type.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
<option value="" selected></option>
|
|
||||||
{{range .ViewState.group_types}}
|
|
||||||
<option value="{{.}}">{{.}}</option>
|
|
||||||
{{end}}
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
<div class="flex justify-end">
|
|
||||||
<a href="/app/group_module/">
|
|
||||||
<button type="button"
|
|
||||||
class="bg-white py-2 px-4 border border-gray-300 rounded-2xl shadow-sm text-sm font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Annuler</button>
|
|
||||||
</a>
|
|
||||||
<button type="submit"
|
|
||||||
class="ml-3 inline-flex justify-center py-2 px-4 border border-transparent shadow-sm text-sm font-medium rounded-2xl text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Créer le groupe</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,254 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Gestion du groupe</h1>
|
|
||||||
|
|
||||||
<div class="sm:flex sm:items-center">
|
|
||||||
<div class="sm:flex-auto">
|
|
||||||
<p class="mt-2 text-sm text-gray-700"></p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<!-- ****************************** -->
|
|
||||||
<div class="mt-8 max-w-3xl mx-auto grid grid-cols-1 gap-6 lg:max-w-7xl lg:grid-flow-col-dense lg:grid-cols-3">
|
|
||||||
<div class="space-y-6 lg:col-start-1 lg:col-span-1 ">
|
|
||||||
<div class="bg-white shadow sm:rounded-2xl">
|
|
||||||
<h2 id="timeline-title" class="text-lg font-medium text-gray-900 p-4 sm:px-6">Info du groupe</h2>
|
|
||||||
<div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
<div>
|
|
||||||
<div class="mt-5 border-gray-200">
|
|
||||||
<dl class="sm:divide-y sm:divide-gray-200">
|
|
||||||
<div class="sm:pb-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Nom</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.group.Data.name}}</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Type</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.group.Data.type}}</dd>
|
|
||||||
</div>
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
<!-- *********************************** -->
|
|
||||||
|
|
||||||
|
|
||||||
<div class="lg:col-start-2 lg:col-span-2" x-data="{
|
|
||||||
state: {{.ViewState.list.JSONWithLimits 0 10}},
|
|
||||||
current: 0,
|
|
||||||
nb_pages() {
|
|
||||||
let nbEl = this.state.count
|
|
||||||
return Math.ceil(nbEl/10)
|
|
||||||
},
|
|
||||||
async paginate(page) {
|
|
||||||
let start = (page-1)*10
|
|
||||||
if(start < 0|| start > this.state.count) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
let resp = await fetch('/api/cache/' + this.state.cache_id + '?limits.min=' + start + '&limits.max=' + (start+10))
|
|
||||||
let data = await resp.json()
|
|
||||||
this.state.beneficiaries = data
|
|
||||||
this.current=page-1
|
|
||||||
this.state.group = data
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<div >
|
|
||||||
<div class="-my-2 -mx-4 overflow-x-auto sm:-mx-6 lg:-mx-8">
|
|
||||||
<div class="inline-block min-w-full py-2 align-middle md:px-6 lg:px-8">
|
|
||||||
<div class="overflow-hidden shadow ring-1 ring-black ring-opacity-5 md:rounded-lg">
|
|
||||||
<table class="min-w-full divide-y divide-gray-300">
|
|
||||||
<thead class="bg-gray-50">
|
|
||||||
<tr>
|
|
||||||
<th scope="col"
|
|
||||||
class="py-3.5 pl-4 pr-3 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Nom du bénéficiaire
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900">
|
|
||||||
Téléphone
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900">
|
|
||||||
Email
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="divide-y divide-gray-200 bg-white">
|
|
||||||
<template x-for="beneficiary in state.beneficiaries">
|
|
||||||
<tr>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="flex items-center">
|
|
||||||
<!-- <div class="h-10 w-10 flex-shrink-0">
|
|
||||||
<img class="h-10 w-10 rounded-co"
|
|
||||||
:src="'/app/beneficiaries/' + beneficiary.id + '/picture'" alt="">
|
|
||||||
</div> -->
|
|
||||||
<div class="ml-4">
|
|
||||||
<div class="font-medium text-gray-900"><span
|
|
||||||
x-text="beneficiary.data.first_name"></span> <span
|
|
||||||
x-text="beneficiary.data.last_name"></span></div>
|
|
||||||
<!-- <div class="text-gray-500" x-text="beneficiary.data.email"></div> -->
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-gray-500">
|
|
||||||
<div class="text-gray-900" x-text="beneficiary.data.phone_number"></div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-gray-500"
|
|
||||||
x-text="beneficiary.data.email">
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-gray-500">
|
|
||||||
<template x-for="tag in beneficiary.data.tags">
|
|
||||||
<span
|
|
||||||
class="inline-flex rounded-full bg-green-100 px-2 text-xs font-semibold leading-5 text-green-800"
|
|
||||||
x-text="tag"></span>
|
|
||||||
</template>
|
|
||||||
</td>
|
|
||||||
<td
|
|
||||||
class="relative whitespace-nowrap py-4 pl-3 pr-4 text-right text-sm font-medium sm:pr-6">
|
|
||||||
<a :href="'/app/beneficiaries/' + beneficiary.id"
|
|
||||||
class="text-co-blue hover:text-co-blue">Voir<span class="sr-only">, <span
|
|
||||||
x-text="beneficiary.data.first_name"></span> <span
|
|
||||||
x-text="beneficiary.data.last_name"></span></a>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<!-- More people... -->
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
<div class="bg-white px-4 py-3 flex items-center justify-between border-t border-gray-200 sm:px-6">
|
|
||||||
<div class="flex-1 flex justify-between sm:hidden">
|
|
||||||
<a href="#" class="relative inline-flex items-center px-4 py-2 border border-gray-300 text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50"
|
|
||||||
@click="paginate(current)"> Previous </a>
|
|
||||||
<a href="#" class="ml-3 relative inline-flex items-center px-4 py-2 border border-gray-300 text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50"
|
|
||||||
@click="paginate(current+2)"> Next </a>
|
|
||||||
</div>
|
|
||||||
<div class="hidden sm:flex-1 sm:flex sm:items-center sm:justify-between">
|
|
||||||
<div>
|
|
||||||
<p class="text-sm text-gray-700">
|
|
||||||
Résultats
|
|
||||||
<span class="font-medium" x-text="Math.min((current * 10)+1, state.count)"></span>
|
|
||||||
à
|
|
||||||
<span class="font-medium" x-text="Math.min((current * 10)+10, state.count)"></span>
|
|
||||||
sur
|
|
||||||
<span class="font-medium" x-text="state.count"></span>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<nav class="relative z-0 inline-flex rounded-md shadow-sm -space-x-px" aria-label="Pagination">
|
|
||||||
<a href="#" class="relative inline-flex items-center px-2 py-2 rounded-l-md border border-gray-300 bg-white text-sm font-medium text-gray-500 hover:bg-gray-50"
|
|
||||||
@click="paginate(current)">
|
|
||||||
<span class="sr-only">Previous</span>
|
|
||||||
<!-- Heroicon name: solid/chevron-left -->
|
|
||||||
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd" d="M12.707 5.293a1 1 0 010 1.414L9.414 10l3.293 3.293a1 1 0 01-1.414 1.414l-4-4a1 1 0 010-1.414l4-4a1 1 0 011.414 0z" clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</a>
|
|
||||||
<template x-for="i in nb_pages">
|
|
||||||
<a href="#" @click="paginate(i)"
|
|
||||||
class="relative inline-flex items-center px-4 py-2 border text-sm font-medium"
|
|
||||||
:class="i == current+1 ? 'z-10 bg-indigo-50 border-co-blue text-co-blue' : 'bg-white border-gray-300 text-gray-500 hover:bg-gray-50'"
|
|
||||||
x-text="i"></a>
|
|
||||||
</template>
|
|
||||||
<!-- Current: "z-10 bg-indigo-50 border-indigo-500 text-indigo-600", Default: "bg-white border-gray-300 text-gray-500 hover:bg-gray-50" -->
|
|
||||||
<a href="#" class="relative inline-flex items-center px-2 py-2 rounded-r-md border border-gray-300 bg-white text-sm font-medium text-gray-500 hover:bg-gray-50"
|
|
||||||
@click="paginate(current+2)">
|
|
||||||
<span class="sr-only">Next</span>
|
|
||||||
<!-- Heroicon name: solid/chevron-right -->
|
|
||||||
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</a>
|
|
||||||
</nav>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
<div class="mt-8 max-w-3xl mx-auto grid grid-cols-1 gap-6 lg:max-w-7xl lg:grid-flow-col-dense lg:grid-cols-3">
|
|
||||||
<div class="space-y-6 lg:col-start-1 lg:col-span-1">
|
|
||||||
<div class="bg-white shadow sm:rounded-2xl">
|
|
||||||
<h2 id="timeline-title" class="text-lg font-medium text-gray-900 p-4 sm:px-6">Ajouter un bénéficiaire</h2>
|
|
||||||
<div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
<form method="GET" >
|
|
||||||
<div x-data="{
|
|
||||||
text: '{{if .ViewState.search}}{{.ViewState.search.beneficiary.Data.first_name}} {{.ViewState.search.beneficiary.Data.last_name}}{{end}}',
|
|
||||||
beneficiariesListOpen: false,
|
|
||||||
beneficiaries: {{json .ViewState.beneficiaries}},
|
|
||||||
filteredBeneficiaries: (text) => {
|
|
||||||
if(text=='') return beneficiaries
|
|
||||||
return this.beneficiaries.filter(b => b['data']['first_name'].includes(text) || b['data']['last_name'].includes(text))
|
|
||||||
},
|
|
||||||
fields: {
|
|
||||||
beneficiaryid: {{if .ViewState.search}}'{{.ViewState.search.beneficiary.ID}}'{{else}}null{{end}},
|
|
||||||
},
|
|
||||||
selectbeneficiary(beneficiary) {
|
|
||||||
console.log(beneficiary)
|
|
||||||
this.fields.beneficiaryid = beneficiary.id
|
|
||||||
this.text = beneficiary.data.first_name + ' ' + beneficiary.data.last_name
|
|
||||||
this.beneficiariesListOpen = false
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<input type="hidden" name="beneficiaryid" x-model="fields.beneficiaryid">
|
|
||||||
<label for="combobox" class="block text-sm font-medium text-gray-700">Bénéficiaire</label>
|
|
||||||
<div class="relative mt-1 mb-4">
|
|
||||||
<input @focus="beneficiariesListOpen = true" x-model="text" id="combobox" type="text" class="w-full rounded-2xl border border-gray-300 bg-white py-2 pl-3 pr-12 shadow-sm focus:border-co-blue focus:outline-none focus:ring-1 focus:ring-co-blue sm:text-sm" role="combobox" aria-controls="options" aria-expanded="false">
|
|
||||||
|
|
||||||
<button @click="beneficiariesListOpen = ! beneficiariesListOpen" type="button" class="absolute inset-y-0 right-0 flex items-center rounded-r-2xl px-2 focus:outline-none">
|
|
||||||
<!-- Heroicon name: solid/selector -->
|
|
||||||
<svg class="h-5 w-5 text-gray-400" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd" d="M10 3a1 1 0 01.707.293l3 3a1 1 0 01-1.414 1.414L10 5.414 7.707 7.707a1 1 0 01-1.414-1.414l3-3A1 1 0 0110 3zm-3.707 9.293a1 1 0 011.414 0L10 14.586l2.293-2.293a1 1 0 011.414 1.414l-3 3a1 1 0 01-1.414 0l-3-3a1 1 0 010-1.414z" clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<ul x-show="beneficiariesListOpen" class="absolute z-10 mt-1 max-h-60 w-full overflow-auto rounded-xl bg-white py-1 text-base shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none sm:text-sm" id="options" role="listbox">
|
|
||||||
<!--
|
|
||||||
Combobox option, manage highlight styles based on mouseenter/mouseleave and keyboard navigation.
|
|
||||||
|
|
||||||
Active: "text-white bg-indigo-600", Not Active: "text-gray-900"
|
|
||||||
-->
|
|
||||||
<template x-for="beneficiary in beneficiaries">
|
|
||||||
<li @click="selectbeneficiary(beneficiary)" class="relative cursor-default hover:bg-gray-100 select-none py-2 pl-3 pr-9 text-gray-900" id="option-0" role="option" tabindex="-1">
|
|
||||||
<!-- Selected: "font-semibold" -->
|
|
||||||
<span class="truncate" x-text="beneficiary.data.first_name"></span> <span class="truncate" x-text="beneficiary.data.last_name"></span>
|
|
||||||
|
|
||||||
<!--
|
|
||||||
Checkmark, only display for selected option.
|
|
||||||
|
|
||||||
Active: "text-white", Not Active: "text-indigo-600"
|
|
||||||
-->
|
|
||||||
<span class="absolute inset-y-0 right-0 flex items-center pr-4 text-co-blue">
|
|
||||||
<!-- Heroicon name: solid/check -->
|
|
||||||
<!-- <svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clip-rule="evenodd" />
|
|
||||||
</svg> -->
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<!-- More items... -->
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<!-- <a href="/app/group_module/groups/{{.ViewState.group.ID}}"> -->
|
|
||||||
<button type="submit"
|
|
||||||
class="rounded-2xl border border-transparent bg-co-blue px-4 py-2 my-4 mt-8 w-full text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
Ajouter
|
|
||||||
</button>
|
|
||||||
<!-- </a> -->
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{{end}}
|
|
||||||
@@ -1,60 +0,0 @@
|
|||||||
|
|
||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mt-10 mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h2 class="text-xl font-semibold text-gray-500">Gestion des groupes</h2>
|
|
||||||
|
|
||||||
<div class="sm:flex sm:items-center">
|
|
||||||
<div class="sm:flex-auto">
|
|
||||||
<p class="mt-2 text-sm text-gray-700"></p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-4 sm:mt-0 sm:ml-16 sm:flex-none">
|
|
||||||
<a href="/app/group_module/groups">
|
|
||||||
<button type="button"
|
|
||||||
class="inline-flex items-center justify-center rounded-2xl border border-transparent bg-co-blue px-4 py-2 text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
{{$.IconSet.Icon "hero:outline/plus-circle" "h-5 w-5 mr-3"}}
|
|
||||||
Ajouter un groupe
|
|
||||||
</button>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="bg-white shadow overflow-hidden sm:rounded-3xl mt-4">
|
|
||||||
<ul role="list" class="divide-y divide-gray-200">
|
|
||||||
{{range .ViewState.groups}}
|
|
||||||
<li>
|
|
||||||
<a href="/app/group_module/groups/{{.ID}}" class="block hover:bg-gray-50">
|
|
||||||
<div class="px-4 py-4 flex items-center sm:px-6">
|
|
||||||
<div class="min-w-0 flex-1 sm:flex sm:items-center sm:justify-between">
|
|
||||||
<div class="truncate">
|
|
||||||
<div class="flex text-sm">
|
|
||||||
<p class="font-medium text-lg text-co-blue truncate">{{.Data.name}}</p>
|
|
||||||
<p class="ml-1 flex-shrink-0 font-normal text-gray-500"></p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-2 flex">
|
|
||||||
<div class="flex items-center text-sm text-gray-500">
|
|
||||||
{{$.IconSet.Icon "hero:outline/user-group" "flex-shrink-0 mr-1.5 h-5 w-5"}}
|
|
||||||
<p>
|
|
||||||
{{ len .Members }} bénéficiaires
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="mt-4 flex-shrink-0 sm:mt-0 sm:ml-5">
|
|
||||||
<div class="flex overflow-hidden -space-x-1">
|
|
||||||
<!-- <img class="inline-block h-6 w-6 rounded-full ring-2 ring-white"
|
|
||||||
src="http://localhost:9000/app/beneficiaries/e7616eac-4a87-4396-a505-23e0421b9c4c/picture"
|
|
||||||
alt="Dries Vincent"> -->
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="ml-5 flex-shrink-0">
|
|
||||||
{{$.IconSet.Icon "hero:solid/chevron-right" "h-5 w-5 text-gray-400"}}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
</li>
|
|
||||||
{{end}}
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,103 +0,0 @@
|
|||||||
{{define "journeys_all"}}
|
|
||||||
<div>
|
|
||||||
{{ if gt (len .ViewState.carpools) 0}}
|
|
||||||
{{$carpool := (index .ViewState.carpools 0)}}
|
|
||||||
<div class="px-4 pt-4 flex text-sm text-grey-900 font-bold">
|
|
||||||
<div class="flex-1">
|
|
||||||
{{.IconSet.Icon "tabler-icons:car" "h-6 w-6 inline-flex mr-4"}} Covoiturage
|
|
||||||
{{if $carpool.days.monday}}
|
|
||||||
entre {{$carpool.outward.monday.mintime}} et {{$carpool.outward.monday.maxtime}}
|
|
||||||
{{else if $carpool.days.tuesday}}
|
|
||||||
entre {{$carpool.outward.tuesday.mintime}} et {{$carpool.outward.tuesday.maxtime}}
|
|
||||||
{{else if $carpool.days.wednesday}}
|
|
||||||
entre {{$carpool.outward.wednesday.mintime}} et {{$carpool.outward.wednesday.maxtime}}
|
|
||||||
{{else if $carpool.days.thursday}}
|
|
||||||
entre {{$carpool.outward.thursday.mintime}} et {{$carpool.outward.thursday.maxtime}}
|
|
||||||
{{else if $carpool.days.friday}}
|
|
||||||
entre {{$carpool.outward.friday.mintime}} et {{$carpool.outward.friday.maxtime}}
|
|
||||||
{{else if $carpool.days.saturday}}
|
|
||||||
entre {{$carpool.outward.saturday.mintime}} et {{$carpool.outward.saturday.maxtime}}
|
|
||||||
{{else if $carpool.days.sunday}}
|
|
||||||
entre {{$carpool.outward.sunday.mintime}} et {{$carpool.outward.sunday.maxtime}}
|
|
||||||
{{end}}
|
|
||||||
(Temps trajet : {{divideFloat64 $carpool.duration 60.0 | printf "%.0f"}} Minutes)
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<span class="ml-2 rounded-xl px-2 py-1 bg-co-blue flex items-center justify-center ring-8 ring-white text-sm text-white whitespace-nowrap">
|
|
||||||
RIDYGO
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex items-center justify-center text-sm my-4">
|
|
||||||
<span class="ml-2 mt-1">
|
|
||||||
{{$carpool.from.address}}, {{$carpool.from.city}}
|
|
||||||
</span>
|
|
||||||
{{$.IconSet.Icon "hero:outline/chevron-right" "h-3 w-3 stroke-gray-800 m-2"}}
|
|
||||||
<span class="ml-2 mt-1">
|
|
||||||
{{$carpool.to.address}}, {{$carpool.to.city}}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="p-4 pb-8 flex items-center justify-center">
|
|
||||||
<span class="text-xs text-md">Conducteur : </span>
|
|
||||||
<span class="ml-2 mt-1 h-5 rounded-xl bg-gray-200 flex items-center justify-center ring-8 ring-white text-black p-2 text-sm">
|
|
||||||
{{$carpool.driver.alias}}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="p-4 text-center">
|
|
||||||
<button class="rounded-xl text-md px-4 py-1 bg-gray-200 text-co-blue" @click="tab = 'carpool'">{{ len .ViewState.carpools}} solutions en covoiturage : les voir toutes</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
{{end}}
|
|
||||||
|
|
||||||
|
|
||||||
<div class="px-4 pt-4 flex text-sm text-grey-900 font-bold border-t-2">
|
|
||||||
<div class="flex-1">
|
|
||||||
{{.IconSet.Icon "tabler-icons:bus" "h-6 w-6 inline-flex mr-4"}}
|
|
||||||
{{(timeFrom (index .ViewState.journeys.Journeys 0).Departure).Format "15:04"}} - {{(timeFrom (index .ViewState.journeys.Journeys 0).Arrival).Format "15:04"}}
|
|
||||||
({{(index .ViewState.journeys.Journeys 0).Duration.Minutes | printf "%.0f"}} Minutes)
|
|
||||||
</div>
|
|
||||||
<div></div>
|
|
||||||
</div>
|
|
||||||
<div class="p-4 pb-8 flex">
|
|
||||||
{{$firstwalk := true}}
|
|
||||||
{{range (index .ViewState.journeys.Journeys 0).Sections}}
|
|
||||||
{{if eq .Type "street_network"}}
|
|
||||||
<span class="ml-2 mt-1 h-5 w-5 rounded-co bg-gray-200 flex items-center justify-center ring-8 ring-white text-white">
|
|
||||||
{{$.IconSet.Icon "tabler-icons:walk" "h-4 w-4 stroke-gray-800"}}
|
|
||||||
</span>
|
|
||||||
{{if $firstwalk}}
|
|
||||||
{{$firstwalk = false}}
|
|
||||||
{{$.IconSet.Icon "hero:outline/chevron-right" "h-3 w-3 stroke-gray-800 m-2"}}
|
|
||||||
{{end}}
|
|
||||||
{{end}}
|
|
||||||
{{if eq .Type "public_transport"}}
|
|
||||||
<span class="ml-2 rounded-xl px-2 py-1 bg-co-blue flex items-center justify-center ring-8 ring-white text-sm text-white whitespace-nowrap">
|
|
||||||
{{if eq .Display.Network "Antibes"}}Envibus{{else}}{{.Display.Network}}{{end}} Ligne {{.Display.Label}}
|
|
||||||
</span>
|
|
||||||
{{$.IconSet.Icon "hero:outline/chevron-right" "h-3 w-3 stroke-gray-800 m-2"}}
|
|
||||||
{{end}}
|
|
||||||
{{end}}
|
|
||||||
<div class="flex-1"></div>
|
|
||||||
<button class="text-sm px-2 py-1 bg-gray-200 text-co-blue rounded-xl" @click="tab = 'public-transit'">Voir le détail</button>
|
|
||||||
</div>
|
|
||||||
<div class="p-4 text-center">
|
|
||||||
<button class="rounded-xl text-md px-4 py-1 bg-gray-200 text-co-blue" @click="tab = 'public-transit'">{{ len .ViewState.journeys.Journeys}} solutions en transports en commun : les voir toutes</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!--VEHICLES-->
|
|
||||||
<div class="px-4 pt-16 flex text-sm text-grey-900 border-t-2">
|
|
||||||
<div class="flex-1">
|
|
||||||
{{.IconSet.Icon "tabler-icons:car" "h-6 w-6 inline-flex mr-4"}}
|
|
||||||
<span class=" font-bold">{{len .ViewState.vehicles}} véhicules</span> partagés disponibles ce jour là et la semaine suivante
|
|
||||||
</div>
|
|
||||||
<div></div>
|
|
||||||
</div>
|
|
||||||
<div class="p-4 text-center">
|
|
||||||
<a href="/app/vehicles/"><button class="text-md px-4 py-1 bg-gray-200 text-co-blue rounded-xl">Réserver un véhicule</button></a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,52 +0,0 @@
|
|||||||
{{define "journeys_carpool"}}
|
|
||||||
|
|
||||||
{{$first := true}}
|
|
||||||
{{range .ViewState.carpools}}
|
|
||||||
{{if $first}}
|
|
||||||
{{$first = false}}
|
|
||||||
<div class="p-4 pb-8">
|
|
||||||
{{else}}
|
|
||||||
<div class="p-4 border-t-2 pb-8">
|
|
||||||
{{end}}
|
|
||||||
<div class="flex text-sm text-grey-900 font-bold">
|
|
||||||
{{if .days.monday}}
|
|
||||||
<div class="flex-1">Départ entre {{.outward.monday.mintime}} et {{.outward.monday.maxtime}}</div>
|
|
||||||
{{else if .days.tuesday}}
|
|
||||||
<div class="flex-1">Départ entre {{.outward.tuesday.mintime}} et {{.outward.tuesday.maxtime}}</div>
|
|
||||||
{{else if .days.wednesday}}
|
|
||||||
<div class="flex-1">Départ entre {{.outward.wednesday.mintime}} et {{.outward.wednesday.maxtime}}</div>
|
|
||||||
{{else if .days.thursday}}
|
|
||||||
<div class="flex-1">Départ entre {{.outward.thursday.mintime}} et {{.outward.thursday.maxtime}}</div>
|
|
||||||
{{else if .days.friday}}
|
|
||||||
<div class="flex-1">Départ entre {{.outward.friday.mintime}} et {{.outward.friday.maxtime}}</div>
|
|
||||||
{{else if .days.saturday}}
|
|
||||||
<div class="flex-1">Départ entre {{.outward.saturday.mintime}} et {{.outward.saturday.maxtime}}</div>
|
|
||||||
{{else if .days.sunday}}
|
|
||||||
<div class="flex-1">Départ entre {{.outward.sunday.mintime}} et {{.outward.sunday.maxtime}}</div>
|
|
||||||
{{end}}
|
|
||||||
<div>{{divideFloat64 .duration 60.0 | printf "%.0f"}} Minutes</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex items-center justify-center text-sm my-4">
|
|
||||||
<span class="ml-2 mt-1">
|
|
||||||
{{.from.address}}, {{.from.city}}
|
|
||||||
</span>
|
|
||||||
{{$.IconSet.Icon "hero:outline/chevron-right" "h-3 w-3 stroke-gray-800 m-2"}}
|
|
||||||
<span class="ml-2 mt-1">
|
|
||||||
{{.to.address}}, {{.to.city}}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="p-4 pb-8 flex items-center justify-center">
|
|
||||||
<span class="text-xs text-md">Avec </span>
|
|
||||||
<span class="ml-2 mt-1 h-5 rounded-xl bg-gray-200 flex items-center justify-center ring-8 ring-white text-black p-2 text-sm">
|
|
||||||
{{.driver.alias}}
|
|
||||||
</span>
|
|
||||||
|
|
||||||
<span class="text-xs text-md"> sur l'application </span>
|
|
||||||
<span class="ml-2 rounded-xl px-2 py-1 bg-co-blue flex items-center justify-center ring-8 ring-white text-sm text-white whitespace-nowrap">
|
|
||||||
RIDYGO
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{end}}
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
{{define "journeys_others"}}
|
|
||||||
|
|
||||||
<!--VEHICLES-->
|
|
||||||
<div class="p-4 flex text-sm text-grey-900">
|
|
||||||
<div class="flex-1">
|
|
||||||
{{.IconSet.Icon "tabler-icons:car" "h-6 w-6 inline-flex mr-4"}}
|
|
||||||
<span class=" font-bold">{{len .ViewState.vehicles}} véhicules</span> partagés disponibles ce jour là et la semaine suivante
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="p-2">
|
|
||||||
<table class="min-w-full divide-y divide-gray-300">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th scope="col" class="py-3.5 pl-4 pr-3 text-left text-sm font-semibold text-gray-900 sm:pl-6 md:pl-0">Véhicule</th>
|
|
||||||
<th scope="col" class="py-3.5 px-3 text-left text-sm font-semibold text-gray-900">Numéro</th>
|
|
||||||
<th scope="col" class="py-3.5 px-3 text-left text-sm font-semibold text-gray-900">Gestionnaire</th>
|
|
||||||
<th scope="col" class="py-3.5 px-3 text-left text-sm font-semibold text-gray-900">Lieu</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="divide-y divide-gray-200">
|
|
||||||
{{range .ViewState.vehicles}}
|
|
||||||
<tr>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm font-medium text-gray-900 sm:pl-6 md:pl-0">{{.Data.name}}</td>
|
|
||||||
<td class="whitespace-nowrap py-4 px-3 text-sm text-gray-500">{{.Data.licence_plate}}</td>
|
|
||||||
<td class="whitespace-nowrap py-4 px-3 text-sm text-gray-500">COOPGO</td>
|
|
||||||
<td class="whitespace-nowrap py-4 px-3 text-sm text-gray-500">{{if .Data.address}}{{.Data.address.properties.label}}{{end}}</td>
|
|
||||||
</tr>
|
|
||||||
{{end}}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
<div class="p-4 text-center">
|
|
||||||
<a href="/app/vehicles/"><button class="text-md px-4 py-1 bg-gray-200 text-co-blue rounded-xl">Réserver un véhicule</button></a>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{{end}}
|
|
||||||
@@ -1,77 +0,0 @@
|
|||||||
{{define "journeys_public_transit"}}
|
|
||||||
|
|
||||||
{{$first := true}}
|
|
||||||
{{range .ViewState.journeys.Journeys}}
|
|
||||||
{{if $first}}
|
|
||||||
{{$first = false}}
|
|
||||||
<div class="p-4 pb-8">
|
|
||||||
{{else}}
|
|
||||||
<div class="p-4 border-t-2 pb-8">
|
|
||||||
{{end}}
|
|
||||||
<div class="flex text-md text-grey-900 font-bold">
|
|
||||||
<div class="flex-1">{{(timeFrom .Departure).Format "15:04"}} - {{(timeFrom .Arrival).Format "15:04"}}</div>
|
|
||||||
<div>{{.Duration.Minutes | printf "%.0f"}} Minutes</div>
|
|
||||||
</div>
|
|
||||||
<div class="flow-root">
|
|
||||||
|
|
||||||
<ul role="list" class="-mb-8">
|
|
||||||
{{$firstwalk := true}}
|
|
||||||
{{range .Sections}}
|
|
||||||
{{if eq .Type "street_network"}}
|
|
||||||
<li>
|
|
||||||
<div class="relative py-4">
|
|
||||||
{{if $firstwalk}}
|
|
||||||
{{$firstwalk = false}}
|
|
||||||
<span class="absolute top-4 left-4 -ml-px h-full w-0.5 bg-gray-200" aria-hidden="true"></span>
|
|
||||||
{{end}}
|
|
||||||
<div class="relative flex space-x-3">
|
|
||||||
<div>
|
|
||||||
<span class="ml-2 h-4 w-4 rounded-co bg-gray-200 flex items-center justify-center ring-8 ring-white text-white">
|
|
||||||
{{$.IconSet.Icon "tabler-icons:walk" "h-3 w-3 stroke-gray-800"}}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div class="flex min-w-0 flex-1 justify-between space-x-4 pt-1.5">
|
|
||||||
<div>
|
|
||||||
<p class="text-xs text-gray-500">Marcher <a href="#" class="font-medium text-gray-900">{{walkingLength .}}m</a></p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
{{end}}
|
|
||||||
{{if eq .Type "public_transport"}}
|
|
||||||
<li>
|
|
||||||
<div class="relative py-4">
|
|
||||||
<span class="absolute top-4 left-4 -ml-px h-full w-0.5 bg-gray-200" aria-hidden="true"></span>
|
|
||||||
<div class="relative flex space-x-3">
|
|
||||||
<div>
|
|
||||||
<span class="h-8 w-8 rounded-co bg-co-blue flex items-center justify-center ring-8 ring-white">
|
|
||||||
{{$.IconSet.Icon "tabler-icons:bus" "h-5 w-5 stroke-white"}}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div class="flex min-w-0 flex-1 justify-between space-x-4 pt-1.5">
|
|
||||||
<div>
|
|
||||||
<p class="text-md text-gray-500">{{if eq .Display.Network "Antibes"}}Envibus{{else}}{{.Display.Network}}{{end}} <a href="#" class="font-medium text-gray-900">Ligne {{.Display.Label}}</a></p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="ml-16 pt-2">
|
|
||||||
<div>
|
|
||||||
<p class="text-sm text-gray-500">Départ <a href="#" class="font-medium text-gray-900">{{(timeFrom .Departure).Format "15:04"}}</a> - Arrivée <a href="#" class="font-medium text-gray-900">{{(timeFrom .Arrival).Format "15:04"}}</a></p>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<p class="text-sm text-gray-500">De <a href="#" class="font-medium text-gray-900">{{.From.Name}}</a> à <a href="#" class="font-medium text-gray-900">{{.To.Name}}</a></p>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<p class="text-sm text-gray-500">Direction <a href="#" class="font-medium text-gray-900">{{.Display.Direction}}</a></p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
{{end}}
|
|
||||||
{{end}}
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{end}}
|
|
||||||
@@ -1,120 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Déplacements</h1>
|
|
||||||
|
|
||||||
|
|
||||||
<div class="mt-8 max-w-3xl mx-auto grid grid-cols-1 gap-6 lg:max-w-7xl lg:grid-flow-col-dense lg:grid-cols-3">
|
|
||||||
<div class="space-y-6 lg:col-start-1 lg:col-span-1">
|
|
||||||
<div class="bg-white shadow sm:rounded-2xl">
|
|
||||||
<h2 id="timeline-title" class="text-lg font-medium text-gray-900 p-4 sm:px-6">Chercher une solution</h2>
|
|
||||||
<div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
<form method="GET">
|
|
||||||
|
|
||||||
{{ $departureField := "departure" }}
|
|
||||||
{{ $departureLabel := "Départ" }}
|
|
||||||
{{ $departure := .ViewState.departure }}
|
|
||||||
{{ template "address_autocomplete" dict "FieldName" $departureField "FieldLabel" $departureLabel "Address" $departure }}
|
|
||||||
|
|
||||||
{{ $destinationField := "destination" }}
|
|
||||||
{{ $destinationLabel := "Destination" }}
|
|
||||||
{{ $destination := .ViewState.destination }}
|
|
||||||
{{ template "address_autocomplete" dict "FieldName" $destinationField "FieldLabel" $destinationLabel "Address" $destination }}
|
|
||||||
|
|
||||||
|
|
||||||
<div class="py-4 grid grid-cols-2">
|
|
||||||
<div class="lg:col-span-1">
|
|
||||||
<label for="departuredate" class="block text-sm font-medium text-gray-700">Le</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<input type="date" id="departuredate" name="departuredate" value="{{.ViewState.departuredate}}"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block w-full sm:text-sm border-gray-300 rounded-l-2xl border-r-1">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="lg:col-span-1">
|
|
||||||
<label for="departuretime" class="block text-sm font-medium text-gray-700">A</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<input type="time" id="departuretime" name="departuretime" value="{{.ViewState.departuretime}}"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block w-full sm:text-sm border-gray-300 rounded-r-2xl border-l-0">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<button type="submit"
|
|
||||||
class="rounded-2xl border border-transparent bg-co-blue px-4 py-2 my-4 mt-8 w-full text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
Chercher
|
|
||||||
</button>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="lg:col-start-2 lg:col-span-2">
|
|
||||||
{{if .ViewState.searched}}
|
|
||||||
<section aria-labelledby="results-title" x-data="{
|
|
||||||
tab: 'all',
|
|
||||||
to(event) {
|
|
||||||
this.tab = event.target.value
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<div class="bg-white shadow sm:rounded-lg sm:overflow-hidden">
|
|
||||||
<div class="divide-y divide-gray-200">
|
|
||||||
<div>
|
|
||||||
<div class="sm:hidden">
|
|
||||||
<label for="tabs" class="sr-only">Select a tab</label>
|
|
||||||
<select id="tabs" name="tabs" @change="to"
|
|
||||||
class="block w-full pl-3 pr-10 py-2 text-base border-gray-300 focus:outline-none focus:ring-indigo-500 focus:border-indigo-500 sm:text-sm rounded-md">
|
|
||||||
<option value="all">Tous modes</option>
|
|
||||||
|
|
||||||
<option value="carpool">Covoiturage</option>
|
|
||||||
|
|
||||||
<option value="public-transit">Transports</option>
|
|
||||||
|
|
||||||
<!-- <option value="active-modes">Modes actifs</option> -->
|
|
||||||
|
|
||||||
<option value="others">Autres</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
<div class="hidden sm:block">
|
|
||||||
<div class="border-b border-gray-200 pl-4">
|
|
||||||
<nav class="-mb-px flex space-x-8" aria-label="Tabs">
|
|
||||||
<!-- Current: "border-indigo-500 text-indigo-600", Default: "border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300" -->
|
|
||||||
<a href="#" @click="tab = 'all'"
|
|
||||||
class="whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"
|
|
||||||
:class="tab == 'all' ? 'border-co-blue text-co-blue' : 'border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300'">
|
|
||||||
Tous modes </a>
|
|
||||||
|
|
||||||
<a href="#" @click="tab = 'carpool'"
|
|
||||||
class="whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"
|
|
||||||
:class="tab == 'carpool' ? 'border-co-blue text-co-blue' : 'border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300'">
|
|
||||||
Covoiturage </a>
|
|
||||||
|
|
||||||
<a href="#" @click="tab = 'public-transit'"
|
|
||||||
class="whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"
|
|
||||||
:class="tab == 'public-transit' ? 'border-co-blue text-co-blue' : 'border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300'">
|
|
||||||
Transports </a>
|
|
||||||
|
|
||||||
<!-- <a href="#" @click="tab = 'active-modes'"
|
|
||||||
class="whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"
|
|
||||||
:class="tab == 'active-modes' ? 'border-co-blue text-co-blue' : 'border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300'">
|
|
||||||
Modes actifs </a> -->
|
|
||||||
|
|
||||||
<a href="#" @click="tab = 'others'"
|
|
||||||
class="whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"
|
|
||||||
:class="tab == 'others' ? 'border-co-blue text-co-blue' : 'border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300'">
|
|
||||||
Autres </a>
|
|
||||||
</nav>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div x-show="tab == 'all'">{{template "journeys_all" .}}</div>
|
|
||||||
<div x-show="tab == 'carpool'">{{template "journeys_carpool" .}}</div>
|
|
||||||
<div x-show="tab == 'public-transit'">{{template "journeys_public_transit" .}}</div>
|
|
||||||
<div x-show="tab == 'others'">{{template "journeys_others" .}}</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
{{end}}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,178 +0,0 @@
|
|||||||
{{define "main"}}
|
|
||||||
<!DOCTYPE html>
|
|
||||||
<html class="h-full bg-gray-100">
|
|
||||||
|
|
||||||
<head>
|
|
||||||
<title>PARCOURSMOB</title>
|
|
||||||
<link rel="stylesheet" href="/public/css/main.css" />
|
|
||||||
<!-- <script defer type="text/javascript" src="/public/js/main.js" defer></script> -->
|
|
||||||
<script src="https://cdn.jsdelivr.net/npm/@kingshott/iodine@8.1.0/dist/iodine.min.umd.js" defer></script>
|
|
||||||
<script defer src="https://unpkg.com/alpinejs@3.x.x/dist/cdn.min.js" defer></script>
|
|
||||||
</head>
|
|
||||||
|
|
||||||
<body class="h-full" x-data="{ offCanvasMenu: false }">
|
|
||||||
<div class="relative z-40 md:hidden" role="dialog" aria-modal="true">
|
|
||||||
|
|
||||||
<div class="fixed inset-0 bg-gray-600 bg-opacity-75" x-show="offCanvasMenu"
|
|
||||||
x-transition:enter="transition-opacity ease-linear duration-300" x-transition:enter-start="opacity-0"
|
|
||||||
x-transition:enter-end="opacity-100" x-transition:leave="transition-opacity ease-linear duration-300"
|
|
||||||
x-transition:leave-start="opacity-100" x-transition:leave-end="opacity-0"></div>
|
|
||||||
|
|
||||||
<div class="fixed inset-0 flex z-40" x-show="offCanvasMenu"
|
|
||||||
x-transition:enter="transition ease-in-out duration-300 transform" x-transition:enter-start="-translate-x-full"
|
|
||||||
x-transition:enter-end="translate-x-0" x-transition:leave="transition ease-in-out duration-300 transform"
|
|
||||||
x-transition:leave-start="translate-x-0" x-transition:leave-end="-translate-x-full">
|
|
||||||
<div class="relative flex-1 flex flex-col max-w-xs w-full pt-5 pb-4 bg-co-blue">
|
|
||||||
<div class="absolute top-0 right-0 -mr-12 pt-2" @click="offCanvasMenu = false">
|
|
||||||
<button type="button"
|
|
||||||
class="ml-1 flex items-center justify-center h-10 w-10 rounded-full focus:outline-none focus:ring-2 focus:ring-inset focus:ring-white">
|
|
||||||
<span class="sr-only">Close sidebar</span>
|
|
||||||
{{ .IconSet.Icon "hero:outline/x" "h-6 w-6" }}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex-shrink-0 flex items-center px-4">
|
|
||||||
<img class="h-8 w-auto" src="/public/images/parcoursmob_logo_whitered.svg" alt="PARCOURSMOB">
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 flex-1 h-0 overflow-y-auto">
|
|
||||||
|
|
||||||
{{ template "mainmenu" . }}
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex-shrink-0 w-14" aria-hidden="true">
|
|
||||||
<!-- Dummy element to force sidebar to shrink to fit close icon -->
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Static sidebar for desktop -->
|
|
||||||
<div class="hidden md:flex md:w-64 md:flex-col md:fixed md:inset-y-0">
|
|
||||||
<!-- Sidebar component, swap this element with another sidebar if you like -->
|
|
||||||
<div class="flex flex-col flex-grow pt-5 bg-co-blue overflow-y-auto">
|
|
||||||
<div class="flex items-center flex-shrink-0 px-4">
|
|
||||||
<img class="h-8 w-auto" src="/public/images/parcoursmob_logo_whitered.svg" alt="PARCOURSMOB">
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 flex-1 flex flex-col">
|
|
||||||
{{ template "mainmenu" . }}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="md:pl-64 flex flex-col flex-1">
|
|
||||||
<div class="sticky top-0 z-10 flex-shrink-0 flex h-16 bg-white shadow">
|
|
||||||
<button @click="offCanvasMenu = true" type="button"
|
|
||||||
class="px-4 border-r border-gray-200 text-gray-500 focus:outline-none focus:ring-2 focus:ring-inset focus:ring-co-blue md:hidden">
|
|
||||||
<span class="sr-only">Open sidebar</span>
|
|
||||||
<!-- Heroicon name: outline/menu-alt-2 -->
|
|
||||||
<svg class="h-6 w-6" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="2"
|
|
||||||
stroke="currentColor" aria-hidden="true">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M4 6h16M4 12h16M4 18h7" />
|
|
||||||
</svg>
|
|
||||||
</button>
|
|
||||||
<div class="flex-1 px-4 flex justify-between">
|
|
||||||
<div class="flex-1 flex">
|
|
||||||
<form class="w-full flex md:ml-0" action="/app/beneficiaries/" method="GET">
|
|
||||||
<label for="search-field" class="sr-only">Search</label>
|
|
||||||
<div class="relative w-full text-gray-400 focus-within:text-gray-600">
|
|
||||||
<div class="absolute inset-y-0 left-0 flex items-center pointer-events-none">
|
|
||||||
<!-- Heroicon name: solid/search -->
|
|
||||||
{{$.IconSet.Icon "hero:solid/search" "h5 w-5"}}
|
|
||||||
</div>
|
|
||||||
<input id="search-field"
|
|
||||||
class="block w-full h-full pl-8 pr-3 py-2 border-transparent text-gray-900 placeholder-gray-500 focus:outline-none focus:placeholder-gray-400 focus:ring-0 focus:border-transparent sm:text-sm"
|
|
||||||
placeholder="Chercher un bénéficiaire" type="search" name="search">
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
<div class="ml-4 flex items-center md:ml-6">
|
|
||||||
|
|
||||||
|
|
||||||
<a href="/app/administration/">
|
|
||||||
{{if and .AdministrationState.Display .AdministrationState.Active}}
|
|
||||||
<button
|
|
||||||
class="max-w-xs bg-co-blue px-4 py-2 text-white flex items-center text-sm rounded-2xl focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">
|
|
||||||
{{.IconSet.Icon "hero:outline/cog" "h-6 w-6"}} Administration
|
|
||||||
</button>
|
|
||||||
{{else if and .AdministrationState.Display}}
|
|
||||||
<button
|
|
||||||
class="max-w-xs bg-white hover:bg-gray-50 border-gray-300 border px-4 py-2 text-gray-700 flex items-center text-sm rounded-2xl focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">
|
|
||||||
{{.IconSet.Icon "hero:outline/cog" "h-6 w-6"}} Administration
|
|
||||||
</button>
|
|
||||||
{{end}}
|
|
||||||
</a>
|
|
||||||
<div class="ml-3 relative" x-data="{ groupMenuOpen: false }">
|
|
||||||
<div>
|
|
||||||
<button @click="groupMenuOpen = ! groupMenuOpen" type="button"
|
|
||||||
class="max-w-xs bg-white hover:bg-gray-50 border-gray-300 border px-4 py-2 text-gray-700 flex items-center text-sm rounded-2xl focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">
|
|
||||||
{{.Group.Data.name}}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="origin-top-right absolute right-0 mt-2 w-48 rounded-md shadow-lg py-1 bg-white ring-1 ring-black ring-opacity-5 focus:outline-none"
|
|
||||||
role="menu" aria-orientation="vertical" aria-labelledby="group-menu-button" tabindex="-1"
|
|
||||||
x-show="groupMenuOpen" x-transition:enter="transition ease-out duration-100"
|
|
||||||
x-transition:enter-start="transform opacity-0 scale-95"
|
|
||||||
x-transition:enter-end="transform opacity-100 scale-100"
|
|
||||||
x-transition:leave="transition ease-in duration-75"
|
|
||||||
x-transition:leave-start="transform opacity-100 scale-100"
|
|
||||||
x-transition:leave-end="transform opacity-0 scale-95">
|
|
||||||
<!-- Active: "bg-gray-100", Not Active: "" -->
|
|
||||||
<a href="/app/group/settings" class="block px-4 py-2 text-sm text-gray-700" role="menuitem" tabindex="-1"
|
|
||||||
id="user-menu-item-1">Paramètres</a>
|
|
||||||
|
|
||||||
<a href="/auth/groups/switch" class="block px-4 py-2 text-sm text-gray-700" role="menuitem" tabindex="-1"
|
|
||||||
id="user-menu-item-2">Changer d'organisation</a>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Profile dropdown -->
|
|
||||||
<div class="ml-3 relative" x-data="{ profileMenuOpen: false }">
|
|
||||||
<div>
|
|
||||||
<!-- <button @click="profileMenuOpen = ! profileMenuOpen" type="button" -->
|
|
||||||
<button type="button"
|
|
||||||
class="max-w-xs bg-white flex items-center text-sm rounded-co focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue"
|
|
||||||
id="user-menu-button" aria-expanded="false" aria-haspopup="true">
|
|
||||||
<span class="sr-only">Open user menu</span>
|
|
||||||
<img class="h-8 w-8 rounded-co"
|
|
||||||
src="http://localhost:9000/app/beneficiaries/e7616eac-4a87-4396-a505-23e0421b9c4c/picture"
|
|
||||||
alt="Menu utilisateur">
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div
|
|
||||||
class="origin-top-right absolute right-0 mt-2 w-48 rounded-md shadow-lg py-1 bg-white ring-1 ring-black ring-opacity-5 focus:outline-none"
|
|
||||||
role="menu" aria-orientation="vertical" aria-labelledby="user-menu-button" tabindex="-1"
|
|
||||||
x-show="profileMenuOpen" x-transition:enter="transition ease-out duration-100"
|
|
||||||
x-transition:enter-start="transform opacity-0 scale-95"
|
|
||||||
x-transition:enter-end="transform opacity-100 scale-100"
|
|
||||||
x-transition:leave="transition ease-in duration-75"
|
|
||||||
x-transition:leave-start="transform opacity-100 scale-100"
|
|
||||||
x-transition:leave-end="transform opacity-0 scale-95">
|
|
||||||
<!-- Active: "bg-gray-100", Not Active: "" -->
|
|
||||||
<a href="/app/profile" class="block px-4 py-2 text-sm text-gray-700" role="menuitem" tabindex="-1"
|
|
||||||
id="user-menu-item-0">Votre profil</a>
|
|
||||||
|
|
||||||
<a href="/app/settings/" class="block px-4 py-2 text-sm text-gray-700" role="menuitem" tabindex="-1"
|
|
||||||
id="user-menu-item-1">Paramètres</a>
|
|
||||||
|
|
||||||
<a href="/app/disconnect" class="block px-4 py-2 text-sm text-gray-700" role="menuitem" tabindex="-1"
|
|
||||||
id="user-menu-item-2">Se déconnecter</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<main>
|
|
||||||
<div class="py-6">
|
|
||||||
{{ template "content" . }}
|
|
||||||
</div>
|
|
||||||
</main>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</body>
|
|
||||||
|
|
||||||
</html>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
|
|
||||||
<div class="mx-auto max-w-7xl px-4 sm:px-6 lg:px-8 space-y-6">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Contact</h1>
|
|
||||||
<div class="bg-white py-2 px-4 shadow sm:rounded-lg sm:px-10">
|
|
||||||
<form action="" method="POST">
|
|
||||||
|
|
||||||
<div class="mb-4 w-full bg-gray-50 rounded-lg border border-gray-200 dark:bg-gray-700 dark:border-gray-600">
|
|
||||||
<div class="py-2 px-4 bg-white rounded-t-lg dark:bg-gray-800">
|
|
||||||
<label class="sr-only">Votre message</label>
|
|
||||||
|
|
||||||
<textarea name="comment" rows="4" class="block w-full resize-none border-0 border-b border-transparent p-0 pb-2 focus:border-indigo-600 focus:ring-0 sm:text-sm" placeholder="Votre message..." required></textarea>
|
|
||||||
</div>
|
|
||||||
<div class="flex justify-center items-center py-2 px-3 border-t dark:border-gray-600">
|
|
||||||
<button type="submit" value="send message" class="px-2 py-2 border border-transparent text-sm font-medium rounded-2xl shadow-sm text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">
|
|
||||||
Envoyer
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,152 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Réservation de véhicule</h1>
|
|
||||||
|
|
||||||
<div class="sm:flex sm:items-center">
|
|
||||||
<div class="sm:flex-auto">
|
|
||||||
<p class="mt-2 text-sm text-gray-700"></p>
|
|
||||||
</div>
|
|
||||||
<!-- <div class="mt-4 sm:mt-0 sm:ml-16 sm:flex-none">
|
|
||||||
<a href="/app/vehicles-management/fleet/add">
|
|
||||||
<button type="button"
|
|
||||||
class="inline-flex items-center justify-center rounded-2xl border border-transparent bg-co-red px-4 py-2 text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-red focus:ring-offset-2 sm:w-auto">
|
|
||||||
Annuler
|
|
||||||
</button>
|
|
||||||
</a>
|
|
||||||
</div> -->
|
|
||||||
</div>
|
|
||||||
<div class="mt-8 max-w-3xl mx-auto grid grid-cols-1 gap-6 lg:max-w-7xl lg:grid-flow-col-dense lg:grid-cols-3">
|
|
||||||
<div class="space-y-6 lg:col-start-1 lg:col-span-1">
|
|
||||||
<div class="bg-white shadow sm:rounded-2xl">
|
|
||||||
<h2 id="timeline-title" class="text-lg font-medium text-gray-900 p-4 sm:px-6">Bénéficiaire</h2>
|
|
||||||
<div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
<div>
|
|
||||||
<div class="mt-5 border-gray-200">
|
|
||||||
<dl class="sm:divide-y sm:divide-gray-200">
|
|
||||||
<div class="sm:pb-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Nom</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.beneficiary.Data.first_name}}
|
|
||||||
{{.ViewState.beneficiary.Data.last_name}}</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Email</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.beneficiary.Data.email}}</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Téléphone</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.beneficiary.Data.phone_number}}</dd>
|
|
||||||
</div>
|
|
||||||
{{if .ViewState.Data.birthdate}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Date de naissance</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">{{(timeFrom
|
|
||||||
.ViewState.Data.birthdate).Format
|
|
||||||
"02/01/2006"}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if and .ViewState.Data.gender (ne .ViewState.Data.gender "0")}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Date de naissance</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">{{genderISO5218
|
|
||||||
.ViewState.Data.gender}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.Data.address}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Adresse</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.Data.address.properties.label}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="lg:col-start-2 lg:col-span-2">
|
|
||||||
<div class="bg-white shadow sm:rounded-2xl sm:px-6">
|
|
||||||
<div class="bg-white px-4 py-5 border-b border-gray-200 sm:px-6">
|
|
||||||
<div class="-ml-4 -mt-4 flex justify-between items-center flex-wrap sm:flex-nowrap">
|
|
||||||
<div class="ml-4 mt-4">
|
|
||||||
<h3 class="text-lg leading-6 font-medium text-gray-900">Réservation</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Informations utiles sur la réservation.</p>
|
|
||||||
</div>
|
|
||||||
<div class="ml-4 mt-4 flex-shrink-0">
|
|
||||||
<button type="button"
|
|
||||||
class="relative inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-xs font-medium rounded-2xl text-co-blue bg-gray-100 hover:bg-co-blue hover:text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">SMS</button>
|
|
||||||
<button type="button"
|
|
||||||
class="relative inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-xs font-medium rounded-2xl text-co-blue bg-gray-100 hover:bg-co-blue hover:text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Email</button>
|
|
||||||
<!-- <button type="button"
|
|
||||||
class="relative inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-xs font-medium rounded-2xl text-co-blue bg-gray-100 hover:bg-co-blue hover:text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Imprimer</button> -->
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="px-4 py-5 sm:px-6">
|
|
||||||
<div>
|
|
||||||
<div class="mt-5 border-gray-200">
|
|
||||||
<dl class="sm:divide-y sm:divide-gray-200">
|
|
||||||
<div class="sm:pb-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Gestionnaire</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.group.Data.name}}
|
|
||||||
</dd>
|
|
||||||
</div>
|
|
||||||
{{if .ViewState.booking.Data.booked_by}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<!-- <div class="sm:pb-5 sm:grid sm:grid-cols-3 sm:gap-4"> -->
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Réservé par</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{if .ViewState.booking.Data.booked_by.user}}
|
|
||||||
<a href="/app/members/{{.ViewState.booking.Data.booked_by.user.id}}" class="flex inline">
|
|
||||||
<img class="h-5 w-5 rounded-co mr-1"
|
|
||||||
src="/app/members/{{.ViewState.booking.Data.booked_by.user.id}}/picture" alt="">
|
|
||||||
{{.ViewState.booking.Data.booked_by.user.display_name}}
|
|
||||||
</a>
|
|
||||||
{{end}}
|
|
||||||
</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Véhicule</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.vehicle.Data.name}}</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Immatriculation</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.vehicle.Data.licence_plate}}</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Type</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
Voiture</dd>
|
|
||||||
</div>
|
|
||||||
{{if .ViewState.vehicle.Data.address}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Lieu de récupération</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">{{.ViewState.vehicle.Data.address.properties.label}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Date de récupération</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">{{(timeFrom .ViewState.booking.Startdate).Format
|
|
||||||
"02/01/2006"}}</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Date de retour</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">{{(timeFrom .ViewState.booking.Enddate).Format
|
|
||||||
"02/01/2006"}}</dd>
|
|
||||||
</div>
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,68 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Réservations</h1>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<div class="mt-8 flex flex-col">
|
|
||||||
<div class="-my-2 -mx-4 overflow-x-auto sm:-mx-6 lg:-mx-8">
|
|
||||||
<div class="inline-block min-w-full py-2 align-middle md:px-6 lg:px-8">
|
|
||||||
<div class="overflow-hidden shadow ring-1 ring-black ring-opacity-5 md:rounded-lg">
|
|
||||||
<table class="min-w-full divide-y divide-gray-300">
|
|
||||||
<thead class="bg-gray-50">
|
|
||||||
<tr>
|
|
||||||
<!-- <th scope="col"
|
|
||||||
class="py-3.5 pl-4 pr-3 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Statut
|
|
||||||
</th> -->
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Type
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Beneficiaire
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Dates
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="relative py-3.5 pl-3 pr-4 sm:pr-6">
|
|
||||||
<span class="sr-only">Actions</span>
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="divide-y divide-gray-200 bg-white">
|
|
||||||
|
|
||||||
{{range .ViewState.bookings}}
|
|
||||||
<tr>
|
|
||||||
<!-- <td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >aa</div>
|
|
||||||
</td> -->
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >Voiture</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" ><img class="h-6 w-6 rounded-co"
|
|
||||||
src="/app/beneficiaries/{{.Driver}}/picture" alt=""></div>
|
|
||||||
</td>
|
|
||||||
<!-- <td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >aa</div>
|
|
||||||
</td> -->
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >Du {{(timeFrom .Startdate).Format "02/01/2006"}} au {{(timeFrom .Enddate).Format "02/01/2006"}}</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<a href="/app/vehicles/bookings/{{.ID}}"
|
|
||||||
class="text-co-blue hover:text-co-blue">Voir</a>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
{{end}}
|
|
||||||
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
{{end}}
|
|
||||||
@@ -1,158 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Véhicules partagés</h1>
|
|
||||||
|
|
||||||
<div class="sm:flex sm:items-center">
|
|
||||||
<div class="sm:flex-auto">
|
|
||||||
<p class="mt-2 text-sm text-gray-700"></p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-4 sm:mt-0 sm:ml-16 sm:flex-none">
|
|
||||||
<a href="/app/vehicles/bookings/">
|
|
||||||
<button type="button"
|
|
||||||
class="inline-flex items-center justify-center rounded-2xl border border-transparent bg-co-blue px-4 py-2 text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
Voir les prêts de véhicules
|
|
||||||
</button>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="mt-8 max-w-3xl mx-auto grid grid-cols-1 gap-6 lg:max-w-7xl lg:grid-flow-col-dense lg:grid-cols-3">
|
|
||||||
<div class="space-y-6 lg:col-start-1 lg:col-span-1">
|
|
||||||
<div class="bg-white shadow sm:rounded-2xl">
|
|
||||||
<h2 id="timeline-title" class="text-lg font-medium text-gray-900 p-4 sm:px-6">Chercher un véhicule</h2>
|
|
||||||
<div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
<form method="GET">
|
|
||||||
|
|
||||||
|
|
||||||
<div x-data="{
|
|
||||||
text: '{{if .ViewState.search}}{{.ViewState.search.beneficiary.Data.first_name}} {{.ViewState.search.beneficiary.Data.last_name}}{{end}}',
|
|
||||||
beneficiariesListOpen: false,
|
|
||||||
beneficiaries: {{json .ViewState.beneficiaries}},
|
|
||||||
filteredBeneficiaries: (text) => {
|
|
||||||
if(text=='') return beneficiaries
|
|
||||||
return this.beneficiaries.filter(b => b['data']['first_name'].includes(text) || b['data']['last_name'].includes(text))
|
|
||||||
},
|
|
||||||
fields: {
|
|
||||||
beneficiaryid: {{if .ViewState.search}}'{{.ViewState.search.beneficiary.ID}}'{{else}}null{{end}},
|
|
||||||
},
|
|
||||||
selectbeneficiary(beneficiary) {
|
|
||||||
console.log(beneficiary)
|
|
||||||
this.fields.beneficiaryid = beneficiary.id
|
|
||||||
this.text = beneficiary.data.first_name + ' ' + beneficiary.data.last_name
|
|
||||||
this.beneficiariesListOpen = false
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<input type="hidden" name="beneficiaryid" x-model="fields.beneficiaryid">
|
|
||||||
<label for="combobox" class="block text-sm font-medium text-gray-700">Bénéficiaire</label>
|
|
||||||
<div class="relative mt-1 mb-4">
|
|
||||||
<input @focus="beneficiariesListOpen = true" x-model="text" id="combobox" type="text" class="w-full rounded-2xl border border-gray-300 bg-white py-2 pl-3 pr-12 shadow-sm focus:border-co-blue focus:outline-none focus:ring-1 focus:ring-co-blue sm:text-sm" role="combobox" aria-controls="options" aria-expanded="false">
|
|
||||||
|
|
||||||
<button @click="beneficiariesListOpen = ! beneficiariesListOpen" type="button" class="absolute inset-y-0 right-0 flex items-center rounded-r-2xl px-2 focus:outline-none">
|
|
||||||
<!-- Heroicon name: solid/selector -->
|
|
||||||
<svg class="h-5 w-5 text-gray-400" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd" d="M10 3a1 1 0 01.707.293l3 3a1 1 0 01-1.414 1.414L10 5.414 7.707 7.707a1 1 0 01-1.414-1.414l3-3A1 1 0 0110 3zm-3.707 9.293a1 1 0 011.414 0L10 14.586l2.293-2.293a1 1 0 011.414 1.414l-3 3a1 1 0 01-1.414 0l-3-3a1 1 0 010-1.414z" clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<ul x-show="beneficiariesListOpen" class="absolute z-10 mt-1 max-h-60 w-full overflow-auto rounded-xl bg-white py-1 text-base shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none sm:text-sm" id="options" role="listbox">
|
|
||||||
<!--
|
|
||||||
Combobox option, manage highlight styles based on mouseenter/mouseleave and keyboard navigation.
|
|
||||||
|
|
||||||
Active: "text-white bg-indigo-600", Not Active: "text-gray-900"
|
|
||||||
-->
|
|
||||||
<template x-for="beneficiary in beneficiaries">
|
|
||||||
<li @click="selectbeneficiary(beneficiary)" class="relative cursor-default hover:bg-gray-100 select-none py-2 pl-3 pr-9 text-gray-900" id="option-0" role="option" tabindex="-1">
|
|
||||||
<!-- Selected: "font-semibold" -->
|
|
||||||
<span class="truncate" x-text="beneficiary.data.first_name"></span> <span class="truncate" x-text="beneficiary.data.last_name"></span>
|
|
||||||
|
|
||||||
<!--
|
|
||||||
Checkmark, only display for selected option.
|
|
||||||
|
|
||||||
Active: "text-white", Not Active: "text-indigo-600"
|
|
||||||
-->
|
|
||||||
<span class="absolute inset-y-0 right-0 flex items-center pr-4 text-co-blue">
|
|
||||||
<!-- Heroicon name: solid/check -->
|
|
||||||
<!-- <svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clip-rule="evenodd" />
|
|
||||||
</svg> -->
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<!-- More items... -->
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
<!-- {{ $fieldName := "address" }}
|
|
||||||
{{ template "address_autocomplete" dict "FieldName" $fieldName }} -->
|
|
||||||
<div class="py-4 grid grid-cols-2">
|
|
||||||
<div class="lg:col-span-1">
|
|
||||||
<label for="startdate" class="block text-sm font-medium text-gray-700">Du</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<input type="date" id="startdate" name="startdate" value="{{.ViewState.search.startdate}}"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block w-full sm:text-sm border-gray-300 rounded-l-2xl border-r-1">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="lg:col-span-1">
|
|
||||||
<label for="enddate" class="block text-sm font-medium text-gray-700">Au</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<input type="date" id="enddate" name="enddate" value="{{.ViewState.search.enddate}}"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block w-full sm:text-sm border-gray-300 rounded-r-2xl border-l-0">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
{{template "vehicle_type_select" .}}
|
|
||||||
|
|
||||||
<button type="submit"
|
|
||||||
class="rounded-2xl border border-transparent bg-co-blue px-4 py-2 my-4 mt-8 w-full text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
Chercher
|
|
||||||
</button>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="lg:col-start-2 lg:col-span-2">
|
|
||||||
{{if .ViewState.searched}}
|
|
||||||
<div class="bg-white px-4 py-5 shadow sm:rounded-2xl sm:px-6">
|
|
||||||
<h2 id="timeline-title" class="text-lg font-medium text-gray-900">Véhicules disponibles</h2>
|
|
||||||
<div class="mt-8 flex flex-col">
|
|
||||||
<div class="-my-2 -mx-4 overflow-x-auto sm:-mx-6 lg:-mx-8">
|
|
||||||
<div class="inline-block min-w-full py-2 align-middle md:px-6 lg:px-8">
|
|
||||||
<table class="min-w-full divide-y divide-gray-300">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th scope="col" class="py-3.5 pl-4 pr-3 text-left text-sm font-semibold text-gray-900 sm:pl-6 md:pl-0">Véhicule</th>
|
|
||||||
<th scope="col" class="py-3.5 px-3 text-left text-sm font-semibold text-gray-900">Numéro</th>
|
|
||||||
<th scope="col" class="py-3.5 px-3 text-left text-sm font-semibold text-gray-900">Gestionnaire</th>
|
|
||||||
<th scope="col" class="py-3.5 px-3 text-left text-sm font-semibold text-gray-900">Lieu</th>
|
|
||||||
<th scope="col" class="relative py-3.5 pl-3 pr-4 sm:pr-6 md:pr-0">
|
|
||||||
<span class="sr-only">Réserver</span>
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="divide-y divide-gray-200">
|
|
||||||
{{range .ViewState.search.vehicles}}
|
|
||||||
<tr>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm font-medium text-gray-900 sm:pl-6 md:pl-0">{{.Data.name}}</td>
|
|
||||||
<td class="whitespace-nowrap py-4 px-3 text-sm text-gray-500">{{.Data.licence_plate}}</td>
|
|
||||||
<td class="whitespace-nowrap py-4 px-3 text-sm text-gray-500">COOPGO</td>
|
|
||||||
<td class="whitespace-nowrap py-4 px-3 text-sm text-gray-500">{{if .Data.address}}{{.Data.address.properties.label}}{{end}}</td>
|
|
||||||
<td class="relative whitespace-nowrap py-4 pl-3 pr-4 text-right text-sm font-medium sm:pr-6 md:pr-0">
|
|
||||||
<a href="/app/vehicles/v/{{.ID}}/b/{{$.ViewState.search.beneficiary.ID}}?startdate={{$.ViewState.search.startdate}}&enddate={{$.ViewState.search.enddate}}" class="text-co-blue hover:text-co-blue">Réserver<span class="sr-only"> pour {{$.ViewState.search.beneficiary.Data.first_name}} {{$.ViewState.search.beneficiary.Data.last_name}}</span></a>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
{{end}}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,75 +0,0 @@
|
|||||||
{{define "bookings_list"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<div class="mt-8 flex flex-col">
|
|
||||||
<div class="-my-2 -mx-4 overflow-x-auto sm:-mx-6 lg:-mx-8">
|
|
||||||
<div class="inline-block min-w-full py-2 align-middle md:px-6 lg:px-8">
|
|
||||||
<div class="overflow-hidden shadow ring-1 ring-black ring-opacity-5 md:rounded-lg">
|
|
||||||
<table class="min-w-full divide-y divide-gray-300">
|
|
||||||
<thead class="bg-gray-50">
|
|
||||||
<tr>
|
|
||||||
<!-- <th scope="col"
|
|
||||||
class="py-3.5 pl-4 pr-3 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Statut
|
|
||||||
</th> -->
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Type
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Immatriculation
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Beneficiaire
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Gestionnaire
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Dates
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="relative py-3.5 pl-3 pr-4 sm:pr-6">
|
|
||||||
<span class="sr-only">Actions</span>
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="divide-y divide-gray-200 bg-white">
|
|
||||||
{{range .ViewState.vehicles}}
|
|
||||||
{{$vehicle := .}}
|
|
||||||
{{range .Bookings}}
|
|
||||||
<tr>
|
|
||||||
<!-- <td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >aa</div>
|
|
||||||
</td> -->
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >Voiture</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >{{$vehicle.Data.licence_plate}}</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" ><img class="h-6 w-6 rounded-co"
|
|
||||||
src="/app/beneficiaries/{{.Driver}}/picture" alt=""></div>
|
|
||||||
</td>
|
|
||||||
<!-- <td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >aa</div>
|
|
||||||
</td> -->
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >COOPGO</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >Du {{(timeFrom .Startdate).Format "02/01/2006"}} au {{(timeFrom .Enddate).Format "02/01/2006"}}</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<a href="/app/vehicles-management/bookings/{{.ID}}"
|
|
||||||
class="text-co-blue hover:text-co-blue">Voir</a>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
{{end}}
|
|
||||||
{{end}}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,259 +0,0 @@
|
|||||||
{{define "calendar"}}
|
|
||||||
<div class="flex items-center">
|
|
||||||
<h2 class="flex-auto font-semibold text-gray-900">Août 2022</h2>
|
|
||||||
<button type="button"
|
|
||||||
class="-my-1.5 flex flex-none items-center justify-center p-1.5 text-gray-400 hover:text-gray-500">
|
|
||||||
<span class="sr-only">Mois précédent</span>
|
|
||||||
<!-- Heroicon name: solid/chevron-left -->
|
|
||||||
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"
|
|
||||||
aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd"
|
|
||||||
d="M12.707 5.293a1 1 0 010 1.414L9.414 10l3.293 3.293a1 1 0 01-1.414 1.414l-4-4a1 1 0 010-1.414l4-4a1 1 0 011.414 0z"
|
|
||||||
clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</button>
|
|
||||||
<button type="button"
|
|
||||||
class="-my-1.5 -mr-1.5 ml-2 flex flex-none items-center justify-center p-1.5 text-gray-400 hover:text-gray-500">
|
|
||||||
<span class="sr-only">Mois suivant</span>
|
|
||||||
<!-- Heroicon name: solid/chevron-right -->
|
|
||||||
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"
|
|
||||||
aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd"
|
|
||||||
d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z"
|
|
||||||
clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="mt-10 grid grid-cols-7 text-center text-xs leading-6 text-gray-500">
|
|
||||||
<div>L</div>
|
|
||||||
<div>M</div>
|
|
||||||
<div>M</div>
|
|
||||||
<div>J</div>
|
|
||||||
<div>V</div>
|
|
||||||
<div>S</div>
|
|
||||||
<div>D</div>
|
|
||||||
</div>
|
|
||||||
<div class="mt-2 grid grid-cols-7 text-sm">
|
|
||||||
<div class="py-2">
|
|
||||||
<!--
|
|
||||||
Always include: "mx-auto flex h-8 w-8 items-center justify-center rounded-full"
|
|
||||||
Is selected, include: "text-white"
|
|
||||||
Is not selected and is today, include: "text-indigo-600"
|
|
||||||
Is not selected and is not today and is current month, include: "text-gray-900"
|
|
||||||
Is not selected and is not today and is not current month, include: "text-gray-400"
|
|
||||||
Is selected and is today, include: "bg-indigo-600"
|
|
||||||
Is selected and is not today, include: "bg-gray-900"
|
|
||||||
Is not selected, include: "hover:bg-gray-200"
|
|
||||||
Is selected or is today, include: "font-semibold"
|
|
||||||
-->
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-01">1</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-02">2</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-03">3</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-04">4</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-05">5</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-06">6</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-07">7</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-08">8</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-09">9</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-10">10</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-11">11</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full bg-gray-900 font-semibold text-white">
|
|
||||||
<time datetime="2022-08-12">12</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-13">13</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-14">14</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-15">15</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-16">16</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-17">17</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-18">18</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-19">19</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-20">20</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-21">21</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-22">22</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-23">23</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-24">24</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-25">25</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-26">26</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-27">27</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-28">28</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-29">29</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-30">30</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-900 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-08-31">31</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-400 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-09-01">1</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-400 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-09-02">2</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-400 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-09-03">3</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 py-2">
|
|
||||||
<button type="button"
|
|
||||||
class="mx-auto flex h-8 w-8 items-center justify-center rounded-full text-gray-400 hover:bg-gray-200">
|
|
||||||
<time datetime="2022-09-04">4</time>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,105 +0,0 @@
|
|||||||
{{define "vehicle_type_select"}}
|
|
||||||
<div x-data="{
|
|
||||||
selectOpen: false,
|
|
||||||
vehicle_type: 'car',
|
|
||||||
vehicle_label: 'Voiture',
|
|
||||||
selectType(type, label) {
|
|
||||||
this.vehicle_type = type,
|
|
||||||
this.vehicle_label = label
|
|
||||||
this.selectOpen = false
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<input type="hidden" name="vehicle_type" x-model="vehicle_type">
|
|
||||||
<label id="listbox-label" class="block text-sm font-medium text-gray-700"> Type de véhicule </label>
|
|
||||||
<div class="mt-1 relative">
|
|
||||||
<button @click="selectOpen = ! selectOpen" type="button"
|
|
||||||
class="relative w-full bg-white border border-gray-300 rounded-2xl shadow-sm pl-3 pr-10 py-2 text-left cursor-default focus:outline-none focus:ring-1 focus:ring-co-blue focus:border-co-blue sm:text-sm"
|
|
||||||
aria-haspopup="listbox" aria-expanded="true" aria-labelledby="listbox-label">
|
|
||||||
<span class="block truncate" x-text="vehicle_label"></span>
|
|
||||||
<span class="absolute inset-y-0 right-0 flex items-center pr-2 pointer-events-none">
|
|
||||||
{{$.IconSet.Icon "hero:solid/selector" "h-5 w-5 text-gray-400"}}
|
|
||||||
</span>
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<ul class="absolute z-10 mt-1 w-full bg-white shadow-lg max-h-60 rounded-xl py-1 text-base ring-1 ring-black ring-opacity-5 overflow-auto focus:outline-none sm:text-sm"
|
|
||||||
tabindex="-1" role="listbox" aria-labelledby="listbox-label" aria-activedescendant="listbox-option-3"
|
|
||||||
x-show="selectOpen" x-transition:leave="transition ease-in duration-100"
|
|
||||||
x-transition:leave-start="transform opacity-100" x-transition:leave-end="transform opacity-0">
|
|
||||||
|
|
||||||
<li class="text-gray-900 cursor-default select-none relative py-2 pl-8 pr-4 hover:bg-co-blue hover:text-white" id="listbox-option-0"
|
|
||||||
role="option" @click="selectType('car', 'Voiture')">
|
|
||||||
<!-- Selected: "font-semibold", Not Selected: "font-normal" -->
|
|
||||||
<span class="font-normal block truncate "> Voiture </span>
|
|
||||||
<span x-show="vehicle_type == 'car'"
|
|
||||||
class="text-co-blue absolute inset-y-0 left-0 flex items-center pl-1.5">
|
|
||||||
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"
|
|
||||||
aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd"
|
|
||||||
d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z"
|
|
||||||
clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
<li class="text-gray-900 cursor-default select-none relative py-2 pl-8 pr-4 hover:bg-co-blue hover:text-white" id="listbox-option-0"
|
|
||||||
role="option" @click="selectType('no_licence_car', 'Voiture sans permis')">
|
|
||||||
<!-- Selected: "font-semibold", Not Selected: "font-normal" -->
|
|
||||||
<span class="font-normal block truncate "> Voiture sans permis </span>
|
|
||||||
<span x-show="vehicle_type == 'no_licence_car'"
|
|
||||||
class="text-co-blue absolute inset-y-0 left-0 flex items-center pl-1.5">
|
|
||||||
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"
|
|
||||||
aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd"
|
|
||||||
d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z"
|
|
||||||
clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
<li class="text-gray-900 cursor-default select-none relative py-2 pl-8 pr-4 hover:bg-co-blue hover:text-white" id="listbox-option-0"
|
|
||||||
role="option" @click="selectType('scooter', 'Scooter')">
|
|
||||||
<!-- Selected: "font-semibold", Not Selected: "font-normal" -->
|
|
||||||
<span class="font-normal block truncate "> Scooter </span>
|
|
||||||
<span x-show="vehicle_type == 'scooter'"
|
|
||||||
class="text-co-blue absolute inset-y-0 left-0 flex items-center pl-1.5">
|
|
||||||
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"
|
|
||||||
aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd"
|
|
||||||
d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z"
|
|
||||||
clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
<li class="text-gray-900 cursor-default select-none relative py-2 pl-8 pr-4 hover:bg-co-blue hover:text-white" id="listbox-option-0"
|
|
||||||
role="option" @click="selectType('trott', 'Trotinette')">
|
|
||||||
<!-- Selected: "font-semibold", Not Selected: "font-normal" -->
|
|
||||||
<span class="font-normal block truncate "> Trotinette </span>
|
|
||||||
<span x-show="vehicle_type == 'trott'"
|
|
||||||
class="text-co-blue absolute inset-y-0 left-0 flex items-center pl-1.5">
|
|
||||||
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"
|
|
||||||
aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd"
|
|
||||||
d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z"
|
|
||||||
clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
<li class="text-gray-900 cursor-default select-none relative py-2 pl-8 pr-4 hover:bg-co-blue hover:text-white" id="listbox-option-0"
|
|
||||||
role="option" @click="selectType('electric_bike', 'Vélo électrique')">
|
|
||||||
<!-- Selected: "font-semibold", Not Selected: "font-normal" -->
|
|
||||||
<span class="font-normal block truncate "> Vélo électrique </span>
|
|
||||||
<span x-show="vehicle_type == 'electric_bike'"
|
|
||||||
class="text-co-blue hover:text-inherit absolute inset-y-0 left-0 flex items-center pl-1.5">
|
|
||||||
<svg class="h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"
|
|
||||||
aria-hidden="true">
|
|
||||||
<path fill-rule="evenodd"
|
|
||||||
d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z"
|
|
||||||
clip-rule="evenodd" />
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
<!-- More items... -->
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{{end}}
|
|
||||||
@@ -1,56 +0,0 @@
|
|||||||
{{define "vehicles_list"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<div class="mt-8 flex flex-col">
|
|
||||||
<div class="-my-2 -mx-4 overflow-x-auto sm:-mx-6 lg:-mx-8">
|
|
||||||
<div class="inline-block min-w-full py-2 align-middle md:px-6 lg:px-8">
|
|
||||||
<div class="overflow-hidden shadow ring-1 ring-black ring-opacity-5 md:rounded-lg">
|
|
||||||
<table class="min-w-full divide-y divide-gray-300">
|
|
||||||
<thead class="bg-gray-50">
|
|
||||||
<tr>
|
|
||||||
<th scope="col"
|
|
||||||
class="py-3.5 pl-4 pr-3 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Numéro (Immat / Bicycode)
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Type
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Modèle
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="px-3 py-3.5 text-left text-sm font-semibold text-gray-900 sm:pl-6">
|
|
||||||
Lieu
|
|
||||||
</th>
|
|
||||||
<th scope="col" class="relative py-3.5 pl-3 pr-4 sm:pr-6">
|
|
||||||
<span class="sr-only">Actions</span>
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="divide-y divide-gray-200 bg-white">
|
|
||||||
{{range .ViewState.vehicles}}
|
|
||||||
<tr>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >{{.Data.licence_plate}}</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >{{if eq .Type "electric_bike"}}Vélo électrique{{else}}Voiture{{end}}</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >{{.Data.name}}</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<div class="text-gray-900" >{{if .Data.address}}{{.Data.address.properties.label}}{{end}}</div>
|
|
||||||
</td>
|
|
||||||
<td class="whitespace-nowrap py-4 pl-4 pr-3 text-sm sm:pl-6">
|
|
||||||
<a href="/app/vehicles-management/fleet/{{.ID}}"
|
|
||||||
class="text-co-blue hover:text-co-blue">Voir</a>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
{{end}}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,227 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Réservation de véhicule</h1>
|
|
||||||
|
|
||||||
<div class="sm:flex sm:items-center">
|
|
||||||
<div class="sm:flex-auto">
|
|
||||||
<p class="mt-2 text-sm text-gray-700"></p>
|
|
||||||
</div>
|
|
||||||
<!-- <div class="mt-4 sm:mt-0 sm:ml-16 sm:flex-none">
|
|
||||||
<button type="button"
|
|
||||||
class="inline-flex items-center justify-center px-4 py-2 border border-gray-300 shadow-sm text-sm font-medium rounded-2xl text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">
|
|
||||||
Changer de véhicule
|
|
||||||
</button>
|
|
||||||
<a href="/app/vehicles-management/bookings/{{.ViewState.booking.ID}}/delete">
|
|
||||||
<button type="button"
|
|
||||||
class="inline-flex items-center justify-center rounded-2xl border border-transparent bg-co-red px-4 py-2 text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-red focus:ring-offset-2 sm:w-auto">
|
|
||||||
Annuler
|
|
||||||
</button>
|
|
||||||
</a>
|
|
||||||
</div> -->
|
|
||||||
</div>
|
|
||||||
<div class="mt-8 max-w-3xl mx-auto grid grid-cols-1 gap-6 lg:max-w-7xl lg:grid-flow-col-dense lg:grid-cols-3">
|
|
||||||
<div class="space-y-6 lg:col-start-1 lg:col-span-1">
|
|
||||||
<div class="bg-white shadow sm:rounded-2xl">
|
|
||||||
<h2 id="timeline-title" class="text-lg font-medium text-gray-900 p-4 sm:px-6">Bénéficiaire</h2>
|
|
||||||
<div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
<div>
|
|
||||||
<div class="mt-5 border-gray-200">
|
|
||||||
<dl class="sm:divide-y sm:divide-gray-200">
|
|
||||||
<div class="sm:pb-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Nom</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.beneficiary.Data.first_name}}
|
|
||||||
{{.ViewState.beneficiary.Data.last_name}}</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Email</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.beneficiary.Data.email}}</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Téléphone</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.beneficiary.Data.phone_number}}</dd>
|
|
||||||
</div>
|
|
||||||
{{if .ViewState.Data.birthdate}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Date de naissance</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">{{(timeFrom
|
|
||||||
.ViewState.Data.birthdate).Format
|
|
||||||
"02/01/2006"}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if and .ViewState.Data.gender (ne .ViewState.Data.gender "0")}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Date de naissance</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">{{genderISO5218
|
|
||||||
.ViewState.Data.gender}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.Data.address}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Adresse</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.Data.address.properties.label}}
|
|
||||||
</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="lg:col-start-2 lg:col-span-2">
|
|
||||||
<div class="bg-white shadow sm:rounded-2xl sm:px-6">
|
|
||||||
<div class="bg-white px-4 py-5 border-b border-gray-200 sm:px-6">
|
|
||||||
<div class="-ml-4 -mt-4 flex justify-between items-center flex-wrap sm:flex-nowrap">
|
|
||||||
<div class="ml-4 mt-4">
|
|
||||||
<h3 class="text-lg leading-6 font-medium text-gray-900">Réservation</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Informations utiles sur la réservation.</p>
|
|
||||||
</div>
|
|
||||||
<div class="ml-4 mt-4 flex-shrink-0">
|
|
||||||
<button type="button"
|
|
||||||
class="relative inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-xs font-medium rounded-2xl text-co-blue bg-gray-100 hover:bg-co-blue hover:text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">SMS</button>
|
|
||||||
<button type="button"
|
|
||||||
class="relative inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-xs font-medium rounded-2xl text-co-blue bg-gray-100 hover:bg-co-blue hover:text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Email</button>
|
|
||||||
<!-- <button type="button"
|
|
||||||
class="relative inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-xs font-medium rounded-2xl text-co-blue bg-gray-100 hover:bg-co-blue hover:text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Imprimer</button> -->
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="px-4 py-5 sm:px-6">
|
|
||||||
<div>
|
|
||||||
<div class="mt-5 border-gray-200">
|
|
||||||
<dl class="sm:divide-y sm:divide-gray-200">
|
|
||||||
<!-- <div class="sm:pb-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Gestionnaire</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
COOPGO
|
|
||||||
</dd>
|
|
||||||
</div> -->
|
|
||||||
{{if .ViewState.booking.Data.booked_by}}
|
|
||||||
<!-- <div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4"> -->
|
|
||||||
<div class="sm:pb-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Prescripteur</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{if .ViewState.booking.Data.booked_by.user}}
|
|
||||||
<a href="/app/members/{{.ViewState.booking.Data.booked_by.user.id}}" class="flex inline">
|
|
||||||
<img class="h-5 w-5 rounded-co mr-1"
|
|
||||||
src="/app/members/{{.ViewState.booking.Data.booked_by.user.id}}/picture" alt="">
|
|
||||||
{{.ViewState.booking.Data.booked_by.user.display_name}}
|
|
||||||
</a>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.booking.Data.booked_by.group}}{{.ViewState.booking.Data.booked_by.group.name}}{{end}}
|
|
||||||
</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Véhicule</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.vehicle.Data.name}}</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Immatriculation</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
{{.ViewState.vehicle.Data.licence_plate}}</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Type</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
|
||||||
Voiture</dd>
|
|
||||||
</div>
|
|
||||||
{{if .ViewState.vehicle.Data.address}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Lieu de récupération</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">{{.ViewState.vehicle.Data.address.properties.label}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Date de récupération</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2 inline-flex"
|
|
||||||
x-data="{ updateOpen: false }">
|
|
||||||
<div class="w-full inline-flex" x-show="!updateOpen">
|
|
||||||
<div class="flex-1">{{(timeFrom .ViewState.booking.Startdate).Format "02/01/2006"}}</div>
|
|
||||||
<a href="#" class="text-co-blue hover:text-co-blue ml-5" @click="updateOpen = ! updateOpen">Modifier</a>
|
|
||||||
</div>
|
|
||||||
<form method="POST" class="inline-flex" x-show="updateOpen">
|
|
||||||
<div class="flex-1">
|
|
||||||
<input type="date" name="startdate" value="{{(timeFrom .ViewState.booking.Startdate).Format "2006-01-02"}}"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block sm:text-sm border-gray-300 rounded-2xl">
|
|
||||||
</div>
|
|
||||||
<button
|
|
||||||
class=" justify-center text-co-blue px-4">
|
|
||||||
OK
|
|
||||||
</button>
|
|
||||||
</form>
|
|
||||||
</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Date de retour</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2"
|
|
||||||
x-data="{ updateOpen: false }">
|
|
||||||
<div class="w-full inline-flex" x-show="!updateOpen">
|
|
||||||
<div class="flex-1">{{(timeFrom .ViewState.booking.Enddate).Format "02/01/2006"}}</div>
|
|
||||||
<a href="#" class="text-co-blue hover:text-co-blue ml-5" @click="updateOpen = ! updateOpen">Modifier</a>
|
|
||||||
</div>
|
|
||||||
<form method="POST" class="inline-flex" x-show="updateOpen">
|
|
||||||
<div class="flex-1">
|
|
||||||
<input type="date" name="enddate" value="{{(timeFrom .ViewState.booking.Enddate).Format "2006-01-02"}}"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block sm:text-sm border-gray-300 rounded-2xl">
|
|
||||||
</div>
|
|
||||||
<button
|
|
||||||
class=" justify-center text-co-blue px-4">
|
|
||||||
OK
|
|
||||||
</button>
|
|
||||||
</form>
|
|
||||||
</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Indisponible à partir du</dt>
|
|
||||||
<dd class="mt-1 text-sm font-bold text-co-red sm:mt-0 sm:col-span-2 inline-flex"
|
|
||||||
x-data="{ updateOpen: false }">
|
|
||||||
<div class="w-full inline-flex" x-show="!updateOpen">
|
|
||||||
<div class="flex-1">{{(timeFrom .ViewState.booking.Unavailablefrom).Format "02/01/2006"}}</div>
|
|
||||||
<a href="#" class="text-co-blue hover:text-co-blue ml-5" @click="updateOpen = ! updateOpen">Modifier</a>
|
|
||||||
</div>
|
|
||||||
<form method="POST" class="inline-flex" x-show="updateOpen">
|
|
||||||
<div class="flex-1">
|
|
||||||
<input type="date" name="unavailablefrom" value="{{(timeFrom .ViewState.booking.Unavailablefrom).Format "2006-01-02"}}"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block sm:text-sm border-gray-300 rounded-2xl">
|
|
||||||
</div>
|
|
||||||
<button
|
|
||||||
class=" justify-center text-co-blue px-4">
|
|
||||||
OK
|
|
||||||
</button>
|
|
||||||
</form>
|
|
||||||
</dd>
|
|
||||||
</div>
|
|
||||||
<div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Sera à nouveau disponible le</dt>
|
|
||||||
<dd class="mt-1 text-sm font-bold text-co-green sm:mt-0 sm:col-span-2 inline-flex"
|
|
||||||
x-data="{ updateOpen: false }">
|
|
||||||
<div class="w-full inline-flex" x-show="!updateOpen">
|
|
||||||
<div class="flex-1">{{(timeFrom .ViewState.booking.Unavailableto).Format "02/01/2006"}}</div>
|
|
||||||
<a href="#" class="text-co-blue hover:text-co-blue ml-5" @click="updateOpen = ! updateOpen">Modifier</a>
|
|
||||||
</div>
|
|
||||||
<form method="POST" class="inline-flex" x-show="updateOpen">
|
|
||||||
<div class="flex-1">
|
|
||||||
<input type="date" name="unavailableto" value="{{(timeFrom .ViewState.booking.Unavailableto).Format "2006-01-02"}}"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block sm:text-sm border-gray-300 rounded-2xl">
|
|
||||||
</div>
|
|
||||||
<button
|
|
||||||
class=" justify-center text-co-blue px-4">
|
|
||||||
OK
|
|
||||||
</button>
|
|
||||||
</dd>
|
|
||||||
</div>
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,106 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Ajouter un véhicule</h1>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8 mt-8" x-data="{
|
|
||||||
fields: {
|
|
||||||
licence_plate: null,
|
|
||||||
name: null,
|
|
||||||
},
|
|
||||||
rules: {
|
|
||||||
licence_plate: ['required', 'regexMatch:^[A-Z]{1,2}-[0-9]{1,3}-[A-Z]{1,2}$'],
|
|
||||||
name: ['required'],
|
|
||||||
},
|
|
||||||
formValidation: {
|
|
||||||
valid: false,
|
|
||||||
fields: {
|
|
||||||
name: {valid: null},
|
|
||||||
licence_plate: {valid: null},
|
|
||||||
}
|
|
||||||
},
|
|
||||||
isFormValid: true,
|
|
||||||
validate() {
|
|
||||||
this.formValidation = Iodine.assert(this.fields, this.rules)
|
|
||||||
},
|
|
||||||
validateField(field) {
|
|
||||||
this.formValidation.fields[field] = Iodine.assert(this.fields[field], this.rules[field])
|
|
||||||
},
|
|
||||||
submit(event) {
|
|
||||||
this.validate()
|
|
||||||
if(!this.formValidation.valid) {
|
|
||||||
this.isFormValid = false
|
|
||||||
event.preventDefault()
|
|
||||||
}
|
|
||||||
return this.formValidation.valid
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<form class="space-y-6" method="POST" @submit="submit">
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Identité du véhicule</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Informations de base sur le véhicule</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 md:mt-0 md:col-span-2">
|
|
||||||
<div class="grid grid-cols-3 md:grid-cols-6 gap-6">
|
|
||||||
<div class="col-span-5">
|
|
||||||
<label for="name" class="block text-sm font-medium text-gray-700">Modèle (ou nom donné au
|
|
||||||
véhicule)</label>
|
|
||||||
<input type="text" name="name" id="name"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.name" @blur="validateField('name')"
|
|
||||||
:class="formValidation.fields.name.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
<div class="col-span-3">
|
|
||||||
{{template "vehicle_type_select" .}}
|
|
||||||
</div>
|
|
||||||
<div class="col-span-3">
|
|
||||||
<label for="licence_plate"
|
|
||||||
class="block text-sm font-medium text-gray-700">Immatriculation</label>
|
|
||||||
<input type="text" name="licence_plate" id="licence_plate" placeholder="XX-123-YY"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.licence_plate"
|
|
||||||
@blur="fields.licence_plate = fields.licence_plate.toUpperCase(); validateField('licence_plate')"
|
|
||||||
:class="formValidation.fields.licence_plate.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Informations pratiques</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Informations pratiques pour la réservation</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 md:mt-0 md:col-span-2">
|
|
||||||
{{ $fieldName := "address" }}
|
|
||||||
{{ template "address_autocomplete" dict "FieldName" $fieldName }}
|
|
||||||
|
|
||||||
<div class="mt-5">
|
|
||||||
<label for="informations" class="block text-sm font-medium text-gray-700">Informations pratiques pour le bénéficiaire</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<textarea rows="4" name="informations" id="informations"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block w-full sm:text-sm border-gray-300 rounded-2xl"></textarea>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex justify-end">
|
|
||||||
<p x-show="! isFormValid" class="px-4 py-2 text-sm text-co-red">Certains champs de sont pas valides.</p>
|
|
||||||
<a href="/app/vehicles-management/">
|
|
||||||
<button type="button"
|
|
||||||
class="bg-white py-2 px-4 border border-gray-300 rounded-2xl shadow-sm text-sm font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Annuler</button>
|
|
||||||
</a>
|
|
||||||
<button type="submit"
|
|
||||||
class="ml-3 inline-flex justify-center py-2 px-4 border border-transparent shadow-sm text-sm font-medium rounded-2xl text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Ajouter
|
|
||||||
le véhicule</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,90 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<main class="py-10">
|
|
||||||
<div class="max-w-3xl mx-auto px-4 sm:px-6 md:flex md:items-center md:justify-between md:space-x-5 lg:max-w-7xl lg:px-8">
|
|
||||||
<div class="flex items-center space-x-5">
|
|
||||||
<!-- <div class="flex-shrink-0">
|
|
||||||
<div class="relative">
|
|
||||||
<img class="h-16 w-16 rounded-co" src="/app/beneficiaries/{{.ViewState.ID}}/picture" alt="">
|
|
||||||
<span class="absolute inset-0 shadow-inner rounded-full" aria-hidden="true"></span>
|
|
||||||
</div>
|
|
||||||
</div> -->
|
|
||||||
<div>
|
|
||||||
<h1 class="text-2xl font-bold text-gray-900">{{.ViewState.vehicle.Data.name}}</h1>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="mt-6 flex flex-col-reverse justify-stretch space-y-4 space-y-reverse sm:flex-row-reverse sm:justify-end sm:space-x-reverse sm:space-y-0 sm:space-x-3 md:mt-0 md:flex-row md:space-x-3">
|
|
||||||
<button type="button"
|
|
||||||
class="inline-flex items-center justify-center px-4 py-2 border border-gray-300 shadow-sm text-sm font-medium rounded-2xl text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">Retirer de la flotte</button>
|
|
||||||
<a href="/app/vehicles-management/fleet/{{.ViewState.vehicle.ID}}/update" class="inline-flex"><button type="button"
|
|
||||||
class="w-full px-4 py-2 border border-transparent text-sm font-medium rounded-2xl shadow-sm text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-co-blue">Modifier</button></a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="mt-8 max-w-3xl mx-auto grid grid-cols-1 gap-6 sm:px-6 lg:max-w-7xl lg:grid-flow-col-dense lg:grid-cols-3">
|
|
||||||
<div class="space-y-6 lg:col-start-1 lg:col-span-2">
|
|
||||||
<section aria-labelledby="vehicle-information-title">
|
|
||||||
<div class="bg-white shadow sm:rounded-lg">
|
|
||||||
<div class="px-4 py-5 sm:px-6">
|
|
||||||
<h2 id="vehicle-information-title" class="text-lg leading-6 font-medium text-gray-900">Informations</h2>
|
|
||||||
<p class="mt-1 max-w-2xl text-sm text-gray-500">Informations sur le véhicule</p>
|
|
||||||
</div>
|
|
||||||
<div class="border-t border-gray-200 px-4 py-5 sm:px-6">
|
|
||||||
<dl class="grid grid-cols-1 gap-x-4 gap-y-8 sm:grid-cols-2">
|
|
||||||
{{if .ViewState.vehicle.Data.type}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Type</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{if eq .ViewState.vehicle.Data.type "electric_bike"}}Vélo électrique{{else}}Voiture{{end}}</dd>
|
|
||||||
</div>
|
|
||||||
{{else}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Type</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">Voiture</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.vehicle.Data.licence_plate}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Numéro (Immatriculation, bicycode, ...)</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.vehicle.Data.licence_plate}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.vehicle.Data.address}}
|
|
||||||
<div class="sm:col-span-1">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Lieu</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.vehicle.Data.address.properties.label}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
{{if .ViewState.vehicle.Data.informations}}
|
|
||||||
<div class="sm:col-span-2">
|
|
||||||
<dt class="text-sm font-medium text-gray-500">Informations pratiques pour le bénéficiaire</dt>
|
|
||||||
<dd class="mt-1 text-sm text-gray-900">{{.ViewState.vehicle.Data.informations}}</dd>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
<section aria-labelledby="timeline-title" class="lg:col-start-3 lg:col-span-1">
|
|
||||||
<div class="bg-white px-4 py-5 shadow sm:rounded-lg sm:px-6">
|
|
||||||
<h2 id="timeline-title" class="text-lg font-medium text-gray-900">Réservations à venir</h2>
|
|
||||||
{{if eq (len .ViewState.vehicle.Bookings) 0}}
|
|
||||||
<p class="p-12 text-gray-500 text-center text-md">Aucune réservation à venir</p>
|
|
||||||
{{end}}
|
|
||||||
<ul role="list" class="divide-y divide-gray-200">
|
|
||||||
{{range .ViewState.vehicle.Bookings}}
|
|
||||||
<li class="py-4 flex">
|
|
||||||
<div class="ml-3">
|
|
||||||
<a href="/app/vehicles-management/bookings/{{.ID}}" class="hover:bg-gray-200">
|
|
||||||
<p class="text-sm font-medium text-gray-900">Du {{(timeFrom .Startdate).Format "02/01/2006"}} au {{(timeFrom .Enddate).Format "02/01/2006"}}</p>
|
|
||||||
<p class="text-sm text-gray-500"></p>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
{{end}}
|
|
||||||
</ul>
|
|
||||||
{{template "calendar" .}}
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
</main>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,106 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Modifier un véhicule</h1>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8 mt-8" x-data="{
|
|
||||||
fields: {
|
|
||||||
licence_plate: '{{ .ViewState.vehicle.Data.licence_plate }}',
|
|
||||||
name: '{{ .ViewState.vehicle.Data.name }}',
|
|
||||||
},
|
|
||||||
rules: {
|
|
||||||
licence_plate: ['required', 'regexMatch:^[A-Z]{1,2}-[0-9]{1,3}-[A-Z]{1,2}$'],
|
|
||||||
name: ['required'],
|
|
||||||
},
|
|
||||||
formValidation: {
|
|
||||||
valid: false,
|
|
||||||
fields: {
|
|
||||||
name: {valid: null},
|
|
||||||
licence_plate: {valid: null},
|
|
||||||
}
|
|
||||||
},
|
|
||||||
isFormValid: true,
|
|
||||||
validate() {
|
|
||||||
this.formValidation = Iodine.assert(this.fields, this.rules)
|
|
||||||
},
|
|
||||||
validateField(field) {
|
|
||||||
this.formValidation.fields[field] = Iodine.assert(this.fields[field], this.rules[field])
|
|
||||||
},
|
|
||||||
submit(event) {
|
|
||||||
this.validate()
|
|
||||||
if(!this.formValidation.valid) {
|
|
||||||
this.isFormValid = false
|
|
||||||
event.preventDefault()
|
|
||||||
}
|
|
||||||
return this.formValidation.valid
|
|
||||||
}
|
|
||||||
}">
|
|
||||||
<form class="space-y-6" method="POST" @submit="submit">
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Identité du véhicule</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Informations de base sur le véhicule</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 md:mt-0 md:col-span-2">
|
|
||||||
<div class="grid grid-cols-3 md:grid-cols-6 gap-6">
|
|
||||||
<div class="col-span-5">
|
|
||||||
<label for="name" class="block text-sm font-medium text-gray-700">Modèle (ou nom donné au
|
|
||||||
véhicule)</label>
|
|
||||||
<input type="text" name="name" id="name"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.name" @blur="validateField('name')"
|
|
||||||
:class="formValidation.fields.name.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
<div class="col-span-3">
|
|
||||||
{{template "vehicle_type_select" .}}
|
|
||||||
</div>
|
|
||||||
<div class="col-span-3">
|
|
||||||
<label for="licence_plate"
|
|
||||||
class="block text-sm font-medium text-gray-700">Immatriculation</label>
|
|
||||||
<input type="text" name="licence_plate" id="licence_plate" placeholder="XX-123-YY"
|
|
||||||
class="mt-1 focus:ring-co-blue focus:border-co-blue block w-full shadow-sm sm:text-sm rounded-2xl"
|
|
||||||
x-model="fields.licence_plate"
|
|
||||||
@blur="fields.licence_plate = fields.licence_plate.toUpperCase(); validateField('licence_plate')"
|
|
||||||
:class="formValidation.fields.licence_plate.valid == false ? 'border-co-red border-2' : 'border-gray-300'">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="bg-white shadow px-4 py-5 sm:rounded-lg sm:p-6">
|
|
||||||
<div class="md:grid md:grid-cols-3 md:gap-6">
|
|
||||||
<div class="md:col-span-1">
|
|
||||||
<h3 class="text-lg font-medium leading-6 text-gray-900">Informations pratiques</h3>
|
|
||||||
<p class="mt-1 text-sm text-gray-500">Informations pratiques pour la réservation</p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-5 md:mt-0 md:col-span-2">
|
|
||||||
{{ $fieldName := "address" }}
|
|
||||||
{{ template "address_autocomplete" (dict "FieldName" $fieldName "Address" .ViewState.vehicle.Data.address) }}
|
|
||||||
|
|
||||||
<div class="mt-5">
|
|
||||||
<label for="informations" class="block text-sm font-medium text-gray-700">Informations pratiques pour le bénéficiaire</label>
|
|
||||||
<div class="mt-1">
|
|
||||||
<textarea rows="4" name="informations" id="informations"
|
|
||||||
class="shadow-sm focus:ring-co-blue focus:border-co-blue block w-full sm:text-sm border-gray-300 rounded-2xl"></textarea>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex justify-end">
|
|
||||||
<p x-show="! isFormValid" class="px-4 py-2 text-sm text-co-red">Certains champs de sont pas valides.</p>
|
|
||||||
<a href="/app/vehicles-management/fleet/{{.ViewState.vehicle.ID}}">
|
|
||||||
<button type="button"
|
|
||||||
class="bg-white py-2 px-4 border border-gray-300 rounded-2xl shadow-sm text-sm font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Annuler</button>
|
|
||||||
</a>
|
|
||||||
<button type="submit"
|
|
||||||
class="ml-3 inline-flex justify-center py-2 px-4 border border-transparent shadow-sm text-sm font-medium rounded-2xl text-white bg-co-blue hover:bg-co-blue focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-co-blue">Modifier
|
|
||||||
le véhicule</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
{{end}}
|
|
||||||
@@ -1,31 +0,0 @@
|
|||||||
{{define "content"}}
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">Gestion des véhicules et réservations</h1>
|
|
||||||
|
|
||||||
<h2 class="text-xl font-semibold text-gray-600 pt-8">Réservations</h2>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{{template "bookings_list" .}}
|
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 md:px-8">
|
|
||||||
<h2 class="text-xl font-semibold text-gray-600 pt-8">Véhicules</h2>
|
|
||||||
|
|
||||||
<div class="sm:flex sm:items-center">
|
|
||||||
<div class="sm:flex-auto">
|
|
||||||
<p class="mt-2 text-sm text-gray-700"></p>
|
|
||||||
</div>
|
|
||||||
<div class="mt-4 sm:mt-0 sm:ml-16 sm:flex-none">
|
|
||||||
<a href="/app/vehicles-management/fleet/add">
|
|
||||||
<button type="button"
|
|
||||||
class="inline-flex items-center justify-center rounded-2xl border border-transparent bg-co-blue px-4 py-2 text-sm font-medium text-white shadow-sm focus:outline-none focus:ring-2 focus:ring-co-blue focus:ring-offset-2 sm:w-auto">
|
|
||||||
{{$.IconSet.Icon "hero:outline/plus-circle" "h-5 w-5 mr-3"}}
|
|
||||||
Ajouter un véhicule
|
|
||||||
</button>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{{template "vehicles_list" .}}
|
|
||||||
|
|
||||||
{{end}}
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1 +0,0 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 374.2 33.2"><defs><style>.cls-1{fill:#ff1300;}.cls-2{fill:#243887;}</style></defs><g id="Calque_2" data-name="Calque 2"><g id="Calque_1-2" data-name="Calque 1"><path class="cls-1" d="M300.9.6a4.7,4.7,0,0,1,4.9,4.9V32.6h-8.6a4.7,4.7,0,0,1-4.9-4.9V14.1h-.9c-.6,0-1,.6-1,1.5v17H282a4.7,4.7,0,0,1-4.9-4.9V14.1h-1a1.4,1.4,0,0,0-.9,1.5v17H263a1.2,1.2,0,0,1-1.2-1.2V5.5A4.7,4.7,0,0,1,266.7.6h17.9l-.7,5a.9.9,0,0,0,.7,1.1.9.9,0,0,0,1-.8l.8-4.6a.9.9,0,0,1,.9-.7Z"/><path class="cls-1" d="M341.1,16.4c0,9.1-1.9,16.8-16.5,16.8S307.8,25,307.8,16.4,310.6,0,324.6,0,341.1,7.8,341.1,16.4Z"/><path class="cls-1" d="M344.3,32.6a1.2,1.2,0,0,1-1.2-1.2V1.8A1.2,1.2,0,0,1,344.3.6h18.8c6.9,0,10.4,2.9,10.4,8.1s-1.6,6.9-4.5,7.5h0c2.9.6,5.2,3.2,5.2,7.7s-3.5,8.6-10.4,8.6Z"/><path class="cls-2" d="M16.2,11.6c0-.6-.4-.9-1-.9h-.8V31.4a1.2,1.2,0,0,1-1.2,1.2H1.2A1.2,1.2,0,0,1,0,31.4V1.8A1.2,1.2,0,0,1,1.2.6H16.8C25.7.6,30.9,4.2,31,10.7s-4.1,9.9-9.4,10H16.1Z"/><path class="cls-2" d="M44.9,23.1H43.2L40,31.9a1.1,1.1,0,0,1-1.2.7H26.9a1.3,1.3,0,0,1-1.3-1.9L37,3.4A4.5,4.5,0,0,1,41.3.6h5.8a4.1,4.1,0,0,1,4,2.6L62.6,30.8c.5,1-.2,1.8-1.3,1.8H41.7Z"/><path class="cls-2" d="M66.1,32.6a1.2,1.2,0,0,1-1.2-1.2V1.8A1.2,1.2,0,0,1,66.1.6H83.2C92.7.6,95.5,7,95.6,12.3c0,3.3-.8,6.9-3.4,8.3l3.4,12H84.4c-2.3,0-5.2-.7-5.9-2.7L71.9,10.7a1,1,0,0,0-1.2-.7,1,1,0,0,0-.5,1.3l7.2,21.3Z"/><path class="cls-2" d="M129.6,17.1c.8,0,1.3.5,1.2,1.2-.2,8.3-2.9,14.9-16.4,14.9-15.8.1-16.8-8.2-16.8-16.8S100.5,0,114.4,0c15.7,0,16.5,8,16.5,15.4H115.3V11.3c0-.8-.4-1.2-1-1.2s-.9.4-.9,1.2V21.9c0,.9.4,1.2.9,1.2s1-.3,1-1.2V17.1Z"/><path class="cls-2" d="M166.2,16.4c0,9.1-2,16.8-16.6,16.8S132.8,25,132.8,16.4,135.7,0,149.6,0,166.2,7.8,166.2,16.4Z"/><path class="cls-2" d="M200.5,1.8V12.2c0,12.7-1.4,21-16.3,21s-16-8.4-16-20.9V1.8A1.2,1.2,0,0,1,169.4.6h13a.9.9,0,0,1,.9.9V21.8c0,.9.4,1.2,1,1.2s.9-.3.9-1.2V1.5a.9.9,0,0,1,.9-.9h13.1A1.2,1.2,0,0,1,200.5,1.8Z"/><path class="cls-2" d="M203.9,32.6a1.2,1.2,0,0,1-1.3-1.2V1.8A1.2,1.2,0,0,1,203.8.6h17.1c9.5,0,12.3,6.4,12.4,11.7,0,3.3-.8,6.9-3.4,8.3l3.4,12H222.1c-2.3,0-5.2-.7-5.9-2.7l-6.6-19.2a1,1,0,0,0-1.2-.7,1,1,0,0,0-.5,1.3l7.2,21.3Z"/><path class="cls-2" d="M235.8,29.2v-10a1,1,0,0,1,1.5-1l11.1,5.1a1,1,0,0,0,1.2-.4.9.9,0,0,0-.6-1.2l-9.3-4.4c-4.5-2.2-4.4-5.5-4.4-8.2,0-7.6,7.4-9.1,14.6-9.1,5.4,0,12.1,1.8,14.3,4.4v9.2a1,1,0,0,1-1.5,1L252.3,9.8a1,1,0,0,0-1.3.4q-.3.8.6,1.2l8.5,4.1c5.3,2.6,4.7,4.9,4.7,9.2s-6.3,8.5-14.9,8.5C244.7,33.2,238,31.6,235.8,29.2Z"/></g></g></svg>
|
|
||||||
|
Before Width: | Height: | Size: 2.4 KiB |
@@ -1 +0,0 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 374.2 33.2"><defs><style>.cls-1{fill:#ff1300;}.cls-2{fill:#fff;}</style></defs><g id="Calque_2" data-name="Calque 2"><g id="Calque_1-2" data-name="Calque 1"><path class="cls-1" d="M300.9.6a4.7,4.7,0,0,1,4.9,4.9V32.6h-8.6a4.7,4.7,0,0,1-4.9-4.9V14.1h-.9c-.6,0-1,.6-1,1.5v17H282a4.7,4.7,0,0,1-4.9-4.9V14.1h-1a1.4,1.4,0,0,0-.9,1.5v17H263a1.2,1.2,0,0,1-1.2-1.2V5.5A4.7,4.7,0,0,1,266.7.6h17.9l-.7,5a.9.9,0,0,0,.7,1.1.9.9,0,0,0,1-.8l.8-4.6a.9.9,0,0,1,.9-.7Z"/><path class="cls-1" d="M341.1,16.4c0,9.1-1.9,16.8-16.5,16.8S307.8,25,307.8,16.4,310.6,0,324.6,0,341.1,7.8,341.1,16.4Z"/><path class="cls-1" d="M344.3,32.6a1.2,1.2,0,0,1-1.2-1.2V1.8A1.2,1.2,0,0,1,344.3.6h18.8c6.9,0,10.4,2.9,10.4,8.1s-1.6,6.9-4.5,7.5h0c2.9.6,5.2,3.2,5.2,7.7s-3.5,8.6-10.4,8.6Z"/><path class="cls-2" d="M16.2,11.6c0-.6-.4-.9-1-.9h-.8V31.4a1.2,1.2,0,0,1-1.2,1.2H1.2A1.2,1.2,0,0,1,0,31.4V1.8A1.2,1.2,0,0,1,1.2.6H16.8C25.7.6,30.9,4.2,31,10.7s-4.1,9.9-9.4,10H16.1Z"/><path class="cls-2" d="M44.9,23.1H43.2L40,31.9a1.1,1.1,0,0,1-1.2.7H26.9a1.3,1.3,0,0,1-1.3-1.9L37,3.4A4.5,4.5,0,0,1,41.3.6h5.8a4.1,4.1,0,0,1,4,2.6L62.6,30.8c.5,1-.2,1.8-1.3,1.8H41.7Z"/><path class="cls-2" d="M66.1,32.6a1.2,1.2,0,0,1-1.2-1.2V1.8A1.2,1.2,0,0,1,66.1.6H83.2C92.7.6,95.5,7,95.6,12.3c0,3.3-.8,6.9-3.4,8.3l3.4,12H84.4c-2.3,0-5.2-.7-5.9-2.7L71.9,10.7a1,1,0,0,0-1.2-.7,1,1,0,0,0-.5,1.3l7.2,21.3Z"/><path class="cls-2" d="M129.6,17.1c.8,0,1.3.5,1.2,1.2-.2,8.3-2.9,14.9-16.4,14.9-15.8.1-16.8-8.2-16.8-16.8S100.5,0,114.4,0c15.7,0,16.5,8,16.5,15.4H115.3V11.3c0-.8-.4-1.2-1-1.2s-.9.4-.9,1.2V21.9c0,.9.4,1.2.9,1.2s1-.3,1-1.2V17.1Z"/><path class="cls-2" d="M166.2,16.4c0,9.1-2,16.8-16.6,16.8S132.8,25,132.8,16.4,135.7,0,149.6,0,166.2,7.8,166.2,16.4Z"/><path class="cls-2" d="M200.5,1.8V12.2c0,12.7-1.4,21-16.3,21s-16-8.4-16-20.9V1.8A1.2,1.2,0,0,1,169.4.6h13a.9.9,0,0,1,.9.9V21.8c0,.9.4,1.2,1,1.2s.9-.3.9-1.2V1.5a.9.9,0,0,1,.9-.9h13.1A1.2,1.2,0,0,1,200.5,1.8Z"/><path class="cls-2" d="M203.9,32.6a1.2,1.2,0,0,1-1.3-1.2V1.8A1.2,1.2,0,0,1,203.8.6h17.1c9.5,0,12.3,6.4,12.4,11.7,0,3.3-.8,6.9-3.4,8.3l3.4,12H222.1c-2.3,0-5.2-.7-5.9-2.7l-6.6-19.2a1,1,0,0,0-1.2-.7,1,1,0,0,0-.5,1.3l7.2,21.3Z"/><path class="cls-2" d="M235.8,29.2v-10a1,1,0,0,1,1.5-1l11.1,5.1a1,1,0,0,0,1.2-.4.9.9,0,0,0-.6-1.2l-9.3-4.4c-4.5-2.2-4.4-5.5-4.4-8.2,0-7.6,7.4-9.1,14.6-9.1,5.4,0,12.1,1.8,14.3,4.4v9.2a1,1,0,0,1-1.5,1L252.3,9.8a1,1,0,0,0-1.3.4q-.3.8.6,1.2l8.5,4.1c5.3,2.6,4.7,4.9,4.7,9.2s-6.3,8.5-14.9,8.5C244.7,33.2,238,31.6,235.8,29.2Z"/></g></g></svg>
|
|
||||||
|
Before Width: | Height: | Size: 2.4 KiB |
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user