Compare commits
6 Commits
Author | SHA1 | Date |
---|---|---|
Jimmy Pellier | 045d1a54a3 | 5 years ago |
Jimmy Pellier | 085c544587 | 5 years ago |
Jimmy Pellier | 4cb28e49de | 5 years ago |
Jimmy Pellier | 04c5230b6d | 5 years ago |
Jimmy Pellier | be9a12a9d8 | 5 years ago |
Jimmy Pellier | 1a8b9302d9 | 5 years ago |
@ -0,0 +1,8 @@ |
||||
VUE_APP_BACK_URL=http://localhost/covid |
||||
VUE_APP_BACK_USER=http://localhost/user |
||||
VUE_APP_IMAGE_STORE=http://localhost/apache/public/uploads/ |
||||
VUE_APP_TEMPERATURE_HUB=http://localhost/covid/batchExecutionHub |
||||
VUE_APP_TIMEOUT_TEMPERATURE_HUB=5000 |
||||
VUE_APP_MSG_TEMPERATURE_HUB=temperature |
||||
VUE_APP_EMIT_MSG_TEMPERATURE=received-temperature |
||||
VUE_APP_MAX_CHART=30 |
@ -0,0 +1,8 @@ |
||||
VUE_APP_BACK_URL=http://localhost/covid |
||||
VUE_APP_BACK_USER=http://localhost/user |
||||
VUE_APP_IMAGE_STORE=http://localhost/apache/public/uploads/ |
||||
VUE_APP_TEMPERATURE_HUB=http://localhost/covid/batchExecutionHub |
||||
VUE_APP_TIMEOUT_TEMPERATURE_HUB=5000 |
||||
VUE_APP_MSG_TEMPERATURE_HUB=temperature |
||||
VUE_APP_EMIT_MSG_TEMPERATURE=received-temperature |
||||
VUE_APP_MAX_CHART=30 |
@ -0,0 +1,21 @@ |
||||
.DS_Store |
||||
node_modules |
||||
/dist |
||||
|
||||
# local env files |
||||
.env.local |
||||
.env.*.local |
||||
|
||||
# Log files |
||||
npm-debug.log* |
||||
yarn-debug.log* |
||||
yarn-error.log* |
||||
|
||||
# Editor directories and files |
||||
.idea |
||||
.vscode |
||||
*.suo |
||||
*.ntvs* |
||||
*.njsproj |
||||
*.sln |
||||
*.sw? |
@ -0,0 +1,13 @@ |
||||
# étape de build |
||||
FROM node:lts-alpine as build-stage |
||||
WORKDIR /app |
||||
COPY package*.json ./ |
||||
RUN npm install |
||||
COPY . . |
||||
RUN npm run build |
||||
|
||||
# étape de production |
||||
FROM nginx:stable-alpine as production-stage |
||||
COPY --from=build-stage /app/dist /usr/share/nginx/html |
||||
EXPOSE 80 |
||||
CMD ["nginx", "-g", "daemon off;"] |
@ -1,2 +1,29 @@ |
||||
# TemperatureCheckerFront |
||||
# clientapp |
||||
|
||||
## Project setup |
||||
``` |
||||
npm install |
||||
``` |
||||
|
||||
### Compiles and hot-reloads for development |
||||
``` |
||||
npm run serve |
||||
``` |
||||
|
||||
### Compiles and minifies for production |
||||
``` |
||||
npm run build |
||||
``` |
||||
|
||||
### Run your tests |
||||
``` |
||||
npm run test |
||||
``` |
||||
|
||||
### Lints and fixes files |
||||
``` |
||||
npm run lint |
||||
``` |
||||
|
||||
### Customize configuration |
||||
See [Configuration Reference](https://cli.vuejs.org/config/). |
||||
|
@ -0,0 +1,6 @@ |
||||
module.exports = { |
||||
presets: [ |
||||
'@vue/cli-plugin-babel/preset' |
||||
], |
||||
|
||||
} |
@ -0,0 +1,34 @@ |
||||
.corps { |
||||
height: 80vh; |
||||
display: block; |
||||
position: relative; |
||||
|
||||
} |
||||
|
||||
.bg-passageOk |
||||
{ |
||||
background-color: seagreen; |
||||
} |
||||
|
||||
.corps::after { |
||||
content: ""; |
||||
background-repeat:no-repeat; |
||||
background-position:center center; |
||||
background-attachment:fixed; |
||||
background-size: 50vh auto; |
||||
background-image: url("../src/assets/logo.png"); |
||||
opacity: 0.5; |
||||
top: 0; |
||||
left: 0; |
||||
bottom: 0; |
||||
right: 0; |
||||
position: absolute; |
||||
z-index: -1; |
||||
} |
||||
|
||||
.fade-enter-active{ |
||||
transition: opacity .5s; |
||||
} |
||||
.fade-enter /* .fade-leave-active below version 2.1.8 */ { |
||||
opacity: 0; |
||||
} |
@ -0,0 +1,296 @@ |
||||
@charset "UTF-8"; |
||||
@import url("https://fonts.googleapis.com/css?family=Ubuntu:400,700|Cabin+Condensed:400,600"); |
||||
|
||||
|
||||
|
||||
body { |
||||
background-color: #FFF; |
||||
margin: 0px; |
||||
font-family: Ubuntu, sans-serif; |
||||
color: #1e1e1e; |
||||
font-weight: normal; |
||||
padding-top: 0; |
||||
} |
||||
|
||||
h1, h2, h3, h4 { |
||||
font-family: "Cabin Condensed", sans-serif; |
||||
} |
||||
|
||||
header { |
||||
padding: 1em; |
||||
} |
||||
|
||||
header .headline { |
||||
max-width: 640px; |
||||
margin: 0 auto; |
||||
} |
||||
|
||||
header .headline h1 { |
||||
font-size: 3em; |
||||
margin-bottom: 0; |
||||
} |
||||
|
||||
header .headline h2 { |
||||
margin-top: 0.2em; |
||||
} |
||||
|
||||
footer { |
||||
padding: 1em 2em 2em; |
||||
} |
||||
|
||||
#container { |
||||
width: 640px; |
||||
margin: 20px auto; |
||||
padding: 10px; |
||||
} |
||||
|
||||
#interactive.viewport { |
||||
width: 640px; |
||||
height: 480px; |
||||
} |
||||
|
||||
|
||||
#interactive.viewport canvas, video { |
||||
float: left; |
||||
width: 640px; |
||||
height: 480px; |
||||
} |
||||
|
||||
#interactive.viewport canvas.drawingBuffer, video.drawingBuffer { |
||||
margin-left: -640px; |
||||
} |
||||
|
||||
.controls fieldset { |
||||
border: none; |
||||
margin: 0; |
||||
padding: 5px; |
||||
} |
||||
|
||||
.controls .input-group { |
||||
float: left; |
||||
} |
||||
|
||||
.controls .input-group input, .controls .input-group button { |
||||
display: block; |
||||
} |
||||
|
||||
.controls .reader-config-group { |
||||
float: right; |
||||
} |
||||
|
||||
.controls .reader-config-group label { |
||||
display: block; |
||||
} |
||||
|
||||
.controls .reader-config-group label span { |
||||
width: 9rem; |
||||
display: inline-block; |
||||
text-align: right; |
||||
} |
||||
|
||||
.controls:after { |
||||
content: ''; |
||||
display: block; |
||||
clear: both; |
||||
} |
||||
|
||||
|
||||
#result_strip { |
||||
margin: 10px 0; |
||||
border-top: 1px solid #EEE; |
||||
border-bottom: 1px solid #EEE; |
||||
padding: 10px 0; |
||||
} |
||||
|
||||
#result_strip > ul { |
||||
padding: 0; |
||||
margin: 0; |
||||
list-style-type: none; |
||||
width: auto; |
||||
overflow-x: auto; |
||||
overflow-y: hidden; |
||||
white-space: nowrap; |
||||
} |
||||
|
||||
#result_strip > ul > li { |
||||
display: inline-block; |
||||
vertical-align: middle; |
||||
width: 160px; |
||||
} |
||||
|
||||
#result_strip > ul > li .thumbnail { |
||||
padding: 5px; |
||||
margin: 4px; |
||||
border: 1px dashed #CCC; |
||||
} |
||||
|
||||
#result_strip > ul > li .thumbnail img { |
||||
max-width: 140px; |
||||
} |
||||
|
||||
#result_strip > ul > li .thumbnail .caption { |
||||
white-space: normal; |
||||
} |
||||
|
||||
#result_strip > ul > li .thumbnail .caption h4 { |
||||
text-align: center; |
||||
word-wrap: break-word; |
||||
height: 40px; |
||||
margin: 0px; |
||||
} |
||||
|
||||
#result_strip > ul:after { |
||||
content: ""; |
||||
display: table; |
||||
clear: both; |
||||
} |
||||
|
||||
|
||||
.scanner-overlay { |
||||
display: none; |
||||
width: 640px; |
||||
height: 510px; |
||||
position: absolute; |
||||
padding: 20px; |
||||
top: 50%; |
||||
margin-top: -275px; |
||||
left: 50%; |
||||
margin-left: -340px; |
||||
background-color: #FFF; |
||||
-moz-box-shadow: #333333 0px 4px 10px; |
||||
-webkit-box-shadow: #333333 0px 4px 10px; |
||||
box-shadow: #333333 0px 4px 10px; |
||||
} |
||||
|
||||
.scanner-overlay > .header { |
||||
position: relative; |
||||
margin-bottom: 14px; |
||||
} |
||||
|
||||
.scanner-overlay > .header h4, .scanner-overlay > .header .close { |
||||
line-height: 16px; |
||||
} |
||||
|
||||
.scanner-overlay > .header h4 { |
||||
margin: 0px; |
||||
padding: 0px; |
||||
} |
||||
|
||||
.scanner-overlay > .header .close { |
||||
position: absolute; |
||||
right: 0px; |
||||
top: 0px; |
||||
height: 16px; |
||||
width: 16px; |
||||
text-align: center; |
||||
font-weight: bold; |
||||
font-size: 14px; |
||||
cursor: pointer; |
||||
} |
||||
|
||||
|
||||
i.icon-24-scan { |
||||
width: 24px; |
||||
height: 24px; |
||||
background-image: url(""); |
||||
display: inline-block; |
||||
background-repeat: no-repeat; |
||||
line-height: 24px; |
||||
margin-top: 1px; |
||||
vertical-align: text-top; |
||||
} |
||||
|
||||
@media (max-width: 603px) { |
||||
|
||||
#container { |
||||
width: 300px; |
||||
margin: 10px auto; |
||||
-moz-box-shadow: none; |
||||
-webkit-box-shadow: none; |
||||
box-shadow: none; |
||||
} |
||||
|
||||
#container form.voucher-form input.voucher-code { |
||||
width: 180px; |
||||
} |
||||
} |
||||
@media (max-width: 603px) { |
||||
|
||||
.reader-config-group { |
||||
width: 100%; |
||||
} |
||||
|
||||
.reader-config-group label > span { |
||||
width: 50%; |
||||
} |
||||
|
||||
.reader-config-group label > select, .reader-config-group label > input { |
||||
max-width: calc(50% - 2px); |
||||
} |
||||
|
||||
#interactive.viewport { |
||||
width: 300px; |
||||
height: 300px; |
||||
overflow: hidden; |
||||
} |
||||
|
||||
|
||||
#interactive.viewport canvas, video { |
||||
margin-top: -50px; |
||||
width: 300px; |
||||
height: 400px; |
||||
} |
||||
|
||||
#interactive.viewport canvas.drawingBuffer, video.drawingBuffer { |
||||
margin-left: -300px; |
||||
} |
||||
|
||||
|
||||
#result_strip { |
||||
margin-top: 5px; |
||||
padding-top: 5px; |
||||
} |
||||
|
||||
#result_strip ul.thumbnails > li { |
||||
width: 150px; |
||||
} |
||||
|
||||
#result_strip ul.thumbnails > li .thumbnail .imgWrapper { |
||||
width: 130px; |
||||
height: 130px; |
||||
overflow: hidden; |
||||
} |
||||
|
||||
#result_strip ul.thumbnails > li .thumbnail .imgWrapper img { |
||||
margin-top: -25px; |
||||
width: 130px; |
||||
height: 180px; |
||||
} |
||||
} |
||||
@media (max-width: 603px) { |
||||
|
||||
.overlay.scanner { |
||||
width: 640px; |
||||
height: 510px; |
||||
padding: 20px; |
||||
margin-top: -275px; |
||||
margin-left: -340px; |
||||
background-color: #FFF; |
||||
-moz-box-shadow: none; |
||||
-webkit-box-shadow: none; |
||||
box-shadow: none; |
||||
} |
||||
|
||||
.overlay.scanner > .header { |
||||
margin-bottom: 14px; |
||||
} |
||||
|
||||
.overlay.scanner > .header h4, .overlay.scanner > .header .close { |
||||
line-height: 16px; |
||||
} |
||||
|
||||
.overlay.scanner > .header .close { |
||||
height: 16px; |
||||
width: 16px; |
||||
} |
||||
} |
@ -0,0 +1 @@ |
||||
tempfront |
@ -0,0 +1,64 @@ |
||||
{ |
||||
"name": "coviiiiid", |
||||
"version": "0.1.0", |
||||
"private": true, |
||||
"scripts": { |
||||
"serve": "node_modules/.bin/vue-cli-service serve", |
||||
"build": "node_modules/.bin/vue-cli-service build", |
||||
"lint": "node_modules/.bin/vue-cli-service lint" |
||||
}, |
||||
"dependencies": { |
||||
"@aspnet/signalr": "^1.1.4", |
||||
"@fortawesome/fontawesome-svg-core": "^1.2.28", |
||||
"@fortawesome/free-solid-svg-icons": "^5.13.0", |
||||
"@fortawesome/vue-fontawesome": "^0.1.9", |
||||
"axios": "^0.19.2", |
||||
"bootstrap-vue": "^2.15.0", |
||||
"chart.js": "^2.9.3", |
||||
"commander": "^4.0.1", |
||||
"core-js": "^3.4.3", |
||||
"debounce": "^1.2.0", |
||||
"es6-promise": "^4.2.8", |
||||
"jquery": "^1.9.1", |
||||
"keycloak-js": "^10.0.2", |
||||
"nimble": "0.0.2", |
||||
"node-datetime": "^2.1.2", |
||||
"request-digest": "^1.0.13", |
||||
"socket.io": "^2.3.0", |
||||
"vue": "^2.6.10", |
||||
"vue-chartjs": "^3.5.0", |
||||
"vue-i18n": "^8.18.1", |
||||
"vue-infinite-loading": "^2.4.5", |
||||
"vue-js-toggle-button": "^1.3.3", |
||||
"vue-router": "^3.3.2", |
||||
"vuex": "^3.4.0", |
||||
"xml2js": "^0.4.22" |
||||
}, |
||||
"devDependencies": { |
||||
"@vue/cli-plugin-babel": "^4.1.0", |
||||
"@vue/cli-plugin-eslint": "^4.1.0", |
||||
"@vue/cli-service": "^4.1.0", |
||||
"babel-eslint": "^10.0.3", |
||||
"eslint": "^5.16.0", |
||||
"eslint-plugin-vue": "^5.0.0", |
||||
"vue-template-compiler": "^2.6.10" |
||||
}, |
||||
"eslintConfig": { |
||||
"root": true, |
||||
"env": { |
||||
"node": true |
||||
}, |
||||
"extends": [ |
||||
"plugin:vue/essential", |
||||
"eslint:recommended" |
||||
], |
||||
"rules": {}, |
||||
"parserOptions": { |
||||
"parser": "babel-eslint" |
||||
} |
||||
}, |
||||
"browserslist": [ |
||||
"> 1%", |
||||
"last 2 versions" |
||||
] |
||||
} |
After Width: | Height: | Size: 4.2 KiB |
@ -0,0 +1,17 @@ |
||||
<!DOCTYPE html> |
||||
<html lang="en"> |
||||
<head> |
||||
<meta charset="utf-8"> |
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge"> |
||||
<meta name="viewport" content="width=device-width,initial-scale=1.0"> |
||||
<link rel="icon" href="<%= BASE_URL %>favicon.ico"> |
||||
<title>covid finder</title> |
||||
</head> |
||||
<body> |
||||
<noscript> |
||||
<strong>We're sorry but clientapp doesn't work properly without JavaScript enabled. Please enable it to continue.</strong> |
||||
</noscript> |
||||
<div id="app"></div> |
||||
<!-- built files will be auto injected --> |
||||
</body> |
||||
</html> |
@ -0,0 +1,10 @@ |
||||
{ |
||||
"realm": "Test", |
||||
"auth-server-url": "https://oauth.apsidetop-devel.net/auth", |
||||
"ssl-required": "external", |
||||
"resource": "test", |
||||
"credentials": { |
||||
"secret": "5cf010f0-b4a9-4db9-a9f2-102bc703c448" |
||||
}, |
||||
"confidential-port": 0 |
||||
} |
@ -0,0 +1,59 @@ |
||||
<template> |
||||
<b-container fluid id="app"> |
||||
<Menu class="header" /> |
||||
<div style="margin-top:60px"> |
||||
<router-view > </router-view> |
||||
</div> |
||||
</b-container> |
||||
</template> |
||||
|
||||
<script> |
||||
import Menu from './components/Menu.vue' |
||||
import Vue from 'vue' |
||||
import { BootstrapVue, IconsPlugin } from 'bootstrap-vue' |
||||
import 'bootstrap/dist/css/bootstrap.css' |
||||
import 'bootstrap-vue/dist/bootstrap-vue.css' |
||||
|
||||
import { library } from '@fortawesome/fontawesome-svg-core' |
||||
import { faUserSecret } from '@fortawesome/free-solid-svg-icons' |
||||
import { FontAwesomeIcon } from '@fortawesome/vue-fontawesome' |
||||
|
||||
library.add(faUserSecret) |
||||
|
||||
Vue.component('font-awesome-icon', FontAwesomeIcon) |
||||
|
||||
// Install BootstrapVue |
||||
Vue.use(BootstrapVue) |
||||
// Optionally install the BootstrapVue icon components plugin |
||||
Vue.use(IconsPlugin) |
||||
|
||||
export default { |
||||
name: 'app', |
||||
components: { |
||||
Menu |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style> |
||||
#app { |
||||
font-family: Avenir, Helvetica, Arial, sans-serif; |
||||
-webkit-font-smoothing: antialiased; |
||||
-moz-osx-font-smoothing: grayscale; |
||||
text-align: center; |
||||
color: #2c3e50; |
||||
margin-top: 0px; |
||||
padding-left: 0; |
||||
padding-right: 0 |
||||
} |
||||
.header { |
||||
position:fixed; /* fixing the position takes it out of html flow - knows |
||||
nothing about where to locate itself except by browser |
||||
coordinates */ |
||||
left:0; /* top left corner should start at leftmost spot */ |
||||
top:0; /* top left corner should start at topmost spot */ |
||||
width:100vw; /* take up the full browser width */ |
||||
z-index:100; /* high z index so other content scrolls underneath */ |
||||
height:50px; /* define height for content */ |
||||
} |
||||
</style> |
After Width: | Height: | Size: 6.7 KiB |
After Width: | Height: | Size: 32 KiB |
After Width: | Height: | Size: 1.5 MiB |
After Width: | Height: | Size: 15 KiB |
After Width: | Height: | Size: 6.9 KiB |
After Width: | Height: | Size: 4.7 KiB |
After Width: | Height: | Size: 19 KiB |
@ -0,0 +1,91 @@ |
||||
<template> |
||||
<div id="videoapp"> |
||||
<b-button v-b-modal.modal-2 > |
||||
<font-awesome-icon icon="camera" /> |
||||
{{$t('user_detail_photo')}} |
||||
</b-button> |
||||
<b-modal static size="xl" id="modal-2" :hide-footer="true" > |
||||
<template v-slot:modal-header> |
||||
Title |
||||
<b-button size="sm" variant="danger" @click="close()"> |
||||
Cancel |
||||
</b-button> |
||||
|
||||
</template> |
||||
<template v-slot:default> |
||||
<div> |
||||
<div><video ref="video" id="video" width="640" height="480" autoplay ></video></div> |
||||
<div><button id="snap" v-on:click="capture()">Snap Photo</button></div> |
||||
<canvas ref="canvas" id="canvas" width="640" height="480"></canvas> |
||||
<ul> |
||||
<li v-for="c in captures" v-bind:key="c.id"> |
||||
<img v-bind:src="c.photo" alt="Photo" height="50" /> |
||||
</li> |
||||
</ul> |
||||
</div> |
||||
</template> |
||||
</b-modal> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
name: "CapturePhoto", |
||||
mounted() { |
||||
this.video = this.$refs.video; |
||||
let video = document.querySelector('video'); |
||||
if(navigator.mediaDevices && navigator.mediaDevices.getUserMedia) { |
||||
navigator.mediaDevices.getUserMedia({ video: true }).then(stream => { |
||||
video.srcObject = stream |
||||
}); |
||||
} |
||||
}, |
||||
data () { |
||||
return { |
||||
video: {}, |
||||
canvas: {}, |
||||
compteur:0, |
||||
captures: [], |
||||
snapshot:{} |
||||
} |
||||
}, |
||||
methods: { |
||||
close() |
||||
{ |
||||
this.snapshot= null; |
||||
this.$bvModal.hide("modal-2"); |
||||
}, |
||||
capture() { |
||||
this.canvas = this.$refs.canvas; |
||||
this.canvas.getContext("2d").drawImage(this.video, 0, 0, 640, 480); |
||||
|
||||
this.compteur++; |
||||
this.snapshot = {photo:this.canvas.toDataURL("image/png"), id:this.compteur++} |
||||
// this.captures.push(this.snapshot); |
||||
this.$emit('update:snapshot', this.snapshot); |
||||
this.$bvModal.hide("modal-2"); |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style> |
||||
body: { |
||||
background-color: #F0F0F0; |
||||
} |
||||
/* #app { |
||||
text-align: center; |
||||
color: #2c3e50; |
||||
margin-top: 60px; |
||||
}*/ |
||||
#video { |
||||
background-color: #000000; |
||||
} |
||||
#canvas { |
||||
display: none; |
||||
} |
||||
li { |
||||
display: inline; |
||||
padding: 5px; |
||||
} |
||||
</style> |
@ -0,0 +1,160 @@ |
||||
<template> |
||||
<div class="MenuNavBar"> |
||||
<b-navbar toggleable="lg" type="dark" variant="info" style="min-height:60px; padding: 0px 0px 0px 0px"> |
||||
|
||||
<b-navbar-brand class="nav-link d-flex align-items-center justify-content-center" @click="doLogin"> |
||||
<b-avatar variant="info" v-bind:src="img" size="3rem" > |
||||
</b-avatar> |
||||
</b-navbar-brand> |
||||
|
||||
<b-navbar-brand class="nav-link d-flex align-items-center justify-content-center" to="/" > |
||||
{{$t('title')}} |
||||
</b-navbar-brand> |
||||
<b-navbar-toggle target="nav-collapse"></b-navbar-toggle> |
||||
<b-collapse id="nav-collapse" is-nav> |
||||
<b-navbar-nav> |
||||
<b-nav-item v-if="isAdmin" active-class="active" class="nav-link d-flex align-items-center justify-content-center" to="/users" >{{$t('user_title')}}</b-nav-item> |
||||
<b-nav-item active-class="active" class="nav-link d-flex align-items-center justify-content-center" @click="logout" >{{$t('logout')}}</b-nav-item> |
||||
</b-navbar-nav> |
||||
|
||||
</b-collapse> |
||||
|
||||
</b-navbar> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
|
||||
import { BNavbar, BNavbarBrand, BNavbarToggle } from 'bootstrap-vue' |
||||
import { BAvatar } from 'bootstrap-vue' |
||||
import UserService from "../services/UserService"; |
||||
import Vue from "vue"; |
||||
import axios from "axios"; |
||||
|
||||
Vue.component('b-avatar', BAvatar) |
||||
|
||||
export default { |
||||
name: "Menu", |
||||
components : |
||||
{ |
||||
BNavbar,BNavbarBrand, BNavbarToggle, BAvatar |
||||
}, |
||||
data:function(){ |
||||
return { |
||||
} |
||||
}, |
||||
computed : |
||||
{ |
||||
user(){ |
||||
return this.$store.state.user; |
||||
}, |
||||
img() |
||||
{ |
||||
return this.$store.state.user != null && this.$store.state.user.image != null ? |
||||
process.env.VUE_APP_IMAGE_STORE + this.$store.state.user.image : |
||||
''; |
||||
}, |
||||
isAdmin() { |
||||
return this.$store.state.roles.find(l => l.name == 'Administrators' && l.isActived) != null; |
||||
}, |
||||
isClient() { |
||||
return this.$store.state.roles.find(l => l.name == 'customer' && l.isActived) != null; |
||||
}, |
||||
isLivreur() { |
||||
return this.$store.state.roles.find(l => l.name == 'delivery' && l.isActived) != null; |
||||
}, |
||||
isManager() { |
||||
return this.$store.state.roles.find(l => l.name == 'manager' && l.isActived) != null; |
||||
}, |
||||
}, |
||||
mounted() { |
||||
let that = this; |
||||
UserService.initKeycloak( function () { |
||||
sessionStorage.setItem("react-token", UserService.getToken()); |
||||
sessionStorage.setItem("react-refresh-token", UserService.getRefreshToken()); |
||||
|
||||
let axiosConfig = { |
||||
withCredentials: true, |
||||
headers: { |
||||
'Authorization': 'Bearer ' + sessionStorage.getItem("react-token"), |
||||
} |
||||
}; |
||||
|
||||
axios.get(process.env.VUE_APP_BACK_USER + "/api/users/bykeycloakid?guid=" + UserService.getUsername().sub, |
||||
axiosConfig |
||||
).then(result => { |
||||
if( result.data.keycloakId !== null) |
||||
that.$store.commit("updateUser", result.data); |
||||
else |
||||
UserService.doLogout(); |
||||
}); |
||||
|
||||
axios.get(process.env.VUE_APP_BACK_USER + "/api/roles/bykeycloakid?keycloakid=" + UserService.getUsername().sub, |
||||
axiosConfig |
||||
).then(result => { |
||||
that.$store.commit("updateRole", result.data); |
||||
}); |
||||
|
||||
}, |
||||
function(){ |
||||
that.$store.commit("updateUser", null); |
||||
that.$store.commit("updateRole", []); |
||||
if(that.$router.currentRoute.path !== "/") |
||||
that.$router.push({path:"/"}); |
||||
}); |
||||
|
||||
}, |
||||
methods : { |
||||
logout: function () |
||||
{ |
||||
UserService.doLogout(); |
||||
sessionStorage.setItem("react-token", ""); |
||||
sessionStorage.setItem("react-refresh-token", ""); |
||||
|
||||
|
||||
}, |
||||
doLogin : function() |
||||
{ |
||||
|
||||
let that = this; |
||||
UserService.initKeycloak( function () { |
||||
|
||||
sessionStorage.setItem("react-token", UserService.getToken()); |
||||
sessionStorage.setItem("react-refresh-token", UserService.getRefreshToken()); |
||||
|
||||
let axiosConfig = { |
||||
withCredentials: true, |
||||
headers: { |
||||
'Authorization': 'Bearer ' + sessionStorage.getItem("react-token"), |
||||
|
||||
} |
||||
}; |
||||
|
||||
axios.get(process.env.VUE_APP_BACK_USER + "/api/users/bykeycloakid?guid=" + UserService.getUsername().sub, |
||||
axiosConfig |
||||
).then(result => { |
||||
if( result.data.keycloakId !== null) |
||||
that.$store.commit("updateUser", result.data); |
||||
else |
||||
UserService.doLogout(); |
||||
}); |
||||
|
||||
axios.get(process.env.VUE_APP_BACK_USER + "/api/roles/bykeycloakid?keycloakid=" + UserService.getUsername().sub, |
||||
axiosConfig |
||||
).then(result => { |
||||
that.$store.commit("updateRole", result.data); |
||||
}); |
||||
|
||||
}, |
||||
UserService.doLogin()) |
||||
} |
||||
} |
||||
|
||||
} |
||||
|
||||
|
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,271 @@ |
||||
<template> |
||||
<b-container fluid id="temperature"> |
||||
<b-row style="padding-top: 10px"> |
||||
<b-col> |
||||
<b-avatar |
||||
v-if="!acquirementInprogress" |
||||
v-b-tooltip.hover :title="$t('sub_title')" |
||||
variant="info" :src="image_avatar" |
||||
size="10rem" style="margin-right: 5px" |
||||
> |
||||
</b-avatar> |
||||
|
||||
<div><video |
||||
v-if="acquirementInprogress" |
||||
ref="video" |
||||
id="video" width="160" height="160" style="border-radius: 80px;" |
||||
:srcObject.prop="stream" |
||||
autoplay /> |
||||
</div> |
||||
</b-col> |
||||
</b-row> |
||||
<b-row> |
||||
|
||||
</b-row> |
||||
<b-row> |
||||
<b-col> |
||||
<label style="font-size: 10px;font-style: italic">{{message}}</label> |
||||
</b-col> |
||||
</b-row> |
||||
<b-row v-if="result != null"> |
||||
<b-col> |
||||
{{result.temperature | formatFloat}} {{$t('temperature_unit')}} |
||||
</b-col> |
||||
</b-row> |
||||
<b-row style="padding-top: 10px"> |
||||
<b-col cols="6" class="d-flex justify-content-start"> |
||||
{{$t('object_temperature')}} : {{received.Element | formatFloat}} {{$t('temperature_unit')}} |
||||
</b-col> |
||||
<b-col cols="6" class="d-flex justify-content-start"> |
||||
{{$t('ambient_temperature')}} : {{received.Ambient | formatFloat}} {{$t('temperature_unit')}} |
||||
</b-col> |
||||
<b-col cols="12" class="d-flex justify-content-start" v-if="isAdmin"> |
||||
{{$t('emissivity')}} : {{received.Emissivity | formatFloat}} |
||||
</b-col> |
||||
</b-row> |
||||
<b-row v-if="isAuth"> |
||||
<test-chart |
||||
:options="options" |
||||
:styles="myStyles" |
||||
:chart-data="datacollection"></test-chart> |
||||
</b-row> |
||||
<b-row> |
||||
<b-col> |
||||
<b-button @click="acquireTemp" variant="danger" :disabled="acquirementInprogress" v-if="isAuth"> |
||||
{{$t('temperature_acq')}} |
||||
<font-awesome-icon icon="question" /> |
||||
|
||||
<b-spinner v-if="acquirementInprogress" type="border" :label="$t('inprogress')"/> |
||||
</b-button> |
||||
</b-col> |
||||
|
||||
</b-row> |
||||
|
||||
</b-container> |
||||
</template> |
||||
|
||||
<script> |
||||
import axios from "axios"; |
||||
import TestChart from "./TestChart"; |
||||
import Vue from "vue"; |
||||
import '../services/TemperatureHub.js' |
||||
import TemperatureHub from "../services/TemperatureHub"; |
||||
|
||||
Vue.use(TestChart); |
||||
|
||||
|
||||
export default { |
||||
name: 'Temperature', |
||||
components: { |
||||
TestChart |
||||
}, |
||||
|
||||
props: { |
||||
msg: String |
||||
}, |
||||
data:function(){ |
||||
return { |
||||
stream:{}, |
||||
datacollection: {}, |
||||
received : {}, |
||||
|
||||
acquirementInprogress:false, |
||||
result: null, |
||||
x:'', |
||||
options: { |
||||
responsive: true, |
||||
maintainAspectRatio: false, |
||||
}, |
||||
roles:[], |
||||
temperatureHub : null |
||||
} |
||||
}, |
||||
computed: { |
||||
image() |
||||
{ |
||||
return this.$store.state.user != null && this.$store.state.user.image != null ? |
||||
process.env.VUE_APP_IMAGE_STORE + this.$store.state.user.image : |
||||
require('@/assets/coviiiiiid.jpg'); |
||||
}, |
||||
myStyles() { |
||||
return { |
||||
height: `300px`, |
||||
width:'100%', |
||||
position: 'relative' |
||||
} |
||||
}, |
||||
message() |
||||
{ |
||||
if(this.result == null) |
||||
return ""; |
||||
else |
||||
return this.result.message; |
||||
}, |
||||
image_avatar() |
||||
{ |
||||
if(this.result == null) |
||||
{ |
||||
if(this.acquirementInprogress) |
||||
return require('@/assets/waiting.jpg'); |
||||
else |
||||
return this.image; |
||||
} |
||||
else |
||||
{ |
||||
|
||||
if(this.result.fever === null) |
||||
return require('@/assets/out_of_range.jpg'); |
||||
else if(this.result.fever) |
||||
return require('@/assets/fever_ok.jpg'); |
||||
else |
||||
return this.image; |
||||
|
||||
|
||||
} |
||||
}, |
||||
isAdmin() { |
||||
|
||||
return this.$store.state.roles.find(l => l.name == 'Administrators' && l.isActived) != null; |
||||
}, |
||||
isAuth() { |
||||
|
||||
return this.$store.state.roles !== undefined && this.$store.state.roles.find(l => l.isActived) != null; |
||||
}, |
||||
}, |
||||
filters: { |
||||
formatFloat : function(value) { |
||||
if (value) { |
||||
return parseFloat(value).toFixed(2); |
||||
} |
||||
} |
||||
}, |
||||
|
||||
created () { |
||||
this.temperatureHub = new TemperatureHub(); |
||||
Vue.prototype.$questionHub = this.temperatureHub.startHub(); |
||||
this.$questionHub.$on(process.env.VUE_APP_EMIT_MSG_TEMPERATURE, this.onChanged); |
||||
|
||||
// Listen to score changes coming from SignalR events |
||||
|
||||
}, |
||||
mounted() { |
||||
|
||||
this.fillData (); |
||||
|
||||
}, |
||||
methods : { |
||||
fillData () { |
||||
this.datacollection = { |
||||
labels: [], |
||||
datasets: [ |
||||
{ |
||||
label: this.$t('temperature_chart_label'), |
||||
backgroundColor: '#f87979', |
||||
data: [] |
||||
}] |
||||
} |
||||
}, |
||||
|
||||
onChanged: function(msg) { |
||||
this.received = msg; |
||||
|
||||
let datacollection = { |
||||
labels: [], |
||||
datasets: [ |
||||
{ |
||||
label: this.$t('temperature_chart_label'), |
||||
backgroundColor: '#f87979', |
||||
data: [] |
||||
}] |
||||
} |
||||
|
||||
if(this.datacollection.labels.length > process.env.VUE_APP_MAX_CHART) |
||||
{ |
||||
this.datacollection.labels.shift(); |
||||
this.datacollection.datasets[0].data.shift(); |
||||
} |
||||
|
||||
|
||||
datacollection.labels.push(...this.datacollection.labels); |
||||
datacollection.datasets[0].data.push(...this.datacollection.datasets[0].data); |
||||
|
||||
datacollection.labels.push(this.x); |
||||
datacollection.datasets[0].data.push(this.received.Element); |
||||
this.datacollection = datacollection; |
||||
|
||||
}, |
||||
acquireTemp : async function() |
||||
{ |
||||
try { |
||||
this.acquirementInprogress = true; |
||||
let that = this; |
||||
this.video = this.$refs.video; |
||||
if(navigator.mediaDevices && navigator.mediaDevices.getUserMedia) { |
||||
navigator.mediaDevices.getUserMedia({ video: true }).then(stream => { |
||||
that.stream = stream |
||||
}); |
||||
} |
||||
|
||||
this.result = null; |
||||
let axiosConfig = { |
||||
withCredentials: true, |
||||
headers:{'Authorization': 'Bearer ' + sessionStorage.getItem("react-token"), |
||||
} |
||||
}; |
||||
let result = await axios.get(process.env.VUE_APP_BACK_URL + "/api/temperature", axiosConfig); |
||||
window.console.log(result); |
||||
this.result = result.data; |
||||
|
||||
} |
||||
finally { |
||||
this.acquirementInprogress = false; |
||||
this.stream = null; |
||||
} |
||||
} |
||||
}, |
||||
beforeDestroy : async function() { |
||||
|
||||
// Make sure to cleanup SignalR event handlers when removing the component |
||||
this.$questionHub.$off(process.env.VUE_APP_EMIT_MSG_TEMPERATURE, this.onChanged); |
||||
await this.temperatureHub.stopHub(); |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<!-- Add "scoped" attribute to limit CSS to this component only --> |
||||
<style scoped> |
||||
h3 { |
||||
margin: 40px 0 0; |
||||
} |
||||
ul { |
||||
list-style-type: none; |
||||
padding: 0; |
||||
} |
||||
li { |
||||
display: inline-block; |
||||
margin: 0 10px; |
||||
} |
||||
a { |
||||
color: #42b983; |
||||
} |
||||
</style> |
@ -0,0 +1,17 @@ |
||||
import { Line, mixins } from 'vue-chartjs' |
||||
const { reactiveProp } = mixins |
||||
|
||||
export default { |
||||
extends: Line, |
||||
mixins: [reactiveProp], |
||||
props: ['options'], |
||||
options: { |
||||
responsive: true, |
||||
maintainAspectRatio: false, |
||||
}, |
||||
mounted () { |
||||
// this.chartData is created in the mixin.
|
||||
// If you want to pass options please create a local options object
|
||||
this.renderChart(this.chartData, this.options) |
||||
} |
||||
} |
@ -0,0 +1,183 @@ |
||||
<template> |
||||
<div> |
||||
<b-card v-if="user.isNew == undefined" no-body class="overflow-hidden cardUser" |
||||
> |
||||
<b-row no-gutters style="height: 200px" > |
||||
<b-col md="8" cols="7" class="border-right" style=";background-color: #EEEEEE"> |
||||
<b-row align-h="center" v-if="user.image !== null"> |
||||
<router-link |
||||
:to="{name: 'User', params: {id: user.id } }" |
||||
v-slot="{ href, route, navigate}" |
||||
v-if="user.image !== '' && user.image !== null" |
||||
> |
||||
<div class="mb-2 imageUser"> |
||||
<b-avatar :href="href" @click="navigate" |
||||
:title="user.nom + ' ' + user.prenom" |
||||
variant="light" |
||||
v-bind:src="img" size="6rem" |
||||
> |
||||
</b-avatar> |
||||
</div> |
||||
</router-link> |
||||
</b-row> |
||||
<b-row align-h="center"> |
||||
<b-col> |
||||
<h4> {{user.prenom}} {{user.nom}}</h4> |
||||
</b-col> |
||||
</b-row> |
||||
<b-row v-for="role in user.roles" v-bind:key="role.id"> |
||||
<b-col class="d-flex justify-content-center align-items-center"> |
||||
<h5> {{role.libelle}}</h5> |
||||
</b-col> |
||||
</b-row> |
||||
</b-col> |
||||
<b-col md="4" cols="5" class="d-flex flex-column justify-content-between"> |
||||
|
||||
<b-row > |
||||
<b-col style="padding-bottom: 15px; padding-top: 15px"> |
||||
<router-link |
||||
:to="{name: 'User', params: {id: user.id } }" |
||||
v-slot="{ href, route, navigate}" |
||||
> |
||||
|
||||
<b-button :href="href" @click="navigate" class="actionButton" > |
||||
<font-awesome-icon icon="edit" /> |
||||
{{$t('edit')}}</b-button> |
||||
</router-link> |
||||
</b-col> |
||||
</b-row> |
||||
<b-row> |
||||
<b-col style="padding-bottom: 15px; padding-top: 15px"> |
||||
<b-button @click="deleteUser(user)" class="actionButton" > |
||||
<font-awesome-icon icon="trash" /> |
||||
{{$t('delete')}} |
||||
</b-button> |
||||
</b-col> |
||||
</b-row> |
||||
</b-col> |
||||
</b-row> |
||||
</b-card> |
||||
|
||||
<b-card v-if="user.isNew" no-body |
||||
class="overflow-hidden rounded Desktopbutton cardUser"> |
||||
<router-link |
||||
:to="{name: 'User', params: {id: 0 } }" |
||||
v-slot="{ href, route, navigate}"> |
||||
<b-button :href="href" @click="navigate" |
||||
style="width:100%; height: 100%" |
||||
class="d-flex justify-content-center align-items-center"> |
||||
<font-awesome-icon icon="plus-circle" style="width: 64px; height:64px" /> |
||||
</b-button> |
||||
</router-link> |
||||
</b-card> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import axios from "axios"; |
||||
|
||||
export default { |
||||
name: "CardUser", |
||||
props: |
||||
{ |
||||
method: {type: Function}, |
||||
user: { |
||||
type: Object |
||||
} |
||||
}, |
||||
mounted() |
||||
{ |
||||
|
||||
}, |
||||
computed : |
||||
{ |
||||
img() |
||||
{ |
||||
return this.user != null && this.user.image!= null ? |
||||
process.env.VUE_APP_IMAGE_STORE + this.user.image : |
||||
''; |
||||
}, |
||||
}, |
||||
methods: |
||||
{ |
||||
deleteUser : function(user) |
||||
{ |
||||
|
||||
let axiosConfig = { |
||||
withCredentials: true, |
||||
headers:{'Authorization': 'Bearer ' + sessionStorage.getItem("react-token"), |
||||
} |
||||
}; |
||||
this.$bvModal.msgBoxConfirm('Vous allez supprimer l\'utilisateur \'' + user.login + '\'. Voulez-vous continuer ?', |
||||
{ |
||||
title: 'Suppression Utilisateur', |
||||
size: 'sm', |
||||
buttonSize: 'sm', |
||||
okVariant: 'success', |
||||
okTitle: 'Oui', |
||||
cancelVariant: 'danger', |
||||
cancelTitle: 'Non', |
||||
footerClass: 'p-2', |
||||
hideHeaderClose: false, |
||||
centered: true |
||||
}) |
||||
.then(value => { |
||||
if(value) { |
||||
axios.delete(process.env.VUE_APP_BACK_USER+ "/api/users/" + user.id, |
||||
axiosConfig |
||||
).then(() => { |
||||
this.$emit('send-message'); |
||||
}); |
||||
} |
||||
}) |
||||
.catch(err => { |
||||
window.console.error(err.messages); |
||||
}) |
||||
|
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
.card-body |
||||
{ |
||||
padding-right: 0px; |
||||
padding-left: 0px; |
||||
padding-top: 0px; |
||||
} |
||||
.cardUser { |
||||
max-width: 540px; |
||||
height: 175px |
||||
} |
||||
.actionButton{ |
||||
width:125px |
||||
} |
||||
|
||||
.imageUser { |
||||
width:96px; |
||||
height:96px |
||||
} |
||||
|
||||
.headerUser { |
||||
position:fixed; /* fixing the position takes it out of html flow - knows |
||||
nothing about where to locate itself except by browser |
||||
coordinates */ |
||||
left:0; /* top left corner should start at leftmost spot */ |
||||
/*top:0; /* top left corner should start at topmost spot */ |
||||
width:100vw; /* take up the full browser width */ |
||||
z-index:50; /* high z index so other content scrolls underneath */ |
||||
height: 100px; |
||||
background-color: white; |
||||
} |
||||
@media (min-width: 50em) { |
||||
.Desktopbutton { display: block; } |
||||
.Mobilebutton { display: none; } |
||||
} |
||||
|
||||
@media (max-width: 50em) { |
||||
.Desktopbutton { display: none; } |
||||
.Mobilebutton { display: block; } |
||||
.MobileEntete {margin-top: 35px;} |
||||
} |
||||
</style> |
@ -0,0 +1,228 @@ |
||||
<template> |
||||
<b-container fluid id="ListeUser" > |
||||
<div class="headerUser" variant="dark"> |
||||
<b-row align-h="center" |
||||
style="padding-top: 15px;" |
||||
> |
||||
<h2>{{$t('user_title')}} </h2> |
||||
</b-row> |
||||
|
||||
<b-row align-v="center"> |
||||
<b-col cols="8" lg="3" > |
||||
<b-form-input v-model="search" |
||||
@input="debounceInput" |
||||
v-bind:placeholder="$t('search')" style="max-width: 400px"></b-form-input> |
||||
</b-col> |
||||
<b-col cols="1" lg="1"> |
||||
<b-button @click="display"> |
||||
<font-awesome-icon icon="search" /> |
||||
</b-button> |
||||
</b-col> |
||||
</b-row> |
||||
|
||||
<b-row> |
||||
<router-link |
||||
:to="{name: 'User', params: {id: 0 } }" |
||||
v-slot="{ href, route, navigate}"> |
||||
<b-button :href="href" @click="navigate" class="Mobilebutton" |
||||
style="width:100%"> |
||||
<font-awesome-icon icon="plus-circle" /> |
||||
</b-button> |
||||
</router-link> |
||||
</b-row> |
||||
|
||||
</div> |
||||
<div style="padding-top: 135px" class="MobileEntete"> |
||||
|
||||
<b-row v-if="!isLoaded" class="text-center d-flex justify-content-center"> |
||||
<b-spinner type="border" label="chargement en cours..."></b-spinner> |
||||
</b-row> |
||||
|
||||
<b-row v-if="isLoaded && users.length > 0" > |
||||
<b-col cols="12" md="4" lg="4" v-for="user in users" v-bind:key="user.id" |
||||
style="padding-right: 5px; padding-left: 5px; padding-top: 5px; padding-bottom: 5px"> |
||||
<div> |
||||
<card-user :user="user" @send-message="handleSendMessage"/> |
||||
</div> |
||||
</b-col> |
||||
|
||||
<infinite-loading @infinite="infiniteHandler" :identifier="infiniteId"> |
||||
<div slot="no-more"></div> |
||||
<div slot="no-results"></div> |
||||
</infinite-loading> |
||||
</b-row> |
||||
<b-row class="mb-4" v-else> |
||||
<p v-if="isLoaded">{{$t('user_list_empty')}} </p> |
||||
</b-row> |
||||
</div> |
||||
</b-container> |
||||
|
||||
|
||||
</template> |
||||
|
||||
<script> |
||||
import axios from 'axios'; |
||||
import InfiniteLoading from 'vue-infinite-loading'; |
||||
import CardUser from "./CardUser"; |
||||
import { debounce } from "debounce"; |
||||
|
||||
export default { |
||||
name: "ListeUser", |
||||
props:['value'], |
||||
components: { |
||||
CardUser, |
||||
InfiniteLoading, |
||||
}, |
||||
|
||||
data:function(){ |
||||
return { |
||||
page: 1, |
||||
pageSize:4, |
||||
search:'', |
||||
infiniteId: + new Date(), |
||||
users: [ |
||||
], |
||||
isLoaded: false |
||||
} |
||||
}, |
||||
methods : |
||||
{ |
||||
handleSendMessage() { |
||||
this.display(); |
||||
}, |
||||
debounceInput: debounce(function () { |
||||
this.display(); |
||||
}, 500,false), |
||||
infiniteHandler : function($state) |
||||
{ |
||||
let that = this; |
||||
let axiosConfig = { |
||||
withCredentials: true, |
||||
headers:{'Authorization': 'Bearer ' + sessionStorage.getItem("react-token"), |
||||
} |
||||
}; |
||||
|
||||
axios.get(process.env.VUE_APP_BACK_USER + "/api/users?page=" + that.page |
||||
+ "&pageSize=" + that.pageSize |
||||
+ "&search=" + that.search, |
||||
axiosConfig |
||||
).then(result => { |
||||
if (result.data.length) { |
||||
this.page += 1; |
||||
this.users.push(...result.data); |
||||
$state.loaded(); |
||||
} else { |
||||
$state.complete(); |
||||
} |
||||
}); |
||||
}, |
||||
display : async function() |
||||
{ |
||||
this.isLoaded = false; |
||||
|
||||
let axiosConfig = { |
||||
withCredentials: true, |
||||
headers:{'Authorization': 'Bearer ' + sessionStorage.getItem("react-token"), |
||||
} |
||||
}; |
||||
|
||||
this.page = 1; |
||||
|
||||
let result = await axios.get(process.env.VUE_APP_BACK_USER + "/api/users?page=" + this.page |
||||
+ "&pageSize=" + this.pageSize |
||||
+ "&search=" + this.search, |
||||
axiosConfig |
||||
); |
||||
|
||||
this.isLoaded = true; |
||||
this.page += 1; |
||||
this.users = [ {isNew:true}]; |
||||
this.users.push(...result.data); |
||||
this.infiniteId += 1; |
||||
}, |
||||
|
||||
deleteUser : function(user) |
||||
{ |
||||
let that = this; |
||||
|
||||
let axiosConfig = { |
||||
withCredentials: true, |
||||
headers:{'Authorization': 'Bearer ' + sessionStorage.getItem("react-token"), |
||||
} |
||||
}; |
||||
this.$bvModal.msgBoxConfirm('Vous allez supprimer l\'utilisateur \'' + user.nom + '\'. Voulez-vous continuer ?', |
||||
{ |
||||
title: 'Suppression Utilisateur', |
||||
size: 'sm', |
||||
buttonSize: 'sm', |
||||
okVariant: 'success', |
||||
okTitle: 'Oui', |
||||
cancelVariant: 'danger', |
||||
cancelTitle: 'Non', |
||||
footerClass: 'p-2', |
||||
hideHeaderClose: false, |
||||
centered: true |
||||
}) |
||||
.then(value => { |
||||
if(value) { |
||||
axios.delete(process.env.VUE_APP_BACK_USER+ "/api/users/" + user.id, |
||||
axiosConfig |
||||
).then(() => { |
||||
that.display(); |
||||
}); |
||||
} |
||||
}) |
||||
.catch(err => { |
||||
window.console.error(err.messages); |
||||
}) |
||||
|
||||
} |
||||
}, |
||||
mounted() { |
||||
this.display(); |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
.card-body |
||||
{ |
||||
padding-right: 0px; |
||||
padding-left: 0px; |
||||
padding-top: 0px; |
||||
} |
||||
.cardUser { |
||||
max-width: 540px; |
||||
height: 175px |
||||
} |
||||
.actionButton{ |
||||
width:125px |
||||
} |
||||
|
||||
.imageUser { |
||||
width:96px; |
||||
height:96px |
||||
} |
||||
|
||||
.headerUser { |
||||
position:fixed; /* fixing the position takes it out of html flow - knows |
||||
nothing about where to locate itself except by browser |
||||
coordinates */ |
||||
left:0; /* top left corner should start at leftmost spot */ |
||||
/*top:0; /* top left corner should start at topmost spot */ |
||||
width:100vw; /* take up the full browser width */ |
||||
z-index:50; /* high z index so other content scrolls underneath */ |
||||
height: 100px; |
||||
background-color: white; |
||||
} |
||||
@media (min-width: 50em) { |
||||
.Desktopbutton { display: block; } |
||||
.Mobilebutton { display: none; } |
||||
} |
||||
|
||||
@media (max-width: 50em) { |
||||
.Desktopbutton { display: none; } |
||||
.Mobilebutton { display: block; } |
||||
.MobileEntete {margin-top: 35px;} |
||||
} |
||||
</style> |
@ -0,0 +1,328 @@ |
||||
<template> |
||||
|
||||
<b-container fluid id="User" > |
||||
<!-- <b-row class="d-flex justify-content-center" style="padding-top: 15px;"> |
||||
<h2>{{titre}} </h2> |
||||
</b-row>--> |
||||
|
||||
<b-row v-if="!isLoaded" class="text-center d-flex justify-content-center"> |
||||
<b-spinner type="border" label="chargement en cours..."></b-spinner> |
||||
</b-row> |
||||
|
||||
<b-row v-if="isLoaded && user !== null"> |
||||
<b-col cols="12" md="2"> |
||||
<b-row align-v="center"> |
||||
<b-col > |
||||
<b-avatar |
||||
variant="light" |
||||
v-bind:src="img" size="10rem" |
||||
> |
||||
</b-avatar> |
||||
</b-col> |
||||
</b-row> |
||||
<b-row align-v="center"> |
||||
<b-col class="d-flex justify-content-center align-items-center"> |
||||
<b-button v-if="hasImage" variant="danger" class="Desktop" @click="clearImage">{{$t('clear')}}</b-button> |
||||
</b-col> |
||||
</b-row> |
||||
</b-col> |
||||
<b-col cols="12" md="10"> |
||||
<b-row align-v="center" style="padding-top: 5px"> |
||||
<b-col md="1" cols="4"> |
||||
<label>{{$t('user_detail_prenom')}}</label> |
||||
</b-col> |
||||
<b-col md="5" cols="8"> |
||||
<b-form-input v-model="user.prenom" |
||||
:state="validFirstName" |
||||
v-bind:placeholder="$t('user_detail_prenom')"></b-form-input> |
||||
</b-col> |
||||
<b-col md="1" cols="4"> |
||||
<label>{{$t('user_detail_nom')}}</label> |
||||
</b-col> |
||||
<b-col md="5" cols="8"> |
||||
<b-form-input number v-model="user.nom" |
||||
:state="validName" |
||||
v-bind:placeholder="$t('user_detail_nom')"></b-form-input> |
||||
</b-col> |
||||
</b-row> |
||||
<b-row> |
||||
<b-col md="1" cols="4"> |
||||
<label>{{$t('user_detail_username')}}</label> |
||||
</b-col> |
||||
<b-col md="5" cols="8"> |
||||
<b-form-input v-model="user.userName" |
||||
:state="validUserName" |
||||
v-bind:placeholder="$t('user_detail_username')"></b-form-input> |
||||
</b-col> |
||||
<!-- <b-col md="1" cols="4"> |
||||
<label>{{$t('user_detail_mail')}}</label> |
||||
</b-col> |
||||
<b-col md="5" cols="8"> |
||||
<b-form-input mail v-model="user.email" v-bind:placeholder="$t('user_detail_mail')"></b-form-input> |
||||
</b-col>--> |
||||
</b-row> |
||||
<b-row align-v="center" style="padding-top: 5px"> |
||||
<b-col md="1" cols="4"> |
||||
<label> {{$t('user_detail_id_keycloak')}}</label> |
||||
</b-col> |
||||
<b-col md="5" cols="8"> |
||||
<b-form-input :disabled="!isAdmin" v-model="user.keycloakId" v-bind:placeholder="$t('user_detail_id_keycloak')"></b-form-input> |
||||
</b-col> |
||||
<b-col md="1" class="Desktop"> |
||||
<label>{{$t('user_detail_image')}}</label> |
||||
</b-col> |
||||
<b-col md="5" class="Desktop"> |
||||
<b-form-file |
||||
v-model="file" |
||||
plain |
||||
></b-form-file> |
||||
</b-col> |
||||
</b-row> |
||||
<b-row class="d-flex justify-content-center align-items-center" style="padding-top: 10px"> |
||||
<h5>{{$t('user_detail_roles')}}</h5> |
||||
</b-row> |
||||
<b-row> |
||||
<b-col> |
||||
<b-row v-for="role in roles" v-bind:key="role.id"> |
||||
<b-col> |
||||
{{role.name}} |
||||
</b-col> |
||||
<b-col> |
||||
<toggle-button |
||||
v-model="role.isActived" |
||||
:value="false" |
||||
:font-size="16" |
||||
:sync="true" |
||||
/> |
||||
</b-col> |
||||
</b-row> |
||||
|
||||
</b-col> |
||||
</b-row> |
||||
|
||||
</b-col> |
||||
</b-row> |
||||
<b-row v-if="isLoaded" > |
||||
<b-col class="d-flex justify-content-between align-items-center"> |
||||
<router-link |
||||
:to="{name: 'ListeUser' }" |
||||
v-slot="{ href, route, navigate}" |
||||
> |
||||
<b-button :href="href" @click="navigate" variant="danger"><font-awesome-icon icon="arrow-alt-circle-left" /> {{$t('back')}}</b-button> |
||||
</router-link> |
||||
|
||||
<photo v-bind:snapshot.sync="photo"></photo> |
||||
|
||||
<b-button @click="saveAndQuit" variant="success" v-bind:disabled="!isValid"><font-awesome-icon icon="check" /> {{$t('valid')}}</b-button> |
||||
</b-col> |
||||
</b-row> |
||||
</b-container> |
||||
</template> |
||||
|
||||
<script> |
||||
import axios from 'axios'; |
||||
import photo from '../Capture/CapturePhoto'; |
||||
import Vue from "vue"; |
||||
|
||||
|
||||
Vue.use(photo); |
||||
|
||||
const base64Encode = data => |
||||
new Promise((resolve, reject) => { |
||||
const reader = new FileReader(); |
||||
reader.readAsDataURL(data); |
||||
reader.onload = () => resolve(reader.result); |
||||
reader.onerror = error => reject(error); |
||||
}); |
||||
|
||||
export default { |
||||
name: "User", |
||||
components: { photo}, |
||||
data:function(){ |
||||
return { |
||||
user: null, |
||||
id: null, |
||||
componentKey:0, |
||||
file : null, |
||||
photo: null, |
||||
img:null, |
||||
roles: null, |
||||
isLoaded : false |
||||
} |
||||
}, |
||||
computed: { |
||||
isAdmin() { |
||||
|
||||
return this.$store.state.roles.find(l => l.name == 'Administrators' && l.isActived) != null; |
||||
}, |
||||
isValid() |
||||
{ |
||||
return this.validFirstName && this.validName && this.validUserName; |
||||
}, |
||||
validFirstName(){ |
||||
return this.user != null && this.user.prenom !== '' && this.user.prenom !== null; |
||||
}, |
||||
validName(){ |
||||
|
||||
return this.user != null && this.user.nom !== '' && this.user.nom !== null; |
||||
}, |
||||
validUserName(){ |
||||
|
||||
return this.user != null && this.user.userName !== '' && this.user.userName !== null && this.user.userName !== undefined; |
||||
}, |
||||
hasImage() { |
||||
return !!this.img; |
||||
}, |
||||
titre(){ |
||||
return (this.user != null) ? this.user.prenom + ' ' + this.user.nom : this.$i18n.t('user_detail_title'); |
||||
} |
||||
}, |
||||
watch : |
||||
{ |
||||
file : function (newValue, oldValue) { |
||||
if (newValue !== oldValue) { |
||||
if (newValue) { |
||||
base64Encode(newValue) |
||||
.then(value => { |
||||
this.user.image= newValue.name; |
||||
this.img = value; |
||||
}) |
||||
.catch(() => { |
||||
this.img = null; |
||||
}); |
||||
} else { |
||||
this.img = null; |
||||
} |
||||
} |
||||
}, |
||||
photo: function(val) |
||||
{ |
||||
this.img =val.photo; |
||||
}, |
||||
|
||||
}, |
||||
methods : |
||||
{ |
||||
clearImage() { |
||||
this.img = null; |
||||
this.file = null; |
||||
this.photo = null; |
||||
|
||||
this.user.image="user_unknown.png"; |
||||
}, |
||||
saveAndQuit : async function() |
||||
{ |
||||
await this.save(); |
||||
await this.$router.push({name: 'ListeUser'}); |
||||
}, |
||||
save : async function() |
||||
{ |
||||
let that = this; |
||||
|
||||
let axiosConfig = { |
||||
withCredentials: true, |
||||
headers:{'Authorization': 'Bearer ' + sessionStorage.getItem("react-token"), |
||||
} |
||||
}; |
||||
let result = await axios.put(process.env.VUE_APP_BACK_USER + "/api/users/" + this.id, this.user, axiosConfig); |
||||
that.user = result.data; |
||||
let data = new FormData(); |
||||
|
||||
if(that.file !== null) { |
||||
data.set('file', that.file) |
||||
await axios.post(process.env.VUE_APP_BACK_USER + '/api/users/' + that.id + '/loadimage', data, { |
||||
headers: { |
||||
'Content-Type': 'multipart/form-data', |
||||
'Authorization': 'Bearer ' + sessionStorage.getItem("react-token"), |
||||
} |
||||
}); |
||||
} |
||||
else if(that.photo != null) |
||||
{ |
||||
data.set('base64', that.img.split(",")[1]); |
||||
await axios.post(process.env.VUE_APP_BACK_USER + "/api/users/" + that.id + '/loadimage64',data, |
||||
{ |
||||
headers: { |
||||
'Content-Type': 'multipart/form-data', |
||||
'Authorization': 'Bearer ' + sessionStorage.getItem("react-token"), |
||||
} |
||||
} |
||||
); |
||||
that.componentKey += 1; |
||||
} |
||||
|
||||
await axios.put(process.env.VUE_APP_BACK_USER + "/api/roles/" + this.user.keycloakId, this.roles, axiosConfig); |
||||
} |
||||
}, |
||||
mounted() { |
||||
this.id = this.$route.params.id; |
||||
|
||||
let axiosConfig = { |
||||
withCredentials: true, |
||||
headers: { |
||||
'Authorization': 'Bearer ' + sessionStorage.getItem("react-token"), |
||||
} |
||||
}; |
||||
this.isLoaded = false; |
||||
if(this.id !== 0) { |
||||
axios.get(process.env.VUE_APP_BACK_USER + "/api/users/" + this.id, |
||||
axiosConfig |
||||
).then(result => { |
||||
this.user = result.data; |
||||
this.img = process.env.VUE_APP_IMAGE_STORE + this.user.image; |
||||
|
||||
axios.get(process.env.VUE_APP_BACK_USER + "/api/roles/bykeycloakid?keycloakid=" + this.user.keycloakId, |
||||
axiosConfig |
||||
).then(result => { |
||||
this.isLoaded = true; |
||||
this.roles = result.data; |
||||
}); |
||||
|
||||
}); |
||||
} |
||||
else |
||||
{ |
||||
this.user = {id:0, nom:'', prenom:'', keycloakId:''} |
||||
|
||||
axios.get(process.env.VUE_APP_BACK_USER + "/api/roles", |
||||
axiosConfig |
||||
).then(result => { |
||||
this.isLoaded = true; |
||||
this.roles = result.data; |
||||
}); |
||||
} |
||||
|
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
.row |
||||
{ |
||||
padding-top: 5px |
||||
} |
||||
.card-body |
||||
{ |
||||
padding-right: 0px; |
||||
padding-left: 0px; |
||||
padding-top: 0px; |
||||
padding-bottom: 0px; |
||||
} |
||||
|
||||
.collapsed > .when-opened, |
||||
:not(.collapsed) > .when-closed { |
||||
display: none; |
||||
} |
||||
|
||||
@media (min-width: 50em) { |
||||
.Desktop { display: block; } |
||||
.Mobile { display: none; } |
||||
} |
||||
|
||||
@media (max-width: 50em) { |
||||
.Desktop { display: none; } |
||||
.Mobile { display: block; } |
||||
} |
||||
|
||||
|
||||
</style> |
@ -0,0 +1,41 @@ |
||||
export default { |
||||
en: { |
||||
}, |
||||
'fr-FR': { |
||||
edit:"Editer", |
||||
delete:"Supprimer", |
||||
add:"ajouter", |
||||
logout:"Logout", |
||||
login:"Login", |
||||
save:"Sauvegarder", |
||||
back:"Retour", |
||||
valid:"Valider", |
||||
unload:"Décharge", |
||||
search:"Recherche", |
||||
clear:"Effacer", |
||||
inprogress:"chargement en cours...", |
||||
|
||||
title:"Coviiiid", |
||||
sub_title:"Coviiiid !!!!! Je sais où tu t'caches ! Viens là que j'te bute !", |
||||
object_temperature : "T° objet", |
||||
ambient_temperature : "T° ambiante", |
||||
emissivity:"Emissivité", |
||||
temperature_unit : "°C", |
||||
temperature_acq:"Coviiiid", |
||||
temperature_chart_label:"Température", |
||||
|
||||
user_title : "Utilisateurs", |
||||
user_list_empty: "Aucun utilisateur", |
||||
|
||||
user_detail_title: "Utilisateur", |
||||
user_detail_nom: "Nom", |
||||
user_detail_prenom: "Prénom", |
||||
user_detail_image:"Image", |
||||
user_detail_id_keycloak:'idKeycloak', |
||||
user_detail_username : 'login', |
||||
user_detail_mail : 'Email', |
||||
user_detail_roles : 'Rôles', |
||||
user_detail_photo:"photo", |
||||
|
||||
} |
||||
} |
@ -0,0 +1,78 @@ |
||||
import Vue from 'vue' |
||||
import App from './App.vue' |
||||
import router from './router/index' |
||||
import VueI18n from 'vue-i18n' |
||||
import messages from './lang/messages' |
||||
import { library } from '@fortawesome/fontawesome-svg-core' |
||||
import { fas } from '@fortawesome/free-solid-svg-icons' |
||||
import { FontAwesomeIcon } from '@fortawesome/vue-fontawesome' |
||||
import axios from "axios"; |
||||
import UserService from "./services/UserService"; |
||||
|
||||
|
||||
import ToggleButton from 'vue-js-toggle-button' |
||||
import Vuex from 'vuex' |
||||
|
||||
|
||||
library.add(fas) |
||||
Vue.component('font-awesome-icon', FontAwesomeIcon) |
||||
Vue.config.productionTip = false; |
||||
Vue.use(VueI18n) |
||||
Vue.use(ToggleButton); |
||||
|
||||
let locale = navigator.language; |
||||
const i18n = new VueI18n({ |
||||
fallbackLocale: 'fr', |
||||
locale: locale, |
||||
messages |
||||
}) |
||||
|
||||
Vue.use(Vuex) |
||||
const store = new Vuex.Store({ |
||||
state : { |
||||
roles: [], |
||||
user:null, |
||||
}, |
||||
mutations: { |
||||
updateRole (state, roles) { |
||||
this.state.roles = roles; |
||||
}, |
||||
updateUser (state, user) { |
||||
this.state.user = user; |
||||
} |
||||
|
||||
} |
||||
}) |
||||
|
||||
new Vue({ |
||||
router, |
||||
i18n, |
||||
store, |
||||
mounted(){ |
||||
axios.interceptors.response.use( (response) => { |
||||
// Return a successful response back to the calling service
|
||||
return response; |
||||
}, (error) => { |
||||
// Return any error which is not due to authentication back to the calling service
|
||||
if (error.response.status !== 401) { |
||||
return new Promise((resolve, reject) => { |
||||
reject(error); |
||||
}); |
||||
} |
||||
else if (error.response.status === 401) { |
||||
return new Promise((resolve, reject) => { |
||||
UserService.doLogout(); |
||||
UserService.initKeycloak( function () { |
||||
|
||||
sessionStorage.setItem("react-token", UserService.getToken()); |
||||
sessionStorage.setItem("react-refresh-token", UserService.getRefreshToken()); |
||||
}, UserService.doLogin); |
||||
reject(error); |
||||
}); |
||||
} |
||||
|
||||
|
||||
}) |
||||
}, |
||||
render: h => h(App), |
||||
}).$mount('#app') |
@ -0,0 +1,33 @@ |
||||
import Vue from 'vue' |
||||
import Router from 'vue-router' |
||||
import Temperature from "../components/Temperature"; |
||||
import User from "../components/User/User"; |
||||
import ListeUser from "../components/User/ListeUser"; |
||||
|
||||
Vue.use(Router) |
||||
|
||||
export default new Router({ |
||||
routes: [ |
||||
{ |
||||
path: '*', |
||||
name: 'Temperature', |
||||
component: Temperature |
||||
}, |
||||
{ |
||||
path: '/', |
||||
name: 'Temperature', |
||||
component: Temperature |
||||
}, |
||||
|
||||
{ |
||||
path: '/user/:id', |
||||
name: 'User', |
||||
component: User |
||||
}, |
||||
{ |
||||
path: '/users', |
||||
name: 'ListeUser', |
||||
component: ListeUser |
||||
}, |
||||
] |
||||
}) |
@ -0,0 +1,52 @@ |
||||
import {HubConnectionBuilder, LogLevel} from "@aspnet/signalr"; |
||||
import Vue from "vue"; |
||||
|
||||
export default class TemperatureHub |
||||
{ |
||||
hubConnection; |
||||
|
||||
constructor() { |
||||
this.hubConnection = new HubConnectionBuilder() |
||||
.withUrl(process.env.VUE_APP_TEMPERATURE_HUB) |
||||
.configureLogging(LogLevel.Information) |
||||
.build(); |
||||
} |
||||
|
||||
start() { |
||||
let startedPromise = null; |
||||
let that = this; |
||||
startedPromise = this.hubConnection.start().catch(() => { |
||||
return new Promise((resolve, reject) => |
||||
setTimeout(() => that.start().then(resolve).catch(reject), process.env.VUE_APP_TIMEOUT_TEMPERATURE_HUB)) |
||||
}) |
||||
return startedPromise; |
||||
} |
||||
|
||||
startHub() |
||||
{ |
||||
let that = this; |
||||
this.hubConnection.onclose(() => |
||||
{ |
||||
that.start(); |
||||
}/**/); |
||||
|
||||
this.start(); |
||||
|
||||
// use new Vue instance as an event bus
|
||||
let questionHub = new Vue() |
||||
// every component will use this.$questionHub to access the event bus
|
||||
|
||||
// Forward server side SignalR events through $questionHub, where components will listen to them
|
||||
this.hubConnection.on(process.env.VUE_APP_MSG_TEMPERATURE_HUB, (msg) => { |
||||
questionHub.$emit(process.env.VUE_APP_EMIT_MSG_TEMPERATURE, JSON.parse(msg) ) |
||||
}) |
||||
|
||||
return questionHub; |
||||
} |
||||
|
||||
async stopHub() |
||||
{ |
||||
await this.hubConnection.stop(); |
||||
|
||||
} |
||||
} |
@ -0,0 +1,50 @@ |
||||
import Keycloak from "keycloak-js"; |
||||
|
||||
const _kc = new Keycloak('/keycloak.json'); |
||||
|
||||
/** |
||||
* Initializes Keycloak instance and calls the provided callback function if successfully authenticated. |
||||
* |
||||
* @param onAuthenticatedCallback |
||||
*/ |
||||
const initKeycloak = (onAuthenticatedCallback, onUnAuthentificatedCallback) => { |
||||
_kc.init({ |
||||
onLoad: 'check-sso', |
||||
flow: 'implicit', |
||||
promiseType: 'native', |
||||
}) |
||||
.then((authenticated) => { |
||||
if (authenticated) { |
||||
onAuthenticatedCallback(); |
||||
} else { |
||||
window.console.warn("not authenticated!"); |
||||
onUnAuthentificatedCallback(); |
||||
} |
||||
}) |
||||
}; |
||||
|
||||
const doLogin = _kc.login; |
||||
|
||||
const doLogout = _kc.logout; |
||||
|
||||
const getToken = () => _kc.token; |
||||
|
||||
const getRefreshToken = () => _kc.refreshToken; |
||||
|
||||
const updateToken = (successCallback) => { |
||||
return _kc.updateToken(5) |
||||
.then(successCallback) |
||||
.catch(doLogin) |
||||
}; |
||||
|
||||
const getUsername = () => _kc.tokenParsed; |
||||
|
||||
export default { |
||||
initKeycloak, |
||||
doLogin, |
||||
doLogout, |
||||
getToken, |
||||
getRefreshToken, |
||||
updateToken, |
||||
getUsername, |
||||
} |
@ -0,0 +1,18 @@ |
||||
import axios from 'axios'; |
||||
import { Promise } from "es6-promise"; |
||||
|
||||
|
||||
export default () => { |
||||
|
||||
axios.interceptors.response.use( (response) => { |
||||
// Return a successful response back to the calling service
|
||||
return response; |
||||
}, (error) => { |
||||
// Return any error which is not due to authentication back to the calling service
|
||||
if (error.response.status !== 401) { |
||||
return new Promise((resolve, reject) => { |
||||
reject(error); |
||||
}); |
||||
} |
||||
}) |
||||
} |
@ -0,0 +1,6 @@ |
||||
module.exports = { |
||||
devServer: { |
||||
https: false, |
||||
port:8080 |
||||
} |
||||
} |