Compare commits
	
		
			64 Commits
		
	
	
		
			fix_cache
			...
			ad1fd45bed
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ad1fd45bed | |||
| f0c4a3a10d | |||
|  | 70658e4fb9 | ||
|  | 2b31753265 | ||
| f8991e90ab | |||
|  | 60290956ec | ||
|  | b9647ce36e | ||
| 8c4b9ceb9d | |||
|  | 84d8d4523b | ||
| 647812576e | |||
|  | 2dfee66958 | ||
|  | 2b1666c949 | ||
|  | 0c724cae7f | ||
|  | 6de45801d2 | ||
| 03897e1139 | |||
| 00a733c03b | |||
| 3dc8131c33 | |||
|  | 7c271d8c47 | ||
| 259d56271c | |||
|  | b9f3bbbe15 | ||
| 14a2a59786 | |||
| 0ae6e7dfda | |||
| 15ccb5630a | |||
| 4ec292cca7 | |||
| 14a953536a | |||
| 288f983816 | |||
| dd6032f3ef | |||
|  | 7e2f5bc506 | ||
|  | 4ef92efd0e | ||
| e769dd6757 | |||
|  | 550a51523f | ||
| 323cb05388 | |||
|  | 7e0851bfef | ||
|  | 98b6d167e8 | ||
| 79baddb8f6 | |||
|  | 0f8c83c2e2 | ||
|  | fad52644d2 | ||
|  | 5f51a1008b | ||
| 279c171ba2 | |||
| 9ba8e3e84e | |||
| 6de38a9725 | |||
| 067eeb9494 | |||
|  | f48b570494 | ||
|  | 0733f8d5af | ||
|  | 8071c01c5d | ||
| b355463dd9 | |||
| 4ee3d9bc44 | |||
| d75d45e204 | |||
| 79e949bdd4 | |||
|  | 9f3754776f | ||
| 651fb2b1a1 | |||
|  | aa5988ce75 | ||
|  | 9ae18e1e4b | ||
| ef8c8e896d | |||
| 22ebb0e1f4 | |||
| 09e4b3262f | |||
| 6a3d4239ab | |||
| 9d71c93b5b | |||
|  | 5145b833ae | ||
|  | 4080cee818 | ||
|  | f4d73654d1 | ||
| 4fda5513a9 | |||
| 32407b0e8f | |||
| b30e1196f4 | 
							
								
								
									
										741
									
								
								Documentation/openapi/main.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										741
									
								
								Documentation/openapi/main.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,741 @@ | |||||||
|  | openapi: 3.0.3 | ||||||
|  | info: | ||||||
|  |   title: MyInpulse Backend Api | ||||||
|  |   description: this document servers as a documentation for the backend api.   | ||||||
|  |   version: 0.0.0 | ||||||
|  |  | ||||||
|  | tags: | ||||||
|  |   - name: Entrepreneurs API | ||||||
|  |     description: La partie de l'api dédiée aux entrepreneurs | ||||||
|  |   - name: Admin API | ||||||
|  |     description: La partie de l'api dédiée aux entrepreneurs | ||||||
|  |   - name: Shared API | ||||||
|  |     description: La partie de l'api dédiée aux entrepreneurs et admins | ||||||
|  |  | ||||||
|  |  | ||||||
|  | components: | ||||||
|  |   schemas: | ||||||
|  |     user: | ||||||
|  |       type: object | ||||||
|  |       properties: | ||||||
|  |         nom: | ||||||
|  |           type: string | ||||||
|  |         prenom: | ||||||
|  |           type: string | ||||||
|  |         email: | ||||||
|  |           type: string | ||||||
|  |           example: "example@exmaple.com" | ||||||
|  |         secondaryEmail: | ||||||
|  |           type: string | ||||||
|  |           example: "example@exmaple.com" | ||||||
|  |         tel: | ||||||
|  |           type: string | ||||||
|  |           example: "0612345678" | ||||||
|  |     user-entrepreneur: | ||||||
|  |       type: object | ||||||
|  |       properties: | ||||||
|  |         user: | ||||||
|  |           $ref: "#/components/schemas/user" | ||||||
|  |         entrepreneur: | ||||||
|  |           type: object | ||||||
|  |           properties: | ||||||
|  |             ecole:  | ||||||
|  |               type: string | ||||||
|  |               example: "enseirb" | ||||||
|  |             filiere: | ||||||
|  |               type: string | ||||||
|  |               example: "info" | ||||||
|  |             status: | ||||||
|  |               type: boolean | ||||||
|  |               example: false | ||||||
|  |     user-admin: | ||||||
|  |       type: object | ||||||
|  |       properties: | ||||||
|  |         admin: | ||||||
|  |           $ref: "#/components/schemas/user" | ||||||
|  |  | ||||||
|  |   securitySchemes: | ||||||
|  |       MyINPulse: | ||||||
|  |         type: oauth2 | ||||||
|  |         flows: | ||||||
|  |           implicit: | ||||||
|  |             authorizationUrl: https://petstore3.swagger.io/oauth/authorize | ||||||
|  |             scopes: | ||||||
|  |               MyINPulse-admin: Administrateur | ||||||
|  |               MyINPulse-entrepreneur: Utilisateur | ||||||
|  |  | ||||||
|  | paths: | ||||||
|  |  | ||||||
|  | #          _    ____  __  __ ___ _   _      _    ____ ___  | ||||||
|  | #         / \  |  _ \|  \/  |_ _| \ | |    / \  |  _ \_ _| | ||||||
|  | #        / _ \ | | | | |\/| || ||  \| |   / _ \ | |_) | |  | ||||||
|  | #       / ___ \| |_| | |  | || || |\  |  / ___ \|  __/| |  | ||||||
|  | #      /_/   \_\____/|_|  |_|___|_| \_| /_/   \_\_|  |___| | ||||||
|  | # | ||||||
|  |  | ||||||
|  |   /admin/projects: | ||||||
|  |     get: | ||||||
|  |       summary: Retourne la liste of projets associés à l'admin | ||||||
|  |       tags:  | ||||||
|  |         - Admin API  | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |             - MyINPulse-admin | ||||||
|  |       description: | ||||||
|  |         JSON array of who's elements are objects containing necessary information for the view  | ||||||
|  |         (project name, entrepreneur names, etc..) | ||||||
|  |         of the projects an admin is watching over. | ||||||
|  |       responses: | ||||||
|  |         "200": | ||||||
|  |           description: OK | ||||||
|  |           content: | ||||||
|  |             application/json: | ||||||
|  |               schema: | ||||||
|  |                 type: array | ||||||
|  |                 items: | ||||||
|  |                   type: object | ||||||
|  |                   properties: | ||||||
|  |                     name: | ||||||
|  |                       type: string | ||||||
|  |                     E_names: | ||||||
|  |                       type: string | ||||||
|  |                       description: entrepreneur names | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is missing or invalid | ||||||
|  |   /admin/projects/pending/decision: | ||||||
|  |     post: | ||||||
|  |       summary: valider un projet en attente de validation | ||||||
|  |       tags: | ||||||
|  |         - Admin API | ||||||
|  |       description: | ||||||
|  |         if the request is accepted the status of the | ||||||
|  |         project is changed to ongoing, entrepreneur | ||||||
|  |         account is confirmed and the project is linked | ||||||
|  |         to the admin accepting the request and the | ||||||
|  |         entrepreneur requesting it. Else the pending | ||||||
|  |         project and user info are deleted. | ||||||
|  |       security: | ||||||
|  |           - MyINPulse: | ||||||
|  |               - MyINPulse-admin | ||||||
|  |       requestBody: | ||||||
|  |         required: true | ||||||
|  |         content: | ||||||
|  |           application/json: | ||||||
|  |             schema: | ||||||
|  |               type: object | ||||||
|  |               properties: | ||||||
|  |                 pedingProjectId:  | ||||||
|  |                   type: integer | ||||||
|  |                 decision: | ||||||
|  |                   type: boolean | ||||||
|  |                  | ||||||
|  |       responses: | ||||||
|  |         "200":  | ||||||
|  |           description: OK | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |             missing or invalid | ||||||
|  |              | ||||||
|  |   /admin/projects/add: | ||||||
|  |     post: | ||||||
|  |       summary: Ajout manuel d'un projet | ||||||
|  |       description: | ||||||
|  |         Adds a project with the | ||||||
|  |         inputed details | ||||||
|  |       tags: | ||||||
|  |         - Admin API | ||||||
|  |       security: | ||||||
|  |           - MyINPulse: | ||||||
|  |               - MyINPulse-admin | ||||||
|  |       requestBody: | ||||||
|  |         required: true | ||||||
|  |         content: | ||||||
|  |           application/json: | ||||||
|  |             schema: | ||||||
|  |               type: object | ||||||
|  |               properties: | ||||||
|  |                 name: | ||||||
|  |                   type: string | ||||||
|  |                 founder: | ||||||
|  |                   $ref: "#/components/schemas/user-entrepreneur" | ||||||
|  |                  | ||||||
|  |       responses: | ||||||
|  |         "200":  | ||||||
|  |           description: OK | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |             missing or invalid | ||||||
|  |    | ||||||
|  |   /admin/appointments/report/{appointmentId}: | ||||||
|  |     put: | ||||||
|  |       summary: enregistrer un rapport du rendez-vous | ||||||
|  |       description: | ||||||
|  |         Generate a PDF file formatted  | ||||||
|  |         from input text and links it  | ||||||
|  |         to the appointement. | ||||||
|  |       tags: | ||||||
|  |         - Admin API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |           - MyINPulse-admin | ||||||
|  |       parameters: | ||||||
|  |         - in: path | ||||||
|  |           name: appointmentId | ||||||
|  |           required: true | ||||||
|  |           schema: | ||||||
|  |             type: integer | ||||||
|  |       requestBody: | ||||||
|  |         required: true | ||||||
|  |         content: | ||||||
|  |           application/json: | ||||||
|  |             schema: | ||||||
|  |               type: object | ||||||
|  |               properties: | ||||||
|  |                 title: | ||||||
|  |                   type: string | ||||||
|  |                 body: | ||||||
|  |                   type: string | ||||||
|  |                 conclusion: | ||||||
|  |                   type: string | ||||||
|  |          | ||||||
|  |       responses: | ||||||
|  |         "200":  | ||||||
|  |           description: OK | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |                     missing or invalid | ||||||
|  |     post: | ||||||
|  |       summary: modifier un rapport déja éxistant du rendez-vous | ||||||
|  |       description: | ||||||
|  |         Modifies the report file to input  | ||||||
|  |         text and links it to the appointement. | ||||||
|  |       tags: | ||||||
|  |         - Admin API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |           - MyINPulse-admin | ||||||
|  |       parameters: | ||||||
|  |         - in: path | ||||||
|  |           name: appointmentId | ||||||
|  |           required: true | ||||||
|  |           schema: | ||||||
|  |             type: integer | ||||||
|  |       requestBody: | ||||||
|  |         required: true | ||||||
|  |         content: | ||||||
|  |           application/json: | ||||||
|  |             schema: | ||||||
|  |               type: object | ||||||
|  |               properties: | ||||||
|  |                 title: | ||||||
|  |                   type: string | ||||||
|  |                 body: | ||||||
|  |                   type: string | ||||||
|  |                 conclusion: | ||||||
|  |                   type: string | ||||||
|  |          | ||||||
|  |       responses: | ||||||
|  |         "200":  | ||||||
|  |           description: OK | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |                     missing or invalid | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   /admin/projects/remove/{projectId}: | ||||||
|  |     delete: | ||||||
|  |       summary: supression d'un project | ||||||
|  |       description: | ||||||
|  |         Removes the project | ||||||
|  |         with the inputed id projectId | ||||||
|  |       tags: | ||||||
|  |         - Admin API | ||||||
|  |       security: | ||||||
|  |           - MyINPulse: | ||||||
|  |               - MyINPulse-admin | ||||||
|  |       parameters: | ||||||
|  |         - in: path | ||||||
|  |           name: projectId | ||||||
|  |           required: true | ||||||
|  |           schema: | ||||||
|  |             type: integer | ||||||
|  |            | ||||||
|  |       responses: | ||||||
|  |         "200":  | ||||||
|  |           description: OK | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |             missing or invalid | ||||||
|  |       | ||||||
|  |  | ||||||
|  |   /admin/projects/pending: | ||||||
|  |     get: | ||||||
|  |       summary: Retourne la liste des projets en attente de validation | ||||||
|  |       tags:  | ||||||
|  |         - Admin API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |             - MyINPulse-admin | ||||||
|  |       description: | ||||||
|  |         JSON array of who's elements are objects containing | ||||||
|  |         necessary information for the view (project name, | ||||||
|  |         entrepreneur names, etc..) of all pending projects. | ||||||
|  |       responses: | ||||||
|  |         "200": | ||||||
|  |           description: OK | ||||||
|  |           content: | ||||||
|  |             application/json: | ||||||
|  |               schema: | ||||||
|  |                 type: array | ||||||
|  |                 items: | ||||||
|  |                   type: object | ||||||
|  |                   properties: | ||||||
|  |                     name: | ||||||
|  |                       type: string | ||||||
|  |                     founder: | ||||||
|  |                       $ref: "#/components/schemas/user-entrepreneur" | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is missing or invalid | ||||||
|  |  | ||||||
|  | # | ||||||
|  | #       ____  _                        _      _    ____ ___  | ||||||
|  | #      / ___|| |__   __ _ _ __ ___  __| |    / \  |  _ \_ _| | ||||||
|  | #      \___ \| '_ \ / _` | '__/ _ \/ _` |   / _ \ | |_) | |  | ||||||
|  | #       ___) | | | | (_| | | |  __/ (_| |  / ___ \|  __/| |  | ||||||
|  | #      |____/|_| |_|\__,_|_|  \___|\__,_| /_/   \_\_|  |___| | ||||||
|  | #                    | ||||||
|  |  | ||||||
|  |   /shared/appointments/upcoming: | ||||||
|  |     get: | ||||||
|  |       summary: Retourne la list des prochains rendez-vous de l'utilisateur | ||||||
|  |       tags:  | ||||||
|  |         - Shared API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |             - MyINPulse-admin | ||||||
|  |             - MyINPulse-entrepreneur | ||||||
|  |       description: | ||||||
|  |         JSON array of upcoming appointment data (name, date, time etc..) for a user. | ||||||
|  |       responses: | ||||||
|  |         "200": | ||||||
|  |           description: OK | ||||||
|  |           content: | ||||||
|  |             application/json: | ||||||
|  |               schema: | ||||||
|  |                 type: array | ||||||
|  |                 items: | ||||||
|  |                   type: object | ||||||
|  |                   properties: | ||||||
|  |                     name: | ||||||
|  |                       type: string | ||||||
|  |                     date: | ||||||
|  |                       type: string | ||||||
|  |                     time: | ||||||
|  |                       type: string | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is missing or invalid         | ||||||
|  |  | ||||||
|  |   /shared/projects/lcsection/{projectId}/{title}/{date}: | ||||||
|  |       get: | ||||||
|  |         summary: Retourne la liste de sections de LC avec un titre donné | ||||||
|  |         tags:  | ||||||
|  |           - Shared API | ||||||
|  |         security: | ||||||
|  |         - MyINPulse: | ||||||
|  |             - MyINPulse-admin | ||||||
|  |             - MyINPulse-entrepreneur | ||||||
|  |         description: | ||||||
|  |           JSON array containing Lean Canvas  | ||||||
|  |           section data with a title for the | ||||||
|  |           current date (or given date if the | ||||||
|  |           date parameter is passed) | ||||||
|  |         parameters: | ||||||
|  |           - in: path | ||||||
|  |             required: true | ||||||
|  |             name: projectId | ||||||
|  |             schema: | ||||||
|  |               type: integer | ||||||
|  |           - in: path | ||||||
|  |             required: true | ||||||
|  |             description: this number can be 1, 2,...,8. It is associated with the title of the lcsection | ||||||
|  |             name: title | ||||||
|  |             schema: | ||||||
|  |               type: integer | ||||||
|  |               enum: [1, 2, 3, 4, 5, 6, 7, 8]  | ||||||
|  |              | ||||||
|  |           - in: path | ||||||
|  |             required: true | ||||||
|  |             name: date | ||||||
|  |             description: the date corresponding to the wanted version of lc section. "Nan" for the latest version | ||||||
|  |             example: "NaN" | ||||||
|  |             schema: | ||||||
|  |               type: string | ||||||
|  |                | ||||||
|  |         responses: | ||||||
|  |           "200": | ||||||
|  |             description: OK | ||||||
|  |             content: | ||||||
|  |               application/json: | ||||||
|  |                 schema: | ||||||
|  |                   type: array | ||||||
|  |                   items: | ||||||
|  |                     type: object | ||||||
|  |                     properties: | ||||||
|  |                       section: | ||||||
|  |                         type: string | ||||||
|  |                       txt: | ||||||
|  |                         type: string | ||||||
|  |                       time: | ||||||
|  |                         type: string | ||||||
|  |           "400": | ||||||
|  |             description: Bad request | ||||||
|  |           "401": | ||||||
|  |             description: Authorization information is missing or invalid | ||||||
|  |    | ||||||
|  |   /shared/projects/entrepreneurs/{projectId}: | ||||||
|  |     get: | ||||||
|  |       summary: Retourne la liste d'entrepreneurs associée à un projet donné | ||||||
|  |       tags:  | ||||||
|  |           - Shared API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |             - MyINPulse-admin | ||||||
|  |             - MyINPulse-entrepreneur | ||||||
|  |       description: | ||||||
|  |         JSON array of entrepreneur | ||||||
|  |         names associated with a project | ||||||
|  |       parameters: | ||||||
|  |           - in: path | ||||||
|  |             name: projectId | ||||||
|  |             schema: | ||||||
|  |               type: integer | ||||||
|  |             required: true | ||||||
|  |       responses: | ||||||
|  |         "200": | ||||||
|  |           description: OK | ||||||
|  |           content: | ||||||
|  |             application/json: | ||||||
|  |               schema: | ||||||
|  |                 type: array | ||||||
|  |                 items: | ||||||
|  |                   $ref: "#/components/schemas/user-entrepreneur" | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is missing or invalid | ||||||
|  |    | ||||||
|  |   /shared/projects/admin/{projectId}: | ||||||
|  |     get: | ||||||
|  |       summary: Retourne les informations de l'admin qui accompagne le projet | ||||||
|  |       tags:  | ||||||
|  |           - Shared API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |             - MyINPulse-admin | ||||||
|  |             - MyINPulse-entrepreneur | ||||||
|  |       description:  | ||||||
|  |         JSON object containing information (name, gmail, tel, etc..)  | ||||||
|  |         the admin supervising the project with id projectID. | ||||||
|  |       parameters: | ||||||
|  |             - in: path | ||||||
|  |               name: projectId | ||||||
|  |               schema: | ||||||
|  |                 type: integer | ||||||
|  |               required: true | ||||||
|  |       responses: | ||||||
|  |         "200": | ||||||
|  |           description: OK | ||||||
|  |           content: | ||||||
|  |             application/json: | ||||||
|  |               schema: | ||||||
|  |                 $ref: "#/components/schemas/user-admin" | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |             missing or invalid | ||||||
|  |    | ||||||
|  |   /shared/projects/appointments/{projectId}: | ||||||
|  |     get: | ||||||
|  |       summary: Retourne les rendez-vous du projet  | ||||||
|  |       tags:  | ||||||
|  |           - Shared API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |             - MyINPulse-admin | ||||||
|  |             - MyINPulse-entrepreneur | ||||||
|  |       description:  | ||||||
|  |         JSON array of upcoming and past appointment  | ||||||
|  |         data for the project with id projectID. | ||||||
|  |       parameters: | ||||||
|  |         - in: path | ||||||
|  |           name: projectId | ||||||
|  |           schema: | ||||||
|  |             type: integer | ||||||
|  |           required: true | ||||||
|  |       responses: | ||||||
|  |         "200": | ||||||
|  |           description: OK | ||||||
|  |           content: | ||||||
|  |             application/json: | ||||||
|  |               schema: | ||||||
|  |                 type: array | ||||||
|  |                 items: | ||||||
|  |                   type: object | ||||||
|  |                   properties: | ||||||
|  |                     appointementId: | ||||||
|  |                       type: integer | ||||||
|  |                     name:  | ||||||
|  |                       type: string | ||||||
|  |                     date:  | ||||||
|  |                       type: string | ||||||
|  |                     time:  | ||||||
|  |                       type: string | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |             missing or invalid | ||||||
|  |   /shared/projects/appointments/report/{apointementId}: | ||||||
|  |     get: | ||||||
|  |       summary: Retourne le rapport pdf du rendez-vous | ||||||
|  |       tags:  | ||||||
|  |           - Shared API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |             - MyINPulse-admin | ||||||
|  |             - MyINPulse-entrepreneur | ||||||
|  |       description: | ||||||
|  |         PDF file containing the ap- | ||||||
|  |         pointment report | ||||||
|  |       parameters: | ||||||
|  |         - in: path | ||||||
|  |           name: apointementId | ||||||
|  |           schema: | ||||||
|  |             type: integer | ||||||
|  |           required: true | ||||||
|  |       responses: | ||||||
|  |         "200": | ||||||
|  |           description: OK | ||||||
|  |           content: | ||||||
|  |             application/pdf: | ||||||
|  |               schema: | ||||||
|  |                 type: string | ||||||
|  |                 format: binary | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |             missing or invalid | ||||||
|  |    | ||||||
|  |   /shared/appointments/request: | ||||||
|  |     post: | ||||||
|  |       summary: demander un rendez-vous | ||||||
|  |       description: | ||||||
|  |         will add an appointement request request by the applicant  | ||||||
|  |         to have an appointment to be confirmed or denied by the | ||||||
|  |         specified participants of the appointement. | ||||||
|  |       tags: | ||||||
|  |         - Shared API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |           - MyINPulse-entrepreneur | ||||||
|  |           - MyINPulse-admin | ||||||
|  |       requestBody: | ||||||
|  |         description: \"participants\" property is an array containing userids of the participants in the appointement | ||||||
|  |         required: true | ||||||
|  |         content: | ||||||
|  |           application/json: | ||||||
|  |             schema: | ||||||
|  |               type: object | ||||||
|  |               properties: | ||||||
|  |                 title: | ||||||
|  |                   type: string | ||||||
|  |                 start_time: | ||||||
|  |                   type: string | ||||||
|  |                 end_time: | ||||||
|  |                   type: string | ||||||
|  |                 place: | ||||||
|  |                   type: string | ||||||
|  |                 applicantId: | ||||||
|  |                   type: integer | ||||||
|  |                 participants: | ||||||
|  |                   #/* */ | ||||||
|  |                   type: array | ||||||
|  |                   items: | ||||||
|  |                     type: integer | ||||||
|  |                      | ||||||
|  |       responses: | ||||||
|  |         "200":  | ||||||
|  |           description: OK | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |                     missing or invalid         | ||||||
|  |    | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #      _____ _   _ _____ ____  _____ ____  ____  _____ _   _ _____ _   _ ____   | ||||||
|  | #     | ____| \ | |_   _|  _ \| ____|  _ \|  _ \| ____| \ | | ____| | | |  _ \  | ||||||
|  | #     |  _| |  \| | | | | |_) |  _| | |_) | |_) |  _| |  \| |  _| | | | | |_) | | ||||||
|  | #     | |___| |\  | | | |  _ <| |___|  __/|  _ <| |___| |\  | |___| |_| |  _ <  | ||||||
|  | #     |_____|_|_\_| |_| |_| \_\_____|_|   |_| \_\_____|_| \_|_____|\___/|_| \_\ | ||||||
|  | #        / \  |  _ \_ _|                                                        | ||||||
|  | #       / _ \ | |_) | |                                                         | ||||||
|  | #      / ___ \|  __/| |                                                         | ||||||
|  | #     /_/   \_\_|  |___|                                                        | ||||||
|  | # | ||||||
|  |  | ||||||
|  |                      | ||||||
|  |   /entrepreneur/projects/request: | ||||||
|  |     post: | ||||||
|  |       summary: demander la création et validation d'un projet | ||||||
|  |       tags: | ||||||
|  |         - Entrepreneurs API | ||||||
|  |       description:  | ||||||
|  |         Adds project to pending projects  | ||||||
|  |         to then be accepted or rejected by | ||||||
|  |         an admin | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |           - MyINPulse-entrepreneur | ||||||
|  |       requestBody: | ||||||
|  |         required: true | ||||||
|  |         content: | ||||||
|  |           application/json: | ||||||
|  |             schema: | ||||||
|  |               type: object | ||||||
|  |               properties: | ||||||
|  |                 name: | ||||||
|  |                   type: string | ||||||
|  |                 founder: | ||||||
|  |                   $ref: "#/components/schemas/user-entrepreneur" | ||||||
|  |       responses: | ||||||
|  |         "200":  | ||||||
|  |           description: OK | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |             missing or invalid | ||||||
|  |              | ||||||
|  |             | ||||||
|  |   /entrepreneur/lcsection/add/{projectId}: | ||||||
|  |     post: | ||||||
|  |       summary: ajouter une sections au LC | ||||||
|  |       description:  | ||||||
|  |         Adds input data to the user's LC  | ||||||
|  |         with a specified title. | ||||||
|  |       tags: | ||||||
|  |        - Entrepreneurs API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |           - MyINPulse-entrepreneur | ||||||
|  |       parameters: | ||||||
|  |         - in: path | ||||||
|  |           name: projectId | ||||||
|  |           schema: | ||||||
|  |             type: integer | ||||||
|  |           required: true | ||||||
|  |       requestBody: | ||||||
|  |         required: true | ||||||
|  |         content: | ||||||
|  |           application/json: | ||||||
|  |             schema: | ||||||
|  |               type: object | ||||||
|  |               properties: | ||||||
|  |                 title: | ||||||
|  |                   type: string | ||||||
|  |                 txt: | ||||||
|  |                   type: string | ||||||
|  |       responses: | ||||||
|  |         "200":  | ||||||
|  |           description: OK | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |             missing or invalid | ||||||
|  |   /entrepreneur/lcsection/modify/{sectionId}: | ||||||
|  |     put: | ||||||
|  |       summary: modifier les données d'une section LC | ||||||
|  |       description:  | ||||||
|  |         Modifies input Lean Canvas section by changing it to  | ||||||
|  |         the information in the request body and changes the | ||||||
|  |         time stamp. | ||||||
|  |       tags: | ||||||
|  |        - Entrepreneurs API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |           - MyINPulse-entrepreneur | ||||||
|  |       parameters: | ||||||
|  |         - in: path | ||||||
|  |           name: sectionId | ||||||
|  |           schema: | ||||||
|  |             type: integer | ||||||
|  |           required: true | ||||||
|  |       requestBody: | ||||||
|  |         required: true | ||||||
|  |         content: | ||||||
|  |           application/json: | ||||||
|  |             schema: | ||||||
|  |               type: object | ||||||
|  |               properties: | ||||||
|  |                 title: | ||||||
|  |                   type: string | ||||||
|  |                 txt: | ||||||
|  |                   type: string | ||||||
|  |       responses: | ||||||
|  |         "200":  | ||||||
|  |           description: OK | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |             missing or invalid | ||||||
|  |              | ||||||
|  |   /entrepreneur/lcsection/remove/{sectionId}: | ||||||
|  |     delete: | ||||||
|  |       summary: supprimer une section LC. | ||||||
|  |       description: | ||||||
|  |         Deletes section from Lean Canvas | ||||||
|  |       tags: | ||||||
|  |         - Entrepreneurs API | ||||||
|  |       security: | ||||||
|  |         - MyINPulse: | ||||||
|  |           - MyINPulse-entrepreneur | ||||||
|  |       parameters: | ||||||
|  |         - in: path | ||||||
|  |           name: sectionId | ||||||
|  |           schema: | ||||||
|  |             type: integer | ||||||
|  |           required: true | ||||||
|  |       responses: | ||||||
|  |         "200":  | ||||||
|  |           description: OK | ||||||
|  |         "400": | ||||||
|  |           description: Bad request | ||||||
|  |         "401": | ||||||
|  |           description: Authorization information is | ||||||
|  |                       missing or invalid | ||||||
|  |    | ||||||
|  |    | ||||||
|  |    | ||||||
|  |    | ||||||
							
								
								
									
										3
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										3
									
								
								Makefile
									
									
									
									
									
								
							| @@ -33,6 +33,8 @@ dev-front: clean vite keycloak | |||||||
| 	@cp config/frontdev.docker-compose.yaml docker-compose.yaml | 	@cp config/frontdev.docker-compose.yaml docker-compose.yaml | ||||||
| 	@docker compose up -d --build | 	@docker compose up -d --build | ||||||
| 	@cd ./front/MyINPulse-front/ && npm run dev | 	@cd ./front/MyINPulse-front/ && npm run dev | ||||||
|  | 	@echo "cd MyINPulse-back" && echo 'export $$(cat .env | xargs)' | ||||||
|  | 	@echo "./gradlew bootRun --args='--server.port=8081'" | ||||||
|  |  | ||||||
| prod: clean keycloak | prod: clean keycloak | ||||||
| 	@cp config/prod.env front/MyINPulse-front/.env | 	@cp config/prod.env front/MyINPulse-front/.env | ||||||
| @@ -43,6 +45,7 @@ prod: clean keycloak | |||||||
| 	 | 	 | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| dev-back: keycloak | dev-back: keycloak | ||||||
| 	@cp config/backdev.env front/MyINPulse-front/.env | 	@cp config/backdev.env front/MyINPulse-front/.env | ||||||
| 	@cp config/backdev.env .env | 	@cp config/backdev.env .env | ||||||
|   | |||||||
							
								
								
									
										0
									
								
								front/Dockerfile
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										0
									
								
								front/Dockerfile
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
								
								
									
										63
									
								
								front/MyINPulse-front/fake_data/db.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								front/MyINPulse-front/fake_data/db.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | |||||||
|  | { | ||||||
|  |   "entrepreneurs": [ | ||||||
|  |     { "id": 1, "name": "Alice", "email": "alice@example.com" }, | ||||||
|  |     { "id": 2, "name": "Bob", "email": "bob@example.com" }, | ||||||
|  |     { "id": 3, "name": "Charlie", "email": "charlie@example.com" } | ||||||
|  |   ], | ||||||
|  |   "data": [ | ||||||
|  |     { | ||||||
|  |       "projectId": 1, | ||||||
|  |       "title": 1, | ||||||
|  |       "title_text": "1. Problème", | ||||||
|  |       "description": "3 problèmes essentiels à résoudre pour le client" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "projectId": 1, | ||||||
|  |       "title": 2, | ||||||
|  |       "title_text": "2. Segments", | ||||||
|  |       "description": "Les segments de clientèle visés" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "projectId": 1, | ||||||
|  |       "title": 3, | ||||||
|  |       "title_text": "3. Valeur", | ||||||
|  |       "description": "La proposition de valeur" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "projectId": 1, | ||||||
|  |       "title": 4, | ||||||
|  |       "title_text": "4. Solution", | ||||||
|  |       "description": "Les solutions proposées" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "projectId": 1, | ||||||
|  |       "title": 5, | ||||||
|  |       "title_text": "5. Avantage", | ||||||
|  |       "description": "Les avantages concurrentiels" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "projectId": 1, | ||||||
|  |       "title": 6, | ||||||
|  |       "title_text": "6. Canaux", | ||||||
|  |       "description": "Les canaux de distribution" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "projectId": 1, | ||||||
|  |       "title": 7, | ||||||
|  |       "title_text": "7. Indicateurs", | ||||||
|  |       "description": "Les indicateurs clés de performance" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "projectId": 1, | ||||||
|  |       "title": 8, | ||||||
|  |       "title_text": "8. Coûts", | ||||||
|  |       "description": "Les coûts associés" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "projectId": 1, | ||||||
|  |       "title": 9, | ||||||
|  |       "title_text": "9. Revenus", | ||||||
|  |       "description": "Les sources de revenus" | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								front/MyINPulse-front/fake_data/open.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										2
									
								
								front/MyINPulse-front/fake_data/open.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,2 @@ | |||||||
|  | #!/usr/bin/bash | ||||||
|  | json-server --watch db.json --port 5000 | ||||||
| @@ -18,7 +18,8 @@ | |||||||
|     "pinia": "^2.3.1", |     "pinia": "^2.3.1", | ||||||
|     "pinia-plugin-persistedstate": "^4.2.0", |     "pinia-plugin-persistedstate": "^4.2.0", | ||||||
|     "vue": "^3.5.13", |     "vue": "^3.5.13", | ||||||
|     "vue-router": "^4.5.0" |     "vue-router": "^4.5.0", | ||||||
|  |     "jwt-decode": "^4.0.0" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@playwright/test": "^1.49.1", |     "@playwright/test": "^1.49.1", | ||||||
|   | |||||||
| @@ -1,47 +1,16 @@ | |||||||
| <script setup lang="ts"> | <script setup lang="ts"> | ||||||
| import { RouterView } from "vue-router"; | import { RouterLink, RouterView } from 'vue-router' | ||||||
| import ErrorWrapper from "@/views/errorWrapper.vue"; | import ErrorWrapper from "@/views/errorWrapper.vue"; | ||||||
| import ProjectComponent from "@/components/ProjectComponent.vue"; |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <template> | <template> | ||||||
|     <HeaderComponent /> |  | ||||||
|     <error-wrapper></error-wrapper> | <Header /> | ||||||
|     <div id="main"> |     <ErrorWrapper /> | ||||||
|         <ProjectComponent |     <!--<RouterLink to="/">Home</RouterLink> | --> | ||||||
|             v-for="(project, index) in projects" |     <!--<RouterLink to="/canvas">Canvas</RouterLink> --> | ||||||
|             :key="index" |  | ||||||
|             :project-name="project.name" |  | ||||||
|         /> |  | ||||||
|     </div> |  | ||||||
|     <RouterView /> |     <RouterView /> | ||||||
| </template> | </template> | ||||||
|  |  | ||||||
| <script lang="ts"> |  | ||||||
| import HeaderComponent from "@/components/HeaderComponent.vue"; |  | ||||||
|  |  | ||||||
| export default { |  | ||||||
|     name: "App", |  | ||||||
|     components: { |  | ||||||
|         HeaderComponent, |  | ||||||
|     }, |  | ||||||
|     data() { |  | ||||||
|         return { |  | ||||||
|             projects: [ |  | ||||||
|                 { |  | ||||||
|                     name: "Projet Alpha", |  | ||||||
|                     //link: './project-alpha.html', |  | ||||||
|                     //members: ['Alice', 'Bob', 'Charlie'], |  | ||||||
|                 }, |  | ||||||
|                 { |  | ||||||
|                     name: "Projet Beta", |  | ||||||
|                     //link: './project-beta.html', |  | ||||||
|                     //members: ['David', 'Eve', 'Frank'], |  | ||||||
|                 }, |  | ||||||
|             ], |  | ||||||
|         }; |  | ||||||
|     }, |  | ||||||
| }; |  | ||||||
| </script> |  | ||||||
|  |  | ||||||
| <style scoped></style> |  | ||||||
|   | |||||||
							
								
								
									
										102
									
								
								front/MyINPulse-front/src/components/AddProjectForm.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								front/MyINPulse-front/src/components/AddProjectForm.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | |||||||
|  | <template> | ||||||
|  |     <form class="add-project-form" @submit.prevent="submitProject"> | ||||||
|  |       <h2>Ajouter un projet</h2> | ||||||
|  |    | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="projectName">Nom du projet</label> | ||||||
|  |         <input | ||||||
|  |           id="projectName" | ||||||
|  |           v-model="project.projectName" | ||||||
|  |           type="text" | ||||||
|  |           required | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |    | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="creationDate">Date de création</label> | ||||||
|  |         <input | ||||||
|  |           id="creationDate" | ||||||
|  |           v-model="project.creationDate" | ||||||
|  |           type="text" | ||||||
|  |           placeholder="JJ-MM-AAAA" | ||||||
|  |           required | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |    | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="logo">Logo</label> | ||||||
|  |         <input | ||||||
|  |           id="logo" | ||||||
|  |           v-model="project.logo" | ||||||
|  |           type="text" | ||||||
|  |           placeholder="(à discuter)" | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |    | ||||||
|  |       <button type="submit">Ajouter</button> | ||||||
|  |     </form> | ||||||
|  |   </template> | ||||||
|  |    | ||||||
|  |   <script setup lang="ts"> | ||||||
|  |   import { ref } from "vue"; | ||||||
|  |   import { postApi } from "@/services/api.ts";  | ||||||
|  |    | ||||||
|  |   const project = ref({ | ||||||
|  |     idProject: 0, | ||||||
|  |     projectName: "", | ||||||
|  |     creationDate: "", | ||||||
|  |     logo: "to be discussed not yet fixed", | ||||||
|  |   }); | ||||||
|  |    | ||||||
|  |   function submitProject() { | ||||||
|  |     postApi("/admin/projects/add", project.value); | ||||||
|  |   } | ||||||
|  |   </script> | ||||||
|  |    | ||||||
|  |   <style scoped> | ||||||
|  |   .add-project-form { | ||||||
|  |     max-width: 500px; | ||||||
|  |     margin: 0 auto; | ||||||
|  |     padding: 20px; | ||||||
|  |     background: #fff; | ||||||
|  |     border-radius: 10px; | ||||||
|  |     box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1); | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   h2 { | ||||||
|  |     margin-bottom: 20px; | ||||||
|  |     font-size: 24px; | ||||||
|  |     color: #333; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .form-group { | ||||||
|  |     margin-bottom: 15px; | ||||||
|  |     display: flex; | ||||||
|  |     flex-direction: column; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   label { | ||||||
|  |     font-weight: bold; | ||||||
|  |     margin-bottom: 5px; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   input { | ||||||
|  |     padding: 8px; | ||||||
|  |     border-radius: 5px; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   button { | ||||||
|  |     background-color: #4caf50; | ||||||
|  |     color: white; | ||||||
|  |     padding: 10px 15px; | ||||||
|  |     border: none; | ||||||
|  |     border-radius: 5px; | ||||||
|  |     cursor: pointer; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   button:hover { | ||||||
|  |     background-color: #45a049; | ||||||
|  |   } | ||||||
|  |   </style> | ||||||
|  |    | ||||||
							
								
								
									
										75
									
								
								front/MyINPulse-front/src/components/Agenda.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								front/MyINPulse-front/src/components/Agenda.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,75 @@ | |||||||
|  | <template> | ||||||
|  |     <div id="agenda"> | ||||||
|  |         <h3>Rendez-vous</h3> | ||||||
|  |         <table> | ||||||
|  |         <tbody> | ||||||
|  |             <tr v-for=" (p, index) in projectRDV" :key="index" > | ||||||
|  |                 <td>{{ p.projectName }} </td> <td>{{ p.date }}</td> <td>{{ p.lieu }}</td> | ||||||
|  |             </tr> | ||||||
|  |         </tbody> | ||||||
|  |         </table> | ||||||
|  |     </div> | ||||||
|  | </template> | ||||||
|  |  | ||||||
|  | <script setup lang="ts"> | ||||||
|  |     import { defineProps } from "vue"; | ||||||
|  |  | ||||||
|  |     interface rendezVous{ | ||||||
|  |         projectName: String, | ||||||
|  |         date: String, | ||||||
|  |         lieu: String, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const props = defineProps<{ | ||||||
|  |         projectRDV: rendezVous[] | ||||||
|  |     }>(); | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <style scoped> | ||||||
|  |  #agenda {    | ||||||
|  |     padding: 20px; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /* Table Styling */ | ||||||
|  |  table { | ||||||
|  |     width: 100%; | ||||||
|  |     border-collapse: collapse; | ||||||
|  |     font-family: Arial, sans-serif; | ||||||
|  |     text-align: left; | ||||||
|  |     margin-top: 20px; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   /* Header Row (if exists) */ | ||||||
|  |   th { | ||||||
|  |     background-color: #f4f4f4; | ||||||
|  |     padding: 12px; | ||||||
|  |     font-weight: bold; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   /* Table Body Rows */ | ||||||
|  |   tbody tr { | ||||||
|  |     border-bottom: 1px solid #ddd; | ||||||
|  |     transition: background-color 0.2s ease; /* Smooth hover effect */ | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   tbody tr:hover { | ||||||
|  |     background-color: #f9f9f9; /* Highlight row on hover */ | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   /* Cells Styling */ | ||||||
|  |   td { | ||||||
|  |     padding: 10px; | ||||||
|  |     border: 1px solid #eee; | ||||||
|  |     font-size: 14px; | ||||||
|  |     vertical-align: middle; /* Align text to middle */ | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   /* First Column Styling */ | ||||||
|  |   td:first-child { | ||||||
|  |     text-align: center; | ||||||
|  |     width: 50px; /* Adjust width as needed */ | ||||||
|  |   } | ||||||
|  |   | ||||||
|  | </style> | ||||||
							
								
								
									
										173
									
								
								front/MyINPulse-front/src/components/LoginComponent.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										173
									
								
								front/MyINPulse-front/src/components/LoginComponent.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,173 @@ | |||||||
|  | <script lang="ts" setup> | ||||||
|  | import { onMounted, ref } from "vue"; | ||||||
|  | import { useRouter } from "vue-router"; | ||||||
|  | import {jwtDecode} from "jwt-decode"; // i hope this doesn't break the code later | ||||||
|  | import { store } from "../main.ts"; | ||||||
|  | import { callApi } from "@/services/api.ts"; | ||||||
|  |  | ||||||
|  | const router = useRouter(); | ||||||
|  |  | ||||||
|  | type TokenPayload = { | ||||||
|  |   realm_access?: { | ||||||
|  |     roles?: string[]; | ||||||
|  |   }; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const customRequest = ref(''); | ||||||
|  |  | ||||||
|  | onMounted(() => { | ||||||
|  |   if (store.authenticated && store.user.token) { | ||||||
|  |     try { | ||||||
|  |       const decoded = jwtDecode<TokenPayload>(store.user.token); | ||||||
|  |       const roles = decoded.realm_access?.roles || []; | ||||||
|  |  | ||||||
|  |       if (roles.includes("MyINPulse-admin")) { | ||||||
|  |         router.push("/"); | ||||||
|  |       } else if (roles.includes("entrepreneur")) { | ||||||
|  |         router.push("/entrepreneur"); | ||||||
|  |       } | ||||||
|  |     } catch (err) { | ||||||
|  |       console.error("Failed to decode token", err); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }); | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | <template> | ||||||
|  |     <error-wrapper></error-wrapper> | ||||||
|  |     <div class="auth-container"> | ||||||
|  |       <div class="auth-card"> | ||||||
|  |         <h1>Bienvenue</h1> | ||||||
|  |  | ||||||
|  |         <div class="status" :class="store.authenticated ? 'success' : 'error'"> | ||||||
|  |           <p> | ||||||
|  |             {{ store.authenticated ? '✅ Authenticated' : '❌ Not Authenticated' }} | ||||||
|  |           </p> | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         <div class="actions"> | ||||||
|  |           <button @click="store.login">Login</button> | ||||||
|  |           <button @click="store.logout">Logout</button> | ||||||
|  |           <button @click="store.signup">Signup-admin</button> | ||||||
|  |           <button @click="store.signup">Signup-Entrepreneur</button> | ||||||
|  |           <button @click="store.refreshUserToken">Refresh Token</button> | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         <div class="token-section" v-if="store.authenticated"> | ||||||
|  |           <p><strong>Access Token:</strong></p> | ||||||
|  |           <pre>{{ store.user.token }}</pre> | ||||||
|  |  | ||||||
|  |           <p><strong>Refresh Token:</strong></p> | ||||||
|  |           <pre>{{ store.user.refreshToken }}</pre> | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         <div class="api-calls"> | ||||||
|  |           <h2>Test API Calls</h2> | ||||||
|  |           <button @click="callApi('random')">Call Entrepreneur API</button> | ||||||
|  |           <button @click="callApi('random2')">Call Admin API</button> | ||||||
|  |           <button @click="callApi('unauth/dev')">Call Unauth API</button> | ||||||
|  |  | ||||||
|  |           <div class="custom-call"> | ||||||
|  |             <input v-model="customRequest" placeholder="Custom endpoint" /> | ||||||
|  |             <button @click="callApi(customRequest)">Call</button> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  | </template> | ||||||
|  |  | ||||||
|  | <style scoped> | ||||||
|  | .auth-container { | ||||||
|  |   display: flex; | ||||||
|  |   justify-content: center; | ||||||
|  |   align-items: center; | ||||||
|  |   padding: 3rem 1rem; | ||||||
|  |   min-height: 100vh; | ||||||
|  |   background-color: #eef1f5; | ||||||
|  |   font-family: Arial, sans-serif; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .auth-card { | ||||||
|  |   background: white; | ||||||
|  |   padding: 2rem; | ||||||
|  |   border-radius: 12px; | ||||||
|  |   box-shadow: 0 5px 15px rgba(0, 0, 0, 0.1); | ||||||
|  |   width: 100%; | ||||||
|  |   max-width: 600px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | h1 { | ||||||
|  |   text-align: center; | ||||||
|  |   margin-bottom: 1rem; | ||||||
|  |   color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .status { | ||||||
|  |   text-align: center; | ||||||
|  |   margin-bottom: 1.5rem; | ||||||
|  |   font-weight: bold; | ||||||
|  | } | ||||||
|  | .success { | ||||||
|  |   color: green; | ||||||
|  | } | ||||||
|  | .error { | ||||||
|  |   color: red; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .actions { | ||||||
|  |   display: flex; | ||||||
|  |   flex-wrap: wrap; | ||||||
|  |   gap: 1rem; | ||||||
|  |   justify-content: center; | ||||||
|  |   margin-bottom: 1.5rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .actions button { | ||||||
|  |   padding: 0.6rem 1rem; | ||||||
|  |   background-color: #4a90e2; | ||||||
|  |   border: none; | ||||||
|  |   color: white; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: background-color 0.2s; | ||||||
|  | } | ||||||
|  | .actions button:hover { | ||||||
|  |   background-color: #357abd; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .token-section pre { | ||||||
|  |   background: #f6f8fa; | ||||||
|  |   padding: 0.5rem; | ||||||
|  |   overflow-x: auto; | ||||||
|  |   border: 1px solid #ddd; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   margin-bottom: 1rem; | ||||||
|  |   font-size: 0.85rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .api-calls { | ||||||
|  |   margin-top: 2rem; | ||||||
|  | } | ||||||
|  | .api-calls h2 { | ||||||
|  |   margin-bottom: 1rem; | ||||||
|  |   color: #444; | ||||||
|  |   font-size: 1.1rem; | ||||||
|  | } | ||||||
|  | .api-calls button { | ||||||
|  |   margin-right: 0.5rem; | ||||||
|  |   margin-bottom: 0.5rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .custom-call { | ||||||
|  |   margin-top: 1rem; | ||||||
|  |   display: flex; | ||||||
|  |   gap: 0.5rem; | ||||||
|  | } | ||||||
|  | .custom-call input { | ||||||
|  |   flex: 1; | ||||||
|  |   padding: 0.5rem; | ||||||
|  |   border: 1px solid #ccc; | ||||||
|  |   border-radius: 6px; | ||||||
|  | } | ||||||
|  | </style> | ||||||
							
								
								
									
										149
									
								
								front/MyINPulse-front/src/components/PendingProjectComponent.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										149
									
								
								front/MyINPulse-front/src/components/PendingProjectComponent.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,149 @@ | |||||||
|  | <template> | ||||||
|  |     <div class="project"> | ||||||
|  |         <div class="project-header"> | ||||||
|  |             <div class="project-title"> | ||||||
|  |                 <h2>{{ projectName }}</h2> | ||||||
|  |                 <p>Projet mis le: {{ creationDate }}</p> | ||||||
|  |             </div> | ||||||
|  |         <div class="project-button"> | ||||||
|  |             <button id="accept" @click="acceptProject">Accepter</button> | ||||||
|  |             <button id="refus" @click="refuseProject">Refuser</button> | ||||||
|  |         </div> | ||||||
|  |     </div> | ||||||
|  |     </div> | ||||||
|  |  | ||||||
|  | </template> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | <script setup lang="ts"> | ||||||
|  | import { defineProps } from "vue"; | ||||||
|  | import { postApi } from "@/services/api"; | ||||||
|  | import { addNewMessage, color } from "@/services/popupDisplayer"; | ||||||
|  |  | ||||||
|  | const props = defineProps<{ | ||||||
|  |   projectName: string; | ||||||
|  |   creationDate: string; | ||||||
|  | }>(); | ||||||
|  |  | ||||||
|  | const URI = "/admin/projects/pending/decision"; | ||||||
|  |  | ||||||
|  | const sendDecision = (decision: "true" | "false") => { | ||||||
|  |   postApi( | ||||||
|  |     URI, | ||||||
|  |     { | ||||||
|  |       projectName: props.projectName, | ||||||
|  |       decision, | ||||||
|  |     }, | ||||||
|  |     () => { | ||||||
|  |       addNewMessage( | ||||||
|  |         `Projet ${props.projectName} ${decision === "true" ? "accepté" : "refusé"}`, | ||||||
|  |         color.Green | ||||||
|  |       ); | ||||||
|  |     }, | ||||||
|  |     (err) => { | ||||||
|  |       addNewMessage(`Erreur lors de la décision`, color.Red); | ||||||
|  |       console.error(err); | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const acceptProject = () => sendDecision("true"); | ||||||
|  | const refuseProject = () => sendDecision("false"); | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | <style scoped> | ||||||
|  | .project { | ||||||
|  |   background: linear-gradient(to right, #f4f4f4, #ffffff); | ||||||
|  |   border: 1px solid #ddd; | ||||||
|  |   border-radius: 10px; | ||||||
|  |   padding: 20px; | ||||||
|  |   margin: 20px 0; | ||||||
|  |   box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1); | ||||||
|  |   font-family: Arial, sans-serif; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Header Styling */ | ||||||
|  | .project-header { | ||||||
|  |   display: flex; | ||||||
|  |   justify-content: space-between; | ||||||
|  |   align-items: flex-start; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .project-title { | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .project-title h2 { | ||||||
|  |   font-size: 20px; | ||||||
|  |   color: #333; | ||||||
|  |   margin: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .project-title p { | ||||||
|  |   font-size: 14px; | ||||||
|  |   color: #777; | ||||||
|  |   margin-top: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Button Container */ | ||||||
|  | .project-buttons { | ||||||
|  |   display: flex; | ||||||
|  |   gap: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #accept { | ||||||
|  |   background-color: #45a049; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #refus { | ||||||
|  |   background-color: red; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | button { | ||||||
|  |   padding: 10px 15px; | ||||||
|  |   color: white; | ||||||
|  |   border: none; | ||||||
|  |   cursor: pointer; | ||||||
|  |   border-radius: 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | button:hover { | ||||||
|  |   transform: scale(1.05); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #accept:hover { | ||||||
|  |   background-color: #3e8e41; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #refus:hover { | ||||||
|  |   background-color: darkred; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Project Body (unchanged) */ | ||||||
|  | .project-body { | ||||||
|  |   margin-top: 15px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .project-body p { | ||||||
|  |   font-size: 16px; | ||||||
|  |   color: #555; | ||||||
|  |   margin-bottom: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .project-body ul { | ||||||
|  |   list-style-type: disc; | ||||||
|  |   margin: 0; | ||||||
|  |   padding-left: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .project-body ul li { | ||||||
|  |   font-size: 14px; | ||||||
|  |   color: #666; | ||||||
|  |   line-height: 1.6; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | </style> | ||||||
| @@ -1,21 +1,130 @@ | |||||||
| <template> | <template> | ||||||
|     <div class="project"> |     <div @click="goToLink" class="project"> | ||||||
|         <div class="project-header"> |         <div class="project-header"> | ||||||
|             <h2>{{ projectName }}</h2> |             <h2 >{{ projectName }}</h2> | ||||||
|  |             <div class="project-buttons"> | ||||||
|  |                 <button class="contact-btn">Contact</button> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="project-body">    | ||||||
|  |             <ul> | ||||||
|  |                 <li v-for="(name, index) in listName" :key="index">{{ name }}</li> | ||||||
|  |             </ul> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
| </template> | </template> | ||||||
|  |  | ||||||
| <script lang="ts"> |  | ||||||
| import type { PropType } from "vue"; |  | ||||||
|  |  | ||||||
| export default { |  | ||||||
|     name: "ProjectComponent", | <script setup lang="ts"> | ||||||
|     props: { | import { defineProps } from "vue"; | ||||||
|         projectName: { | import { useRouter } from 'vue-router' | ||||||
|             type: Object as PropType<string>, |  | ||||||
|             required: true, |  | ||||||
|         }, | const props = defineProps<{ | ||||||
|     }, |     projectName: string; | ||||||
|  |     listName: string[]; | ||||||
|  |     projectLink: string; | ||||||
|  | }>(); | ||||||
|  |  | ||||||
|  | const router = useRouter(); | ||||||
|  |  | ||||||
|  | const goToLink = () => { | ||||||
|  |   if (props.projectLink) { | ||||||
|  |     router.push(props.projectLink); | ||||||
|  |   } | ||||||
|  |   | ||||||
| }; | }; | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | <style scoped> | ||||||
|  | .project { | ||||||
|  |     background: linear-gradient(to right, #f4f4f4, #ffffff); | ||||||
|  |     border: 1px solid #ddd; | ||||||
|  |     border-radius: 10px; | ||||||
|  |     padding: 20px; | ||||||
|  |     margin: 20px 0; | ||||||
|  |     box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1); | ||||||
|  |     font-family: Arial, sans-serif; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   /* Header Styling */ | ||||||
|  |   .project-header { | ||||||
|  |     display: flex; | ||||||
|  |     justify-content: space-between; | ||||||
|  |     align-items: center; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .project-header h2 { | ||||||
|  |     font-size: 20px; | ||||||
|  |     color: #333; | ||||||
|  |     margin: 0; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   /* Button Container */ | ||||||
|  |   .project-buttons { | ||||||
|  |     display: flex; | ||||||
|  |     gap: 10px; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   .info-btn { | ||||||
|  |     background-color: #4CAF50; | ||||||
|  |     color: #fff; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .info-btn:hover { | ||||||
|  |     background-color: #45a049; | ||||||
|  |     transform: scale(1.05); | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .contact-btn { | ||||||
|  |     background-color: #007BFF; | ||||||
|  |     color: #fff; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .contact-btn:hover { | ||||||
|  |     background-color: #0056b3; | ||||||
|  |     transform: scale(1.05); | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   .project-body { | ||||||
|  |     margin-top: 15px; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .project-body p { | ||||||
|  |     font-size: 16px; | ||||||
|  |     color: #555; | ||||||
|  |     margin-bottom: 10px; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .project-body ul { | ||||||
|  |     list-style-type: disc; | ||||||
|  |     margin: 0; | ||||||
|  |     padding-left: 20px; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .project-body ul li { | ||||||
|  |     font-size: 14px; | ||||||
|  |     color: #666; | ||||||
|  |     line-height: 1.6; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   button { | ||||||
|  |   padding: 10px 15px; | ||||||
|  |   background-color: #007bff; | ||||||
|  |   color: white; | ||||||
|  |   border: none; | ||||||
|  |   cursor: pointer; | ||||||
|  |   border-radius: 5px; | ||||||
|  | } | ||||||
|  |   button:hover { | ||||||
|  |     background-color: #0056b3; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | </style> | ||||||
							
								
								
									
										394
									
								
								front/MyINPulse-front/src/components/canvas/CanvasItem.vue
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										394
									
								
								front/MyINPulse-front/src/components/canvas/CanvasItem.vue
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,394 @@ | |||||||
|  | <template> | ||||||
|  |   <div :class="['cell', { expanded }]" @click="handleClick"> | ||||||
|  |     <h3 class="fs-5 fw-medium">{{ title_text }}</h3> | ||||||
|  |  | ||||||
|  |     <div class="section-bloc" v-for="(desc, index) in currentDescriptions" :key="index"> | ||||||
|  |  | ||||||
|  | <!-- ADMIN --------------------------------------------------------------------------------------------> | ||||||
|  |  | ||||||
|  |       <template v-if="IS_ADMIN"> | ||||||
|  |         <div class="description"> | ||||||
|  |           <p class="m-0">{{ desc }}</p> | ||||||
|  |         </div> | ||||||
|  |       </template> | ||||||
|  |        | ||||||
|  | <!-- ENTREP -------------------------------------------------------------------------------------------> | ||||||
|  |        | ||||||
|  |       <template v-if="!IS_ADMIN"> | ||||||
|  |         <!-- Mode affichage --> | ||||||
|  |         <template v-if="!isEditing[index]"> | ||||||
|  |           <div class="description"> | ||||||
|  |             <p class="m-0">{{ desc }}</p> | ||||||
|  |           </div> | ||||||
|  |           <div class="button-container"> | ||||||
|  |             <button v-if="expanded" @click.stop="startEditing(index)" class="edit-button">Éditer</button> | ||||||
|  |           </div> | ||||||
|  |         </template> | ||||||
|  |  | ||||||
|  |         <!-- Mode édition --> | ||||||
|  |         <template v-else> | ||||||
|  |           <textarea v-model="editedDescriptions[index]" class="edit-input"></textarea> | ||||||
|  |           <div class="button-container"> | ||||||
|  |             <button @click.stop="saveEdit(index)" class="save-button">Enregistrer</button> | ||||||
|  |             <button @click.stop="cancelEdit(index)" class="cancel-button">Annuler</button> | ||||||
|  |           </div> | ||||||
|  |         </template> | ||||||
|  |       </template> | ||||||
|  |     </div> | ||||||
|  | <!----------------------------------------------------------------------------------------------------> | ||||||
|  |     <template v-if="expanded"> | ||||||
|  |       <div class="canvas-exit-hint"> | ||||||
|  |         Cliquez n'importe où pour quitter le canvas | ||||||
|  |       </div> | ||||||
|  |     </template> | ||||||
|  |   </div> | ||||||
|  | </template> | ||||||
|  |  | ||||||
|  | <script setup lang="ts"> | ||||||
|  | import { ref, defineProps, onMounted } from "vue"; | ||||||
|  | import axios from "axios"; | ||||||
|  | import { axiosInstance } from "@/services/api.ts"; | ||||||
|  |  | ||||||
|  | const IS_MOCK_MODE = true;  | ||||||
|  | const IS_ADMIN = false; | ||||||
|  |  | ||||||
|  | const props = defineProps<{ | ||||||
|  |   projectId: number; | ||||||
|  |   title: number; | ||||||
|  |   title_text: string; | ||||||
|  |   description: string; | ||||||
|  | }>(); | ||||||
|  |  | ||||||
|  | const expanded = ref(false); | ||||||
|  | const currentDescriptions = ref<string[]>([]); | ||||||
|  | currentDescriptions.value[0] = props.description; | ||||||
|  | const editedDescriptions = ref<string[]>([]); | ||||||
|  | const isEditing = ref<boolean[]>([]); | ||||||
|  |  | ||||||
|  | onMounted(() => { | ||||||
|  |   fetchData(props.projectId, props.title, "NaN", IS_MOCK_MODE); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /* FOR LOCAL DATABASE | ||||||
|  | const fetchData = async () => { | ||||||
|  |   try { | ||||||
|  |     const response = await axios.get("http://localhost:5000/data"); // Met à jour l'URL | ||||||
|  |     if (response.data.length > 0) { | ||||||
|  |       currentDescription.value = response.data[0].canva_data; | ||||||
|  |       editedDescription.value = response.data[0].canva_data; | ||||||
|  |     } else { | ||||||
|  |       console.warn("Aucune donnée reçue."); | ||||||
|  |     } | ||||||
|  |   } catch (error) { | ||||||
|  |     console.error("Erreur lors de la récupération des données :", error); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | // Fonction fetchData avec possibilité d'utiliser le mock | ||||||
|  | /* FOR FETCHING WITH AXIOS DIRECTLY | ||||||
|  | const fetchData = async (projectId: number, title: number, date: string, useMock = false) => { | ||||||
|  |   try { | ||||||
|  |     const responseData = useMock | ||||||
|  |       ? await mockFetch(projectId, title, date) | ||||||
|  |       : (await axios.get<{ txt: string }[]>( | ||||||
|  |           `http://localhost:5000/shared/projects/lcsection/${projectId}/${title}/${date}` | ||||||
|  |         )).data; | ||||||
|  |     if (responseData.length > 0) { | ||||||
|  |       currentDescriptions.value = responseData.map((item) => item.txt); | ||||||
|  |       editedDescriptions.value = [...currentDescriptions.value]; | ||||||
|  |       isEditing.value = Array(responseData.length).fill(false); | ||||||
|  |     } else { | ||||||
|  |       console.warn("Aucune donnée reçue."); | ||||||
|  |     } | ||||||
|  |   } catch (error) { | ||||||
|  |     console.error("Erreur lors de la récupération des données :", error); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | // Fonction fetchData avec possibilité d'utiliser le mock | ||||||
|  | const fetchData = async (projectId: number, title: number, date: string, useMock = false) => { | ||||||
|  |   try { | ||||||
|  |     const responseData = useMock | ||||||
|  |       ? await mockFetch(projectId, title, date) | ||||||
|  |       : (await axiosInstance.get<{ txt: string }[]>( | ||||||
|  |           `/shared/projects/lcsection/${projectId}/${title}/${date}` | ||||||
|  |         )).data; | ||||||
|  |  | ||||||
|  |     if (responseData.length > 0) { | ||||||
|  |       currentDescriptions.value = responseData.map((item) => item.txt); | ||||||
|  |       editedDescriptions.value = [...currentDescriptions.value]; | ||||||
|  |       isEditing.value = Array(responseData.length).fill(false); | ||||||
|  |     } else { | ||||||
|  |       console.warn("Aucune donnée reçue."); | ||||||
|  |     } | ||||||
|  |   } catch (error) { | ||||||
|  |     console.error("Erreur lors de la récupération des données :", error); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Fonction de simulation de l'API | ||||||
|  | const mockFetch = async (projectId: number, title: number, date: string) => { | ||||||
|  |   console.log(`Mock fetch pour projectId: ${projectId}, title: ${title}, date: ${date}`); | ||||||
|  |  | ||||||
|  |   return new Promise<{ txt: string }[]>((resolve) => { | ||||||
|  |     setTimeout(() => { | ||||||
|  |       resolve([ | ||||||
|  |       {txt: "Ceci est une description 1 pour tester le front."}, | ||||||
|  |       {txt: "Deuxième description."}, | ||||||
|  |       {txt: "Troisième description."} | ||||||
|  |       ]); | ||||||
|  |     }, 500); // Simule un délai réseau de 500ms | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Utilisation du mock dans handleClick pour tester sans serveur | ||||||
|  | const handleClick = async () => { | ||||||
|  |   if (!expanded.value) { | ||||||
|  |     await fetchData(props.projectId, props.title, "NaN", IS_MOCK_MODE); | ||||||
|  |   } else if (!isEditing.value.includes(true)) { | ||||||
|  |     // Réinitialiser les descriptions si aucune édition n'est en cours | ||||||
|  |     currentDescriptions.value = [props.description]; | ||||||
|  |     editedDescriptions.value = [props.description]; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (!isEditing.value.includes(true)) { | ||||||
|  |     expanded.value = !expanded.value; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | const startEditing = (index: number) => { | ||||||
|  |   isEditing.value[index] = true; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  | const saveEdit = async (index: number) => { | ||||||
|  |   try { | ||||||
|  |     const id = index + 1; // À adapter selon l'ID réel des données | ||||||
|  |     await axios.put(`http://localhost:5000/data/${id}`, { | ||||||
|  |       canva_data: editedDescriptions.value[index] | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     // Mettre à jour l'affichage local après la mise à jour réussie | ||||||
|  |     currentDescriptions.value[index] = editedDescriptions.value[index]; | ||||||
|  |     isEditing.value[index] = false; | ||||||
|  |   } catch (error) { | ||||||
|  |     console.error("Erreur lors de la mise à jour des données :", error); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | const saveEdit = async (index: number) => { | ||||||
|  |   if (IS_MOCK_MODE) { | ||||||
|  |     await mockSaveEdit(index); | ||||||
|  |   } else { | ||||||
|  |     try { | ||||||
|  |       const id = index + 1; | ||||||
|  |       await axios.put(`http://localhost:5000/data/${id}`, { | ||||||
|  |         canva_data: editedDescriptions.value[index] | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       // Mettre à jour l'affichage local après la mise à jour réussie | ||||||
|  |       currentDescriptions.value[index] = editedDescriptions.value[index]; | ||||||
|  |       isEditing.value[index] = false; | ||||||
|  |     } catch (error) { | ||||||
|  |       console.error("Erreur lors de la mise à jour des données :", error); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Fonction de mock pour l'enregistrement | ||||||
|  | const mockSaveEdit = async (index: number) => { | ||||||
|  |   try { | ||||||
|  |     const id = index + 1; | ||||||
|  |     console.log(`Mock save pour l'ID ${id} avec la description : ${editedDescriptions.value[index]}`); | ||||||
|  |  | ||||||
|  |     await new Promise((resolve) => setTimeout(resolve, 500)); // Simulation de délai réseau | ||||||
|  |  | ||||||
|  |     // Mettre à jour l'affichage local après la mise à jour réussie | ||||||
|  |     currentDescriptions.value[index] = editedDescriptions.value[index]; | ||||||
|  |     isEditing.value[index] = false; | ||||||
|  |   } catch (error) { | ||||||
|  |     console.error("Erreur lors de la mise à jour des données mockées :", error); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const cancelEdit = (index: number) => { | ||||||
|  |   editedDescriptions.value[index] = currentDescriptions.value[index]; | ||||||
|  |   isEditing.value[index] = false; | ||||||
|  | }; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <style scoped> | ||||||
|  | @import "@/components/canvas/style-project.css"; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .cell { | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  |   align-items: center; | ||||||
|  |   justify-content: center; | ||||||
|  |   text-align: center; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  |   cursor: pointer; | ||||||
|  |   box-shadow: 0 4px 5px rgba(0, 0, 0, 0.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .expanded-content { | ||||||
|  |   justify-content: flex-start !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .cell:not(.expanded):hover { | ||||||
|  |   transform: scale(1.05); | ||||||
|  |   box-shadow: 0 8px 9px rgba(0, 0, 0, 0.2); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .cell h3 { | ||||||
|  |   font-size: 15px; | ||||||
|  |   font-weight: 500; | ||||||
|  |   font-family: 'Arial', sans-serif; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .p { | ||||||
|  |   font-size: 10px; | ||||||
|  |   color: #666; | ||||||
|  |   font-family: 'Arial', sans-serif; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .expanded { | ||||||
|  |   padding-top: 10%; | ||||||
|  |   position: fixed; | ||||||
|  |   top: 0; | ||||||
|  |   left: 0; | ||||||
|  |   width: 100%; | ||||||
|  |   height: 100%; | ||||||
|  |   background: white; | ||||||
|  |   z-index: 10; | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   justify-content: flex-start; | ||||||
|  |   box-shadow: 0 0 10px rgba(0, 0, 0, 0.2); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .description { | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center;  | ||||||
|  |   justify-content: center; | ||||||
|  |   text-align: center; | ||||||
|  |   width: 100%; | ||||||
|  |   height: 100%; | ||||||
|  |   font-size: 16px; | ||||||
|  |   margin-top: 10px; | ||||||
|  |   margin-left: 2%; | ||||||
|  |   margin-right: 4%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .description + .p { | ||||||
|  |   align-items: center; | ||||||
|  |   justify-content: center; | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .edit-input { | ||||||
|  |   width: 100%; | ||||||
|  |   height: 100%; | ||||||
|  |   padding: 10px; | ||||||
|  |   border: 1px solid #ccc; | ||||||
|  |   border-radius: 5px; | ||||||
|  |   margin-top: 10px; | ||||||
|  |   box-sizing: border-box; | ||||||
|  |   margin-left: 2%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .button-container { | ||||||
|  |   display: block; | ||||||
|  |   margin-top: 20px; | ||||||
|  |   justify-content: center; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 10px; | ||||||
|  |   padding-right: 1%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .section-bloc ,.editing-section-bloc { | ||||||
|  |   width: 100%; | ||||||
|  |   justify-content: center; | ||||||
|  |   align-items: center; | ||||||
|  |   display: flex; | ||||||
|  |   margin-right: 10%; | ||||||
|  |   margin: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .edit-button { | ||||||
|  |   width: 100px; | ||||||
|  |   height: 40px; | ||||||
|  |   border: none; | ||||||
|  |   border-radius: 5px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: background 0.3s ease; | ||||||
|  |   font-size: 12px; | ||||||
|  |   margin-right: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .save-button, .cancel-button { | ||||||
|  |   width: 100px; | ||||||
|  |   height: 40px; | ||||||
|  |   border: none; | ||||||
|  |   border-radius: 5px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: background 0.3s ease; | ||||||
|  |   font-size: 12px; | ||||||
|  |   margin-bottom: 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .edit-button { | ||||||
|  |   background-color: #007bff; | ||||||
|  |   color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .save-button { | ||||||
|  |   background-color: #28a745; | ||||||
|  |   color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .cancel-button { | ||||||
|  |   background-color: #dc3545; | ||||||
|  |   color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .edit-button:hover { | ||||||
|  |   background-color: #0056b3; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .save-button:hover { | ||||||
|  |   background-color: #218838; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .cancel-button:hover { | ||||||
|  |   background-color: #c82333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .canvas-exit-hint { | ||||||
|  |   font-size: 0.75rem; | ||||||
|  |   color: #666; | ||||||
|  |   position: fixed; | ||||||
|  |   bottom: 10px; | ||||||
|  |   left: 0; | ||||||
|  |   width: 100%; | ||||||
|  |   text-align: center; | ||||||
|  |   z-index: 1000; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | </style> | ||||||
							
								
								
									
										200
									
								
								front/MyINPulse-front/src/components/canvas/HeaderCanvas.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										200
									
								
								front/MyINPulse-front/src/components/canvas/HeaderCanvas.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,200 @@ | |||||||
|  | <template> | ||||||
|  |   <header class="header"> | ||||||
|  |     <img src="../icons/logo inpulse.png" alt="INPulse Logo" class="logo" /> | ||||||
|  |  | ||||||
|  |     <div class="header-actions"> | ||||||
|  |       <div class="dropdown-wrapper"> | ||||||
|  |         <button class="contact-button" @click="toggleDropdown">Contact</button> | ||||||
|  |         <div class="contact-dropdown" :class="{ 'dropdown-visible': isDropdownOpen }"> | ||||||
|  |           <button @click="contactAll">Contacter tous</button> | ||||||
|  |           <button | ||||||
|  |             v-for="(email, index) in entrepreneurEmails" | ||||||
|  |             :key="index" | ||||||
|  |             @click="copyToClipboard(email)" | ||||||
|  |           > | ||||||
|  |             {{ email }} | ||||||
|  |           </button> | ||||||
|  |  | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |       <RouterLink to="/" class="return-button">Retour</RouterLink> | ||||||
|  |     </div> | ||||||
|  |   </header> | ||||||
|  | </template> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | <script setup lang="ts"> | ||||||
|  | import { ref, onMounted } from "vue"; | ||||||
|  | import axios from "axios"; | ||||||
|  |  | ||||||
|  | const IS_MOCK_MODE = true; | ||||||
|  |  | ||||||
|  | const props = defineProps<{ | ||||||
|  |   projectId: number; | ||||||
|  | }>(); | ||||||
|  |  | ||||||
|  | type Entrepreneur = { | ||||||
|  |   idUser: number; | ||||||
|  |   userSurname: string; | ||||||
|  |   userName: string; | ||||||
|  |   primaryMail: string; | ||||||
|  |   secondaryMail: string; | ||||||
|  |   phoneNumber: string; | ||||||
|  |   school: string; | ||||||
|  |   course: string; | ||||||
|  |   sneeStatus: boolean; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const isDropdownOpen = ref(false); | ||||||
|  | const entrepreneurEmails = ref<string[]>([]); | ||||||
|  |  | ||||||
|  | const toggleDropdown = () => { | ||||||
|  |   isDropdownOpen.value = !isDropdownOpen.value; | ||||||
|  |   console.log("Dropdown toggled:", isDropdownOpen.value); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const fetchEntrepreneurs = async (projectId: number, useMock = IS_MOCK_MODE) => { | ||||||
|  |   try { | ||||||
|  |     const responseData: Entrepreneur[] = useMock | ||||||
|  |       ? await mockFetchEntrepreneurs(projectId) | ||||||
|  |       : (await axios.get(`http://localhost:5000/shared/projects/entrepreneurs/${projectId}`)).data; | ||||||
|  |  | ||||||
|  |     if (responseData.length > 0) { | ||||||
|  |       entrepreneurEmails.value = responseData.map((item: Entrepreneur) => item.primaryMail); | ||||||
|  |     } else { | ||||||
|  |       console.warn("Aucun entrepreneur trouvé."); | ||||||
|  |     } | ||||||
|  |   } catch (error) { | ||||||
|  |     console.error("Erreur lors de la récupération des entrepreneurs :", error); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Fonction de simulation de l'API | ||||||
|  | const mockFetchEntrepreneurs = async (projectId :number) => { | ||||||
|  |   console.log(`Mock fetch pour projectId: ${projectId}`); | ||||||
|  |  | ||||||
|  |   return new Promise((resolve) => { | ||||||
|  |     setTimeout(() => { | ||||||
|  |       resolve([ | ||||||
|  |         { | ||||||
|  |           idUser: 1, | ||||||
|  |           userSurname: "Doe", | ||||||
|  |           userName: "John", | ||||||
|  |           primaryMail: "john.doe@example.com", | ||||||
|  |           secondaryMail: "johndoe@backup.com", | ||||||
|  |           phoneNumber: "612345678", | ||||||
|  |           school: "ENSEIRB", | ||||||
|  |           course: "Info", | ||||||
|  |           sneeStatus: false | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           idUser: 2, | ||||||
|  |           userSurname: "Smith", | ||||||
|  |           userName: "Jane", | ||||||
|  |           primaryMail: "jane.smith@example.com", | ||||||
|  |           secondaryMail: "janesmith@backup.com", | ||||||
|  |           phoneNumber: "698765432", | ||||||
|  |           school: "ENSEIRB", | ||||||
|  |           course: "Info", | ||||||
|  |           sneeStatus: true | ||||||
|  |         } | ||||||
|  |       ]); | ||||||
|  |     }, 500); | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const contactAll = () => { | ||||||
|  |   alert("Contacter tous les entrepreneurs : " + entrepreneurEmails.value.join(", ")); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const copyToClipboard = (email: string) => { | ||||||
|  |   navigator.clipboard.writeText(email).then(() => { | ||||||
|  |     alert(`Adresse copiée : ${email}`); | ||||||
|  |   }).catch(err => { | ||||||
|  |     console.error("Erreur lors de la copie :", err); | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | onMounted(() => fetchEntrepreneurs(props.projectId, IS_MOCK_MODE)); | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <style scoped> | ||||||
|  | @import "@/components/canvas/style-project.css"; | ||||||
|  |  | ||||||
|  | .header { | ||||||
|  |   display: flex; | ||||||
|  |   justify-content: space-between; | ||||||
|  |   align-items: center; | ||||||
|  |   padding: 15px 30px; | ||||||
|  |   background-color: #f9f9f9; | ||||||
|  |   box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .logo { | ||||||
|  |   height: 50px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .header-actions { | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 20px; | ||||||
|  |   position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .contact-button, | ||||||
|  | .return-button { | ||||||
|  |   background-color: #009CDE; | ||||||
|  |   color: white; | ||||||
|  |   border: none; | ||||||
|  |   padding: 10px 15px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   font-size: 14px; | ||||||
|  |   border-radius: 5px; | ||||||
|  |   text-decoration: none; | ||||||
|  |   transition: background-color 0.2s ease; | ||||||
|  |   font-family: Arial, sans-serif; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .return-button:hover, | ||||||
|  | .contact-button:hover { | ||||||
|  |   background-color: #007bad; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .contact-dropdown { | ||||||
|  |   position: absolute; | ||||||
|  |   top: 100%; | ||||||
|  |   left: 0; | ||||||
|  |   background-color: #000; | ||||||
|  |   color: white; | ||||||
|  |   box-shadow: 0px 4px 8px rgba(255, 255, 255, 0.2); | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 10px; | ||||||
|  |   margin-top: 5px; | ||||||
|  |   z-index: 1000; | ||||||
|  |   min-width: 200px; | ||||||
|  |   display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .contact-dropdown button { | ||||||
|  |   display: block; | ||||||
|  |   width: 100%; | ||||||
|  |   padding: 5px; | ||||||
|  |   text-align: left; | ||||||
|  |   border: none; | ||||||
|  |   background: none; | ||||||
|  |   cursor: pointer; | ||||||
|  |   color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .contact-dropdown button:hover { | ||||||
|  |   background-color: #009CDE; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .contact-dropdown.dropdown-visible { | ||||||
|  |   display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | </style> | ||||||
							
								
								
									
										77
									
								
								front/MyINPulse-front/src/components/canvas/LeanCanvas.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								front/MyINPulse-front/src/components/canvas/LeanCanvas.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | |||||||
|  | <template> | ||||||
|  |   <div class="canvas container-fluid">   | ||||||
|  |     <CanvasItem | ||||||
|  |       v-for="(item, index) in items" | ||||||
|  |       :key="index" | ||||||
|  |       :title="item.title" | ||||||
|  |       :title_text="item.title_text" | ||||||
|  |       :description="item.description" | ||||||
|  |       :projectId="item.projectId" | ||||||
|  |       :class="['canvas-item', item.class, 'card', 'shadow', 'p-3']" | ||||||
|  |     /> | ||||||
|  |   </div> | ||||||
|  | </template> | ||||||
|  |  | ||||||
|  | <script setup lang="ts"> | ||||||
|  | import { ref, onMounted } from "vue"; | ||||||
|  | import CanvasItem from "@/components/canvas/CanvasItem.vue"; | ||||||
|  |  | ||||||
|  | const items = ref([ | ||||||
|  |   { projectId: 1, title: 1, title_text: "1. Problème", description: "3 problèmes essentiels à résoudre pour le client", class: "Probleme" }, | ||||||
|  |   { projectId: 1, title: 2, title_text: "2. Segments", description: "Les segments de clientèle visés", class: "Segments" }, | ||||||
|  |   { projectId: 1, title: 3, title_text: "3. Valeur", description: "La proposition de valeur", class: "Valeur" }, | ||||||
|  |   { projectId: 1, title: 4, title_text: "4. Solution", description: "Les solutions proposées", class: "Solution" }, | ||||||
|  |   { projectId: 1, title: 5, title_text: "5. Avantage", description: "Les avantages concurrentiels", class: "Avantage" }, | ||||||
|  |   { projectId: 1, title: 6, title_text: "6. Canaux", description: "Les canaux de distribution", class: "Canaux" }, | ||||||
|  |   { projectId: 1, title: 7, title_text: "7. Indicateurs", description: "Les indicateurs clés de performance", class: "Indicateurs" }, | ||||||
|  |   { projectId: 1, title: 8, title_text: "8. Coûts", description: "Les coûts associés", class: "Couts" }, | ||||||
|  |   { projectId: 1, title: 9, title_text: "9. Revenus", description: "Les sources de revenus", class: "Revenus" } | ||||||
|  | ]); | ||||||
|  |  | ||||||
|  | onMounted(() => { | ||||||
|  |   const bootstrapCss = document.createElement('link') | ||||||
|  |   bootstrapCss.rel = 'stylesheet' | ||||||
|  |   bootstrapCss.href = 'https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/css/bootstrap.min.css' | ||||||
|  |   bootstrapCss.integrity = 'sha384-QWTKZyjpPEjISv5WaRU9OFeRpok6YctnYmDr5pNlyT2bRjXh0JMhjY6hW+Fpc+NC' | ||||||
|  |   bootstrapCss.crossOrigin = 'anonymous' | ||||||
|  |   document.head.appendChild(bootstrapCss) | ||||||
|  |  | ||||||
|  |   const bootstrapJs = document.createElement('script') | ||||||
|  |   bootstrapJs.src = 'https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/js/bootstrap.bundle.min.js' | ||||||
|  |   bootstrapJs.integrity = 'sha384-mQ93S0EhrF4Z1nM+fTflmYf0DyzsY5j7F5H3WlClDD6H3WUJh6kxBkF3GDW8n1j6' | ||||||
|  |   bootstrapJs.crossOrigin = 'anonymous' | ||||||
|  |   document.body.appendChild(bootstrapJs) | ||||||
|  | }) | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <style scoped> | ||||||
|  | @import "@/components/canvas/style-project.css"; | ||||||
|  |  | ||||||
|  | .canvas { | ||||||
|  |   display: grid; | ||||||
|  |   grid-template-columns: repeat(10, 1fr); | ||||||
|  |   grid-template-rows: repeat(6, 1fr); | ||||||
|  |   gap: 12px; | ||||||
|  |   padding: 30px; | ||||||
|  |   /*background-color: #f8f9fa;*/ | ||||||
|  |   position: relative; | ||||||
|  |   height: 90vh; | ||||||
|  |   overflow: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .Probleme { grid-column: 1 / 3; grid-row: 1 / 5; } | ||||||
|  | .Segments { grid-column: 9 / 11; grid-row: 1 / 5; } | ||||||
|  | .Valeur { grid-column: 5 / 7; grid-row: 1 / 5; } | ||||||
|  | .Solution { grid-column: 3 / 5; grid-row: 1 / 3; } | ||||||
|  | .Avantage { grid-column: 7 / 9; grid-row: 1 / 3; } | ||||||
|  | .Canaux { grid-column: 7 / 9; grid-row: 3 / 5; } | ||||||
|  | .Indicateurs { grid-column: 3 / 5; grid-row: 3 / 5; } | ||||||
|  | .Couts { grid-column: 1 / 6; grid-row: 5 / 7; } | ||||||
|  | .Revenus { grid-column: 6 / 11; grid-row: 5 / 7; } | ||||||
|  |  | ||||||
|  | .canvas-item { | ||||||
|  |   /*background-color: white;*/ | ||||||
|  |   border: 1px solid #dee2e6; | ||||||
|  |   border-radius: 0.5rem; | ||||||
|  | } | ||||||
|  | </style> | ||||||
							
								
								
									
										156
									
								
								front/MyINPulse-front/src/components/canvas/style-project.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								front/MyINPulse-front/src/components/canvas/style-project.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | |||||||
|  | body { | ||||||
|  |     font-family: Arial, sans-serif; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 10px; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .row { | ||||||
|  |     display: flex; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .cell { | ||||||
|  |     flex: 1; | ||||||
|  |     border: 1px solid #ddd; | ||||||
|  |     padding: 10px; | ||||||
|  |     text-align: center; | ||||||
|  |     background-color: #f1f1f1; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .produit { | ||||||
|  |     background-color: #f9e4e4; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .marche { | ||||||
|  |     background-color: #e4f1f9; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .valeur { | ||||||
|  |     background-color: #f9f4e4; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   h3 { | ||||||
|  |     margin: 0; | ||||||
|  |     font-size: 18px; | ||||||
|  |     color: #333; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   p { | ||||||
|  |     margin: 5px 0 0; | ||||||
|  |     font-size: 14px; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |  | ||||||
|  |   body { | ||||||
|  |     font-family: Arial, sans-serif; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  |     background-color: #f9f9f9; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   h1 img { | ||||||
|  |     height: 80px; | ||||||
|  |     margin: 40px; | ||||||
|  |     text-align: center; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .row { | ||||||
|  |     display: flex; | ||||||
|  |     margin-bottom: 10px; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   #ade { | ||||||
|  |     max-width: 1200px; | ||||||
|  |     margin: 20px auto; | ||||||
|  |     padding: 20px; | ||||||
|  |     text-align: center; | ||||||
|  |     background-color: #e8f5e9; | ||||||
|  |     border: 2px solid #4caf50; | ||||||
|  |     border-radius: 10px; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   #ade h3 { | ||||||
|  |     color: #2e7d32; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   #ade p { | ||||||
|  |     margin: 10px 0; | ||||||
|  |     font-size: 16px; | ||||||
|  |     color: #333; | ||||||
|  |   } | ||||||
|  |   header { | ||||||
|  |     display: flex; | ||||||
|  |     justify-content: space-between; | ||||||
|  |     align-items: center; | ||||||
|  |     padding: 10px 20px; | ||||||
|  |     background-color: #fff; | ||||||
|  |     border-bottom: 2px solid #ddd; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   header img { | ||||||
|  |     height: 60px; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   header .contact-menu { | ||||||
|  |     position: relative; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .contact-button, .return { | ||||||
|  |     padding: 10px 15px; | ||||||
|  |     border: none; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     background-color: #2196f3; | ||||||
|  |     color: #fff; | ||||||
|  |     cursor: pointer; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  .contact-button:hover, .return:hover { | ||||||
|  |     background-color: #1976d2; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /* Dropdown styling */ | ||||||
|  |   .contact-dropdown { | ||||||
|  |     position: absolute; | ||||||
|  |     right: 0; | ||||||
|  |     top: 50px; | ||||||
|  |     display: none; | ||||||
|  |     flex-direction: column; | ||||||
|  |     gap: 10px; | ||||||
|  |     padding: 15px; | ||||||
|  |     background-color: #fff; | ||||||
|  |     border: 1px solid #ddd; | ||||||
|  |     border-radius: 8px; | ||||||
|  |     box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1); | ||||||
|  |     z-index: 10; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .contact-dropdown button { | ||||||
|  |     padding: 8px 12px; | ||||||
|  |     border: none; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     background-color: #4caf50; | ||||||
|  |     color: #fff; | ||||||
|  |     cursor: pointer; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .contact-dropdown button:hover { | ||||||
|  |     background-color: #388e3c; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .return { | ||||||
|  |     background-color: #f44336; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .return:hover { | ||||||
|  |     background-color: #d32f2f; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .header-buttons { | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |     gap: 15px; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   a{ | ||||||
|  |     color: white; | ||||||
|  |   } | ||||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 5.1 KiB After Width: | Height: | Size: 39 KiB | 
| @@ -26,6 +26,52 @@ keycloakService.CallInit(() => { | |||||||
|         console.error(e); |         console.error(e); | ||||||
|         createApp(App).mount("#app"); |         createApp(App).mount("#app"); | ||||||
|     } |     } | ||||||
| }); |  | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | // this shit made by me so i can run the canva vue app | ||||||
|  | //createApp(App).use(router).mount('#app'); | ||||||
|  |  | ||||||
|  | // TODO: fix the comment | ||||||
|  | /* | ||||||
|  | function tokenInterceptor () { | ||||||
|  |     axios.interceptors.request.use(config => { | ||||||
|  |         const keycloak = useKeycloak() | ||||||
|  |         if (keycloak.authenticated) { | ||||||
|  |             // Note that this is a simple example. | ||||||
|  |             // you should be careful not to leak tokens to third parties. | ||||||
|  |             // in this example the token is added to all usage of axios. | ||||||
|  |             config.headers.Authorization = `Bearer ${keycloak.token}` | ||||||
|  |         } | ||||||
|  |         return config | ||||||
|  |     }, error => { | ||||||
|  |         console.error("tokenInterceptor: Rejected") | ||||||
|  |         return Promise.reject(error) | ||||||
|  |     }) | ||||||
|  | } | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | /* | ||||||
|  | app.use(VueKeyCloak,{ | ||||||
|  |     onReady: (keycloak) => { | ||||||
|  |         console.log("Ready !") | ||||||
|  |         tokenInterceptor() | ||||||
|  |     }, | ||||||
|  |     init: { | ||||||
|  |         onLoad: 'login-required', | ||||||
|  |         checkLoginIframe: false, | ||||||
|  |  | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     config: { | ||||||
|  |         realm: 'test', | ||||||
|  |         url: 'http://localhost:7080', | ||||||
|  |         clientId: 'myinpulse' | ||||||
|  |     } | ||||||
|  | }  ); | ||||||
|  | */ | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| export { store }; | export { store }; | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								front/MyINPulse-front/src/plugins/authStore.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								front/MyINPulse-front/src/plugins/authStore.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | // file: src/plugins/authStore.js | ||||||
|  |  | ||||||
|  | import { useAuthStore } from "@/stores/authStore.ts"; | ||||||
|  | import keycloakService from '@/services/keycloak'; | ||||||
|  | // Setup auth store as a plugin so it can be accessed globally in our FE | ||||||
|  | const authStorePlugin = { | ||||||
|  |     install(app: any, option: any) { | ||||||
|  |         const store = useAuthStore(option.pinia); | ||||||
|  |         app.config.globalProperties.$store = store; | ||||||
|  |         keycloakService.CallInitStore(store); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export default authStorePlugin; | ||||||
| @@ -4,14 +4,37 @@ const router = createRouter({ | |||||||
|   history: createWebHistory(import.meta.env.BASE_URL), |   history: createWebHistory(import.meta.env.BASE_URL), | ||||||
|   routes: [ |   routes: [ | ||||||
|     { |     { | ||||||
|             path: "/test", |       path: '/test', | ||||||
|             name: "test", |       name: 'test', | ||||||
|       // route level code-splitting |       // route level code-splitting | ||||||
|       // this generates a separate chunk (About.[hash].js) for this route |       // this generates a separate chunk (About.[hash].js) for this route | ||||||
|       // which is lazy-loaded when the route is visited. |       // which is lazy-loaded when the route is visited. | ||||||
|             component: () => import("../views/testComponent.vue"), |       component: () => import('../views/testComponent.vue'), | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       path: '/login', | ||||||
|  |       name: 'login', | ||||||
|  |       component: () => import('../components/LoginComponent.vue'), | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       path: '/', | ||||||
|  |       name: 'Admin-main', | ||||||
|  |       component: () => import('../views/AdminMain.vue'), | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  | // route pour les canvas (made by adnane), in fact the two vue apps are separated for now   | ||||||
|  |     { | ||||||
|  |       path: '/canvas', | ||||||
|  |       name: 'canvas', | ||||||
|  |       component: () => import('../views/CanvasView.vue'), | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     { | ||||||
|  |       path: '/signup', | ||||||
|  |       name: 'signup', | ||||||
|  |       component: () => import('../views/EntrepSignUp.vue'), | ||||||
|     }, |     }, | ||||||
|   ], |   ], | ||||||
| }); | }) | ||||||
|  |  | ||||||
| export default router; | export default router; | ||||||
|   | |||||||
| @@ -65,4 +65,28 @@ function callApi( | |||||||
|         ); |         ); | ||||||
| } | } | ||||||
|  |  | ||||||
| export { callApi }; | function postApi( | ||||||
|  |     endpoint: string, | ||||||
|  |     data: any, | ||||||
|  |     onSuccessHandler?: (response: AxiosResponse) => void, | ||||||
|  |     onErrorHandler?: (error: AxiosError) => void | ||||||
|  | ): void { | ||||||
|  |     axiosInstance | ||||||
|  |         .post(endpoint, data) | ||||||
|  |         .then(onSuccessHandler ?? defaultApiSuccessHandler) | ||||||
|  |         .catch(onErrorHandler ?? defaultApiErrorHandler); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function deleteApi( | ||||||
|  |     endpoint: string, | ||||||
|  |     onSuccessHandler?: (response: AxiosResponse) => void, | ||||||
|  |     onErrorHandler?: (error: AxiosError) => void | ||||||
|  | ): void { | ||||||
|  |     axiosInstance | ||||||
|  |         .delete(endpoint) | ||||||
|  |         .then(onSuccessHandler ?? defaultApiSuccessHandler) | ||||||
|  |         .catch(onErrorHandler ?? defaultApiErrorHandler); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | export { axiosInstance, callApi, postApi, deleteApi }; | ||||||
|   | |||||||
| @@ -54,7 +54,7 @@ const useAuthStore = defineStore("storeAuth", { | |||||||
|         async logout() { |         async logout() { | ||||||
|             try { |             try { | ||||||
|                 await keycloakService.CallLogout( |                 await keycloakService.CallLogout( | ||||||
|                     import.meta.env.VITE_APP_URL + "/test" |                     import.meta.env.VITE_APP_URL + "/login" //redirect to login page instead of test... | ||||||
|                 ); |                 ); | ||||||
|                 await this.clearUserData(); |                 await this.clearUserData(); | ||||||
|             } catch (error) { |             } catch (error) { | ||||||
|   | |||||||
							
								
								
									
										139
									
								
								front/MyINPulse-front/src/views/AdminMain.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										139
									
								
								front/MyINPulse-front/src/views/AdminMain.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,139 @@ | |||||||
|  | <template> | ||||||
|  |     <Header /> | ||||||
|  |     <error-wrapper></error-wrapper> | ||||||
|  |     <div id="container">   | ||||||
|  |       <div id="main"> | ||||||
|  |         <h3> Projet en cours </h3> | ||||||
|  |           <ProjectComp  | ||||||
|  |             v-for="(project, index) in projects"  | ||||||
|  |             :key="index" | ||||||
|  |             :projectName="project.name" | ||||||
|  |             :listName="project.members" | ||||||
|  |             :projectLink="project.link" | ||||||
|  |           /> | ||||||
|  |  | ||||||
|  |         <div id ="main"> | ||||||
|  |           <h3> Projet en attente </h3> | ||||||
|  |            | ||||||
|  |           <PendingProjectComponent | ||||||
|  |             v-for="( project, index) in pendingProjects" | ||||||
|  |             :key="index" | ||||||
|  |             :projectName="project.name" | ||||||
|  |             :creationDate="project.creationDate" | ||||||
|  |           /> | ||||||
|  |         </div>  | ||||||
|  |  | ||||||
|  |          | ||||||
|  |       </div>   | ||||||
|  |        | ||||||
|  |       <Agenda :projectRDV="rendezVous" /> | ||||||
|  |     </div> | ||||||
|  |     <AddProjectForm/> | ||||||
|  | </template> | ||||||
|  |  | ||||||
|  | <script setup lang="ts"> | ||||||
|  | import { ref, onMounted } from "vue"; | ||||||
|  | import { callApi } from "@/services/api"; | ||||||
|  |  | ||||||
|  | import Header from "../components/HeaderComponent.vue"; | ||||||
|  | import Agenda from "../components/Agenda.vue"; | ||||||
|  | import ProjectComp from "../components/ProjectComponent.vue"; | ||||||
|  | import PendingProjectComponent from "@/components/PendingProjectComponent.vue"; | ||||||
|  | import AddProjectForm from "@/components/AddProjectForm.vue"; | ||||||
|  |  | ||||||
|  | const PORT = "8081"; | ||||||
|  | const URI = `http://localhost:${PORT}`; | ||||||
|  |  | ||||||
|  | //const projects = ref<{ name: string; link: string; members: string[] }[]>([]); | ||||||
|  |  | ||||||
|  | /* const fetchProjects = () => { | ||||||
|  |   callApi( | ||||||
|  |     `${URI}/admin/projects`, | ||||||
|  |     async (response) => { | ||||||
|  |       console.log(response); | ||||||
|  |       const projectList = response.data; | ||||||
|  |  | ||||||
|  |       const projectPromises = projectList.map((project: any) => { | ||||||
|  |         return new Promise(async (resolve) => { | ||||||
|  |           callApi( | ||||||
|  |             `${URI}/shared/projects/entrepreneurs/${project.idProject}`, | ||||||
|  |             (memberResponse) => { | ||||||
|  |               const members = memberResponse.data.map((m: any) => m.userName); | ||||||
|  |               resolve({ | ||||||
|  |                 name: project.projectName, | ||||||
|  |                 link: `/project/${project.idProject}`, | ||||||
|  |                 members, | ||||||
|  |               }); | ||||||
|  |             }, | ||||||
|  |             () => { | ||||||
|  |               // Error fetching members, still resolve with empty members | ||||||
|  |               resolve({ | ||||||
|  |                 name: project.projectName, | ||||||
|  |                 link: `/project/${project.idProject}`, | ||||||
|  |                 members: [], | ||||||
|  |               }); | ||||||
|  |             } | ||||||
|  |           ); | ||||||
|  |         }); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       projects.value = await Promise.all(projectPromises); | ||||||
|  |     }, | ||||||
|  |     (error) => { | ||||||
|  |       console.error("Error fetching projects:", error); | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | onMounted(fetchProjects); | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | const projects = ref([ | ||||||
|  |   { | ||||||
|  |     name: "Projet Alpha", | ||||||
|  |     link: "/canvas", // to test | ||||||
|  |     members: ["Alice", "Bob", "Charlie"], | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     name: "Projet Beta", | ||||||
|  |     link: "./canvas", // to test | ||||||
|  |     members: ["David", "Eve", "Frank"], | ||||||
|  |   }, | ||||||
|  | ]); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | const pendingProjects = ref ([ | ||||||
|  |   { name: "l'eau", creationDate: "26-02-2024" }, | ||||||
|  |   { name: "l'air", creationDate: "09-03-2023" }, | ||||||
|  | ]) | ||||||
|  |  | ||||||
|  | const rendezVous = ref([ | ||||||
|  |   { projectName: "Projet Alpha", date: "2025-03-10", lieu: "P106" }, | ||||||
|  |   { projectName: "Projet Beta", date: "2025-04-15", lieu: "Td10" }, | ||||||
|  | ]); | ||||||
|  |  | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <style scoped> | ||||||
|  |  | ||||||
|  | #container { | ||||||
|  |     margin: 0; | ||||||
|  |     display: grid; | ||||||
|  |     grid-template-columns: 3fr 1fr; /* Main body takes 3/4, agenda 1/4 */ | ||||||
|  |     height: 100vh; /* Full viewport height */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | button { | ||||||
|  |   padding: 10px 15px; | ||||||
|  |   background-color: #007bff; | ||||||
|  |   color: white; | ||||||
|  |   border: none; | ||||||
|  |   cursor: pointer; | ||||||
|  |   border-radius: 5px; | ||||||
|  | } | ||||||
|  | button:hover { | ||||||
|  |   background-color: #0056b3; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | </style> | ||||||
							
								
								
									
										114
									
								
								front/MyINPulse-front/src/views/CanvasView.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								front/MyINPulse-front/src/views/CanvasView.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,114 @@ | |||||||
|  | <template> | ||||||
|  |     <div> | ||||||
|  |       <header> | ||||||
|  |         <HeaderCanvas :projectId="1" /> | ||||||
|  |       </header> | ||||||
|  |     </div> | ||||||
|  |     <div> | ||||||
|  |       <h1 class="page-title">PAGE CANVAS</h1> | ||||||
|  |  | ||||||
|  |       <p class="canvas-help-text"> | ||||||
|  |         Cliquez sur un champ du tableau pour afficher son contenu en détail ci-dessous. | ||||||
|  |       </p> | ||||||
|  |       <LeanCanvas /> | ||||||
|  |  | ||||||
|  |       <div class="info-box"> | ||||||
|  |         <p> | ||||||
|  |           Responsable : <strong>{{ admin.userName }} {{ admin.userSurname }}</strong><br /> | ||||||
|  |           Contact : <a href="mailto:{{ admin.primaryMail }}">{{ admin.primaryMail }}</a> | | ||||||
|  |           <a href="tel:{{ admin.phoneNumber }}">{{ admin.phoneNumber }}</a> | ||||||
|  |         </p> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  | </template> | ||||||
|  |    | ||||||
|  | <script setup lang="ts"> | ||||||
|  | // @ts-ignore | ||||||
|  | import HeaderCanvas from "../components/canvas/HeaderCanvas.vue"; | ||||||
|  | import LeanCanvas from '../components/canvas/LeanCanvas.vue'; | ||||||
|  | import { ref, onMounted } from "vue"; | ||||||
|  | import { axiosInstance } from "@/services/api.ts"; | ||||||
|  |  | ||||||
|  | const IS_MOCK_MODE = true; | ||||||
|  |  | ||||||
|  | // Variables pour les informations de l'administrateur | ||||||
|  | const admin = ref({ | ||||||
|  |   idUser: 0, | ||||||
|  |   userSurname: "", | ||||||
|  |   userName: "", | ||||||
|  |   primaryMail: "", | ||||||
|  |   secondaryMail: "", | ||||||
|  |   phoneNumber: "" | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | const mockAdminData = { | ||||||
|  |   idUser: 1, | ||||||
|  |   userSurname: "ALAMI", | ||||||
|  |   userName: "Adnane", | ||||||
|  |   primaryMail: "mock.admin@example.com", | ||||||
|  |   secondaryMail: "admin.backup@example.com", | ||||||
|  |   phoneNumber: "0600000000" | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Fonction pour récupérer les données de l'administrateur | ||||||
|  | const fetchAdminData = async (projectId: number, useMock = IS_MOCK_MODE) => { | ||||||
|  |   try { | ||||||
|  |     if (useMock) { | ||||||
|  |       console.log("Utilisation des données mockées pour l'administrateur"); | ||||||
|  |       admin.value = mockAdminData; | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const response = await axiosInstance.get(`/shared/projects/admin/${projectId}`); | ||||||
|  |     admin.value = response.data; | ||||||
|  |   } catch (error) { | ||||||
|  |     console.error("Erreur lors de la récupération des données de l'administrateur :", error); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Appeler la fonction fetch au montage du composant | ||||||
|  | onMounted(() => { | ||||||
|  |   const projectId = 1; | ||||||
|  |   fetchAdminData(projectId); | ||||||
|  | }); | ||||||
|  | </script> | ||||||
|  |    | ||||||
|  | <style scoped> | ||||||
|  | .page-title { | ||||||
|  |   text-align: center; | ||||||
|  |   font-size: 2.5rem; | ||||||
|  |   margin-top: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .canvas-help-text { | ||||||
|  |   text-align: center; | ||||||
|  |   font-size: 0.7rem; | ||||||
|  |   color: #666; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .info-box { | ||||||
|  |   background-color: #f9f9f9; | ||||||
|  |   padding: 15px;              | ||||||
|  |   border-radius: 8px;         | ||||||
|  |   box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);  | ||||||
|  |   font-family: Arial, sans-serif;  | ||||||
|  |   width: 30%;                 | ||||||
|  |   max-width: 600px;           | ||||||
|  |   margin: 20px auto;          | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .info-box p { | ||||||
|  |   font-size: 16px; | ||||||
|  |   line-height: 1.5;   | ||||||
|  |   color: #333;            | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .info-box a { | ||||||
|  |   color: #007bff;         | ||||||
|  |   text-decoration: none;  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .info-box a:hover { | ||||||
|  |   text-decoration: underline; | ||||||
|  | } | ||||||
|  | </style> | ||||||
							
								
								
									
										186
									
								
								front/MyINPulse-front/src/views/EntrepSignUp.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										186
									
								
								front/MyINPulse-front/src/views/EntrepSignUp.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,186 @@ | |||||||
|  | <template> | ||||||
|  |     <form class="add-project-form" @submit.prevent="submitForm"> | ||||||
|  |       <h2>Ajouter un projet</h2> | ||||||
|  |    | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="name">Nom du projet</label> | ||||||
|  |         <input | ||||||
|  |           id="name" | ||||||
|  |           v-model="form.name" | ||||||
|  |           type="text" | ||||||
|  |           required | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |    | ||||||
|  |       <h3>Entrepreneur</h3> | ||||||
|  |    | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="founderName">Nom</label> | ||||||
|  |         <input | ||||||
|  |           id="founderName" | ||||||
|  |           v-model="form.founder.userName" | ||||||
|  |           type="text" | ||||||
|  |           required | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="founderSurname">Prénom</label> | ||||||
|  |         <input | ||||||
|  |           id="founderSurname" | ||||||
|  |           v-model="form.founder.userSurname" | ||||||
|  |           type="text" | ||||||
|  |           required | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="founderPrimaryMail">Email Principal</label> | ||||||
|  |         <input | ||||||
|  |           id="founderPrimaryMail" | ||||||
|  |           v-model="form.founder.primaryMail" | ||||||
|  |           type="email" | ||||||
|  |           required | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="founderSecondaryMail">Email Secondaire</label> | ||||||
|  |         <input | ||||||
|  |           id="founderSecondaryMail" | ||||||
|  |           v-model="form.founder.secondaryMail" | ||||||
|  |           type="email" | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="founderPhoneNumber">Numéro de téléphone</label> | ||||||
|  |         <input | ||||||
|  |           id="founderPhoneNumber" | ||||||
|  |           v-model="form.founder.phoneNumber" | ||||||
|  |           type="tel" | ||||||
|  |           required | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="founderSchool">École</label> | ||||||
|  |         <input | ||||||
|  |           id="founderSchool" | ||||||
|  |           v-model="form.founder.school" | ||||||
|  |           type="text" | ||||||
|  |           required | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="founderCourse">Département</label> | ||||||
|  |         <input | ||||||
|  |           id="founderCourse" | ||||||
|  |           v-model="form.founder.course" | ||||||
|  |           type="text" | ||||||
|  |           required | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <div class="form-group"> | ||||||
|  |         <label for="founderSneeStatus">Statut étudiant entrepreneur</label> | ||||||
|  |         <input | ||||||
|  |           id="founderSneeStatus" | ||||||
|  |           v-model="form.founder.sneeStatus" | ||||||
|  |           type="checkbox" | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |    | ||||||
|  |       <button type="submit">Soumettre</button> | ||||||
|  |     </form> | ||||||
|  |   </template> | ||||||
|  |    | ||||||
|  |   <script setup lang="ts"> | ||||||
|  |   import { ref } from "vue"; | ||||||
|  |   import { postApi } from "@/services/api";  | ||||||
|  |    | ||||||
|  |   const form = ref({ | ||||||
|  |     name: '', | ||||||
|  |     founder: { | ||||||
|  |       userSurname: '', | ||||||
|  |       userName: '', | ||||||
|  |       primaryMail: '', | ||||||
|  |       secondaryMail: '', | ||||||
|  |       phoneNumber: '', | ||||||
|  |       school: '', | ||||||
|  |       course: '', | ||||||
|  |       sneeStatus: false | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |    | ||||||
|  |   function submitForm() { | ||||||
|  |     postApi("/entrepreneur/projects/request", form.value); | ||||||
|  |   } | ||||||
|  |   </script> | ||||||
|  |    | ||||||
|  |   <style scoped> | ||||||
|  |   .add-project-form { | ||||||
|  |     max-width: 600px; | ||||||
|  |     margin: 0 auto; | ||||||
|  |     padding: 20px; | ||||||
|  |     background: #fff; | ||||||
|  |     border-radius: 10px; | ||||||
|  |     box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1); | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   h2 { | ||||||
|  |     margin-bottom: 20px; | ||||||
|  |     font-size: 24px; | ||||||
|  |     color: #333; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   h3 { | ||||||
|  |     margin-top: 20px; | ||||||
|  |     font-size: 20px; | ||||||
|  |     color: #555; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .form-group { | ||||||
|  |     margin-bottom: 15px; | ||||||
|  |     display: flex; | ||||||
|  |     flex-direction: column; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   label { | ||||||
|  |     font-weight: bold; | ||||||
|  |     margin-bottom: 5px; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   input { | ||||||
|  |     padding: 8px; | ||||||
|  |     border-radius: 5px; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     font-size: 1em; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   input[type="checkbox"] { | ||||||
|  |     width: auto; | ||||||
|  |     margin-right: 10px; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   button { | ||||||
|  |     background-color: #4caf50; | ||||||
|  |     color: white; | ||||||
|  |     padding: 10px 15px; | ||||||
|  |     border: none; | ||||||
|  |     border-radius: 5px; | ||||||
|  |     cursor: pointer; | ||||||
|  |     font-size: 1em; | ||||||
|  |     width: 100%; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   button:hover { | ||||||
|  |     background-color: #45a049; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   button:active { | ||||||
|  |     background-color: #388e3c; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   input[type="text"]:focus, | ||||||
|  |   input[type="email"]:focus, | ||||||
|  |   input[type="tel"]:focus { | ||||||
|  |     border-color: #4CAF50; | ||||||
|  |     outline: none; | ||||||
|  |   } | ||||||
|  |   </style> | ||||||
|  |    | ||||||
| @@ -15,11 +15,13 @@ import ErrorModal from "@/components/errorModal.vue"; | |||||||
| </template> | </template> | ||||||
|  |  | ||||||
| <style scoped> | <style scoped> | ||||||
| .error-wrapper { |  | ||||||
|  | .error-wrapper{ | ||||||
|   position: absolute; |   position: absolute; | ||||||
|   left: 70%; |   left: 70%; | ||||||
|     //background-color: blue; |   /*background-color: blue;*/ | ||||||
|   height: 100%; |   height: 100%; | ||||||
|   width: 30%; |   width: 30%; | ||||||
|  |  | ||||||
| } | } | ||||||
| </style> | </style> | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user