Compare commits
	
		
			23 Commits
		
	
	
		
			ed8edbd0c0
			...
			master
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 1d558c0296 | |||
| d91e74f61e | |||
| c5c1456e80 | |||
| c2174b27c0 | |||
| 6f53afa8bf | |||
| ac51fc987c | |||
| f44fbd8d06 | |||
| e85851bd4d | |||
| bfb39a0259 | |||
| e7a5e163ff | |||
| 95e1b02760 | |||
| b12af44825 | |||
| d90aa2726b | |||
| 7387ce7f50 | |||
| 7792005403 | |||
| b538e754d2 | |||
| d805ef35b1 | |||
| a1916e3219 | |||
| 9093caa557 | |||
| 2ca44f2550 | |||
| dffd179bc9 | |||
| 6ecf055060 | |||
| 8b2938de0e | 
							
								
								
									
										190
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										190
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							| @@ -2369,17 +2369,17 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "drupal/block_class", | ||||
|             "version": "3.0.0", | ||||
|             "version": "2.0.12", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://git.drupalcode.org/project/block_class.git", | ||||
|                 "reference": "3.0.0" | ||||
|                 "reference": "2.0.12" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://ftp.drupal.org/files/projects/block_class-3.0.0.zip", | ||||
|                 "reference": "3.0.0", | ||||
|                 "shasum": "41993c9ee6b925ec30e927cb557bafd9d5c05461" | ||||
|                 "url": "https://ftp.drupal.org/files/projects/block_class-2.0.12.zip", | ||||
|                 "reference": "2.0.12", | ||||
|                 "shasum": "cc3945f8038bcc9bb48eff897df332aa952d62e6" | ||||
|             }, | ||||
|             "require": { | ||||
|                 "drupal/core": "^9 || ^10 || ^11" | ||||
| @@ -2387,8 +2387,8 @@ | ||||
|             "type": "drupal-module", | ||||
|             "extra": { | ||||
|                 "drupal": { | ||||
|                     "version": "3.0.0", | ||||
|                     "datestamp": "1723241676", | ||||
|                     "version": "2.0.12", | ||||
|                     "datestamp": "1723243074", | ||||
|                     "security-coverage": { | ||||
|                         "status": "covered", | ||||
|                         "message": "Covered by Drupal's security advisory policy" | ||||
| @@ -2692,30 +2692,27 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "drupal/config_ignore", | ||||
|             "version": "3.3.0", | ||||
|             "version": "2.4.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://git.drupalcode.org/project/config_ignore.git", | ||||
|                 "reference": "8.x-3.3" | ||||
|                 "reference": "8.x-2.4" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://ftp.drupal.org/files/projects/config_ignore-8.x-3.3.zip", | ||||
|                 "reference": "8.x-3.3", | ||||
|                 "shasum": "4446811ecb023820a57c227d35c034e0d4363a70" | ||||
|                 "url": "https://ftp.drupal.org/files/projects/config_ignore-8.x-2.4.zip", | ||||
|                 "reference": "8.x-2.4", | ||||
|                 "shasum": "e0e45dde2d6927c5d26de59f352792fb6cf26554" | ||||
|             }, | ||||
|             "require": { | ||||
|                 "drupal/core": "^8.8 || ^9 || ^10 || ^11" | ||||
|             }, | ||||
|             "require-dev": { | ||||
|                 "drupal/config_filter": "^1.8||^2.2", | ||||
|                 "drush/drush": "^10 || ^11 || ^12" | ||||
|                 "drupal/config_filter": "^1 || ^2", | ||||
|                 "drupal/core": "^8 || ^9 || ^10" | ||||
|             }, | ||||
|             "type": "drupal-module", | ||||
|             "extra": { | ||||
|                 "drupal": { | ||||
|                     "version": "8.x-3.3", | ||||
|                     "datestamp": "1713299496", | ||||
|                     "version": "8.x-2.4", | ||||
|                     "datestamp": "1676045435", | ||||
|                     "security-coverage": { | ||||
|                         "status": "covered", | ||||
|                         "message": "Covered by Drupal's security advisory policy" | ||||
| @@ -2743,11 +2740,12 @@ | ||||
|                     "homepage": "https://www.drupal.org/user/413139" | ||||
|                 } | ||||
|             ], | ||||
|             "description": "Ignore certain configuration during import and export.", | ||||
|             "description": "Ignore certain configuration during import.", | ||||
|             "homepage": "http://drupal.org/project/config_ignore", | ||||
|             "support": { | ||||
|                 "source": "https://git.drupalcode.org/project/config_ignore", | ||||
|                 "issues": "http://drupal.org/project/config_ignore" | ||||
|                 "issues": "https://drupal.org/project/config_ignore", | ||||
|                 "irc": "irc://irc.freenode.org/drupal-contribute" | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
| @@ -4371,26 +4369,26 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "drupal/extlink", | ||||
|             "version": "2.0.4", | ||||
|             "version": "1.7.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://git.drupalcode.org/project/extlink.git", | ||||
|                 "reference": "2.0.4" | ||||
|                 "reference": "8.x-1.7" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://ftp.drupal.org/files/projects/extlink-2.0.4.zip", | ||||
|                 "reference": "2.0.4", | ||||
|                 "shasum": "0331ef3457d3a1701f01e04f8256bdd823ea3512" | ||||
|                 "url": "https://ftp.drupal.org/files/projects/extlink-8.x-1.7.zip", | ||||
|                 "reference": "8.x-1.7", | ||||
|                 "shasum": "38650688b5a58496db59f40b2a7f36c4bedcbfb4" | ||||
|             }, | ||||
|             "require": { | ||||
|                 "drupal/core": "^10 || ^11" | ||||
|                 "drupal/core": "^8 || ^9 || ^10" | ||||
|             }, | ||||
|             "type": "drupal-module", | ||||
|             "extra": { | ||||
|                 "drupal": { | ||||
|                     "version": "2.0.4", | ||||
|                     "datestamp": "1732565828", | ||||
|                     "version": "8.x-1.7", | ||||
|                     "datestamp": "1665770295", | ||||
|                     "security-coverage": { | ||||
|                         "status": "covered", | ||||
|                         "message": "Covered by Drupal's security advisory policy" | ||||
| @@ -5879,17 +5877,17 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "drupal/leaflet", | ||||
|             "version": "10.2.41", | ||||
|             "version": "10.2.42", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://git.drupalcode.org/project/leaflet.git", | ||||
|                 "reference": "10.2.41" | ||||
|                 "reference": "10.2.42" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://ftp.drupal.org/files/projects/leaflet-10.2.41.zip", | ||||
|                 "reference": "10.2.41", | ||||
|                 "shasum": "aec6763b23e035d2c92b7f8374ac4f7626e26756" | ||||
|                 "url": "https://ftp.drupal.org/files/projects/leaflet-10.2.42.zip", | ||||
|                 "reference": "10.2.42", | ||||
|                 "shasum": "7480d4b3c5ea8952191b9fea9b55528ad2ac9f0b" | ||||
|             }, | ||||
|             "require": { | ||||
|                 "drupal/core": "^9.3 || ^10 || ^11", | ||||
| @@ -5898,8 +5896,8 @@ | ||||
|             "type": "drupal-module", | ||||
|             "extra": { | ||||
|                 "drupal": { | ||||
|                     "version": "10.2.41", | ||||
|                     "datestamp": "1739919738", | ||||
|                     "version": "10.2.42", | ||||
|                     "datestamp": "1740753317", | ||||
|                     "security-coverage": { | ||||
|                         "status": "covered", | ||||
|                         "message": "Covered by Drupal's security advisory policy" | ||||
| @@ -6568,7 +6566,7 @@ | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://git.drupalcode.org/project/menu_block.git", | ||||
|                 "reference": "b798eff46f2ee54dbb4510cb0cbb873cc7f1aacf" | ||||
|                 "reference": "217779758c6b6c729155595fc90e86375ca6bc5b" | ||||
|             }, | ||||
|             "require": { | ||||
|                 "drupal/core": "^10.1 || ^11" | ||||
| @@ -6579,8 +6577,8 @@ | ||||
|                     "dev-1.x": "1.x-dev" | ||||
|                 }, | ||||
|                 "drupal": { | ||||
|                     "version": "8.x-1.13+6-dev", | ||||
|                     "datestamp": "1739495736", | ||||
|                     "version": "8.x-1.13+8-dev", | ||||
|                     "datestamp": "1740622280", | ||||
|                     "security-coverage": { | ||||
|                         "status": "not-covered", | ||||
|                         "message": "Dev releases are not covered by Drupal security advisories." | ||||
| @@ -8248,21 +8246,20 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "drupal/views_bulk_edit", | ||||
|             "version": "3.0.0", | ||||
|             "version": "2.9.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://git.drupalcode.org/project/views_bulk_edit.git", | ||||
|                 "reference": "3.0.0" | ||||
|                 "reference": "8.x-2.9" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://ftp.drupal.org/files/projects/views_bulk_edit-3.0.0.zip", | ||||
|                 "reference": "3.0.0", | ||||
|                 "shasum": "3b16079aa95fb4834561fcfd1197cce73b7f4b88" | ||||
|                 "url": "https://ftp.drupal.org/files/projects/views_bulk_edit-8.x-2.9.zip", | ||||
|                 "reference": "8.x-2.9", | ||||
|                 "shasum": "db45a8cc9ac629859374b24974eafcef257e4387" | ||||
|             }, | ||||
|             "require": { | ||||
|                 "drupal/core": "^9.4 || ^10 || ^11", | ||||
|                 "php": ">=8.1" | ||||
|                 "drupal/core": "^9.4 || ^10" | ||||
|             }, | ||||
|             "require-dev": { | ||||
|                 "drupal/views_bulk_operations": "~4.2.4" | ||||
| @@ -8273,8 +8270,8 @@ | ||||
|             "type": "drupal-module", | ||||
|             "extra": { | ||||
|                 "drupal": { | ||||
|                     "version": "3.0.0", | ||||
|                     "datestamp": "1725358398", | ||||
|                     "version": "8.x-2.9", | ||||
|                     "datestamp": "1690222256", | ||||
|                     "security-coverage": { | ||||
|                         "status": "covered", | ||||
|                         "message": "Covered by Drupal's security advisory policy" | ||||
| @@ -11106,16 +11103,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/dependency-injection", | ||||
|             "version": "v6.4.16", | ||||
|             "version": "v6.4.19", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/dependency-injection.git", | ||||
|                 "reference": "7a379d8871f6a36f01559c14e11141cc02eb8dc8" | ||||
|                 "reference": "b343c3b2f1539fe41331657b37d5c96c1d1ea842" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/dependency-injection/zipball/7a379d8871f6a36f01559c14e11141cc02eb8dc8", | ||||
|                 "reference": "7a379d8871f6a36f01559c14e11141cc02eb8dc8", | ||||
|                 "url": "https://api.github.com/repos/symfony/dependency-injection/zipball/b343c3b2f1539fe41331657b37d5c96c1d1ea842", | ||||
|                 "reference": "b343c3b2f1539fe41331657b37d5c96c1d1ea842", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -11167,7 +11164,7 @@ | ||||
|             "description": "Allows you to standardize and centralize the way objects are constructed in your application", | ||||
|             "homepage": "https://symfony.com", | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/dependency-injection/tree/v6.4.16" | ||||
|                 "source": "https://github.com/symfony/dependency-injection/tree/v6.4.19" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -11183,7 +11180,7 @@ | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2024-11-25T14:52:46+00:00" | ||||
|             "time": "2025-02-20T10:02:49+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/deprecation-contracts", | ||||
| @@ -11254,16 +11251,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/error-handler", | ||||
|             "version": "v6.4.18", | ||||
|             "version": "v6.4.19", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/error-handler.git", | ||||
|                 "reference": "e8d3b5b1975e67812a54388b1ba8e9ec28eb770e" | ||||
|                 "reference": "3d4e55cd2b8f1979a65eba9ab749d6466c316f71" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/error-handler/zipball/e8d3b5b1975e67812a54388b1ba8e9ec28eb770e", | ||||
|                 "reference": "e8d3b5b1975e67812a54388b1ba8e9ec28eb770e", | ||||
|                 "url": "https://api.github.com/repos/symfony/error-handler/zipball/3d4e55cd2b8f1979a65eba9ab749d6466c316f71", | ||||
|                 "reference": "3d4e55cd2b8f1979a65eba9ab749d6466c316f71", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -11309,7 +11306,7 @@ | ||||
|             "description": "Provides tools to manage errors and ease debugging PHP code", | ||||
|             "homepage": "https://symfony.com", | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/error-handler/tree/v6.4.18" | ||||
|                 "source": "https://github.com/symfony/error-handler/tree/v6.4.19" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -11325,7 +11322,7 @@ | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-01-06T09:38:16+00:00" | ||||
|             "time": "2025-02-02T20:16:33+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/event-dispatcher", | ||||
| @@ -11692,16 +11689,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/http-kernel", | ||||
|             "version": "v6.4.18", | ||||
|             "version": "v6.4.19", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/http-kernel.git", | ||||
|                 "reference": "fca7197bfe9e99dfae7fb1ad3f7f5bd9ef80e1b7" | ||||
|                 "reference": "88f2c9f7feff86bb7b9105c5151bc2c1404cd64c" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/http-kernel/zipball/fca7197bfe9e99dfae7fb1ad3f7f5bd9ef80e1b7", | ||||
|                 "reference": "fca7197bfe9e99dfae7fb1ad3f7f5bd9ef80e1b7", | ||||
|                 "url": "https://api.github.com/repos/symfony/http-kernel/zipball/88f2c9f7feff86bb7b9105c5151bc2c1404cd64c", | ||||
|                 "reference": "88f2c9f7feff86bb7b9105c5151bc2c1404cd64c", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -11786,7 +11783,7 @@ | ||||
|             "description": "Provides a structured process for converting a Request into a Response", | ||||
|             "homepage": "https://symfony.com", | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/http-kernel/tree/v6.4.18" | ||||
|                 "source": "https://github.com/symfony/http-kernel/tree/v6.4.19" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -11802,7 +11799,7 @@ | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-01-29T07:25:58+00:00" | ||||
|             "time": "2025-02-26T10:51:37+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/mailer", | ||||
| @@ -11886,16 +11883,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/mime", | ||||
|             "version": "v6.4.18", | ||||
|             "version": "v6.4.19", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/mime.git", | ||||
|                 "reference": "917d77981eb1ea963608d5cda4d9c0cf72eaa68e" | ||||
|                 "reference": "ac537b6c55ccc2c749f3c979edfa9ec14aaed4f3" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/mime/zipball/917d77981eb1ea963608d5cda4d9c0cf72eaa68e", | ||||
|                 "reference": "917d77981eb1ea963608d5cda4d9c0cf72eaa68e", | ||||
|                 "url": "https://api.github.com/repos/symfony/mime/zipball/ac537b6c55ccc2c749f3c979edfa9ec14aaed4f3", | ||||
|                 "reference": "ac537b6c55ccc2c749f3c979edfa9ec14aaed4f3", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -11951,7 +11948,7 @@ | ||||
|                 "mime-type" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/mime/tree/v6.4.18" | ||||
|                 "source": "https://github.com/symfony/mime/tree/v6.4.19" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -11967,7 +11964,7 @@ | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-01-23T13:10:52+00:00" | ||||
|             "time": "2025-02-17T21:23:52+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/polyfill-ctype", | ||||
| @@ -12604,16 +12601,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/process", | ||||
|             "version": "v6.4.15", | ||||
|             "version": "v6.4.19", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/process.git", | ||||
|                 "reference": "3cb242f059c14ae08591c5c4087d1fe443564392" | ||||
|                 "reference": "7a1c12e87b08ec9c97abdd188c9b3f5a40e37fc3" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/process/zipball/3cb242f059c14ae08591c5c4087d1fe443564392", | ||||
|                 "reference": "3cb242f059c14ae08591c5c4087d1fe443564392", | ||||
|                 "url": "https://api.github.com/repos/symfony/process/zipball/7a1c12e87b08ec9c97abdd188c9b3f5a40e37fc3", | ||||
|                 "reference": "7a1c12e87b08ec9c97abdd188c9b3f5a40e37fc3", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -12645,7 +12642,7 @@ | ||||
|             "description": "Executes commands in sub-processes", | ||||
|             "homepage": "https://symfony.com", | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/process/tree/v6.4.15" | ||||
|                 "source": "https://github.com/symfony/process/tree/v6.4.19" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -12661,7 +12658,7 @@ | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2024-11-06T14:19:14+00:00" | ||||
|             "time": "2025-02-04T13:35:48+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/psr-http-message-bridge", | ||||
| @@ -12831,16 +12828,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/serializer", | ||||
|             "version": "v6.4.18", | ||||
|             "version": "v6.4.19", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/serializer.git", | ||||
|                 "reference": "6ad986f62276da4c8c69754decfaa445a89cb6e3" | ||||
|                 "reference": "a221b2f6066af304d760cff7a26f201b4fab4aef" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/serializer/zipball/6ad986f62276da4c8c69754decfaa445a89cb6e3", | ||||
|                 "reference": "6ad986f62276da4c8c69754decfaa445a89cb6e3", | ||||
|                 "url": "https://api.github.com/repos/symfony/serializer/zipball/a221b2f6066af304d760cff7a26f201b4fab4aef", | ||||
|                 "reference": "a221b2f6066af304d760cff7a26f201b4fab4aef", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -12909,7 +12906,7 @@ | ||||
|             "description": "Handles serializing and deserializing data structures, including object graphs, into array structures or other formats like XML and JSON.", | ||||
|             "homepage": "https://symfony.com", | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/serializer/tree/v6.4.18" | ||||
|                 "source": "https://github.com/symfony/serializer/tree/v6.4.19" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -12925,7 +12922,7 @@ | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-01-28T18:47:02+00:00" | ||||
|             "time": "2025-02-24T08:42:36+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/service-contracts", | ||||
| @@ -13176,16 +13173,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/validator", | ||||
|             "version": "v6.4.18", | ||||
|             "version": "v6.4.19", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/validator.git", | ||||
|                 "reference": "ce20367d07b2592202e9c266b16a93fa50145207" | ||||
|                 "reference": "f3e853dffe7c5db675686b8216d6d890dad8c885" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/validator/zipball/ce20367d07b2592202e9c266b16a93fa50145207", | ||||
|                 "reference": "ce20367d07b2592202e9c266b16a93fa50145207", | ||||
|                 "url": "https://api.github.com/repos/symfony/validator/zipball/f3e853dffe7c5db675686b8216d6d890dad8c885", | ||||
|                 "reference": "f3e853dffe7c5db675686b8216d6d890dad8c885", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -13253,7 +13250,7 @@ | ||||
|             "description": "Provides tools to validate values", | ||||
|             "homepage": "https://symfony.com", | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/validator/tree/v6.4.18" | ||||
|                 "source": "https://github.com/symfony/validator/tree/v6.4.19" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -13269,7 +13266,7 @@ | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-01-27T16:05:44+00:00" | ||||
|             "time": "2025-02-19T13:12:02+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/var-dumper", | ||||
| @@ -13358,16 +13355,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/var-exporter", | ||||
|             "version": "v6.4.13", | ||||
|             "version": "v6.4.19", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/var-exporter.git", | ||||
|                 "reference": "0f605f72a363f8743001038a176eeb2a11223b51" | ||||
|                 "reference": "be6e71b0c257884c1107313de5d247741cfea172" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/var-exporter/zipball/0f605f72a363f8743001038a176eeb2a11223b51", | ||||
|                 "reference": "0f605f72a363f8743001038a176eeb2a11223b51", | ||||
|                 "url": "https://api.github.com/repos/symfony/var-exporter/zipball/be6e71b0c257884c1107313de5d247741cfea172", | ||||
|                 "reference": "be6e71b0c257884c1107313de5d247741cfea172", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -13415,7 +13412,7 @@ | ||||
|                 "serialize" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/var-exporter/tree/v6.4.13" | ||||
|                 "source": "https://github.com/symfony/var-exporter/tree/v6.4.19" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -13431,7 +13428,7 @@ | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2024-09-25T14:18:03+00:00" | ||||
|             "time": "2025-02-13T09:33:32+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/yaml", | ||||
| @@ -13757,6 +13754,7 @@ | ||||
|         "drupal/advanced_text_formatter": 5, | ||||
|         "drupal/bulkdelete": 20, | ||||
|         "drupal/config_devel": 20, | ||||
|         "drupal/config_ignore": 5, | ||||
|         "drupal/config_update": 15, | ||||
|         "drupal/context": 5, | ||||
|         "drupal/date_range_formatter": 20, | ||||
|   | ||||
							
								
								
									
										11
									
								
								config/sync/caravane.settings.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								config/sync/caravane.settings.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| features: | ||||
|   node_user_picture: false | ||||
|   comment_user_picture: true | ||||
|   comment_user_verification: true | ||||
|   favicon: 0 | ||||
| logo: | ||||
|   use_default: 0 | ||||
|   path: 'public://favicon-caravane_1.png' | ||||
| favicon: | ||||
|   use_default: 0 | ||||
|   path: '' | ||||
| @@ -1,4 +1,4 @@ | ||||
| uuid: b0c240c8-beea-4be3-96b8-6543b6734b25 | ||||
| uuid: 39f683b9-ea9f-4e84-8dec-c30162c19b72 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: {  } | ||||
| @@ -12,6 +12,6 @@ context: | ||||
|   fallback: | ||||
|     language: '' | ||||
| menu: | ||||
|   path: /edit/partenaire | ||||
|   path: /edit/partenaires | ||||
|   weight: 0 | ||||
|   description: '' | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| uuid: 86c7ea1e-a93e-4ddf-9887-e1f302a99ea9 | ||||
| uuid: 67828fd5-674e-4e52-9f2f-3457e06c0baa | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: {  } | ||||
| id: intro_partenaires | ||||
| label: 'Intro partenaires' | ||||
| id: intro_ressource | ||||
| label: 'intro ressource' | ||||
| token: false | ||||
| context: | ||||
|   show_warning: true | ||||
| @@ -12,6 +12,6 @@ context: | ||||
|   fallback: | ||||
|     language: '' | ||||
| menu: | ||||
|   path: /edit/partenaires | ||||
|   path: /edit/ressource | ||||
|   weight: 0 | ||||
|   description: '' | ||||
| @@ -1,4 +1,4 @@ | ||||
| uuid: c8c782e1-e597-4e4d-ac96-65142dbadedb | ||||
| uuid: 3d095235-90fe-4b5f-ae32-ef10c26c1f47 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   | ||||
| @@ -1,24 +0,0 @@ | ||||
| uuid: 03e534f2-ceae-4f84-89f2-a0f9997f8a9d | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   config: | ||||
|     - config_pages.type.intro_partenaires | ||||
|     - field.field.config_pages.intro_partenaires.field_intro | ||||
|   module: | ||||
|     - text | ||||
| id: config_pages.intro_partenaires.default | ||||
| targetEntityType: config_pages | ||||
| bundle: intro_partenaires | ||||
| mode: default | ||||
| content: | ||||
|   field_intro: | ||||
|     type: text_textarea | ||||
|     weight: 0 | ||||
|     region: content | ||||
|     settings: | ||||
|       rows: 5 | ||||
|       placeholder: '' | ||||
|     third_party_settings: {  } | ||||
| hidden: | ||||
|   label: true | ||||
| @@ -0,0 +1,33 @@ | ||||
| uuid: c5c09811-6a65-4cff-9936-78dc896b9adf | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   config: | ||||
|     - config_pages.type.intro_ressource | ||||
|     - field.field.config_pages.intro_ressource.field_intro | ||||
|     - field.field.config_pages.intro_ressource.field_titre | ||||
|   module: | ||||
|     - text | ||||
| id: config_pages.intro_ressource.default | ||||
| targetEntityType: config_pages | ||||
| bundle: intro_ressource | ||||
| mode: default | ||||
| content: | ||||
|   field_intro: | ||||
|     type: text_textarea | ||||
|     weight: 1 | ||||
|     region: content | ||||
|     settings: | ||||
|       rows: 5 | ||||
|       placeholder: '' | ||||
|     third_party_settings: {  } | ||||
|   field_titre: | ||||
|     type: string_textfield | ||||
|     weight: 0 | ||||
|     region: content | ||||
|     settings: | ||||
|       size: 60 | ||||
|       placeholder: '' | ||||
|     third_party_settings: {  } | ||||
| hidden: | ||||
|   label: true | ||||
| @@ -1,3 +1,4 @@ | ||||
| uuid: 4b684ae9-1a47-4712-bfbe-54da6f8555da | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   | ||||
| @@ -7,6 +7,7 @@ dependencies: | ||||
|     - field.field.node.ressource.field_date_ressource | ||||
|     - field.field.node.ressource.field_etape | ||||
|     - field.field.node.ressource.field_introduction | ||||
|     - field.field.node.ressource.field_mis_en_avant | ||||
|     - field.field.node.ressource.field_parties_ressource | ||||
|     - field.field.node.ressource.field_thematiques | ||||
|     - field.field.node.ressource.field_type_de_ressource | ||||
| @@ -22,11 +23,12 @@ third_party_settings: | ||||
|   field_group: | ||||
|     group_details: | ||||
|       children: | ||||
|         - field_type_de_ressource | ||||
|         - field_thematiques | ||||
|         - field_etape | ||||
|         - status | ||||
|         - path | ||||
|         - field_mis_en_avant | ||||
|         - field_type_de_ressource | ||||
|       label: Details | ||||
|       region: content | ||||
|       parent_name: '' | ||||
| @@ -41,6 +43,56 @@ third_party_settings: | ||||
|         description: '' | ||||
|         required_fields: true | ||||
|         weight: -100 | ||||
|     group_tabs: | ||||
|       children: | ||||
|         - group_infos | ||||
|         - group_parties | ||||
|       label: Tabs | ||||
|       region: content | ||||
|       parent_name: '' | ||||
|       weight: 1 | ||||
|       format_type: tabs | ||||
|       format_settings: | ||||
|         classes: '' | ||||
|         show_empty_fields: false | ||||
|         id: '' | ||||
|         label_as_html: false | ||||
|         direction: horizontal | ||||
|         width_breakpoint: 640 | ||||
|     group_infos: | ||||
|       children: | ||||
|         - field_autheurice | ||||
|         - field_date_ressource | ||||
|         - field_introduction | ||||
|       label: Infos | ||||
|       region: content | ||||
|       parent_name: group_tabs | ||||
|       weight: 20 | ||||
|       format_type: tab | ||||
|       format_settings: | ||||
|         classes: '' | ||||
|         show_empty_fields: false | ||||
|         id: '' | ||||
|         label_as_html: false | ||||
|         formatter: closed | ||||
|         description: '' | ||||
|         required_fields: true | ||||
|     group_parties: | ||||
|       children: | ||||
|         - field_parties_ressource | ||||
|       label: Parties | ||||
|       region: content | ||||
|       parent_name: group_tabs | ||||
|       weight: 21 | ||||
|       format_type: tab | ||||
|       format_settings: | ||||
|         classes: '' | ||||
|         show_empty_fields: false | ||||
|         id: '' | ||||
|         label_as_html: false | ||||
|         formatter: closed | ||||
|         description: '' | ||||
|         required_fields: true | ||||
| id: node.ressource.default | ||||
| targetEntityType: node | ||||
| bundle: ressource | ||||
| @@ -48,7 +100,7 @@ mode: default | ||||
| content: | ||||
|   field_autheurice: | ||||
|     type: string_textfield | ||||
|     weight: 1 | ||||
|     weight: 2 | ||||
|     region: content | ||||
|     settings: | ||||
|       size: 60 | ||||
| @@ -56,13 +108,13 @@ content: | ||||
|     third_party_settings: {  } | ||||
|   field_date_ressource: | ||||
|     type: datetime_default | ||||
|     weight: 2 | ||||
|     weight: 3 | ||||
|     region: content | ||||
|     settings: {  } | ||||
|     third_party_settings: {  } | ||||
|   field_etape: | ||||
|     type: entity_reference_autocomplete | ||||
|     weight: 5 | ||||
|     weight: 28 | ||||
|     region: content | ||||
|     settings: | ||||
|       match_operator: CONTAINS | ||||
| @@ -72,12 +124,19 @@ content: | ||||
|     third_party_settings: {  } | ||||
|   field_introduction: | ||||
|     type: text_textarea | ||||
|     weight: 3 | ||||
|     weight: 4 | ||||
|     region: content | ||||
|     settings: | ||||
|       rows: 5 | ||||
|       placeholder: '' | ||||
|     third_party_settings: {  } | ||||
|   field_mis_en_avant: | ||||
|     type: boolean_checkbox | ||||
|     weight: 31 | ||||
|     region: content | ||||
|     settings: | ||||
|       display_label: true | ||||
|     third_party_settings: {  } | ||||
|   field_parties_ressource: | ||||
|     type: paragraphs | ||||
|     weight: 5 | ||||
| @@ -98,14 +157,14 @@ content: | ||||
|     third_party_settings: {  } | ||||
|   field_thematiques: | ||||
|     type: autocomplete_deluxe | ||||
|     weight: 4 | ||||
|     weight: 27 | ||||
|     region: content | ||||
|     settings: | ||||
|       match_operator: CONTAINS | ||||
|       autocomplete_route_name: autocomplete_deluxe.autocomplete | ||||
|       size: 60 | ||||
|       selection_handler: default | ||||
|       limit: 10 | ||||
|       match_limit: 10 | ||||
|       min_length: 0 | ||||
|       delimiter: '' | ||||
|       not_found_message_allow: false | ||||
| @@ -114,24 +173,20 @@ content: | ||||
|       no_empty_message: 'No terms could be found. Please type in order to add a new term.' | ||||
|     third_party_settings: {  } | ||||
|   field_type_de_ressource: | ||||
|     type: entity_reference_autocomplete | ||||
|     weight: 3 | ||||
|     type: options_select | ||||
|     weight: 32 | ||||
|     region: content | ||||
|     settings: | ||||
|       match_operator: CONTAINS | ||||
|       match_limit: 10 | ||||
|       size: 60 | ||||
|       placeholder: '' | ||||
|     settings: {  } | ||||
|     third_party_settings: {  } | ||||
|   path: | ||||
|     type: path | ||||
|     weight: 7 | ||||
|     weight: 30 | ||||
|     region: content | ||||
|     settings: {  } | ||||
|     third_party_settings: {  } | ||||
|   status: | ||||
|     type: boolean_checkbox | ||||
|     weight: 6 | ||||
|     weight: 29 | ||||
|     region: content | ||||
|     settings: | ||||
|       display_label: true | ||||
|   | ||||
| @@ -5,6 +5,7 @@ dependencies: | ||||
|   config: | ||||
|     - field.field.paragraph.document.field_autheurice_s | ||||
|     - field.field.paragraph.document.field_date | ||||
|     - field.field.paragraph.document.field_document | ||||
|     - field.field.paragraph.document.field_sous_titre | ||||
|     - field.field.paragraph.document.field_titre | ||||
|     - field.field.paragraph.document.field_vignette | ||||
| @@ -12,6 +13,7 @@ dependencies: | ||||
|     - paragraphs.paragraphs_type.document | ||||
|   module: | ||||
|     - datetime | ||||
|     - file | ||||
|     - image | ||||
| id: paragraph.document.default | ||||
| targetEntityType: paragraph | ||||
| @@ -32,6 +34,13 @@ content: | ||||
|     region: content | ||||
|     settings: {  } | ||||
|     third_party_settings: {  } | ||||
|   field_document: | ||||
|     type: file_generic | ||||
|     weight: 5 | ||||
|     region: content | ||||
|     settings: | ||||
|       progress_indicator: throbber | ||||
|     third_party_settings: {  } | ||||
|   field_sous_titre: | ||||
|     type: string_textfield | ||||
|     weight: 2 | ||||
|   | ||||
| @@ -20,4 +20,5 @@ content: | ||||
|     weight: 0 | ||||
|     region: content | ||||
| hidden: | ||||
|   field_titre: true | ||||
|   search_api_excerpt: true | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| uuid: a79b589c-3286-425b-b508-38744f9ebeb1 | ||||
| uuid: cad2e853-7706-4477-a3cc-d810dcd76c96 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
| @@ -18,7 +18,7 @@ content: | ||||
|     label: visually_hidden | ||||
|     settings: {  } | ||||
|     third_party_settings: {  } | ||||
|     weight: 1 | ||||
|     weight: 0 | ||||
|     region: content | ||||
|   field_titre: | ||||
|     type: string | ||||
| @@ -26,7 +26,7 @@ content: | ||||
|     settings: | ||||
|       link_to_entity: false | ||||
|     third_party_settings: {  } | ||||
|     weight: 0 | ||||
|     weight: 1 | ||||
|     region: content | ||||
| hidden: | ||||
|   search_api_excerpt: true | ||||
|   | ||||
| @@ -1,23 +0,0 @@ | ||||
| uuid: ac925def-bc83-4add-a5c9-88bc02e771b5 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   config: | ||||
|     - config_pages.type.intro_partenaires | ||||
|     - field.field.config_pages.intro_partenaires.field_intro | ||||
|   module: | ||||
|     - text | ||||
| id: config_pages.intro_partenaires.default | ||||
| targetEntityType: config_pages | ||||
| bundle: intro_partenaires | ||||
| mode: default | ||||
| content: | ||||
|   field_intro: | ||||
|     type: text_default | ||||
|     label: visually_hidden | ||||
|     settings: {  } | ||||
|     third_party_settings: {  } | ||||
|     weight: 0 | ||||
|     region: content | ||||
| hidden: | ||||
|   search_api_excerpt: true | ||||
| @@ -0,0 +1,32 @@ | ||||
| uuid: 342c8cdf-4085-4e41-941f-b503beb0ad1f | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   config: | ||||
|     - config_pages.type.intro_ressource | ||||
|     - field.field.config_pages.intro_ressource.field_intro | ||||
|     - field.field.config_pages.intro_ressource.field_titre | ||||
|   module: | ||||
|     - text | ||||
| id: config_pages.intro_ressource.default | ||||
| targetEntityType: config_pages | ||||
| bundle: intro_ressource | ||||
| mode: default | ||||
| content: | ||||
|   field_intro: | ||||
|     type: text_default | ||||
|     label: visually_hidden | ||||
|     settings: {  } | ||||
|     third_party_settings: {  } | ||||
|     weight: 0 | ||||
|     region: content | ||||
|   field_titre: | ||||
|     type: string | ||||
|     label: above | ||||
|     settings: | ||||
|       link_to_entity: false | ||||
|     third_party_settings: {  } | ||||
|     weight: 0 | ||||
|     region: content | ||||
| hidden: | ||||
|   search_api_excerpt: true | ||||
| @@ -7,6 +7,7 @@ dependencies: | ||||
|     - field.field.node.ressource.field_date_ressource | ||||
|     - field.field.node.ressource.field_etape | ||||
|     - field.field.node.ressource.field_introduction | ||||
|     - field.field.node.ressource.field_mis_en_avant | ||||
|     - field.field.node.ressource.field_parties_ressource | ||||
|     - field.field.node.ressource.field_thematiques | ||||
|     - field.field.node.ressource.field_type_de_ressource | ||||
| @@ -14,6 +15,7 @@ dependencies: | ||||
|   module: | ||||
|     - datetime | ||||
|     - entity_reference_revisions | ||||
|     - options | ||||
|     - text | ||||
|     - user | ||||
| id: node.ressource.default | ||||
| @@ -53,6 +55,16 @@ content: | ||||
|     third_party_settings: {  } | ||||
|     weight: 107 | ||||
|     region: content | ||||
|   field_mis_en_avant: | ||||
|     type: boolean | ||||
|     label: above | ||||
|     settings: | ||||
|       format: default | ||||
|       format_custom_false: '' | ||||
|       format_custom_true: '' | ||||
|     third_party_settings: {  } | ||||
|     weight: 109 | ||||
|     region: content | ||||
|   field_parties_ressource: | ||||
|     type: entity_reference_revisions_entity_view | ||||
|     label: above | ||||
| @@ -71,12 +83,11 @@ content: | ||||
|     weight: 4 | ||||
|     region: content | ||||
|   field_type_de_ressource: | ||||
|     type: entity_reference_label | ||||
|     type: list_default | ||||
|     label: above | ||||
|     settings: | ||||
|       link: true | ||||
|     settings: {  } | ||||
|     third_party_settings: {  } | ||||
|     weight: 102 | ||||
|     weight: 108 | ||||
|     region: content | ||||
|   links: | ||||
|     settings: {  } | ||||
|   | ||||
| @@ -8,6 +8,7 @@ dependencies: | ||||
|     - field.field.node.ressource.field_date_ressource | ||||
|     - field.field.node.ressource.field_etape | ||||
|     - field.field.node.ressource.field_introduction | ||||
|     - field.field.node.ressource.field_mis_en_avant | ||||
|     - field.field.node.ressource.field_parties_ressource | ||||
|     - field.field.node.ressource.field_thematiques | ||||
|     - field.field.node.ressource.field_type_de_ressource | ||||
| @@ -29,6 +30,7 @@ hidden: | ||||
|   field_date_ressource: true | ||||
|   field_etape: true | ||||
|   field_introduction: true | ||||
|   field_mis_en_avant: true | ||||
|   field_parties_ressource: true | ||||
|   field_thematiques: true | ||||
|   field_type_de_ressource: true | ||||
|   | ||||
| @@ -5,12 +5,14 @@ dependencies: | ||||
|   config: | ||||
|     - field.field.paragraph.document.field_autheurice_s | ||||
|     - field.field.paragraph.document.field_date | ||||
|     - field.field.paragraph.document.field_document | ||||
|     - field.field.paragraph.document.field_sous_titre | ||||
|     - field.field.paragraph.document.field_titre | ||||
|     - field.field.paragraph.document.field_vignette | ||||
|     - paragraphs.paragraphs_type.document | ||||
|   module: | ||||
|     - datetime | ||||
|     - file | ||||
|     - image | ||||
| id: paragraph.document.default | ||||
| targetEntityType: paragraph | ||||
| @@ -34,6 +36,14 @@ content: | ||||
|     third_party_settings: {  } | ||||
|     weight: 4 | ||||
|     region: content | ||||
|   field_document: | ||||
|     type: file_default | ||||
|     label: above | ||||
|     settings: | ||||
|       use_description_as_link_text: true | ||||
|     third_party_settings: {  } | ||||
|     weight: 5 | ||||
|     region: content | ||||
|   field_sous_titre: | ||||
|     type: string | ||||
|     label: visually_hidden | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| uuid: 3035b470-e4d9-4807-acd6-24a09525f537 | ||||
| uuid: 884f84e5-faeb-4719-a083-32cb215fd311 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| uuid: bbddf173-169d-4977-ba3f-d1ec9f5aa452 | ||||
| uuid: 00096414-fab5-4bed-ac99-b819ad041ed9 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   | ||||
| @@ -1,17 +1,17 @@ | ||||
| uuid: 52fa2fc6-d3ca-4fef-b91b-06be4145e8e3 | ||||
| uuid: e669e2a0-7ad6-4744-9bcf-efd428aa8070 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   config: | ||||
|     - config_pages.type.intro_partenaires | ||||
|     - config_pages.type.intro_ressource | ||||
|     - field.storage.config_pages.field_intro | ||||
|     - filter.format.wysiwyg | ||||
|   module: | ||||
|     - text | ||||
| id: config_pages.intro_partenaires.field_intro | ||||
| id: config_pages.intro_ressource.field_intro | ||||
| field_name: field_intro | ||||
| entity_type: config_pages | ||||
| bundle: intro_partenaires | ||||
| bundle: intro_ressource | ||||
| label: Intro | ||||
| description: '' | ||||
| required: false | ||||
| @@ -0,0 +1,19 @@ | ||||
| uuid: 1a5ca231-32de-4adc-b066-22a8377bf323 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   config: | ||||
|     - config_pages.type.intro_ressource | ||||
|     - field.storage.config_pages.field_titre | ||||
| id: config_pages.intro_ressource.field_titre | ||||
| field_name: field_titre | ||||
| entity_type: config_pages | ||||
| bundle: intro_ressource | ||||
| label: titre | ||||
| description: '' | ||||
| required: false | ||||
| translatable: false | ||||
| default_value: {  } | ||||
| default_value_callback: '' | ||||
| settings: {  } | ||||
| field_type: string | ||||
| @@ -9,7 +9,7 @@ id: node.ressource.field_autheurice | ||||
| field_name: field_autheurice | ||||
| entity_type: node | ||||
| bundle: ressource | ||||
| label: Autheur⋅ice | ||||
| label: Auteur⋅ice | ||||
| description: '' | ||||
| required: false | ||||
| translatable: false | ||||
|   | ||||
| @@ -0,0 +1,23 @@ | ||||
| uuid: 3e824533-8b04-4c3c-acb4-cdf87f40e281 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   config: | ||||
|     - field.storage.node.field_mis_en_avant | ||||
|     - node.type.ressource | ||||
| id: node.ressource.field_mis_en_avant | ||||
| field_name: field_mis_en_avant | ||||
| entity_type: node | ||||
| bundle: ressource | ||||
| label: 'mis en avant' | ||||
| description: '' | ||||
| required: false | ||||
| translatable: false | ||||
| default_value: | ||||
|   - | ||||
|     value: 0 | ||||
| default_value_callback: '' | ||||
| settings: | ||||
|   on_label: Activé | ||||
|   off_label: Désactivé | ||||
| field_type: boolean | ||||
| @@ -1,29 +1,21 @@ | ||||
| uuid: 2c08a730-15f1-48fb-99b5-9b3abb866321 | ||||
| uuid: e439dc6f-3fc5-40cb-b615-f95c5a22a6f9 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   config: | ||||
|     - field.storage.node.field_type_de_ressource | ||||
|     - node.type.ressource | ||||
|     - taxonomy.vocabulary.type_de_ressource | ||||
|   module: | ||||
|     - options | ||||
| id: node.ressource.field_type_de_ressource | ||||
| field_name: field_type_de_ressource | ||||
| entity_type: node | ||||
| bundle: ressource | ||||
| label: 'Type de ressource' | ||||
| description: '' | ||||
| required: false | ||||
| required: true | ||||
| translatable: false | ||||
| default_value: {  } | ||||
| default_value_callback: '' | ||||
| settings: | ||||
|   handler: 'default:taxonomy_term' | ||||
|   handler_settings: | ||||
|     target_bundles: | ||||
|       type_de_ressource: type_de_ressource | ||||
|     sort: | ||||
|       field: name | ||||
|       direction: asc | ||||
|     auto_create: false | ||||
|     auto_create_bundle: '' | ||||
| field_type: entity_reference | ||||
| settings: {  } | ||||
| field_type: list_string | ||||
|   | ||||
| @@ -0,0 +1,27 @@ | ||||
| uuid: 68276064-4601-4b82-a6b5-336faf39a572 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   config: | ||||
|     - field.storage.paragraph.field_document | ||||
|     - paragraphs.paragraphs_type.document | ||||
|   module: | ||||
|     - file | ||||
| id: paragraph.document.field_document | ||||
| field_name: field_document | ||||
| entity_type: paragraph | ||||
| bundle: document | ||||
| label: Document | ||||
| description: '' | ||||
| required: false | ||||
| translatable: false | ||||
| default_value: {  } | ||||
| default_value_callback: '' | ||||
| settings: | ||||
|   handler: 'default:file' | ||||
|   handler_settings: {  } | ||||
|   file_directory: '[date:custom:Y]-[date:custom:m]' | ||||
|   file_extensions: 'pdf doc docx odt zip' | ||||
|   max_filesize: '' | ||||
|   description_field: true | ||||
| field_type: file | ||||
							
								
								
									
										18
									
								
								config/sync/field.storage.node.field_mis_en_avant.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								config/sync/field.storage.node.field_mis_en_avant.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| uuid: e92021e0-c606-4e50-b042-012920aa6341 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   module: | ||||
|     - node | ||||
| id: node.field_mis_en_avant | ||||
| field_name: field_mis_en_avant | ||||
| entity_type: node | ||||
| type: boolean | ||||
| settings: {  } | ||||
| module: core | ||||
| locked: false | ||||
| cardinality: 1 | ||||
| translatable: true | ||||
| indexes: {  } | ||||
| persist_with_no_fields: false | ||||
| custom_storage: false | ||||
| @@ -1,17 +1,30 @@ | ||||
| uuid: 9dedf9d1-5a9a-4c71-9896-4d4f2b2b5387 | ||||
| uuid: be6b9eb9-d827-4adc-9106-e7c31b94aff6 | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   module: | ||||
|     - node | ||||
|     - taxonomy | ||||
|     - options | ||||
| id: node.field_type_de_ressource | ||||
| field_name: field_type_de_ressource | ||||
| entity_type: node | ||||
| type: entity_reference | ||||
| type: list_string | ||||
| settings: | ||||
|   target_type: taxonomy_term | ||||
| module: core | ||||
|   allowed_values: | ||||
|     - | ||||
|       value: cartes_blanches | ||||
|       label: 'Cartes blanches' | ||||
|     - | ||||
|       value: reportages | ||||
|       label: Reportages | ||||
|     - | ||||
|       value: videos | ||||
|       label: Videos | ||||
|     - | ||||
|       value: documents | ||||
|       label: Documents | ||||
|   allowed_values_function: '' | ||||
| module: options | ||||
| locked: false | ||||
| cardinality: 1 | ||||
| translatable: true | ||||
|   | ||||
							
								
								
									
										23
									
								
								config/sync/field.storage.paragraph.field_document.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								config/sync/field.storage.paragraph.field_document.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| uuid: 781e5561-6299-4ce5-a4e4-72211d29087a | ||||
| langcode: fr | ||||
| status: true | ||||
| dependencies: | ||||
|   module: | ||||
|     - file | ||||
|     - paragraphs | ||||
| id: paragraph.field_document | ||||
| field_name: field_document | ||||
| entity_type: paragraph | ||||
| type: file | ||||
| settings: | ||||
|   target_type: file | ||||
|   display_field: true | ||||
|   display_default: true | ||||
|   uri_scheme: public | ||||
| module: file | ||||
| locked: false | ||||
| cardinality: 1 | ||||
| translatable: true | ||||
| indexes: {  } | ||||
| persist_with_no_fields: false | ||||
| custom_storage: false | ||||
| @@ -14,7 +14,7 @@ smtp_from: '' | ||||
| smtp_fromname: '' | ||||
| smtp_client_hostname: '' | ||||
| smtp_client_helo: '' | ||||
| smtp_allowhtml: '' | ||||
| smtp_allowhtml: false | ||||
| smtp_test_address: '' | ||||
| smtp_reroute_address: '' | ||||
| smtp_debugging: false | ||||
|   | ||||
| @@ -4,7 +4,7 @@ favicon: | ||||
|   mimetype: image/vnd.microsoft.icon | ||||
|   path: '' | ||||
|   url: '' | ||||
|   use_default: true | ||||
|   use_default: false | ||||
| features: | ||||
|   comment_user_picture: true | ||||
|   comment_user_verification: true | ||||
|   | ||||
| @@ -20,4 +20,5 @@ permissions: | ||||
|   - 'restful get rest_menu_item' | ||||
|   - 'view intro_gouvernance config page entity' | ||||
|   - 'view intro_partenaire config page entity' | ||||
|   - 'view intro_ressource config page entity' | ||||
|   - 'view media' | ||||
|   | ||||
| @@ -10,6 +10,7 @@ dependencies: | ||||
|     - node.type.etape | ||||
|     - node.type.gallerie_photo | ||||
|     - node.type.partenaire | ||||
|     - node.type.ressource | ||||
|     - node.type.static | ||||
|     - taxonomy.vocabulary.saisons | ||||
|     - taxonomy.vocabulary.thematiques | ||||
| @@ -38,12 +39,14 @@ permissions: | ||||
|   - 'create etape content' | ||||
|   - 'create gallerie_photo content' | ||||
|   - 'create partenaire content' | ||||
|   - 'create ressource content' | ||||
|   - 'create terms in saisons' | ||||
|   - 'create terms in thematiques' | ||||
|   - 'delete equipe revisions' | ||||
|   - 'delete etape revisions' | ||||
|   - 'delete gallerie_photo revisions' | ||||
|   - 'delete partenaire revisions' | ||||
|   - 'delete ressource revisions' | ||||
|   - 'delete static revisions' | ||||
|   - 'delete terms in saisons' | ||||
|   - 'delete terms in thematiques' | ||||
| @@ -51,6 +54,7 @@ permissions: | ||||
|   - 'edit any etape content' | ||||
|   - 'edit any gallerie_photo content' | ||||
|   - 'edit any partenaire content' | ||||
|   - 'edit any ressource content' | ||||
|   - 'edit any static content' | ||||
|   - 'edit intro_gouvernance config page entity' | ||||
|   - 'edit intro_partenaires config page entity' | ||||
| @@ -58,6 +62,7 @@ permissions: | ||||
|   - 'edit own etape content' | ||||
|   - 'edit own gallerie_photo content' | ||||
|   - 'edit own partenaire content' | ||||
|   - 'edit own ressource content' | ||||
|   - 'edit own static content' | ||||
|   - 'edit terms in saisons' | ||||
|   - 'edit terms in thematiques' | ||||
| @@ -65,6 +70,7 @@ permissions: | ||||
|   - 'revert etape revisions' | ||||
|   - 'revert gallerie_photo revisions' | ||||
|   - 'revert partenaire revisions' | ||||
|   - 'revert ressource revisions' | ||||
|   - 'revert static revisions' | ||||
|   - 'use text format footnote' | ||||
|   - 'use text format simple' | ||||
| @@ -73,6 +79,7 @@ permissions: | ||||
|   - 'view etape revisions' | ||||
|   - 'view gallerie_photo revisions' | ||||
|   - 'view partenaire revisions' | ||||
|   - 'view ressource revisions' | ||||
|   - 'view static revisions' | ||||
|   - 'view the administration theme' | ||||
|   - 'view unpublished paragraphs' | ||||
|   | ||||
| @@ -284,7 +284,7 @@ display: | ||||
|             popupAnchor: | ||||
|               x: '' | ||||
|               'y': '' | ||||
|             html: "<div></div>\r\n<div></div>\r\n<div></div>\r\n<div class=\"url\">[node:url]</div>\r\n<div class=\"couleur\">[node:field_couleur]</div>" | ||||
|             html: "<div class=\"url\">[node:url]</div>\r\n<div class=\"couleur\">[node:field_couleur]</div>" | ||||
|             html_class: 'leaflet-map-divicon ' | ||||
|             circle_marker_options: '{"radius":100,"color":"red","fillColor":"#f03","fillOpacity":0.5}' | ||||
|           leaflet_markercluster: | ||||
| @@ -299,6 +299,7 @@ display: | ||||
|           geocoder: | ||||
|             control: false | ||||
|             settings: | ||||
|               popup: false | ||||
|               autocomplete: | ||||
|                 placeholder: 'Search Address' | ||||
|                 title: 'Search an Address on the Map' | ||||
| @@ -311,7 +312,6 @@ display: | ||||
|               min_terms: 4 | ||||
|               delay: 800 | ||||
|               zoom: 16 | ||||
|               popup: false | ||||
|               options: '' | ||||
|           map_lazy_load: | ||||
|             lazy_load: true | ||||
| @@ -1195,7 +1195,18 @@ display: | ||||
|       defaults: | ||||
|         fields: false | ||||
|       display_description: '' | ||||
|       display_extenders: {  } | ||||
|       display_extenders: | ||||
|         matomo: | ||||
|           enabled: false | ||||
|           keyword_gets: '' | ||||
|           keyword_behavior: first | ||||
|           keyword_concat_separator: ' ' | ||||
|           category_behavior: none | ||||
|           category_gets: '' | ||||
|           category_concat_separator: ' ' | ||||
|           category_fallback: '' | ||||
|           category_facets: {  } | ||||
|           category_facets_concat_separator: ', ' | ||||
|     cache_metadata: | ||||
|       max-age: -1 | ||||
|       contexts: | ||||
|   | ||||
| @@ -200,7 +200,7 @@ display: | ||||
|         type: normal | ||||
|         title: 'Les partenaires' | ||||
|         description: '' | ||||
|         weight: -48 | ||||
|         weight: -47 | ||||
|         expanded: false | ||||
|         menu_name: main | ||||
|         parent: '' | ||||
|   | ||||
							
								
								
									
										4
									
								
								readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| ## Routing flow | ||||
|  | ||||
|  | ||||
|  | ||||
							
								
								
									
										1
									
								
								testpush.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								testpush.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| juste pour tester un git push ssh | ||||
 Submodule web/profiles/drupal-starterkit-profile updated: 9eab6ebc8d...18c6b02464
									
								
							| @@ -1,6 +1,6 @@ | ||||
| import { initVueContentModale } from './utils/vue-setup'; | ||||
| import { processClickableElements } from './utils/process-clickable-elements'; | ||||
| import { handleReactiveness, setMenuToggle, setHamburgerWhenLogged } from './utils/layout-setup'; | ||||
| import { handleReactiveness, setMenuToggle, setRightSectionsWhenLogged } from './utils/layout-setup'; | ||||
| import { initFirstLoadRouting, handleClickableElements, handleBrowserNavigation } from './utils/handle-navigation'; | ||||
| import { setupMapStore, preloadEtapesTiles } from './utils/map-setup'; | ||||
|  | ||||
| @@ -17,11 +17,11 @@ import '../scss/main.scss' | ||||
|              | ||||
|             const baseUrl = window.location.protocol + "//" + window.location.host; | ||||
|             const siteName = document.querySelector('#site_name').innerText; | ||||
|             const { store, mapStore, router, route } = initVueContentModale(); | ||||
|             const router = initVueContentModale(); | ||||
|  | ||||
|             handleReactiveness(); | ||||
|             setMenuToggle(); | ||||
|             setHamburgerWhenLogged(drupalSettings); | ||||
|             setRightSectionsWhenLogged(drupalSettings); | ||||
|  | ||||
|             // https://www.drupal.org/docs/extending-drupal/contributed-modules/contributed-module-documentation/leaflet/leaflet-api | ||||
|  | ||||
| @@ -39,25 +39,25 @@ import '../scss/main.scss' | ||||
|                             generalListLinks, | ||||
|                             logoLink, | ||||
|                             mapIcons, | ||||
|                             mapContainer, | ||||
|                         } = processClickableElements(); | ||||
|                         const clickableElements = [...etapeListLinks, ...generalListLinks, logoLink, ...mapIcons]; | ||||
|                         const clickableElements = [...etapeListLinks, ...generalListLinks, logoLink, ...mapIcons, mapContainer];                         | ||||
|  | ||||
|                         setupMapStore(mapStore, map, settings); | ||||
|                         setupMapStore(map, settings); | ||||
|  | ||||
|                         // preloadEtapesTiles(mapStore, map); | ||||
|                         // preloadEtapesTiles(map); | ||||
|  | ||||
|                         initFirstLoadRouting(store, router, baseUrl, siteName); | ||||
|                         initFirstLoadRouting(router, baseUrl, siteName); | ||||
|  | ||||
|                         handleClickableElements(clickableElements, store, router, baseUrl, siteName, mapStore); | ||||
|                         handleClickableElements(clickableElements, router, baseUrl, siteName); | ||||
|  | ||||
|                         window.addEventListener("popstate", () => { | ||||
|                           handleBrowserNavigation(store, baseUrl, siteName, mapStore); | ||||
|                           handleBrowserNavigation(baseUrl, siteName); | ||||
|                         }); | ||||
|                     }); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         init() | ||||
|     } | ||||
|     CaravaneTheme() | ||||
|   | ||||
| @@ -1,616 +1,80 @@ | ||||
| // query et traitement des contenus | ||||
|  | ||||
| import { defineStore } from 'pinia'; | ||||
| import REST from '../api/rest-axios'; | ||||
|  | ||||
| import { useLayoutStore } from './layout'; | ||||
| import { findContentByPath } from '../utils/content/findContentByPath'; | ||||
| import { fetchSingletonFullContent, fetchSingletonPartialContent } from '../utils/content/fetchSingleton'; | ||||
| import { fetchMultipleFullContent, fetchMultiplePartialContent } from '../utils/content/fetchMultiple'; | ||||
|  | ||||
| export const useContentStore = defineStore('content', { | ||||
|     state: () => ({ | ||||
|         contentType: '', | ||||
|         rawContent: {}, | ||||
|         pageTitle: '', | ||||
|         content: { | ||||
|             contentTitle: '', | ||||
|             coordinates: {}, | ||||
|             adresse: {}, | ||||
|             etape_number: '', | ||||
|             couleur: '', | ||||
|             dates: {}, | ||||
|             previous : {}, | ||||
|             next: {}, | ||||
|             vignette: {}, | ||||
|             parties: [], | ||||
|             liens: [], | ||||
|             pieces_jointes: [], | ||||
|  | ||||
|             intro: '', | ||||
|             partenaires: [], | ||||
|             gouvernances: [], | ||||
|         }, | ||||
|         content: {}, | ||||
|         partialLoading: false, | ||||
|         loading: false, | ||||
|         error: null, | ||||
|     }), | ||||
|     actions: { | ||||
|         async fetchContentData(path) { | ||||
|             console.log('start fetch content'); | ||||
|         // pages etape, statiques et ressource ont un seul item par page (singuliers) | ||||
|         // pages gouvernance (contact), ressources et partenaire ont plusieurs items par pages (multiples) | ||||
|         async fetchPartialContentData(path) { | ||||
|             this.resetStore(false); | ||||
|             const contentTypes = [ 'etape', 'static', 'gouvernance', 'partenaire' ]; | ||||
|             const contentTypes = ['etape', 'static', 'gouvernance', 'partenaire', 'ressource']; | ||||
|  | ||||
|             try { | ||||
|                 const findContentByPath = async (contentTypes, path) => { | ||||
|                     for (let type of contentTypes) { | ||||
|                         const response = await REST.get(`/jsonapi/node/${type}/`); | ||||
|                          | ||||
|                         const content = response.data.data.find(content =>  | ||||
|                             content.attributes.metatag.some(tag =>  | ||||
|                                 tag.tag === "link" && tag.attributes.href === path | ||||
|                             ) | ||||
|                         ); | ||||
|                      | ||||
|                         if (content) { | ||||
|                             console.log('content found'); | ||||
|                              | ||||
|                             return { | ||||
|                                 contentType: type, | ||||
|                                 rawContent: content, | ||||
|                             }; | ||||
|                         } | ||||
|                      | ||||
|                         // Handle special case for governance/partners (multiple items per page) | ||||
|                         const pageRequested = window.location.href.split('/').pop().replace(/s?$/, ''); | ||||
|                         if (type === pageRequested | ||||
|                             || (type === 'gouvernance' && pageRequested === 'contact') | ||||
|                         ) { | ||||
|                             return { | ||||
|                                 contentType: type, | ||||
|                                 rawContent: response.data.data, | ||||
|                             }; | ||||
|                         } | ||||
|                     } | ||||
|                     return null; | ||||
|                 };                 | ||||
|                  | ||||
|                 const { contentType, rawContent } = await findContentByPath(contentTypes, path); | ||||
|                 this.contentType = contentType; | ||||
|                 this.rawContent = rawContent; | ||||
|  | ||||
|                 // console.log(`current type: ${contentType}`); | ||||
|  | ||||
|                 // TO DEBUG | ||||
|                 const cleanContentMethod = 'original'; | ||||
|                  | ||||
|                 if (this.contentType !== 'gouvernance' && this.contentType !== 'partenaire' | ||||
|                     && cleanContentMethod === 'original' | ||||
|                 if ( | ||||
|                     this.contentType === 'etape' | ||||
|                     || this.contentType === 'static' | ||||
|                     || this.contentType === 'ressourceItem' | ||||
|                 ) { | ||||
|                     console.time('etape content processing'); | ||||
|                     // pageTitle | ||||
|                     for (let tag of rawContent.attributes.metatag) { | ||||
|                         if (tag.tag === "meta") { | ||||
|                             this.pageTitle = tag.attributes.content; | ||||
|                             break; | ||||
|                         } | ||||
|                     } | ||||
|                     // contentTitle | ||||
|                     this.content.contentTitle = rawContent.attributes.title; | ||||
|  | ||||
|                     // vignette | ||||
|                     const vignetteFetch = await this.fetchFromRelationships('field_vignette', rawContent.relationships);                     | ||||
|                     if (vignetteFetch) { | ||||
|                         this.content.vignette = { | ||||
|                             url: { | ||||
|                                 original: vignetteFetch.attributes.uri.url, | ||||
|                                 small: vignetteFetch.attributes.image_style_uri.content_small, | ||||
|                                 medium: vignetteFetch.attributes.image_style_uri.content_medium, | ||||
|                                 large: vignetteFetch.attributes.image_style_uri.content_large, | ||||
|                             }, | ||||
|                             alt: rawContent.relationships.field_vignette.data.meta.alt | ||||
|                         }; | ||||
|                     } | ||||
|  | ||||
|                     // liens | ||||
|                     if (rawContent.attributes.field_liens.length) { | ||||
|                         this.content.liens = []; | ||||
|                         for (let lien of rawContent.attributes.field_liens) { | ||||
|                             this.content.liens.push({ | ||||
|                                 title: lien.title, | ||||
|                                 url: lien.uri, | ||||
|                             }); | ||||
|                         }                             | ||||
|                     } | ||||
|                     // pièces jointes | ||||
|                     if (rawContent.relationships.field_pieces_jointes.data.length) { | ||||
|                         this.content.pieces_jointes = []; | ||||
|                         for (let pieceJointe of rawContent.relationships.field_pieces_jointes.data) { | ||||
|                             if (pieceJointe.meta.display) { | ||||
|                                 const uuid = pieceJointe.id; | ||||
|                                 const response = await REST.get(`/jsonapi/file/file/${uuid}`); | ||||
|                                 this.content.pieces_jointes.push({ | ||||
|                                     title: pieceJointe.meta.description, | ||||
|                                     url: response.data.data.attributes.uri.url, | ||||
|                                 }); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     if (contentType === 'etape') { | ||||
|                         // coordinates | ||||
|                         this.content.coordinates = { | ||||
|                             lat: rawContent.attributes.field_geofield.lat, | ||||
|                             lon: rawContent.attributes.field_geofield.lon, | ||||
|                         }; | ||||
|                         // adresse | ||||
|                         this.content.adresse = rawContent.attributes.field_adresse; | ||||
|                         // étape number | ||||
|                         this.content.etape_number = rawContent.attributes.field_arret_numero; | ||||
|                         // couleur | ||||
|                         this.content.couleur = rawContent.attributes.field_couleur; | ||||
|                         // dates | ||||
|                         this.content.dates = { | ||||
|                             start: this.getCleanDate(rawContent.attributes.field_dates.value), | ||||
|                             end: this.getCleanDate(rawContent.attributes.field_dates.end_value), | ||||
|                         } | ||||
|                         // previous / next | ||||
|                         await this.getRelatedEtape('previous', path); | ||||
|                         await this.getRelatedEtape('next', path); | ||||
|                     } | ||||
|  | ||||
|                     // parties | ||||
|                     const fieldParties = contentType === 'etape' ? 'field_parties' : 'field_parties_static'; | ||||
|                     const partiesFetch = await this.fetchFromRelationships(fieldParties, rawContent.relationships); | ||||
|                     console.log(this.content); | ||||
|                     console.timeEnd('etape content processing'); | ||||
|                     if (partiesFetch) { | ||||
|                         this.content.parties = []; | ||||
|                         for (let partie of partiesFetch) { | ||||
|                             const partieType = partie.type.replace(/^paragraph--/, ""); | ||||
|                             let partieContent = { | ||||
|                                 type: partieType, | ||||
|                             }; | ||||
|  | ||||
|                             switch (partieType) { | ||||
|                                 case 'carte_sensible': | ||||
|                                     const carteSensibleFetch = await this.fetchFromRelationships('field_image_carte', partie.relationships); | ||||
|                                     if (carteSensibleFetch) { | ||||
|                                         partieContent.carteSensible = { | ||||
|                                             url: { | ||||
|                                                 original: carteSensibleFetch.attributes.uri.url, | ||||
|                                                 small: carteSensibleFetch.attributes.image_style_uri.content_small, | ||||
|                                                 medium: carteSensibleFetch.attributes.image_style_uri.content_medium, | ||||
|                                                 large: carteSensibleFetch.attributes.image_style_uri.content_large, | ||||
|                                                 xlarge: carteSensibleFetch.attributes.image_style_uri.content_x_large, | ||||
|                                             }, | ||||
|                                             alt: partie.relationships.field_image_carte.data.meta.alt, | ||||
|                                         }; | ||||
|                                     } | ||||
|                                     break; | ||||
|                                 case 'titre_texte': | ||||
|                                     console.time('get images from text original') | ||||
|                                     partieContent.titre = partie.attributes.field_titre; | ||||
|                                     partieContent.texte = partie.attributes.field_texte.value;                                     | ||||
|  | ||||
|                                     // get the resized images from the text | ||||
|                                     const imgRegex = /<img[^>]+>/g; | ||||
|                                     const uuidRegex = /data-entity-uuid="([^"]+)"/; | ||||
|  | ||||
|                                     const imgTags = partieContent.texte.match(imgRegex); | ||||
|  | ||||
|                                     if (imgTags) { | ||||
|                                         for (const imgTag of imgTags) { | ||||
|                                             const uuidMatch = imgTag.match(uuidRegex); | ||||
|                                             if (uuidMatch && uuidMatch[1]) { | ||||
|                                                 const uuid = uuidMatch[1]; | ||||
|  | ||||
|                                                 const response = await REST.get(`/jsonapi/file/file/${uuid}`); | ||||
|                                                 const imagesFetch = response.data.data; | ||||
|  | ||||
|                                                 const newImgTag = imgTag | ||||
|                                                     .replace(/src="[^"]+"/,`src="${imagesFetch.attributes.image_style_uri.content_medium}"`) | ||||
|                                                     .replace('>',' data-large-src="' + imagesFetch.attributes.image_style_uri.content_large + '">'); | ||||
|                                                 partieContent.texte = partieContent.texte.replace(imgTag, newImgTag); | ||||
|                                             } | ||||
|                                         } | ||||
|                                     } | ||||
|                                      | ||||
|                                      | ||||
|                                     console.timeEnd('get images from text original') | ||||
|                                     break; | ||||
|                                 case 'chiffres_cles': | ||||
|                                     const chiffresClesFetch = await this.fetchFromRelationships('field_chiffres_clefs', partie.relationships); | ||||
|                                     if (chiffresClesFetch) { | ||||
|                                         partieContent.chiffresCles = []; | ||||
|                                         for (let chiffre of chiffresClesFetch) { | ||||
|                                             partieContent.chiffresCles.push({ | ||||
|                                                 chiffre: chiffre.attributes.field_chiffre, | ||||
|                                                 description: chiffre.attributes.field_description, | ||||
|                                             }); | ||||
|                                         } | ||||
|                                     } | ||||
|                                     break; | ||||
|                                 case 'diaporama': | ||||
|                                     const diaporamaFetch = await this.fetchFromRelationships('field_diaporama', partie.relationships); | ||||
|                                      | ||||
|                                     if (diaporamaFetch) { | ||||
|                                         partieContent.diaporama = []; | ||||
|                                         for (let [index, image] of diaporamaFetch.entries()) { | ||||
|                                             partieContent.diaporama.push({ | ||||
|                                                 url: { | ||||
|                                                     original: image.attributes.uri.url, | ||||
|                                                     small: image.attributes.image_style_uri.content_small, | ||||
|                                                     medium: image.attributes.image_style_uri.content_medium, | ||||
|                                                     large: image.attributes.image_style_uri.content_large, | ||||
|                                                 }, | ||||
|                                                 alt: partie.relationships.field_diaporama.data[index].meta.alt, | ||||
|                                             });                                            | ||||
|                                         } | ||||
|                                          | ||||
|                                     } | ||||
|                                     break; | ||||
|                                 case 'entretien': | ||||
|                                     partieContent.entretien = {}; | ||||
|                                     partieContent.entretien.titre = partie.attributes.field_titre; | ||||
|                                     const personnesFetch = await this.fetchFromRelationships('field_personne_s', partie.relationships); | ||||
|                                     const questionsReponsesFetch = await this.fetchFromRelationships('field_questions_reponses', partie.relationships); | ||||
|                                     if (personnesFetch && questionsReponsesFetch) { | ||||
|                                         partieContent.entretien.personnes = []; | ||||
|                                         for (let personne of personnesFetch) { | ||||
|                                             const portraitFetch = await this.fetchFromRelationships('field_portrait', personne.relationships); | ||||
|                                             if (portraitFetch) { | ||||
|                                                 partieContent.entretien.personnes.push({ | ||||
|                                                     portrait: { | ||||
|                                                         original: portraitFetch.attributes.uri.url, | ||||
|                                                         small: portraitFetch.attributes.image_style_uri.content_small, | ||||
|                                                         medium: portraitFetch.attributes.image_style_uri.content_medium, | ||||
|                                                         large: portraitFetch.attributes.image_style_uri.content_large, | ||||
|                                                     }, | ||||
|                                                     alt: personne.relationships.field_portrait.data.meta.alt, | ||||
|                                                     description: personne.attributes.field_description, | ||||
|                                                 }); | ||||
|                                             } | ||||
|                                         } | ||||
|                                         partieContent.entretien.questionsReponses = []; | ||||
|                                         for (let qr of questionsReponsesFetch) { | ||||
|                                             partieContent.entretien.questionsReponses.push({ | ||||
|                                                 question: qr.attributes.field_question, | ||||
|                                                 reponse: qr.attributes.field_reponse.value, | ||||
|                                             }); | ||||
|                                         } | ||||
|                                     } | ||||
|                                     break; | ||||
|                                 case 'exergue': | ||||
|                                     partieContent.exergue = partie.attributes.field_texte_exergue.value; | ||||
|                                     break; | ||||
|                                 case 'video': | ||||
|                                     partieContent.videos = []; | ||||
|                                     for (let video of partie.attributes.field_videos) { | ||||
|                                         const videoId = video.split('?v=')[1]; | ||||
|                                         const videoUrl = `https://www.youtube.com/embed/${videoId}`; | ||||
|                                         partieContent.videos.push(videoUrl); | ||||
|                                     } | ||||
|                                     break; | ||||
|                             } | ||||
|                             this.content.parties.push(partieContent); | ||||
|                         } | ||||
|                     } | ||||
|                     console.log('content cleaned'); | ||||
|                 } else if (this.contentType !== 'gouvernance' && this.contentType !== 'partenaire' | ||||
|                     && cleanContentMethod === 'promise' | ||||
|                 ) { | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|                     console.time('etape content promise processing'); | ||||
|  | ||||
|  | ||||
|                     const vignettePromise = this.fetchFromRelationships('field_vignette', rawContent.relationships); | ||||
|                     const partiesPromise = this.fetchFromRelationships('field_parties', rawContent.relationships); | ||||
|  | ||||
|                     const previousEtapePromise = contentType === 'etape'? this.getRelatedEtape('previous', path) : null; | ||||
|                     const nextEtapePromise = contentType === 'etape' ? this.getRelatedEtape('next', path) : null; | ||||
|  | ||||
|                     if (contentType === 'etape') { | ||||
|                         // coordinates | ||||
|                         this.content.coordinates = { | ||||
|                             lat: rawContent.attributes.field_geofield.lat, | ||||
|                             lon: rawContent.attributes.field_geofield.lon, | ||||
|                         }; | ||||
|                         // adresse | ||||
|                         this.content.adresse = rawContent.attributes.field_adresse; | ||||
|                         // étape number | ||||
|                         this.content.etape_number = rawContent.attributes.field_arret_numero; | ||||
|                         // couleur | ||||
|                         this.content.couleur = rawContent.attributes.field_couleur; | ||||
|                         // dates | ||||
|                         this.content.dates = { | ||||
|                             start: this.getCleanDate(rawContent.attributes.field_dates.value), | ||||
|                             end: this.getCleanDate(rawContent.attributes.field_dates.end_value), | ||||
|                         }                         | ||||
|                     } | ||||
|  | ||||
|                     // pageTitle | ||||
|                     for (let tag of rawContent.attributes.metatag) { | ||||
|                         if (tag.tag === "meta") { | ||||
|                             this.pageTitle = tag.attributes.content; | ||||
|                             break; | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     // contentTitle | ||||
|                     this.content.contentTitle = rawContent.attributes.title; | ||||
|  | ||||
|                     const [vignetteData, partiesData] = await Promise.all([vignettePromise, partiesPromise]); | ||||
|  | ||||
|                     if (vignetteData) { | ||||
|                         this.content.vignette = { | ||||
|                             url: { | ||||
|                                 original: vignetteData.attributes.uri.url, | ||||
|                                 small: vignetteData.attributes.image_style_uri.content_small, | ||||
|                                 medium: vignetteData.attributes.image_style_uri.content_medium, | ||||
|                                 large: vignetteData.attributes.image_style_uri.content_large, | ||||
|                             }, | ||||
|                             alt: rawContent.relationships.field_vignette.data.meta.alt | ||||
|                         }; | ||||
|                     } | ||||
|  | ||||
|                     if (partiesData) { | ||||
|                         const partiesPromises = partiesData.map(async (partie) => { | ||||
|                             const partieType = partie.type.replace(/^paragraph--/, ""); | ||||
|                             let partieContent = { type: partieType }; | ||||
|  | ||||
|                             switch(partieType) { | ||||
|                                 case 'carte_sensible': | ||||
|                                     const carteSensiblePromise = this.fetchFromRelationships('field_image_carte', partie.relationships); | ||||
|  | ||||
|                                     const carteSensibleData = await carteSensiblePromise; | ||||
|                                     if (carteSensibleData) { | ||||
|                                         partieContent.carteSensible = { | ||||
|                                             url: { | ||||
|                                                 original: carteSensibleData.attributes.uri.url, | ||||
|                                                 small: carteSensibleData.attributes.image_style_uri.content_small, | ||||
|                                                 medium: carteSensibleData.attributes.image_style_uri.content_medium, | ||||
|                                                 large: carteSensibleData.attributes.image_style_uri.content_large, | ||||
|                                                 xlarge: carteSensibleData.attributes.image_style_uri.content_x_large, | ||||
|                                             }, | ||||
|                                             alt: partie.relationships.field_image_carte.data.meta.alt, | ||||
|                                         }; | ||||
|                                     } | ||||
|                                      | ||||
|                                     break; | ||||
|                                 case 'titre_texte': | ||||
|                                     partieContent.titre = partie.attributes.field_titre; | ||||
|                                     partieContent.texte = partie.attributes.field_texte.value;                                     | ||||
|                                      | ||||
|                                     // get the resized images from the text | ||||
|                                     const imgRegex = /<img[^>]+>/g; | ||||
|                                     const uuidRegex = /data-entity-uuid="([^"]+)"/; | ||||
|                                     const imgTags = partieContent.texte.match(imgRegex); | ||||
|                                      | ||||
|                                     if (imgTags) { | ||||
|                                         const imagePromises = imgTags.map(imgTag => { | ||||
|                                             const uuidMatch = imgTag.match(uuidRegex); | ||||
|                                             if (uuidMatch && uuidMatch[1]) { | ||||
|                                                 return REST.get(`/jsonapi/file/file/${uuidMatch[1]}`) | ||||
|                                                     .then(response => ({ | ||||
|                                                         originalTag: imgTag, | ||||
|                                                         imageData: response.data.data | ||||
|                                                     })); | ||||
|                                             } | ||||
|                                         }); | ||||
|  | ||||
|                                         const images = await Promise.all(imagePromises); | ||||
|                                         images.forEach(({originalTag, imageData}) => { | ||||
|                                             const newImgTag = originalTag | ||||
|                                                 .replace(/src="[^"]+"/,`src="${imageData.attributes.image_style_uri.content_medium}"`) | ||||
|                                                 .replace('>',' data-large-src="' + imageData.attributes.image_style_uri.content_large + '">'); | ||||
|                                             partieContent.texte = partieContent.texte.replace(originalTag, newImgTag); | ||||
|                                         }); | ||||
|                                     } | ||||
|                                     break; | ||||
|                                 case 'chiffres_cles': | ||||
|                                     const chiffresClesFetch = await this.fetchFromRelationships('field_chiffres_clefs', partie.relationships); | ||||
|                                     if (chiffresClesFetch) { | ||||
|                                         partieContent.chiffresCles = []; | ||||
|                                         for (let chiffre of chiffresClesFetch) { | ||||
|                                             partieContent.chiffresCles.push({ | ||||
|                                                 chiffre: chiffre.attributes.field_chiffre, | ||||
|                                                 description: chiffre.attributes.field_description, | ||||
|                                             }); | ||||
|                                         } | ||||
|                                     } | ||||
|                                     break; | ||||
|                                 case 'diaporama': | ||||
|                                     break; | ||||
|                                 case 'entretien': | ||||
|                                     break; | ||||
|                                 case 'exergue': | ||||
|                                     break; | ||||
|                                 case 'video': | ||||
|                                     break; | ||||
|                             } | ||||
|                             return partieContent; | ||||
|                         }); | ||||
|  | ||||
|                         this.content.parties = await Promise.all(partiesPromises); | ||||
|                     } | ||||
|  | ||||
|                     // related étapes | ||||
|                     if (contentType === 'etape') await Promise.all([previousEtapePromise, nextEtapePromise]); | ||||
|  | ||||
|                     console.log(this.content); | ||||
|                     console.timeEnd('etape content promise processing'); | ||||
|                     let { pageTitle, partialContent } = fetchSingletonPartialContent(this.contentType, this.rawContent); | ||||
|                     this.pageTitle = pageTitle; | ||||
|                     this.content = partialContent; | ||||
|                 } else { | ||||
|                     // pages gouvernance (contact) et partenaire | ||||
|                     // ont plusieurs items par pages | ||||
|                     const intro = await REST.get(`/jsonapi/config_pages/intro_${this.contentType}/`); | ||||
|                     const introContent = intro.data.data[0]; | ||||
|  | ||||
|                     this.pageTitle =  | ||||
|                         `${introContent.attributes.field_titre} ${introContent.attributes.metatag.find(tag => tag.tag === "meta")?.attributes.content}`; | ||||
|  | ||||
|                     this.content.contentTitle = introContent.attributes.field_titre; | ||||
|                     this.content.intro = introContent.attributes.field_intro?.value; | ||||
|  | ||||
|                     const multiItemPageArray = []; | ||||
|  | ||||
|                     if (this.contentType === 'partenaire') { | ||||
|                         for (let item of rawContent) {        | ||||
|                             const logoFetch = await REST.get(item.relationships.field_logo.links.related.href); | ||||
|      | ||||
|                             multiItemPageArray.push({ | ||||
|                                 title: item.attributes.title, | ||||
|                                 description: item.attributes.body.value, | ||||
|                                 weight: item.attributes.field_poid, | ||||
|                                 link_url: item.attributes.field_lien.uri , | ||||
|                                 logo_alt: item.relationships.field_logo.data.meta.alt, | ||||
|                                 logo_url: { | ||||
|                                     original: logoFetch.data.data.attributes.uri.url, | ||||
|                                     small: logoFetch.data.data.attributes.image_style_uri.content_small, | ||||
|                                     medium: logoFetch.data.data.attributes.image_style_uri.content_medium, | ||||
|                                     large: logoFetch.data.data.attributes.image_style_uri.content_large, | ||||
|                                 }, | ||||
|                             }); | ||||
|                         } | ||||
|                     } else if (this.contentType === 'gouvernance') { | ||||
|                         for (let item of rawContent) { | ||||
|                             const personnesFetch = await REST.get(item.relationships.field_personne_s.links.related.href); | ||||
|                             let personnes = []; | ||||
|                             for (let personne of personnesFetch.data.data) { | ||||
|                                 const portraitFetch = await REST.get(personne.relationships.field_portrait.links.related.href); | ||||
|                                 personnes.push({ | ||||
|                                     nom: personne.attributes.field_nom, | ||||
|                                     prenom: personne.attributes.field_prenom, | ||||
|                                     description: personne.attributes.field_description, | ||||
|                                     photo_meta: personne.relationships.field_portrait.data?.meta.alt, | ||||
|                                     photo_url: portraitFetch.data.data ? { | ||||
|                                         original: portraitFetch.data.data.attributes.uri.url, | ||||
|                                         small: portraitFetch.data.data.attributes.image_style_uri.content_small, | ||||
|                                         medium: portraitFetch.data.data.attributes.image_style_uri.content_medium, | ||||
|                                         large: portraitFetch.data.data.attributes.image_style_uri.content_large, | ||||
|                                     } : null | ||||
|                                 }); | ||||
|                             } | ||||
|                             multiItemPageArray.push({ | ||||
|                                 title: item.attributes.title, | ||||
|                                 weight: item.attributes.field_poid, | ||||
|                                 personnes: personnes, | ||||
|                             }); | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     this.content[`${this.contentType}s`] = multiItemPageArray; | ||||
|                      | ||||
|                     // console.log(this.content); | ||||
|                     let { pageTitle, partialContent } = await fetchMultiplePartialContent(this.contentType); | ||||
|                     this.pageTitle = pageTitle; | ||||
|                     this.content = partialContent; | ||||
|                 } | ||||
|             } catch (error) { | ||||
|                 this.error = 'Failed to fetch data'; | ||||
|                 console.error('Issue with getNodeData', error); | ||||
|             } catch(error) { | ||||
|                 this.error = 'Failed to fetch partial data'; | ||||
|                 console.error('Issue with fetchPartialContentData', error); | ||||
|             } finally { | ||||
|                 this.partialLoading = false; | ||||
|             } | ||||
|         }, | ||||
|         async fetchFullContentData(path) { | ||||
|             try { | ||||
|                 if ( | ||||
|                     this.contentType === 'etape' | ||||
|                     || this.contentType === 'static' | ||||
|                     || this.contentType === 'ressourceItem' | ||||
|                 ) { | ||||
|                     this.content = { ...this.content, ...await fetchSingletonFullContent(this.contentType, this.rawContent, path) }; | ||||
|                 } else { | ||||
|                     this.content = { ...this.content, ...await fetchMultipleFullContent(this.contentType, this.rawContent) }; | ||||
|                 } | ||||
|             } catch(error) { | ||||
|                 this.error = 'Failed to fetch full data'; | ||||
|                 console.error('Issue with fetchFullContentData', error); | ||||
|             } finally { | ||||
|                 this.loading = false; | ||||
|             } | ||||
|         }, | ||||
|         getCleanDate(date) { | ||||
|             return { | ||||
|               d: date.split('-')[2], | ||||
|               m: new Intl.DateTimeFormat('fr-FR', { month: 'long' }).format(new Date(date)), | ||||
|               y: date.split('-')[0], | ||||
|             } | ||||
|         }, | ||||
|         async getRelatedEtape(direction, path) { | ||||
|              const getRelatedEtapeContent = async (relatedEtapeData) => { | ||||
|                 if (relatedEtapeData) { | ||||
|                     const vignetteFetch = await REST.get(relatedEtapeData.relationships.field_vignette.links.related.href); | ||||
|                     if (vignetteFetch.data.data) { | ||||
|                         this.content[direction] = { | ||||
|                             url: relatedEtapeData.attributes.metatag.find(tag => tag.tag === "link")?.attributes.href, | ||||
|                             couleur: relatedEtapeData.attributes.field_couleur, | ||||
|                             title: relatedEtapeData.attributes.title, | ||||
|                             postalCode: relatedEtapeData.attributes.field_adresse.postal_code, | ||||
|                             dates: { | ||||
|                                 start: this.getCleanDate(relatedEtapeData.attributes.field_dates.value), | ||||
|                                 end: this.getCleanDate(relatedEtapeData.attributes.field_dates.end_value), | ||||
|                             }, | ||||
|                             vignette: { | ||||
|                                 url: { | ||||
|                                     original: vignetteFetch.data.data.attributes.uri.url, | ||||
|                                     small: vignetteFetch.data.data.attributes.image_style_uri.content_small, | ||||
|                                     medium: vignetteFetch.data.data.attributes.image_style_uri.content_medium, | ||||
|                                     large: vignetteFetch.data.data.attributes.image_style_uri.content_large, | ||||
|                                 }, | ||||
|                                 alt: relatedEtapeData.relationships.field_vignette.data.meta.alt, | ||||
|                             }, | ||||
|                         } | ||||
|                     } | ||||
|                      | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             const allEtapesData = await REST.get('/jsonapi/views/etapes/block_1/'); | ||||
|             for (let [index, etape] of allEtapesData.data.data.entries()) { | ||||
|                 if (etape.attributes.metatag.some(tag =>  | ||||
|                     tag.tag === "link" && tag.attributes.href === path | ||||
|                 )) { | ||||
|                     const relatedEtapeIndex = direction === 'next' ? index + 1 : index - 1; | ||||
|                     await getRelatedEtapeContent(allEtapesData.data.data[relatedEtapeIndex]); | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         async getRelatedEtape(direction, path) { | ||||
|             const getRelatedEtapeContent = (relatedEtapeData) => { | ||||
|                 if (relatedEtapeData) { | ||||
|                     return this.fetchFromRelationships('field_vignette', relatedEtapeData.relationships) | ||||
|                         .then(vignetteFetch => { | ||||
|                             if (vignetteFetch) { | ||||
|                                 this.content[direction] = { | ||||
|                                     url: relatedEtapeData.attributes.metatag.find(tag => tag.tag === "link")?.attributes.href, | ||||
|                                     couleur: relatedEtapeData.attributes.field_couleur, | ||||
|                                     title: relatedEtapeData.attributes.title, | ||||
|                                     postalCode: relatedEtapeData.attributes.field_adresse.postal_code, | ||||
|                                     dates: { | ||||
|                                         start: this.getCleanDate(relatedEtapeData.attributes.field_dates.value), | ||||
|                                         end: this.getCleanDate(relatedEtapeData.attributes.field_dates.end_value), | ||||
|                                     }, | ||||
|                                     vignette: { | ||||
|                                         url: { | ||||
|                                             original: vignetteFetch.attributes.uri.url, | ||||
|                                             small: vignetteFetch.attributes.image_style_uri.content_small, | ||||
|                                             medium: vignetteFetch.attributes.image_style_uri.content_medium, | ||||
|                                             large: vignetteFetch.attributes.image_style_uri.content_large, | ||||
|                                         }, | ||||
|                                         alt: relatedEtapeData.relationships.field_vignette.data.meta.alt, | ||||
|                                     }, | ||||
|                                 } | ||||
|                             } | ||||
|                         }); | ||||
|                 } | ||||
|             } | ||||
|          | ||||
|             const allEtapesPromise = REST.get('/jsonapi/views/etapes/block_1/'); | ||||
|              | ||||
|             return allEtapesPromise.then(allEtapesData => { | ||||
|                 for (let [index, etape] of allEtapesData.data.data.entries()) { | ||||
|                     if (etape.attributes.metatag.some(tag =>  | ||||
|                         tag.tag === "link" && tag.attributes.href === path | ||||
|                     )) { | ||||
|                         const relatedEtapeIndex = direction === 'next' ? index + 1 : index - 1; | ||||
|                         return getRelatedEtapeContent(allEtapesData.data.data[relatedEtapeIndex]); | ||||
|                     } | ||||
|                 } | ||||
|             }); | ||||
|         }, | ||||
|         async fetchFromRelationships(field, relationships) { | ||||
|             if (relationships[field].links) { | ||||
|                 const contentLink = relationships[field].links.related.href; | ||||
|                 return REST.get(contentLink) | ||||
|                     .then(contentFetch => contentFetch.data.data) | ||||
|                     .catch(error => { | ||||
|                         this.error = 'Failed to fetch data'; | ||||
|                         console.error('Issue with getNodeData', error); | ||||
|                     }); | ||||
|             } | ||||
|             return null; | ||||
|         },         | ||||
|         resetStore(forFrontDisplay) { | ||||
|             this.contentType = ''; | ||||
|             this.pageTitle = ''; | ||||
|             this.content = {}; | ||||
|             this.partialLoading = !forFrontDisplay; | ||||
|             this.loading = !forFrontDisplay; | ||||
|             this.error = null; | ||||
|             useLayoutStore().hideEtapeList(false); | ||||
|         } | ||||
|     }, | ||||
| }); | ||||
|   | ||||
| @@ -46,6 +46,30 @@ export const useLayoutStore = defineStore('layout', { | ||||
|  | ||||
|           this.toggleEtapeListScroll(isIntersecting, listeEtape, column, headerRect.height, animationToggleRect.top); | ||||
|         }, | ||||
|         hideEtapeList(souldListHide) { | ||||
|           const etapeList = document.querySelector(window.innerWidth >= this.minDesktopWidth ? '#etapes-liste' : '.layout__region--third'); | ||||
|           const animationButton = document.querySelector('#animation-toggle'); | ||||
|           const listContainer = etapeList.parentNode; | ||||
|           if (souldListHide) { | ||||
|             listContainer.style.minWidth = '30vw'; | ||||
|             etapeList.style.opacity = '0'; | ||||
|             setTimeout(() => { | ||||
|               etapeList.style.display = 'none'; | ||||
|             }, 300); | ||||
|             if (window.innerWidth >= this.minDesktopWidth) { | ||||
|               animationButton.style.display = 'none'; | ||||
|             } | ||||
|           } else { | ||||
|             listContainer.style.minWidth = 'unset'; | ||||
|             etapeList.style.display = 'block'; | ||||
|             setTimeout(() => { | ||||
|               etapeList.style.opacity = '1'; | ||||
|             }, 10); | ||||
|             if (window.innerWidth >= this.minDesktopWidth) { | ||||
|               animationButton.style.display = 'block'; | ||||
|             } | ||||
|           } | ||||
|         }, | ||||
|         toggleEtapeListScroll(isIntersecting, listeEtape, column, headerHeight, animationToggleTop) { | ||||
|           if (isIntersecting && !this.isEtapeListeScrollable  | ||||
|           || !isIntersecting && this.isEtapeListeScrollable) { | ||||
| @@ -104,6 +128,14 @@ export const useLayoutStore = defineStore('layout', { | ||||
|         setHeaderPosition(currentPageIsIndex) { | ||||
|           const header = document.querySelector('.layout-container > header'); | ||||
|           header.style.position = currentPageIsIndex ? 'fixed' : 'relative'; | ||||
|         } | ||||
|         }, | ||||
|         toggleModaleTransition(shouldModaleTransition, enterDelay) { | ||||
|           if (shouldModaleTransition) { | ||||
|             document.documentElement.style.setProperty('--modale-enter-delay', `${enterDelay}s`); | ||||
|           } else { | ||||
|             document.documentElement.style.setProperty('margin-top', '0'); | ||||
|             document.documentElement.style.setProperty('transition', 'none'); | ||||
|           } | ||||
|         }, | ||||
|     }, | ||||
| }) | ||||
|   | ||||
| @@ -16,6 +16,39 @@ export const useMapStore = defineStore('mapState', { | ||||
|         animationDuration: 3, | ||||
|     }), | ||||
|     actions: { | ||||
|         handleMapMovement(isModaleEtape, wasModaleEtape, lat = this.defaultMapCenter.lat, lon = this.defaultMapCenter.lng) { | ||||
|           if (this.animationsAreEnabled) { | ||||
|             if (isModaleEtape) { | ||||
|               if (!wasModaleEtape) { | ||||
|                 // national -> détail | ||||
|                 useLayoutStore().toggleModaleTransition(true, this.animationDuration); | ||||
|                 this.zoomToPlace(lat, lon); | ||||
|               } else { | ||||
|                 // détail -> détail | ||||
|                 useLayoutStore().toggleModaleTransition(true, this.animationDuration); | ||||
|                 this.zoomToPlace(lat, lon); | ||||
|               } | ||||
|             } else { | ||||
|               if (wasModaleEtape) { | ||||
|                 // détail -> national | ||||
|                 useLayoutStore().toggleModaleTransition(true, this.animationDuration); | ||||
|                 this.resetMap(); | ||||
|               } else { | ||||
|                 // national -> national | ||||
|                 useLayoutStore().toggleModaleTransition(true, 0); | ||||
|               } | ||||
|             } | ||||
|           } else { | ||||
|             if (isModaleEtape) { | ||||
|               // ? -> détail | ||||
|               this.zoomToPlace(lat, lon); | ||||
|             } else { | ||||
|               // ? -> national | ||||
|               this.resetMap(); | ||||
|             } | ||||
|             useLayoutStore().toggleModaleTransition(false); | ||||
|           } | ||||
|         }, | ||||
|         zoomToPlace(lat, long) {           | ||||
|             if (useLayoutStore().isDesktop) long = long - 0.03;            | ||||
|             this.map.flyTo( | ||||
| @@ -32,17 +65,13 @@ export const useMapStore = defineStore('mapState', { | ||||
|             this.currentZoom = useLayoutStore().isDesktop ? this.defaultZoomDesktop : this.defaultZoomMobile; | ||||
|         }, | ||||
|         lockMap() {           | ||||
|             setTimeout(() => { | ||||
|                 this.map.options.minZoom = this.currentZoom; | ||||
|                 this.map.options.maxZoom = this.currentZoom; | ||||
|             }, this.animationDuration * 1000 + 100); | ||||
|             this.map.dragging.disable(); | ||||
|             this.map.touchZoom.disable(); | ||||
|             this.map.doubleClickZoom.disable(); | ||||
|             this.map.scrollWheelZoom.disable(); | ||||
|             this.map.boxZoom.disable(); | ||||
|             this.map.keyboard.disable(); | ||||
|             // map.tap.disable(); | ||||
|             this.map._controlContainer.style.display = 'none';             | ||||
|         }, | ||||
|         unlockMap() { | ||||
|           this.map.options.minZoom = useLayoutStore().isDesktop ? this.defaultZoomDesktop : this.defaultZoomMobile; | ||||
| @@ -53,7 +82,7 @@ export const useMapStore = defineStore('mapState', { | ||||
|           this.map.scrollWheelZoom.enable(); | ||||
|           this.map.boxZoom.enable(); | ||||
|           this.map.keyboard.enable(); | ||||
|           // map.tap.enable(); | ||||
|           this.map._controlContainer.style.display = 'block'; | ||||
|         }, | ||||
|         toggleAnimation() { | ||||
|             this.animationsAreEnabled = !this.animationsAreEnabled; | ||||
|   | ||||
| @@ -0,0 +1,176 @@ | ||||
| import REST from '../../api/rest-axios'; | ||||
| import { fetchFromRelationships, getCleanDate } from './contentFetchUtils'; | ||||
|  | ||||
|  | ||||
| export async function getCarteSensible(partie) { | ||||
|     const carteSensiblePromise = fetchFromRelationships('field_image_carte', partie.relationships); | ||||
|  | ||||
|     const carteSensibleData = await carteSensiblePromise; | ||||
|     if (carteSensibleData) { | ||||
|         return { | ||||
|             url: { | ||||
|                 original: carteSensibleData.attributes.uri.url, | ||||
|                 small: carteSensibleData.attributes.image_style_uri.content_small, | ||||
|                 medium: carteSensibleData.attributes.image_style_uri.content_medium, | ||||
|                 large: carteSensibleData.attributes.image_style_uri.content_large, | ||||
|                 xlarge: carteSensibleData.attributes.image_style_uri.content_x_large, | ||||
|             }, | ||||
|             alt: partie.relationships.field_image_carte.data.meta.alt, | ||||
|         }; | ||||
|     } | ||||
| } | ||||
|  | ||||
| export async function getTitreTexte(partie) { | ||||
|     let titre = partie.attributes.field_titre; | ||||
|     let texte = partie.attributes.field_texte.value;                                     | ||||
|      | ||||
|     // get the resized images from the text | ||||
|     const imgRegex = /<img[^>]+>/g; | ||||
|     const uuidRegex = /data-entity-uuid="([^"]+)"/; | ||||
|     const imgTags = texte.match(imgRegex); | ||||
|      | ||||
|     if (imgTags) { | ||||
|         const imagePromises = imgTags.map(imgTag => { | ||||
|             const uuidMatch = imgTag.match(uuidRegex); | ||||
|             if (uuidMatch && uuidMatch[1]) { | ||||
|                 return REST.get(`/jsonapi/file/file/${uuidMatch[1]}`) | ||||
|                     .then(response => ({ | ||||
|                         originalTag: imgTag, | ||||
|                         imageData: response.data.data | ||||
|                     })); | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         const images = await Promise.all(imagePromises); | ||||
|         images.forEach(({originalTag, imageData}) => { | ||||
|             const newImgTag = originalTag | ||||
|                 .replace(/src="[^"]+"/,`src="${imageData.attributes.image_style_uri.content_medium}"`) | ||||
|                 .replace('>',' data-large-src="' + imageData.attributes.image_style_uri.content_large + '">'); | ||||
|             texte = texte.replace(originalTag, newImgTag); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     return { titre, texte }; | ||||
| } | ||||
|  | ||||
| export async function getChiffresCles(partie) { | ||||
|     const chiffresClesFetch = await fetchFromRelationships('field_chiffres_clefs', partie.relationships); | ||||
|     if (chiffresClesFetch) { | ||||
|         let chiffresCles = []; | ||||
|         for (let chiffre of chiffresClesFetch) { | ||||
|             chiffresCles.push({ | ||||
|                 chiffre: chiffre.attributes.field_chiffre, | ||||
|                 description: chiffre.attributes.field_description, | ||||
|             }); | ||||
|         } | ||||
|         return chiffresCles; | ||||
|     }     | ||||
| } | ||||
|  | ||||
| export async function getDiaporama(partie) { | ||||
|     const diaporamaFetch = await fetchFromRelationships('field_diaporama', partie.relationships); | ||||
|      | ||||
|     if (diaporamaFetch) { | ||||
|         const diaporamaPromises = diaporamaFetch.map((image, index) => { | ||||
|             return { | ||||
|                 url: { | ||||
|                     original: image.attributes.uri.url, | ||||
|                     small: image.attributes.image_style_uri.content_small, | ||||
|                     medium: image.attributes.image_style_uri.content_medium, | ||||
|                     large: image.attributes.image_style_uri.content_large, | ||||
|                 }, | ||||
|                 alt: partie.relationships.field_diaporama.data[index].meta.alt, | ||||
|             }; | ||||
|         }); | ||||
|  | ||||
|         return await Promise.all(diaporamaPromises); | ||||
|     } | ||||
| } | ||||
|  | ||||
| export async function getEntretien(partie) { | ||||
|     const [personnesFetch, questionsReponsesFetch] = await Promise.all([ | ||||
|         fetchFromRelationships('field_personne_s', partie.relationships), | ||||
|         fetchFromRelationships('field_questions_reponses', partie.relationships) | ||||
|     ]); | ||||
|  | ||||
|     if (personnesFetch && questionsReponsesFetch) { | ||||
|         const personnesPromises = personnesFetch.map(async (personne) => { | ||||
|             const portraitFetch = await fetchFromRelationships('field_portrait', personne.relationships); | ||||
|             if (portraitFetch) { | ||||
|                 return { | ||||
|                     portrait: { | ||||
|                         original: portraitFetch.attributes.uri.url, | ||||
|                         small: portraitFetch.attributes.image_style_uri.content_small, | ||||
|                         medium: portraitFetch.attributes.image_style_uri.content_medium, | ||||
|                         large: portraitFetch.attributes.image_style_uri.content_large, | ||||
|                     }, | ||||
|                     alt: personne.relationships.field_portrait.data.meta.alt, | ||||
|                     description: personne.attributes.field_description, | ||||
|                 }; | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         const questionsReponses = questionsReponsesFetch.map(qr => ({ | ||||
|             question: qr.attributes.field_question, | ||||
|             reponse: qr.attributes.field_reponse.value, | ||||
|         })); | ||||
|  | ||||
|         return { | ||||
|             titre: partie.attributes.field_titre, | ||||
|             personnes: await Promise.all(personnesPromises), | ||||
|             questionsReponses: questionsReponses | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| export function getVideos(partie) { | ||||
|     let videos = []; | ||||
|     for (let video of partie.attributes.field_videos) { | ||||
|         const videoId = video.split('?v=')[1]; | ||||
|         const videoUrl = `https://www.youtube.com/embed/${videoId}`; | ||||
|         videos.push(videoUrl); | ||||
|     } | ||||
|     return videos; | ||||
| } | ||||
|  | ||||
| export async function getDocument(partie) { | ||||
|     // const documentFetch = await fetchFromRelationships('field_document', partie.relationships); | ||||
|     const uuid = partie.relationships.field_document.data?.id; | ||||
|     const documentFetch = await REST.get(`/jsonapi/file/file/${uuid}`); | ||||
|     const url = documentFetch.data.data?.attributes.uri.url; | ||||
|  | ||||
|     const titre = partie.attributes.field_titre; | ||||
|     const sousTitre = partie.attributes.field_sous_titre; | ||||
|     const date = partie.attributes.field_date ? getCleanDate(partie.attributes.field_date) : null; | ||||
|     const auteurice = partie.relationships.field_autheurice_s; | ||||
|     const description = partie.relationships.field_document.data?.meta.description; | ||||
|  | ||||
|     const vignetteFetch = await REST.get(`/jsonapi/file/file/${partie.relationships.field_vignette.data?.id}`); | ||||
|     const vignette = { url: vignetteFetch.data.data?.attributes.image_style_uri.content_small, alt: partie.relationships.field_vignette.data?.meta.alt }; | ||||
|      | ||||
|     return { url, titre, sousTitre, date, auteurice, description, vignette }; | ||||
| } | ||||
|  | ||||
| export async function getGallerie(partie) { | ||||
|     const gallerieFetch = await fetchFromRelationships('field_gallerie', partie.relationships); | ||||
|  | ||||
|     if (gallerieFetch) { | ||||
|         const titre = gallerieFetch.attributes.title; | ||||
|         const introduction = gallerieFetch.attributes.body?.processed; | ||||
|          | ||||
|         const imagesFetch = await fetchFromRelationships('field_images', gallerieFetch.relationships); | ||||
|         let images = []; | ||||
|         imagesFetch.forEach((image, index) => { | ||||
|             images.push({ | ||||
|                 url: { | ||||
|                     original: image.attributes.uri.url, | ||||
|                     small: image.attributes.image_style_uri.content_small, | ||||
|                     medium: image.attributes.image_style_uri.content_medium, | ||||
|                     large: image.attributes.image_style_uri.content_large, | ||||
|                 }, | ||||
|                 alt: gallerieFetch.relationships.field_images.data[index].meta.alt, | ||||
|             }); | ||||
|         }); | ||||
|         return { titre, introduction, images }; | ||||
|     }     | ||||
| } | ||||
| @@ -0,0 +1,150 @@ | ||||
| import REST from '../../api/rest-axios'; | ||||
|  | ||||
| export async function fetchFromRelationships(field, relationships) { | ||||
|     field = relationships[field] ? field : `${field}_static`; | ||||
|      | ||||
|     if (relationships[field]?.links) { | ||||
|         const contentLink = relationships[field].links.related.href; | ||||
|         return REST.get(contentLink) | ||||
|             .then(contentFetch => contentFetch.data.data) | ||||
|             .catch(error => { | ||||
|                 this.error = 'Failed to fetch data'; | ||||
|                 console.error('Issue with getNodeData', error); | ||||
|             }); | ||||
|     } | ||||
|     return null; | ||||
| } | ||||
|  | ||||
| export function getCleanDate(date) { | ||||
|     return { | ||||
|       d: date.split('-')[2], | ||||
|       m: new Intl.DateTimeFormat('fr-FR', { month: 'long' }).format(new Date(date)), | ||||
|       y: date.split('-')[0], | ||||
|     } | ||||
| } | ||||
|  | ||||
| export async function getRelatedEtape(direction, path) { | ||||
|     const getRelatedEtapeContent = (relatedEtapeData) => { | ||||
|         if (relatedEtapeData) { | ||||
|             return fetchFromRelationships('field_vignette', relatedEtapeData.relationships) | ||||
|                 .then(vignetteFetch => { | ||||
|                     if (vignetteFetch) { | ||||
|                         return { | ||||
|                             url: relatedEtapeData.attributes.metatag.find(tag => tag.tag === "link")?.attributes.href, | ||||
|                             couleur: relatedEtapeData.attributes.field_couleur, | ||||
|                             title: relatedEtapeData.attributes.title, | ||||
|                             postalCode: relatedEtapeData.attributes.field_adresse.postal_code, | ||||
|                             dates: { | ||||
|                                 start: getCleanDate(relatedEtapeData.attributes.field_dates.value), | ||||
|                                 end: getCleanDate(relatedEtapeData.attributes.field_dates.end_value), | ||||
|                             }, | ||||
|                             vignette: { | ||||
|                                 url: { | ||||
|                                     original: vignetteFetch.attributes.uri.url, | ||||
|                                     small: vignetteFetch.attributes.image_style_uri.content_small, | ||||
|                                     medium: vignetteFetch.attributes.image_style_uri.content_medium, | ||||
|                                     large: vignetteFetch.attributes.image_style_uri.content_large, | ||||
|                                 }, | ||||
|                                 alt: relatedEtapeData.relationships.field_vignette.data.meta.alt, | ||||
|                             }, | ||||
|                         } | ||||
|                     } | ||||
|                 }); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     const allEtapesPromise = REST.get('/jsonapi/views/etapes/block_1/'); | ||||
|      | ||||
|     return allEtapesPromise.then(allEtapesData => { | ||||
|         for (let [index, etape] of allEtapesData.data.data.entries()) { | ||||
|             if (etape.attributes.metatag.some(tag =>  | ||||
|                 tag.tag === "link" && tag.attributes.href === path | ||||
|             )) { | ||||
|                 const relatedEtapeIndex = direction ? (direction === 'next' ? index + 1 : index - 1) : index; | ||||
|                 return getRelatedEtapeContent(allEtapesData.data.data[relatedEtapeIndex]); | ||||
|             } | ||||
|         } | ||||
|     });     | ||||
| } | ||||
|  | ||||
| export async function getRessourceItemCard(item) { | ||||
|     try { | ||||
|         const ressourceFetch = await REST.get(item.links.self.href); | ||||
|          | ||||
|         const partiesFetch = await REST.get(item.relationships.field_parties_ressource?.links.related.href); | ||||
|         const parties = partiesFetch.data.data; | ||||
|          | ||||
|         const vignettePartie = parties.find(partie => partie.type !== "paragraph--titre_texte"); | ||||
|          | ||||
|         let vignette = null; | ||||
|         if (vignettePartie) { | ||||
|             let alt; | ||||
|             switch (vignettePartie.type) { | ||||
|                 case 'paragraph--diaporama': | ||||
|                     alt = vignettePartie.relationships.field_diaporama?.data[0]?.meta.alt; | ||||
|                     const diaporamaFetch = await REST.get(vignettePartie.relationships.field_diaporama?.links.related.href); | ||||
|                     vignette = { | ||||
|                         url: diaporamaFetch.data.data[0].attributes.image_style_uri?.content_small, | ||||
|                         alt | ||||
|                     }; | ||||
|                     break; | ||||
|                  | ||||
|                 case 'paragraph--video': | ||||
|                     const videoId = vignettePartie.attributes.field_videos[0]?.split('?v=')[1]; | ||||
|                     vignette = { | ||||
|                         url: `https://img.youtube.com/vi/${videoId}/0.jpg`, | ||||
|                         alt: item.attributes.title | ||||
|                     }; | ||||
|                     break; | ||||
|                  | ||||
|                 case 'paragraph--galleries': | ||||
|                     const gallerieFetch = await REST.get(vignettePartie.relationships.field_gallerie?.links.related.href); | ||||
|                     const galleryAlt = gallerieFetch.data.data.relationships.field_images?.data[0].meta.alt; | ||||
|                     const gallerieImageFetch = await REST.get(gallerieFetch.data.data.relationships.field_images?.links.related.href); | ||||
|                     vignette = { | ||||
|                         url: gallerieImageFetch.data.data[0].attributes.image_style_uri?.content_small, | ||||
|                         alt: galleryAlt | ||||
|                     }; | ||||
|                     break; | ||||
|                  | ||||
|                 case 'paragraph--document': | ||||
|                     alt = vignettePartie.relationships.field_vignette?.data?.meta.alt; | ||||
|                     const documentFetch = await REST.get(vignettePartie.relationships.field_vignette?.links.related.href); | ||||
|                     vignette = { | ||||
|                         url: documentFetch.data.data?.attributes.image_style_uri?.content_small, | ||||
|                         alt | ||||
|                     }; | ||||
|                     break; | ||||
|                  | ||||
|                 default: | ||||
|                     vignette = null; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         const relatedEtape = await REST.get(item.relationships.field_etape.links.related.href); | ||||
|         // console.log(item); | ||||
|          | ||||
|          | ||||
|         return { | ||||
|             ressourceType: item.attributes.field_type_de_ressource, | ||||
|             title: item.attributes.title, | ||||
|             auteurice: item.attributes.field_autheurice, | ||||
|             promoted: item.attributes.field_mis_en_avant, | ||||
|             date: item.attributes.field_date_ressource ? getCleanDate(item.attributes.field_date_ressource) : null, | ||||
|             url: ressourceFetch.data.data.attributes.metatag.find(tag => tag.tag === "link")?.attributes.href, | ||||
|             relatedEtape: relatedEtape.data.data?.attributes.title, | ||||
|             vignette | ||||
|         }; | ||||
|     } catch (error) { | ||||
|         console.error('Error fetching resource:', error); | ||||
|         return null; | ||||
|     } | ||||
| } | ||||
|  | ||||
| export async function getRelatedRessources(etapeId) { | ||||
|     const ressources = await REST.get(`/jsonapi/node/ressource/`); | ||||
|     const ressourcesRelatedToEtape = ressources.data.data.filter(ressource => ressource.relationships.field_etape.data?.id === etapeId); | ||||
|     const ressourcesRelatedPromises = ressourcesRelatedToEtape.map(ressource => getRessourceItemCard(ressource)); | ||||
|  | ||||
|     return await Promise.all(ressourcesRelatedPromises); | ||||
| } | ||||
| @@ -0,0 +1,43 @@ | ||||
| import REST from '../../api/rest-axios'; | ||||
| import { useLayoutStore } from '../../stores/layout'; | ||||
| import { getPartenaires, getGouvernance, getRessources } from './multiItemPages'; | ||||
|  | ||||
| export async function fetchMultiplePartialContent(contentType) { | ||||
|     let partialContent = {}; | ||||
|     let pageTitle = ''; | ||||
|  | ||||
|     const intro = await REST.get(`/jsonapi/config_pages/intro_${contentType}/`); | ||||
|     const introContent = intro.data.data[0]; | ||||
|  | ||||
|     pageTitle = | ||||
|         `${introContent.attributes.field_titre} ${introContent.attributes.metatag.find(tag => tag.tag === "meta")?.attributes.content}`;                     | ||||
|  | ||||
|     partialContent.contentTitle = introContent.attributes.field_titre; | ||||
|     partialContent.intro = introContent.attributes.field_intro?.value; | ||||
|  | ||||
|     return { pageTitle, partialContent }; | ||||
| } | ||||
|  | ||||
| export async function fetchMultipleFullContent(contentType, rawContent) { | ||||
|     let content = {}; | ||||
|      | ||||
|     let multiItemPageArray = [];                     | ||||
|  | ||||
|     switch (contentType) { | ||||
|         case 'ressource': | ||||
|             multiItemPageArray = await getRessources(rawContent);                             | ||||
|             content.ressourceTypes = new Set(multiItemPageArray.map(item => item.ressourceType)); | ||||
|             useLayoutStore().hideEtapeList(true); | ||||
|             break; | ||||
|         case 'partenaire': | ||||
|             multiItemPageArray = await getPartenaires(rawContent); | ||||
|             break; | ||||
|         case 'gouvernance': | ||||
|             multiItemPageArray = await getGouvernance(rawContent); | ||||
|             break; | ||||
|     } | ||||
|  | ||||
|     content[`${contentType}s`] = multiItemPageArray; | ||||
|      | ||||
|     return content; | ||||
| } | ||||
| @@ -0,0 +1,153 @@ | ||||
| import REST from '../../api/rest-axios'; | ||||
|  | ||||
| import { getCleanDate, fetchFromRelationships, getRelatedEtape, getRelatedRessources } from './contentFetchUtils'; | ||||
| import { getCarteSensible, getTitreTexte, getChiffresCles, getDiaporama, getEntretien, getVideos, getDocument, getGallerie } from './cleanParties'; | ||||
| import { useLayoutStore } from '../../stores/layout'; | ||||
|  | ||||
| export function fetchSingletonPartialContent(contentType, rawContent) { | ||||
|     let partialContent = {}; | ||||
|  | ||||
|     let pageTitle = rawContent.attributes.metatag.find(tag => tag.tag === "meta")?.attributes.content; | ||||
|  | ||||
|     partialContent.contentTitle = rawContent.attributes.title; | ||||
|  | ||||
|     if (contentType === 'etape') { | ||||
|         partialContent.coordinates = { | ||||
|             lat: rawContent.attributes.field_geofield.lat, | ||||
|             lon: rawContent.attributes.field_geofield.lon, | ||||
|         }; | ||||
|         partialContent.adresse = rawContent.attributes.field_adresse; | ||||
|         partialContent.etape_number = rawContent.attributes.field_arret_numero; | ||||
|         partialContent.couleur = rawContent.attributes.field_couleur; | ||||
|         partialContent.dates = { | ||||
|             start: getCleanDate(rawContent.attributes.field_dates.value), | ||||
|             end: getCleanDate(rawContent.attributes.field_dates.end_value), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (contentType === 'ressourceItem') {    | ||||
|         partialContent.ressourceType = rawContent.attributes.field_type_de_ressource; | ||||
|         partialContent.auteurice = rawContent.attributes.field_autheurice; | ||||
|         partialContent.date = rawContent.attributes.field_date_ressource ? getCleanDate(rawContent.attributes.field_date_ressource) : null; | ||||
|     } | ||||
|      | ||||
|     return { pageTitle, partialContent }; | ||||
| } | ||||
|  | ||||
| export async function fetchSingletonFullContent(contentType, rawContent, path) { | ||||
|     let content = {}; | ||||
|  | ||||
|     const vignettePromise = fetchFromRelationships('field_vignette', rawContent.relationships); | ||||
|     const partiesPromise = fetchFromRelationships(contentType === 'ressourceItem' ? 'field_parties_ressource' : 'field_parties', rawContent.relationships); | ||||
|  | ||||
|     let previousEtapePromise, nextEtapePromise; | ||||
|  | ||||
|     if (contentType === 'etape') { | ||||
|         previousEtapePromise = getRelatedEtape('previous', path); | ||||
|         nextEtapePromise = getRelatedEtape('next', path); | ||||
|         content.relatedRessources = await getRelatedRessources(rawContent.id); | ||||
|     } | ||||
|  | ||||
|     if (contentType === 'ressourceItem') { | ||||
|         content.introduction = rawContent.attributes.field_introduction?.processed;                         | ||||
|         if (rawContent.relationships.field_etape.data) { | ||||
|             const relatedEtapeFetch = fetchFromRelationships('field_etape', rawContent.relationships); | ||||
|             const relatedEtape = await Promise.resolve(relatedEtapeFetch); | ||||
|             const relatedEtapeUrl = relatedEtape.attributes.metatag.find(tag => tag.tag === "link")?.attributes.href; | ||||
|             content.relatedEtape = await getRelatedEtape('', relatedEtapeUrl);                         | ||||
|         } | ||||
|          | ||||
|         useLayoutStore().hideEtapeList(true); | ||||
|     } | ||||
|  | ||||
|     const [vignetteData, partiesData] = await Promise.all([vignettePromise, partiesPromise]); | ||||
|  | ||||
|     if (vignetteData) { | ||||
|         content.vignette = { | ||||
|             url: { | ||||
|                 original: vignetteData.attributes.uri.url, | ||||
|                 small: vignetteData.attributes.image_style_uri.content_small, | ||||
|                 medium: vignetteData.attributes.image_style_uri.content_medium, | ||||
|                 large: vignetteData.attributes.image_style_uri.content_large, | ||||
|             }, | ||||
|             alt: rawContent.relationships.field_vignette.data.meta.alt | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     if (partiesData) { | ||||
|         const partiesPromises = partiesData.map(async (partie) => { | ||||
|             const partieType = partie.type.replace(/^paragraph--/, ""); | ||||
|             let partieContent = { type: partieType };                             | ||||
|  | ||||
|             switch (partieType) { | ||||
|                 case 'carte_sensible': | ||||
|                     partieContent.carteSensible = await getCarteSensible(partie); | ||||
|                     break; | ||||
|                 case 'titre_texte': | ||||
|                     const { titre, texte } = await getTitreTexte(partie); | ||||
|                     partieContent.titre = titre; | ||||
|                     partieContent.texte = texte; | ||||
|                     break; | ||||
|                 case 'chiffres_cles': | ||||
|                     partieContent.chiffresCles = await getChiffresCles(partie); | ||||
|                     break; | ||||
|                 case 'diaporama': | ||||
|                     partieContent.diaporama = await getDiaporama(partie); | ||||
|                     break; | ||||
|                 case 'entretien': | ||||
|                     partieContent.entretien = await getEntretien(partie); | ||||
|                     break; | ||||
|                 case 'exergue': | ||||
|                     partieContent.exergue = partie.attributes.field_texte_exergue.value; | ||||
|                     break; | ||||
|                 case 'video': | ||||
|                     partieContent.videos = getVideos(partie); | ||||
|                     break; | ||||
|                 case 'document': | ||||
|                     partieContent.document = await getDocument(partie); | ||||
|                     break; | ||||
|                 case 'galleries': | ||||
|                     partieContent.gallerie = await getGallerie(partie); | ||||
|                     break; | ||||
|             } | ||||
|             return partieContent; | ||||
|         }); | ||||
|  | ||||
|         // liens | ||||
|         if (rawContent.attributes.field_liens?.length) { | ||||
|             content.liens = []; | ||||
|             for (let lien of rawContent.attributes.field_liens) { | ||||
|                 content.liens.push({ | ||||
|                     title: lien.title, | ||||
|                     url: lien.uri, | ||||
|                 }); | ||||
|             }                             | ||||
|         } | ||||
|         // pièces jointes | ||||
|         if (rawContent.relationships.field_pieces_jointes?.data.length) { | ||||
|             content.pieces_jointes = []; | ||||
|             for (let pieceJointe of rawContent.relationships.field_pieces_jointes.data) { | ||||
|                 if (pieceJointe.meta.display) { | ||||
|                     const uuid = pieceJointe.id; | ||||
|                     const response = await REST.get(`/jsonapi/file/file/${uuid}`); | ||||
|                     content.pieces_jointes.push({ | ||||
|                         title: pieceJointe.meta.description, | ||||
|                         url: response.data.data.attributes.uri.url, | ||||
|                     }); | ||||
|                 } | ||||
|             } | ||||
|         }  | ||||
|  | ||||
|         content.parties = await Promise.all(partiesPromises); | ||||
|     } | ||||
|  | ||||
|     // related étapes | ||||
|     if (contentType === 'etape') { | ||||
|         const [prevContent, nextContent] = await Promise.all([previousEtapePromise, nextEtapePromise]); | ||||
|         content.previous = prevContent; | ||||
|         content.next = nextContent; | ||||
|     } | ||||
|  | ||||
|     return content; | ||||
| } | ||||
| @@ -0,0 +1,32 @@ | ||||
| import REST from '../../api/rest-axios'; | ||||
|  | ||||
| export async function findContentByPath(contentTypes, path) { | ||||
|     for (let type of contentTypes) { | ||||
|         const response = await REST.get(`/jsonapi/node/${type}/`); | ||||
|          | ||||
|         const content = response.data.data.find(content =>  | ||||
|             content.attributes.metatag.some(tag =>  | ||||
|                 tag.tag === "link" && tag.attributes.href === path | ||||
|             ) | ||||
|         ); | ||||
|      | ||||
|         if (content) {             | ||||
|             return { | ||||
|                 contentType: content.type === 'node--ressource' ? 'ressourceItem' : type, | ||||
|                 rawContent: content, | ||||
|             }; | ||||
|         } | ||||
|          | ||||
|         // Handle special case for gouvernance, ressources, partenaires (multiple items per page) | ||||
|         let pageRequested = window.location.href.split('/').pop().replace(/s?$/, ''); | ||||
|         if (type === pageRequested | ||||
|             || (type === 'gouvernance' && pageRequested === 'contact') | ||||
|         ) { | ||||
|             return { | ||||
|                 contentType: type, | ||||
|                 rawContent: response.data.data, | ||||
|             }; | ||||
|         } | ||||
|     } | ||||
|     return null; | ||||
| } | ||||
| @@ -0,0 +1,62 @@ | ||||
| import REST from '../../api/rest-axios'; | ||||
| import { getRessourceItemCard } from './contentFetchUtils'; | ||||
|  | ||||
| export async function getPartenaires(rawContent) { | ||||
|     const logoPromises = rawContent.map(item =>  | ||||
|         REST.get(item.relationships.field_logo?.links.related.href) | ||||
|             .then(logoFetch => ({ | ||||
|                 title: item.attributes?.title, | ||||
|                 description: item.attributes?.body?.value, | ||||
|                 weight: item.attributes?.field_poid, | ||||
|                 link_url: item.attributes?.field_lien?.uri, | ||||
|                 logo_alt: item.relationships?.field_logo?.data?.meta.alt, | ||||
|                 logo_url: { | ||||
|                     original: logoFetch.data.data?.attributes.uri.url, | ||||
|                     small: logoFetch.data.data?.attributes.image_style_uri.content_small, | ||||
|                     medium: logoFetch.data.data?.attributes.image_style_uri.content_medium, | ||||
|                     large: logoFetch.data.data?.attributes.image_style_uri.content_large, | ||||
|                 } | ||||
|             })) | ||||
|     ); | ||||
|      | ||||
|     return await Promise.all(logoPromises); | ||||
| } | ||||
|  | ||||
| export async function getGouvernance(rawContent) { | ||||
|     const itemPromises = rawContent.map(item => | ||||
|         REST.get(item.relationships.field_personne_s?.links.related.href) | ||||
|             .then(async personnesFetch => { | ||||
|                 const portraitPromises = personnesFetch.data.data.map(personne => | ||||
|                     REST.get(personne.relationships.field_portrait?.links.related.href) | ||||
|                         .then(portraitFetch => ({ | ||||
|                             nom: personne.attributes.field_nom, | ||||
|                             prenom: personne.attributes.field_prenom, | ||||
|                             description: personne.attributes.field_description, | ||||
|                             photo_meta: personne.relationships.field_portrait.data?.meta.alt, | ||||
|                             photo_url: portraitFetch.data.data ? { | ||||
|                                 original: portraitFetch.data.data?.attributes.uri.url, | ||||
|                                 small: portraitFetch.data.data?.attributes.image_style_uri.content_small, | ||||
|                                 medium: portraitFetch.data.data?.attributes.image_style_uri.content_medium, | ||||
|                                 large: portraitFetch.data.data?.attributes.image_style_uri.content_large, | ||||
|                             } : null | ||||
|                         })) | ||||
|                 ); | ||||
|                  | ||||
|                 return Promise.all(portraitPromises) | ||||
|                     .then(personnes => ({ | ||||
|                         title: item.attributes?.title, | ||||
|                         weight: item.attributes?.field_poid, | ||||
|                         personnes | ||||
|                     })); | ||||
|             }) | ||||
|     ); | ||||
|      | ||||
|     return await Promise.all(itemPromises); | ||||
| } | ||||
|  | ||||
| export async function getRessources(rawContent) { | ||||
|      | ||||
|     const ressourcesPromises = rawContent.map(item => getRessourceItemCard(item));         | ||||
|  | ||||
|     return await Promise.all(ressourcesPromises); | ||||
| } | ||||
| @@ -1,53 +1,57 @@ | ||||
| import { setActiveNavItem } from "./set-active-nav-item"; | ||||
| import { useContentStore } from "../stores/content"; | ||||
| import { useMapStore } from "../stores/map"; | ||||
| import { useLayoutStore } from '../stores/layout'; | ||||
|  | ||||
| export async function initFirstLoadRouting(store, router, baseUrl, siteName) { | ||||
| export async function initFirstLoadRouting(router, baseUrl, siteName) { | ||||
|     const store = useContentStore(); | ||||
|     const decoupled_origin = JSON.parse(window.localStorage.getItem('decoupled_origin')); | ||||
|  | ||||
|     if(decoupled_origin) { | ||||
|         router.push(decoupled_origin.url); | ||||
|         await store.fetchContentData(baseUrl + decoupled_origin.url); | ||||
|         await store.fetchPartialContentData(baseUrl + decoupled_origin.url); | ||||
|         window.localStorage.removeItem("decoupled_origin"); | ||||
|         document.title = store.pageTitle; | ||||
|         setActiveNavItem(store.contentType, decoupled_origin.url); | ||||
|         useLayoutStore().setHeaderPosition(false); | ||||
|         await store.fetchFullContentData(baseUrl + decoupled_origin.url); | ||||
|     } else { | ||||
|         document.title = siteName; | ||||
|         useLayoutStore().setHeaderPosition(true); | ||||
|     } | ||||
| } | ||||
|  | ||||
| export function handleClickableElements(clickableElements, store, router, baseUrl, siteName, mapStore) { | ||||
| export function handleClickableElements(clickableElements, router, baseUrl, siteName) { | ||||
|     for (const link of clickableElements) { | ||||
|         let href = link.href || link.dataset.href; | ||||
|         if (href.startsWith(baseUrl)) href = href.replace(baseUrl, ''); | ||||
|  | ||||
|         link.onclick = async function (e) {             | ||||
|             console.log('click on link, route push'); | ||||
|              | ||||
|             router.push(href); | ||||
|             if (href !== window.location.pathname) { | ||||
|               pageChange(href, store, siteName, mapStore, baseUrl); | ||||
|                 router.push(href); | ||||
|                 pageChange(href, siteName, baseUrl); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| export async function handleBrowserNavigation(store, baseUrl, siteName, mapStore) { | ||||
|   let href = window.location.pathname; | ||||
|   if (href.startsWith(baseUrl)) href = href.replace(baseUrl, ''); | ||||
|   pageChange(href, store, siteName, mapStore, baseUrl) | ||||
| export async function handleBrowserNavigation(baseUrl, siteName) { | ||||
|     let href = window.location.pathname; | ||||
|     if (href.startsWith(baseUrl)) href = href.replace(baseUrl, ''); | ||||
|     pageChange(href, siteName, baseUrl) | ||||
| } | ||||
|  | ||||
| export async function pageChange(href, store, siteName, mapStore, baseUrl) { | ||||
|   console.log('trigger page change'); | ||||
| export async function pageChange(href, siteName, baseUrl) { | ||||
|     const store = useContentStore(); | ||||
|     const mapStore = useMapStore(); | ||||
|  | ||||
|   if (href === '/') { | ||||
|       store.resetStore(true); | ||||
|       document.title = siteName; | ||||
|       mapStore.resetMap(); | ||||
|       useLayoutStore().setHeaderPosition(true); | ||||
|   } else { | ||||
|       await store.fetchContentData(baseUrl + href); | ||||
|       await store.fetchPartialContentData(baseUrl + href); | ||||
|       document.title = store.pageTitle; | ||||
|       useLayoutStore().setHeaderPosition(false); | ||||
|   } | ||||
| @@ -56,4 +60,8 @@ export async function pageChange(href, store, siteName, mapStore, baseUrl) { | ||||
|   const listeEtape = document.querySelector('#etapes-liste'); | ||||
|   const animationToggle = document.querySelector('#animation-toggle'); | ||||
|   if (!useLayoutStore().isDesktop) useLayoutStore().collapseEtapeListe(listeEtape, animationToggle); | ||||
|  | ||||
|   if (href !== '/') { | ||||
|       await store.fetchFullContentData(baseUrl + href); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -56,7 +56,7 @@ export function setMenuToggle() { | ||||
|     layoutStore.setUpHamburgerToggle(menuBurger, menuContainer); | ||||
| } | ||||
|  | ||||
| export function setHamburgerWhenLogged(drupalSettings) { | ||||
| export function setRightSectionsWhenLogged(drupalSettings) { | ||||
|     if (drupalSettings.user.uid != 0) { | ||||
|       const menuBurger = document.querySelector('#hamburger'); | ||||
|       const menuTitle = document.querySelector('#menu-title'); | ||||
| @@ -65,7 +65,11 @@ export function setHamburgerWhenLogged(drupalSettings) { | ||||
|       const headerTop = header.getBoundingClientRect().top; | ||||
|  | ||||
|       menuTitle.style.top = `${headerTop}px`; | ||||
|       menuBurger.style.top = `${headerTop}px`; | ||||
|       menuContainer.style.paddingTop = `${headerTop}px`; | ||||
|       menuContainer.style.paddingTop = `${headerTop + 10}px`; | ||||
|       menuBurger.style.top = `${headerTop + 2}px`; | ||||
|  | ||||
|       const etapesListContainer = document.querySelector('.block-region-third'); | ||||
|       etapesListContainer.style.paddingTop = `50px`; | ||||
|  | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,7 +1,10 @@ | ||||
| import { useLayoutStore } from '../stores/layout'; | ||||
| import { useMapStore } from '../stores/map'; | ||||
| import REST from '../api/rest-axios'; | ||||
|  | ||||
| export function setupMapStore(mapStore, map, settings) { | ||||
| export function setupMapStore(map, settings) { | ||||
|     const mapStore = useMapStore(); | ||||
|  | ||||
|     mapStore.map = map; | ||||
|     mapStore.defaultMapCenter = map.getCenter(); | ||||
|     mapStore.maxZoom = settings.settings.maxZoom; | ||||
| @@ -13,7 +16,9 @@ export function setupMapStore(mapStore, map, settings) { | ||||
|  | ||||
| // not working | ||||
| // may or may not rework on it later | ||||
| export async function preloadEtapesTiles(mapStore, map) { | ||||
| export async function preloadEtapesTiles(map) { | ||||
|     const mapStore = useMapStore(); | ||||
|      | ||||
|     function waitForEvent(el, eventName) { | ||||
|         return new Promise((resolve) => { | ||||
|             el.once(eventName, resolve); | ||||
|   | ||||
| @@ -4,6 +4,7 @@ export function processClickableElements() { | ||||
|         generalListLinks: processStaticLinks(), | ||||
|         logoLink: processLogoLink(), | ||||
|         mapIcons: processMapIcons(), | ||||
|         mapContainer: processMapContainer(), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| @@ -14,9 +15,9 @@ function processEtapeLinks() { | ||||
|         etape_link.addEventListener('click', (e) => e.preventDefault()); | ||||
|         const couleur = etape_link.dataset.couleur; | ||||
|         li.dataset.href = etape_link.attributes.href.value; | ||||
|         const iconElements = li.querySelectorAll('.icone-arret > div'); | ||||
|         const iconElements = li.querySelectorAll('.icone-arret'); | ||||
|         for (let element of iconElements) { | ||||
|           element.style.backgroundColor = couleur; | ||||
|             element.style.backgroundColor = couleur; | ||||
|         } | ||||
|     }); | ||||
|  | ||||
| @@ -46,16 +47,10 @@ function processMapIcons() { | ||||
|  | ||||
|         const hrefContainer = icon.querySelector('.url'); | ||||
|         icon.dataset.href = hrefContainer.innerText; | ||||
|         hrefContainer.style.display = "none"; | ||||
|  | ||||
|         const colorContainer = icon.querySelector('.couleur'); | ||||
|         let color = colorContainer.innerText; | ||||
|         colorContainer.style.display = "none"; | ||||
|  | ||||
|         const iconElements = icon.querySelectorAll('div'); | ||||
|         for (let iconElement of iconElements) { | ||||
|           iconElement.style.backgroundColor = color; | ||||
|         } | ||||
|         icon.style.backgroundColor = color; | ||||
|  | ||||
|         icon.addEventListener('mouseenter', () => { | ||||
|             icon.style.transform = `${icon.style.transform} scale(1.1)`; | ||||
| @@ -70,3 +65,11 @@ function processMapIcons() { | ||||
|  | ||||
|     return icons; | ||||
| } | ||||
|  | ||||
| function processMapContainer() { | ||||
|     let mapContainer = document.querySelector('.leaflet-layer'); | ||||
|     mapContainer.style.height = "100vh"; | ||||
|     mapContainer.style.width = "100vw"; | ||||
|     mapContainer.dataset.href = "/"; | ||||
|     return mapContainer; | ||||
| } | ||||
| @@ -11,13 +11,19 @@ export function setActiveNavItem(contentType, href) { | ||||
|     } | ||||
|  | ||||
|     if (href === '/' || href === '') { | ||||
|         staticNavItems[0].classList.add('is-active'); | ||||
|         staticNavItems[1].classList.add('is-active'); | ||||
|         for (let item of etapeNavItems) { | ||||
|             item.closest('li').classList.remove('inactive'); | ||||
|         } | ||||
|  | ||||
|     } else { | ||||
|         if (contentType === 'static') { | ||||
|          | ||||
|         if ( | ||||
|             contentType === 'static' | ||||
|             || contentType === 'partenaire' | ||||
|             || contentType === 'gouvernance' | ||||
|             || contentType === 'ressource' | ||||
|         ) { | ||||
|             for (let item of staticNavItems) { | ||||
|                 if (item.getAttribute('href') === href) { | ||||
|                     item.classList.add('is-active'); | ||||
| @@ -27,8 +33,11 @@ export function setActiveNavItem(contentType, href) { | ||||
|             for (let item of etapeNavItems) { | ||||
|                 if (item.getAttribute('href') === href) { | ||||
|                     item.closest('li').classList.remove('inactive');                     | ||||
|                     document.querySelector('#etapes-liste').scrollTo(0, item.closest('li').offsetTop); | ||||
|                 } | ||||
|             } | ||||
|         } else if (contentType === "ressourceItem") { | ||||
|             staticNavItems[2].classList.add('is-active'); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -6,9 +6,6 @@ import AnimationToggle from '../vuejs/AnimationToggle.vue'; | ||||
| import VueImageZoomer from 'vue-image-zoomer'; | ||||
| import 'vue-image-zoomer/dist/style.css'; | ||||
|  | ||||
| import { useContentStore } from '../stores/content'; | ||||
| import { useMapStore } from '../stores/map'; | ||||
|  | ||||
| export function initVueContentModale() { | ||||
|     const pinia = createPinia(); | ||||
|  | ||||
| @@ -17,13 +14,11 @@ export function initVueContentModale() { | ||||
|       .use(router) | ||||
|       .use(VueImageZoomer); | ||||
|        | ||||
|     const store = useContentStore(); | ||||
|     const mapStore = useMapStore(); | ||||
|     app.mount('#content-modale'); | ||||
|  | ||||
|     const animationToggle = createApp(AnimationToggle) | ||||
|       .use(pinia) | ||||
|       .mount('#animation-toggle'); | ||||
|  | ||||
|   return { store, mapStore, router }; | ||||
|   return router; | ||||
| } | ||||
|   | ||||
| @@ -3,19 +3,24 @@ | ||||
|       :enter-active-class="animationsAreEnabled ? 'v-enter-active' : 'no-transition'" | ||||
|       :leave-active-class="animationsAreEnabled ? 'v-leave-active' : 'no-transition'" | ||||
|     > | ||||
|         <div v-if="!loading && ( | ||||
|           contentType === 'etape' | ||||
|           || contentType === 'static' | ||||
|           || contentType === 'gouvernance' | ||||
|           || contentType === 'partenaire' | ||||
|         )"> | ||||
|             <div class="content-wrapper"> | ||||
|         <div v-if="!partialLoading && contentType != ''"> | ||||
|             <div | ||||
|               class="content-wrapper" | ||||
|               :class=" | ||||
|                 contentType === 'ressource' || contentType === 'ressourceItem'  | ||||
|                 ? 'ressource' : ''" | ||||
|               > | ||||
|                 <ModaleHeader | ||||
|                     :loading="loading" | ||||
|                     :contentType="contentType" | ||||
|                     :content="content" | ||||
|                     :couleur="content.couleur || brandColor" /> | ||||
|                 <main> | ||||
|                     <div v-for="partie in content.parties" class="partie"> | ||||
|                     <RessourceItemHeader | ||||
|                         v-if="contentType === 'ressourceItem'" | ||||
|                         :content="content" | ||||
|                         :couleur="brandColor" /> | ||||
|                     <div v-if="!loading" v-for="partie in content.parties" class="partie"> | ||||
|                         <ModaleCarteSensible | ||||
|                             v-if="partie.type === 'carte_sensible'" | ||||
|                             :partie="partie" /> | ||||
| @@ -42,31 +47,52 @@ | ||||
|                         <ModaleVideos | ||||
|                             v-if="partie.type === 'video'" | ||||
|                             :partie="partie" /> | ||||
|                           <ModaleGallerie | ||||
|                             v-if="partie.type === 'galleries'" | ||||
|                             :partie="partie" | ||||
|                             :couleur="content.couleur || brandColor" /> | ||||
|                           <ModaleDocument | ||||
|                             v-if="partie.type === 'document'" | ||||
|                             :partie="partie" | ||||
|                             :couleur="content.couleur || brandColor" /> | ||||
|                     </div> | ||||
|                     <EquipeContent | ||||
|                     <div class="content-loading" v-else> | ||||
|                       <div></div> | ||||
|                       <p>Chargement du contenu...</p> | ||||
|                     </div> | ||||
|                     <template v-if="!loading"> | ||||
|                       <EquipeContent | ||||
|                         v-if="contentType === 'gouvernance'" | ||||
|                         :content="content" | ||||
|                         :couleur="content.couleur || brandColor" /> | ||||
|                     <PartenairesContent | ||||
|                         :couleur="brandColor" /> | ||||
|                       <PartenairesContent | ||||
|                         v-if="contentType === 'partenaire'" | ||||
|                         :content="content" /> | ||||
|                       <CentreDeRessource | ||||
|                         v-if="contentType === 'ressource'" | ||||
|                         :content="content" | ||||
|                         :couleur="brandColor" /> | ||||
|                       <RelatedRessources | ||||
|                         v-if="contentType === 'etape' && content.relatedRessources?.length" | ||||
|                         :relatedRessources="content.relatedRessources" | ||||
|                         :couleur="content.couleur || brandColor" /> | ||||
|                     </template> | ||||
|                 </main> | ||||
|                 <PiecesJointes | ||||
|                   v-if="content.pieces_jointes || content.liens" | ||||
|                   :content="content" | ||||
|                   :couleur="content.couleur || brandColor" | ||||
|                   /> | ||||
|                   :couleur="content.couleur || brandColor" /> | ||||
|                 <ModaleFooter | ||||
|                 :contentType="contentType" | ||||
|                 :content="content" | ||||
|                 :couleur="content.couleur || brandColor" | ||||
|                 /> | ||||
|                   :contentType="contentType" | ||||
|                   :content="content" | ||||
|                   :couleur="content.couleur || brandColor" /> | ||||
|             </div> | ||||
|         </div> | ||||
|     </Transition> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { watch, onMounted } from 'vue'; | ||||
| import { watch, onMounted, nextTick } from 'vue'; | ||||
|  | ||||
| import { storeToRefs } from 'pinia'; | ||||
| import { useContentStore } from '../stores/content'; | ||||
| @@ -76,7 +102,10 @@ import ModaleHeader from './components/ModaleHeader.vue'; | ||||
| import ModaleFooter from './components/ModaleFooter.vue'; | ||||
| import EquipeContent from './components/EquipeContent.vue'; | ||||
| import PartenairesContent from './components/PartenairesContent.vue'; | ||||
| import CentreDeRessource from './components/CentreDeRessource.vue'; | ||||
| import RessourceItemHeader from './components/RessourceItemHeader.vue'; | ||||
| import PiecesJointes from './components/PiecesJointes.vue'; | ||||
| import RelatedRessources from './components/RelatedRessources.vue'; | ||||
|  | ||||
| import ModaleCarteSensible from './components/parties/ModaleCarteSensible.vue'; | ||||
| import ModaleTitreTexte from './components/parties/ModaleTitreTexte.vue'; | ||||
| @@ -85,6 +114,8 @@ import ModaleDiaporama from './components/parties/ModaleDiaporama.vue'; | ||||
| import ModaleEntretien from './components/parties/ModaleEntretien.vue'; | ||||
| import ModaleExergue from './components/parties/ModaleExergue.vue'; | ||||
| import ModaleVideos from './components/parties/ModaleVideos.vue'; | ||||
| import ModaleGallerie from './components/parties/ModaleGallerie.vue'; | ||||
| import ModaleDocument from './components/parties/ModaleDocument.vue'; | ||||
|  | ||||
| const store = useContentStore(); | ||||
| const mapState = useMapStore(); | ||||
| @@ -92,96 +123,56 @@ const mapState = useMapStore(); | ||||
| const { | ||||
|     contentType, | ||||
|     content, | ||||
|     partialLoading, | ||||
|     loading, | ||||
|     error, | ||||
| } = storeToRefs(store); | ||||
|  | ||||
| const { defaultMapCenter, animationDuration, animationsAreEnabled } = storeToRefs(mapState); | ||||
| const { animationsAreEnabled } = storeToRefs(mapState); | ||||
|  | ||||
| let isModaleEtape, wasModaleEtape; | ||||
|  | ||||
| const brandColor = "#80c8bf"; | ||||
|  | ||||
| const handleColorChange = () => { | ||||
|   watch( | ||||
|     () => content.value.couleur, | ||||
|     () => { | ||||
| onMounted(() => { | ||||
|   nextTick(() => { | ||||
|     watch( | ||||
|       () => content.value.couleur, | ||||
|       () => { | ||||
|         if (contentType.value === 'etape' && content.value.couleur) { | ||||
|             document.documentElement.style.setProperty('--etape-couleur', content.value.couleur || brandColor); | ||||
|           document.documentElement.style.setProperty('--etape-couleur', content.value.couleur || brandColor); | ||||
|         } | ||||
|     } | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| const handleMapMovement = () => { | ||||
|   watch( | ||||
|     () => loading.value, | ||||
|     () => { | ||||
|         if (!loading.value) { | ||||
|           console.log('loading done'); | ||||
|       } | ||||
|     ); | ||||
|     isModaleEtape = contentType.value === 'etape'; | ||||
|     wasModaleEtape = isModaleEtape; | ||||
|     watch( | ||||
|       () => partialLoading.value, | ||||
|       () => { | ||||
|         if (!partialLoading.value) { | ||||
|           isModaleEtape = contentType.value === 'etape'; | ||||
|  | ||||
|           // Define helper functions in variables | ||||
|           const disableModaleTransition = () => { | ||||
|             document.documentElement.style.setProperty('margin-top', '0'); | ||||
|             document.documentElement.style.setProperty('transition', 'none'); | ||||
|           } | ||||
|           const setModaleTransition = (enterDelay) => { | ||||
|             document.documentElement.style.setProperty('--modale-enter-delay', `${enterDelay}s`); | ||||
|           }; | ||||
|  | ||||
|           const zoomToContentPlace = () => { | ||||
|             mapState.zoomToPlace( | ||||
|               content.value.coordinates.lat ? content.value.coordinates.lat : defaultMapCenter.value.lat, | ||||
|               content.value.coordinates.lon ? content.value.coordinates.lon : defaultMapCenter.value.lng | ||||
|             ); | ||||
|           }; | ||||
|  | ||||
|           if (animationsAreEnabled.value) { | ||||
|  | ||||
|             if (isModaleEtape) { | ||||
|               if (!wasModaleEtape) { | ||||
|                 // national -> détail | ||||
|                 setModaleTransition(animationDuration.value); | ||||
|                 zoomToContentPlace(); | ||||
|               } else { | ||||
|                 // détail -> détail | ||||
|                 setModaleTransition(animationDuration.value); | ||||
|                 zoomToContentPlace(); | ||||
|               } | ||||
|             } else { | ||||
|               if (wasModaleEtape) { | ||||
|                 // détail -> national | ||||
|                 setModaleTransition(animationDuration.value); | ||||
|                 mapState.resetMap(); | ||||
|               } else { | ||||
|                 // national -> national | ||||
|                 setModaleTransition(0); | ||||
|               } | ||||
|             } | ||||
|           } else { | ||||
|             if (isModaleEtape) { | ||||
|               zoomToContentPlace(); | ||||
|             } else { | ||||
|               mapState.resetMap(); | ||||
|             } | ||||
|             disableModaleTransition(); | ||||
|           } | ||||
|  | ||||
|           mapState.handleMapMovement( | ||||
|             isModaleEtape, | ||||
|             wasModaleEtape, | ||||
|             content.value.coordinates?.lat, | ||||
|             content.value.coordinates?.lon | ||||
|           ); | ||||
|           scrollTo(0, 0); | ||||
|  | ||||
|           wasModaleEtape = isModaleEtape; | ||||
|         } | ||||
|     }, | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| onMounted(() => { | ||||
|   console.log('modale mounted'); | ||||
|   isModaleEtape = contentType.value === 'etape'; | ||||
|   wasModaleEtape = isModaleEtape; | ||||
|   handleColorChange(); | ||||
|   handleMapMovement(); | ||||
|       } | ||||
|     ); | ||||
|     watch( | ||||
|       () => contentType.value, | ||||
|       () => { | ||||
|         if (contentType.value === '') { | ||||
|           mapState.unlockMap(); | ||||
|         } else { | ||||
|           mapState.lockMap(); | ||||
|         } | ||||
|       } | ||||
|     ); | ||||
|   }); | ||||
| }); | ||||
| </script> | ||||
|  | ||||
|   | ||||
| @@ -0,0 +1,162 @@ | ||||
| <template> | ||||
|     <div id="centre-de-ressource" :style="{ '--couleur': couleur }"> | ||||
|         <div class="filters"> | ||||
|             <input type="text" v-model="searchQuery" placeholder="Rechercher..." class="search-bar"> | ||||
|             <select v-model="selectedType"> | ||||
|                 <option value="">Tous les types</option> | ||||
|                 <option v-for="type in content.ressourceTypes" :key="type" :value="type"> | ||||
|                     {{ type.replace(/_/g, ' ').replace(/^\w/, char => char.toUpperCase()) }} | ||||
|                 </option> | ||||
|             </select> | ||||
|             <select v-model="selectedEtape"> | ||||
|                 <option value="">Toutes les étapes</option> | ||||
|                 <option v-for="etape in allRelatedEtapes" :key="etape" :value="etape"> | ||||
|                     {{ etape }} | ||||
|                 </option> | ||||
|             </select> | ||||
|         </div> | ||||
|         <div v-if="content.intro" v-html="content.intro" class="intro"></div> | ||||
|         <template v-for="(type, typeIndex) in filteredTypes" :key="type"> | ||||
|             <div class="type-section" v-if="ressourcesToDisplay[type] && ressourcesToDisplay[type].length > 0"> | ||||
|                 <h3>{{ type.replace(/_/g, ' ').replace(/^\w/, char => char.toUpperCase()) }}</h3> | ||||
|                 <div class="ressource-list"> | ||||
|                     <TransitionGroup name="itemFade" tag="div" appear> | ||||
|                         <div class="ressource-item" | ||||
|                             v-for="(ressource, ressourceIndex) in ressourcesToDisplay[type]" | ||||
|                             :key="`${type}-${ressourceIndex}`" | ||||
|                             :style="{ '--index': ressourceIndex - visibleItemsPerSection }"> | ||||
|                             <RessourceCard :ressource="ressource" :index="`${typeIndex}-${ressourceIndex}`" /> | ||||
|                         </div> | ||||
|                     </TransitionGroup> | ||||
|                 </div> | ||||
|                 <div class="button-container"> | ||||
|                     <div v-if="ressourcesByType(type).length > visibleItemsPerSection | ||||
|                         && ressourcesToDisplay[type].length === visibleItemsPerSection" | ||||
|                          class="ressource-button" | ||||
|                          @click="loadMore(type)"> | ||||
|                         Voir plus ↓ | ||||
|                     </div> | ||||
|                     <div v-if="ressourcesToDisplay[type].length > visibleItemsPerSection" | ||||
|                          class="ressource-button" | ||||
|                          @click="showLess(type)"> | ||||
|                         Voir moins ↑ | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </template> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import RessourceCard from './RessourceCard.vue'; | ||||
| import useParseDate from '../composables/useParseDates'; | ||||
|  | ||||
| import { ref, computed, watch, nextTick } from 'vue'; | ||||
|  | ||||
| const props = defineProps({ | ||||
|   content: Object, | ||||
|   couleur: String, | ||||
| }); | ||||
|  | ||||
| const searchQuery = ref(''); | ||||
| const selectedType = ref(''); | ||||
| const ressourcesToDisplay = ref({}); | ||||
| const visibleItemsPerSection = 4; | ||||
| const selectedEtape = ref(''); | ||||
| const allRelatedEtapes = new Set(); | ||||
|  | ||||
| props.content.ressources.forEach(ressource => { | ||||
|     if (ressource.relatedEtape) { | ||||
|         allRelatedEtapes.add(ressource.relatedEtape); | ||||
|     } | ||||
| }) | ||||
|  | ||||
| const filteredTypes = computed(() => { | ||||
|     return selectedType.value ? [selectedType.value] : props.content.ressourceTypes; | ||||
| }); | ||||
|  | ||||
| const ressourcesByType = (type) => { | ||||
|     return props.content.ressources | ||||
|         .filter(ressource => !selectedEtape.value || ressource.relatedEtape === selectedEtape.value) | ||||
|         .filter(ressource => ressource.ressourceType === type) | ||||
|         .filter(ressource => | ||||
|             searchQuery.value === '' || | ||||
|             ressource.title.toLowerCase().includes(searchQuery.value.toLowerCase()) | ||||
|         ).sort((a, b) => { | ||||
|             if (a.promoted && !b.promoted) return -1; | ||||
|             if (!a.promoted && b.promoted) return 1; | ||||
|  | ||||
|             const dateA = useParseDate(a.date); | ||||
|             const dateB = useParseDate(b.date); | ||||
|              | ||||
|             return dateB - dateA; // descending order (newest first) | ||||
|         });         | ||||
| }; | ||||
|  | ||||
| const initializeRessources = (type) => { | ||||
|     ressourcesToDisplay.value[type] = ressourcesByType(type).slice(0, visibleItemsPerSection); | ||||
| }; | ||||
|  | ||||
| props.content.ressourceTypes.forEach(type => initializeRessources(type)); | ||||
|  | ||||
| const loadMore = (type) => { | ||||
|     const currentLength = ressourcesToDisplay.value[type].length; | ||||
|  | ||||
|     if (currentLength < ressourcesByType(type).length) { | ||||
|         ressourcesToDisplay.value[type] = ressourcesByType(type).slice(0, currentLength + visibleItemsPerSection); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const showLess = (type) => { | ||||
|     ressourcesToDisplay.value[type] = ressourcesByType(type).slice(0, visibleItemsPerSection); | ||||
| }; | ||||
|  | ||||
| watch(searchQuery, () => { | ||||
|     props.content.ressourceTypes.forEach(type => { | ||||
|         initializeRessources(type); | ||||
|     }); | ||||
| }); | ||||
|  | ||||
| watch(ressourcesToDisplay.value, async () => { | ||||
|     await nextTick(); | ||||
|     document.querySelectorAll('.ressource-item > div').forEach(el => { | ||||
|         const card = el.__vueParentComponent.exposed; | ||||
|         if (card && card.setClickableElements) { | ||||
|             card.setClickableElements(); | ||||
|         } | ||||
|     }); | ||||
| }, { deep: true }); | ||||
|  | ||||
| watch(selectedType, async () => { | ||||
|     await nextTick(); | ||||
|     document.querySelectorAll('.ressource-item > div').forEach(el => { | ||||
|         const card = el.__vueParentComponent.exposed; | ||||
|         if (card && card.setClickableElements) { | ||||
|             card.setClickableElements(); | ||||
|         } | ||||
|     }); | ||||
| }); | ||||
|  | ||||
| watch(selectedEtape, () => { | ||||
|   props.content.ressourceTypes.forEach(type => { | ||||
|     initializeRessources(type); | ||||
|   }); | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .itemFade-enter-active, .itemFade-leave-active { | ||||
|   transition: all 0.3s ease !important; | ||||
|   transition-delay: calc(0.1s * var(--index)) !important; | ||||
| } | ||||
|  | ||||
| .itemFade-enter-from, .itemFade-leave-to { | ||||
|   opacity: 0; | ||||
|   transform: translateY(-20px) !important; | ||||
| } | ||||
|  | ||||
| .itemFade-enter-to, .itemFade-leave-from { | ||||
|   opacity: 1; | ||||
|   transform: translateY(0px); | ||||
| } | ||||
| </style> | ||||
| @@ -0,0 +1,34 @@ | ||||
| <template> | ||||
|     <div class="card" :class="direction ? (direction === 'previous' ? 'previous' : 'next') : 'solo'" :data-href="relatedContent.url"> | ||||
|         <div class="icon" :style="{ backgroundColor: relatedContent.couleur }"></div> | ||||
|         <div class="card-content"> | ||||
|             <div class="infos"> | ||||
|                 <div class="titre">{{ relatedContent.title }} <span>({{ relatedContent.postalCode.slice(0, 2) }})</span></div> | ||||
|                 <div class="date">Du {{ relatedContent.dates.start.d }} {{ relatedContent.dates.start.m }}<br>au {{ relatedContent.dates.end.d }} {{ relatedContent.dates.end.m }} {{ relatedContent.dates.end.y }}</div> | ||||
|             </div> | ||||
|             <div class="vignette"> | ||||
|                 <img :src="relatedContent.vignette.url.small" :alt="relatedContent.vignette.alt"> | ||||
|             </div> | ||||
|         </div> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| const props = defineProps({ | ||||
|   relatedContent: Object, | ||||
|   direction: String, | ||||
| }); | ||||
|  | ||||
| import { onMounted } from 'vue'; | ||||
| import router from '../../router/router'; | ||||
|  | ||||
| import { handleClickableElements } from '../../utils/handle-navigation.js'; | ||||
|  | ||||
| const siteName = document.querySelector('#site_name').innerText; | ||||
|  | ||||
| onMounted(() => { | ||||
|     const relatedEtapesCards = document.querySelectorAll('.card'); | ||||
|     const baseUrl = window.location.protocol + "//" + window.location.host; | ||||
|     handleClickableElements(relatedEtapesCards, router, baseUrl, siteName); | ||||
| }); | ||||
| </script> | ||||
| @@ -16,6 +16,7 @@ | ||||
|             :navigation="true" | ||||
|             :pagination="true" | ||||
|             :initialSlide="currentSlideIndex" | ||||
|             :keyboard="{ enabled: true }" | ||||
|             :injectStyles="[` | ||||
|               .swiper-button-next, .swiper-button-prev {  | ||||
|                 color: white; | ||||
|   | ||||
| @@ -1,56 +1,24 @@ | ||||
| <template> | ||||
|     <footer> | ||||
|     <footer :class="contentType === 'ressourceItem' ? 'footer-ressource' : ''"> | ||||
|         <div class="brand-pattern pattern-bottom" :style="{ backgroundColor: couleur }"> | ||||
|             <div class="pattern"></div> | ||||
|         </div> | ||||
|  | ||||
|         <div v-if="contentType === 'etape' && (content.previous || content.next)" class="related-etape-links"> | ||||
|             <div v-if="content.previous" class="card previous" @click="goToRelatedElement(content.previous.url)"> | ||||
|                 <div class="icon"> | ||||
|                     <div :style="{ backgroundColor: content.previous.couleur }"></div> | ||||
|                     <div :style="{ backgroundColor: content.previous.couleur }"></div> | ||||
|                     <div :style="{ backgroundColor: content.previous.couleur }"></div> | ||||
|                 </div> | ||||
|                 <div class="card-content"> | ||||
|                     <div class="infos"> | ||||
|                         <div class="titre">{{ content.previous.title }} <span>({{ content.previous.postalCode.slice(0, 2) }})</span></div> | ||||
|                         <div class="date">Du {{ content.previous.dates.start.d }} {{ content.previous.dates.start.m }}<br>au {{ content.previous.dates.end.d }} {{ content.previous.dates.end.m }} {{ content.previous.dates.end.y }}</div> | ||||
|                     </div> | ||||
|                     <div class="vignette"> | ||||
|                         <img :src="content.previous.vignette.url.small" :alt="content.previous.vignette.alt"> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|             <div v-if="content.next" class="card next" @click="goToRelatedElement(content.next.url)"> | ||||
|                 <div class="icon"> | ||||
|                     <div :style="{ backgroundColor: content.next.couleur }"></div> | ||||
|                     <div :style="{ backgroundColor: content.next.couleur }"></div> | ||||
|                     <div :style="{ backgroundColor: content.next.couleur }"></div> | ||||
|                 </div> | ||||
|                 <div class="card-content"> | ||||
|                     <div class="infos"> | ||||
|                         <div class="titre">{{ content.next.title }} <span>({{ content.next.postalCode.slice(0, 2) }})</span></div> | ||||
|                         <div class="date">Du {{ content.next.dates.start.d }} {{ content.next.dates.start.m }}<br>au {{ content.next.dates.end.d }} {{ content.next.dates.end.m }} {{ content.next.dates.end.y }}</div> | ||||
|                     </div> | ||||
|                     <div class="vignette"> | ||||
|                         <img :src="content.next.vignette.url.small" :alt="content.next.vignette.alt"> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             <EtapeCard v-if="content.previous" :relatedContent="content.previous" :direction="'previous'"/> | ||||
|             <EtapeCard v-if="content.next" :relatedContent="content.next" :direction="'next'"/> | ||||
|         </div> | ||||
|         <div v-if="contentType === 'ressourceItem' && content.relatedEtape" > | ||||
|             <div class="related-etape-label" :style="{ backgroundColor: couleur }">Étape de la ressource</div> | ||||
|             <div class="related-etape-links"> | ||||
|                 <EtapeCard :relatedContent="content.relatedEtape" :direction="''" /> | ||||
|             </div> | ||||
|         </div> | ||||
|     </footer> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { useContentStore } from '../../stores/content'; | ||||
| import { pageChange } from '../../utils/handle-navigation.js'; | ||||
|  | ||||
| const brandColor = "#80c8bf"; | ||||
|  | ||||
| const store = useContentStore(); | ||||
| const mapStore = useContentStore(); | ||||
| const siteName = document.querySelector('#site_name').innerText; | ||||
|  | ||||
| import EtapeCard from './EtapeCard.vue'; | ||||
|  | ||||
| const props = defineProps({ | ||||
|   contentType: String, | ||||
| @@ -58,11 +26,4 @@ const props = defineProps({ | ||||
|   couleur: String, | ||||
|   map: Object, | ||||
| }); | ||||
|  | ||||
| async function goToRelatedElement(href) { | ||||
|   const baseUrl = window.location.protocol + "//" + window.location.host; | ||||
|   if (href.startsWith(baseUrl)) href = href.replace(baseUrl, ''); | ||||
|   pageChange(href, store, siteName, mapStore, baseUrl) | ||||
| } | ||||
|  | ||||
| </script> | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
|     <header :class="{ 'not-etape': contentType !== 'etape' }"> | ||||
|         <div class="cover"> | ||||
|             <img v-if="content.vignette" :src="content.vignette.url.medium" :alt="content.vignette.alt"> | ||||
|         <div v-if="contentType === 'etape'" class="cover" :style="{ backgroundColor: `${couleur.substring(0, 7)}99` }"> | ||||
|             <img v-if="!loading && content.vignette" :src="content.vignette.url.medium" :alt="content.vignette.alt"> | ||||
|         </div> | ||||
|         <div v-if="contentType === 'etape' && content.dates" class="cartouche" :style="{ backgroundColor: couleur }"> | ||||
|             <p>Étape n°{{content.etape_number}}</p> | ||||
| @@ -14,7 +14,7 @@ | ||||
|             <div class="locality"> | ||||
|                 <div class="top-triangle" v-if="contentType === 'etape'"></div> | ||||
|                 <div class="locality-title"> | ||||
|                     <h1>{{content.contentTitle}} <em v-if="content.adresse">({{ content.adresse.postal_code.slice(0, 2) }})</em></h1> | ||||
|                     <h1>{{ contentType === 'ressourceItem' ? 'Centre de ressources' : content.contentTitle }} <em v-if="content.adresse">({{ content.adresse.postal_code.slice(0, 2) }})</em></h1> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
| @@ -23,6 +23,7 @@ | ||||
|  | ||||
| <script setup> | ||||
| const props = defineProps({ | ||||
|   loading: Boolean, | ||||
|   contentType: String, | ||||
|   content: Object, | ||||
|   couleur: String, | ||||
|   | ||||
| @@ -4,13 +4,13 @@ | ||||
|       <p v-html="content.intro"></p> | ||||
|     </div> | ||||
|     <div v-for="partenaire in [...content.partenaires].sort((a, b) => a.weight - b.weight)" class="partenaire"> | ||||
|       <figure> | ||||
|       <figure v-if="partenaire.logo_url && partenaire.link_url"> | ||||
|         <a :href="partenaire.link_url" target="_blank"> | ||||
|           <img :src="partenaire.logo_url.small" :alt="partenaire.logo_alt"> | ||||
|         </a> | ||||
|       </figure> | ||||
|       <div class="title"><p v-html="partenaire.title"></p></div> | ||||
|       <div class="description"><p v-html="partenaire.description"></p></div> | ||||
|       <div class="title"><p v-html="partenaire.title || ''"></p></div> | ||||
|       <div class="description"><p v-html="partenaire.description || ''"></p></div> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
|   | ||||
| @@ -0,0 +1,23 @@ | ||||
| <template> | ||||
|     <div class="partie related-ressources"> | ||||
|         <h3> | ||||
|             <p :style="{ background: `linear-gradient(transparent 70%, ${couleur} 70%)` }"> | ||||
|                 Ressources liées | ||||
|             </p> | ||||
|         </h3> | ||||
|         <div class="ressource-list sm-ressource-list"> | ||||
|             <div class="ressource-item" v-for="(relatedRessource, index) in relatedRessources"> | ||||
|                 <RessourceCard :ressource="relatedRessource" :index="index" /> | ||||
|             </div> | ||||
|         </div> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import RessourceCard from './RessourceCard.vue'; | ||||
|  | ||||
| const props = defineProps({ | ||||
|   relatedRessources: Object, | ||||
|   couleur: String, | ||||
| }); | ||||
| </script> | ||||
| @@ -0,0 +1,43 @@ | ||||
| <template> | ||||
|     <div  | ||||
|     :data-href="ressource.url" | ||||
|     :id="`ressource-${index}`" | ||||
|     :class="ressource.promoted ? 'promoted' : ''"> | ||||
|         <figure> | ||||
|             <img :src="ressource?.vignette.url" :alt="ressource?.vignette.alt" /> | ||||
|         </figure> | ||||
|         <div> | ||||
|             <h4>{{ ressource.title }}</h4> | ||||
|             <p v-if="ressource.date">Le {{ ressource?.date.d }} {{ ressource?.date.m }} {{ ressource?.date.y }}</p> | ||||
|             <p v-if="ressource.auteurice">Par {{ ressource?.auteurice }}</p> | ||||
|         </div> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { onMounted } from 'vue'; | ||||
| import router from '../../router/router'; | ||||
| import { handleClickableElements } from '../../utils/handle-navigation.js'; | ||||
|  | ||||
| const siteName = document.querySelector('#site_name').innerText; | ||||
|  | ||||
| let relatedItemCards, baseUrl; | ||||
|  | ||||
| onMounted(() => { | ||||
|     baseUrl = window.location.protocol + "//" + window.location.host; | ||||
|     setClickableElements(); | ||||
| }); | ||||
|  | ||||
| const setClickableElements = () => { | ||||
|     relatedItemCards = document.querySelector(`#ressource-${props.index}`); | ||||
|     handleClickableElements([relatedItemCards], router, baseUrl, siteName); | ||||
| } | ||||
|  | ||||
| defineExpose({ | ||||
|     setClickableElements, | ||||
| }); | ||||
| const props = defineProps({ | ||||
|   ressource: Object, | ||||
|   index: String, | ||||
| }); | ||||
| </script> | ||||
| @@ -0,0 +1,60 @@ | ||||
| <template> | ||||
|   <header id="ressource-item-header"> | ||||
|     <div class="retour"> | ||||
|       <p data-href="/ressources">← Retour au centre de ressources</p> | ||||
|     </div> | ||||
|     <div class="type">{{ content?.displayedType }}</div> | ||||
|     <div class="title"> | ||||
|       <h2 | ||||
|       :style="{ background: `linear-gradient(transparent 70%, ${couleur} 70%)` }"> | ||||
|         {{ content?.contentTitle }} | ||||
|       </h2> | ||||
|     </div> | ||||
|     <div v-if="content.auteurice && content.date" class="meta"> | ||||
|       Par {{ content?.auteurice }}, le {{ content?.date.d }} {{ content?.date.m }} {{ content?.date.y }} | ||||
|     </div> | ||||
|   </header> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { onMounted } from 'vue'; | ||||
| import router from '../../router/router'; | ||||
|  | ||||
| import { handleClickableElements } from '../../utils/handle-navigation.js'; | ||||
|  | ||||
| const siteName = document.querySelector('#site_name')?.innerText; | ||||
|  | ||||
| const props = defineProps({ | ||||
|   content: Object, | ||||
|   couleur: String, | ||||
| }); | ||||
|  | ||||
| function setDisplayedType() { | ||||
|     const ressourceType = props.content?.ressourceType; | ||||
|     switch (ressourceType) { | ||||
|       case 'cartes_blanches': | ||||
|         props.content.displayedType = 'Carte blanche'; | ||||
|         break; | ||||
|       case 'documents': | ||||
|         props.content.displayedType = 'Document'; | ||||
|         break; | ||||
|       case 'galleries': | ||||
|         props.content.displayedType = 'Galerie'; | ||||
|         break; | ||||
|       case 'videos': | ||||
|         props.content.displayedType = 'Vidéo'; | ||||
|         break; | ||||
|       case 'reportages': | ||||
|         props.content.displayedType = 'Reportage'; | ||||
|         break; | ||||
|     } | ||||
|      | ||||
| } | ||||
|  | ||||
| onMounted(() => { | ||||
|     const backToRessourcesLink = document.querySelectorAll('.retour > p'); | ||||
|     const baseUrl = window.location.protocol + "//" + window.location.host; | ||||
|     handleClickableElements(backToRessourcesLink, router, baseUrl, siteName); | ||||
|     setDisplayedType(); | ||||
| }); | ||||
| </script> | ||||
| @@ -0,0 +1,50 @@ | ||||
| <template> | ||||
|     <div class="document" | ||||
|     :style="{ '--couleur': couleur }"> | ||||
|         <div  | ||||
|         class="intro"  | ||||
|         v-html="partie.document.description"> | ||||
|         </div> | ||||
|         <div class="document-grid"> | ||||
|             <figure> | ||||
|                 <a :href="partie.document.url" target="_blank"> | ||||
|                     <img :src="partie.document.vignette.url" :alt="partie.document.vignette.alt" /> | ||||
|                 </a> | ||||
|             </figure> | ||||
|             <div class="infos"> | ||||
|                 <p>{{ partie.document.titre }}</p> | ||||
|                 <p>{{ partie.document.sousTitre }}</p> | ||||
|                 <p>{{ partie.document.auteurice }}</p> | ||||
|                 <p>{{ partie.document.date?.m }} {{ partie.document.date?.y }}</p> | ||||
|             </div> | ||||
|             <div class="download"> | ||||
|                 <a :href="partie.document.url" target="_blank">Télécharger le document</a> | ||||
|             </div> | ||||
|         </div> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| const props = defineProps({ | ||||
|     partie: Object, | ||||
|     couleur: String, | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .intro { | ||||
|     position: relative; | ||||
|     padding-left: 1.8rem; | ||||
|     margin: 3rem 0; | ||||
|     &::before { | ||||
|         content: ''; | ||||
|         display: block; | ||||
|         position: absolute; | ||||
|         background-color: var(--couleur); | ||||
|         width: 0.8rem; | ||||
|         height: 100%; | ||||
|         margin-right: 1rem; | ||||
|         left: 0; | ||||
|     } | ||||
| } | ||||
| </style> | ||||
| @@ -0,0 +1,84 @@ | ||||
| <template> | ||||
|     <div class="gallerie"> | ||||
|         <h3><p :style="{ background: `linear-gradient(transparent 70%, ${couleur} 70%)` }">{{ partie.gallerie.titre }}</p></h3> | ||||
|         <div class="intro" v-html="partie.gallerie.introduction"></div> | ||||
|         <div class="images-grid"> | ||||
|             <figure v-for="image in partie.gallerie.images"> | ||||
|                 <img :src="image.url.medium" :alt="image.alt" @click="handleImageClick"> | ||||
|                 <figcaption class="caption">{{ image.alt }}</figcaption> | ||||
|             </figure> | ||||
|         </div> | ||||
|     </div> | ||||
|  | ||||
|     <ImageModale | ||||
|         :isOpen="isModaleOpen" | ||||
|         :image="currentImage" | ||||
|         :swiperContent="swiperPopupContent" | ||||
|         @close="closeImageModale" /> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { onMounted } from 'vue'; | ||||
| import { useImageModal } from '../../composables/useImageModale'; | ||||
| import ImageModale from '../ImageModale.vue'; | ||||
| // WebComponent | ||||
| // https://swiperjs.com/element | ||||
| import { register } from 'swiper/element/bundle'; | ||||
| register(); | ||||
|  | ||||
| const props = defineProps({ | ||||
|     partie: Object, | ||||
|     couleur: String, | ||||
| }); | ||||
|  | ||||
| const {  | ||||
|   isModaleOpen,  | ||||
|   currentImage,  | ||||
|   swiperPopupContent,  | ||||
|   openImageModale,  | ||||
|   closeImageModale  | ||||
| } = useImageModal(); | ||||
|  | ||||
| const handleImageClick = (event) => { | ||||
|     const clickedImg = event.target; | ||||
|     if (clickedImg.tagName === 'IMG') { | ||||
|         let swiperMedia = []; | ||||
|         const imgGrid = clickedImg.closest('.images-grid'); | ||||
|  | ||||
|         imgGrid.querySelectorAll('figure').forEach((figure) => { | ||||
|             const img = figure.querySelector('img'); | ||||
|             const altText = figure.querySelector('figcaption')?.textContent || ''; | ||||
|             Object.values(props.partie.gallerie.images).forEach((image) => { | ||||
|                 if (img.src === image.url.medium) { | ||||
|                     swiperMedia.push({ src: image.url.large, alt: altText }); | ||||
|                 } | ||||
|             }); | ||||
|         }); | ||||
|         openImageModale(clickedImg.src, clickedImg.alt, swiperMedia);         | ||||
|     } | ||||
| } | ||||
|  | ||||
| onMounted(() => { | ||||
|     setVerticalImages(); | ||||
|     document.documentElement.style.setProperty('--etape-couleur', props.couleur); | ||||
| }); | ||||
|  | ||||
| function setVerticalImages() { | ||||
|     const images = document.querySelectorAll('.images-grid figure img'); | ||||
|     images.forEach((img) => { | ||||
|         if (img.naturalHeight > img.naturalWidth) { | ||||
|             img.classList.add('vertical'); | ||||
|             img.closest('figure').querySelector('figcaption').style.textAlign = 'center'; | ||||
|         } | ||||
|     }); | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| :root { | ||||
|   --swiper-navigation-color: #1a1918; /* cf main.scss */ | ||||
|   --swiper-pagination-color: var(--etape-couleur); | ||||
|   --swiper-navigation-top-offset: calc(100% - 1.5rem); | ||||
|   --swiper-navigation-sides-offset: 5vw; /* cf main.scss */ | ||||
| } | ||||
| </style> | ||||
| @@ -0,0 +1,26 @@ | ||||
| const frenchMonthMap = { | ||||
|     janvier: 0, | ||||
|     février: 1, | ||||
|     mars: 2, | ||||
|     avril: 3, | ||||
|     mai: 4, | ||||
|     juin: 5, | ||||
|     juillet: 6, | ||||
|     août: 7, | ||||
|     septembre: 8, | ||||
|     octobre: 9, | ||||
|     novembre: 10, | ||||
|     décembre: 11 | ||||
|   }; | ||||
|    | ||||
| export default function parseDate(dateObj) { | ||||
|     if (!dateObj || !frenchMonthMap[dateObj.m?.toLowerCase()]) return new Date(0); | ||||
|    | ||||
|     const day = Number(dateObj.d); | ||||
|     const month = frenchMonthMap[dateObj.m.toLowerCase()]; | ||||
|     const year = Number(dateObj.y); | ||||
|    | ||||
|     if (isNaN(day) || isNaN(month) || isNaN(year)) return new Date(0); | ||||
|    | ||||
|     return new Date(year, month, day); | ||||
|   } | ||||
							
								
								
									
										1
									
								
								web/themes/custom/caravane/assets/pictograms/90-ring.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								web/themes/custom/caravane/assets/pictograms/90-ring.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| <svg width="24" height="24" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><style>.spinner_P7sC{transform-origin:center;animation:spinner_svv2 .75s infinite linear}@keyframes spinner_svv2{100%{transform:rotate(360deg)}}</style><path d="M10.14,1.16a11,11,0,0,0-9,8.92A1.59,1.59,0,0,0,2.46,12,1.52,1.52,0,0,0,4.11,10.7a8,8,0,0,1,6.66-6.61A1.42,1.42,0,0,0,12,2.69h0A1.57,1.57,0,0,0,10.14,1.16Z" class="spinner_P7sC"/></svg> | ||||
| After Width: | Height: | Size: 428 B | 
| @@ -0,0 +1 @@ | ||||
| <svg width="20" xmlns="http://www.w3.org/2000/svg" height="24" fill="none"><path d="M19.996,13.728L19.992,17.434L9.964,24.000L0.000,17.500L0.000,13.541L0.000,10.459L0.000,6.500L9.964,0.000L19.992,6.566L19.996,10.272L20.000,10.272L19.998,12.000L19.996,13.728ZL19.996,13.728ZL19.998,12.000" style="fill: rgb(0, 0, 0); fill-opacity: 1;" class="fills"/></svg> | ||||
| After Width: | Height: | Size: 355 B | 
| @@ -15,6 +15,9 @@ $modale-width-desktop: 50vw; | ||||
|  | ||||
| $brand-pattern-height: 110px; | ||||
|  | ||||
| $xsm-font-size-mobile: 0.5rem; | ||||
| $xsm-font-size-desktop: 0.6rem; | ||||
|  | ||||
| $sm-font-size-mobile: 0.6rem; | ||||
| $sm-font-size-desktop: 0.8rem; | ||||
|  | ||||
| @@ -163,7 +166,7 @@ body{ | ||||
|             } | ||||
|             > #hamburger { | ||||
|               position: fixed; | ||||
|               height: 5vh; | ||||
|               height: max(5vh, 40px); | ||||
|               right: 0; | ||||
|               top: 0; | ||||
|               margin-right: $body-margin-x; | ||||
| @@ -343,27 +346,15 @@ body{ | ||||
|                   left: 0; | ||||
|                 } | ||||
|                 .leaflet-map-divicon { | ||||
|                   width: 10px; | ||||
|                   height: 30px; | ||||
|                   display: flex; | ||||
|                   flex-direction: column; | ||||
|                   justify-content: center; | ||||
|                   align-items: center; | ||||
|                   width: 20px !important; | ||||
|                   height: 20px !important; | ||||
|                   transition: transform 0.3s ease-out; | ||||
|                   > div { | ||||
|                     background-color: red; | ||||
|                     display: block; | ||||
|                     width: 20px; | ||||
|                     height: 10px; | ||||
|                     } | ||||
|                   > div:first-of-type { | ||||
|                     height: 8px; | ||||
|                     clip-path: polygon(0 0, 100% 0, 50% 100%); | ||||
|                     transform: rotate(180deg); | ||||
|                   } | ||||
|                   > div:nth-of-type(3) { | ||||
|                     height: 8px; | ||||
|                     clip-path: polygon(0 0, 100% 0, 50% 100%); | ||||
|                   transform-origin: center; | ||||
|                   mask-image: url("/themes/custom/caravane/assets/pictograms/hexagone.svg"); | ||||
|                   mask-size: contain; | ||||
|                   mask-repeat: no-repeat; | ||||
|                   > .url, .couleur { | ||||
|                     display: none; | ||||
|                   } | ||||
|                 } | ||||
|               } | ||||
| @@ -689,24 +680,12 @@ body{ | ||||
|                       } | ||||
|                     } | ||||
|                     > .icone-arret { | ||||
|                       width: 10px; | ||||
|                       height: 30px; | ||||
|                       display: flex; | ||||
|                       flex-direction: column; | ||||
|                       justify-content: center; | ||||
|                       align-items: center; | ||||
|                       > div { | ||||
|                         display: block; | ||||
|                         width: 20px; | ||||
|                         height: 10px; | ||||
|                         &:first-of-type, &:last-of-type { | ||||
|                           height: 8px; | ||||
|                           clip-path: polygon(0 0, 100% 0, 50% 100%); | ||||
|                         } | ||||
|                         &:first-of-type { | ||||
|                           transform: rotate(180deg); | ||||
|                         } | ||||
|                       } | ||||
|                       min-width: 20px; | ||||
|                       height: 20px; | ||||
|                       display: block; | ||||
|                       mask-image: url("/themes/custom/caravane/assets/pictograms/hexagone.svg"); | ||||
|                       mask-size: contain; | ||||
|                       mask-repeat: no-repeat; | ||||
|                     } | ||||
|                   } | ||||
|                   > li:hover { | ||||
| @@ -775,6 +754,7 @@ body{ | ||||
|               } | ||||
|             } | ||||
|             > #content-modale { | ||||
|               pointer-events: none; | ||||
|               padding-bottom: 20vh; | ||||
|               @media screen and (min-width: $desktop-min-width) { | ||||
|                 z-index: 6; | ||||
| @@ -782,6 +762,7 @@ body{ | ||||
|               > div:not(.image-viewer-wrapper, .image-modale) { | ||||
|                 padding-bottom: 5vh; | ||||
|                 > .content-wrapper { | ||||
|                   pointer-events: all; | ||||
|                   left: 1.5vw; | ||||
|                   width: calc($modale-width-mobile); | ||||
|                   top: 15vh; | ||||
| @@ -789,11 +770,29 @@ body{ | ||||
|                   position: relative; | ||||
|                   background-color: white; | ||||
|                   font-size: $labeur-font-size-mobile; | ||||
|                   //padding-bottom: $modale-bottom-padding; | ||||
|                   &.ressource { | ||||
|                     left: 5vw; | ||||
|                   } | ||||
|                   @media screen and (min-width: $desktop-min-width) { | ||||
|                     font-size: $labeur-font-size-desktop; | ||||
|                     width: $modale-width-desktop; | ||||
|                   } | ||||
|                   &.ressource { | ||||
|                     @media screen and (min-width: $tablet-min-width) { | ||||
|                       left: 8vw; | ||||
|                       width: 84vw; | ||||
|                       .locality-title { | ||||
|                         width: 42vw !important; | ||||
|                         margin-left: 21vw !important; | ||||
|                       } | ||||
|                     } | ||||
|                     @media screen and (min-width: $desktop-min-width) { | ||||
|                       .locality-title { | ||||
|                         width: 30vw; | ||||
|                         margin-left: 27vw; | ||||
|                       } | ||||
|                     } | ||||
|                   } | ||||
|                   img { | ||||
|                     width: 100%; | ||||
|                     height: auto; | ||||
| @@ -809,6 +808,7 @@ body{ | ||||
|                     grid-template-rows: auto auto auto auto; | ||||
|                     > .cover { | ||||
|                       grid-row: 1 / span 1; | ||||
|                       min-height: 150px; | ||||
|                       max-height: 60vh; | ||||
|                       display: flex; | ||||
|                       justify-content: center; | ||||
| @@ -926,10 +926,41 @@ body{ | ||||
|                     } | ||||
|                   } | ||||
|                   > main { | ||||
|                     min-height: 30vh; | ||||
|                     z-index: 1; | ||||
|                     position: relative; | ||||
|                     width: 100%; | ||||
|                     padding: 0 $modale-x-padding; | ||||
|                     padding-bottom: 5vh; | ||||
|                     box-sizing: border-box; | ||||
|                     &:has(#ressource-item-header) { | ||||
|                       // padding-right: 50%; | ||||
|                       margin-bottom: 3rem; | ||||
|                       @media screen and (min-width: $desktop-min-width) { | ||||
|                         margin-bottom: 6rem; | ||||
|                         padding: 0 calc($modale-x-padding * 4); | ||||
|                       } | ||||
|                     } | ||||
|                     .content-loading { | ||||
|                       width: 100%; | ||||
|                       text-align: center; | ||||
|                       padding-top: 2rem; | ||||
|                       display: flex; | ||||
|                       flex-direction: column; | ||||
|                       align-items: center; | ||||
|                       > div { | ||||
|                         display: block; | ||||
|                         width: 24px; | ||||
|                         height: 24px; | ||||
|                         background-image: url(/themes/custom/caravane/assets/pictograms/90-ring.svg); | ||||
|                         background-size: 24px; | ||||
|                         background-size: no-repeat; | ||||
|                         margin-bottom: 1rem; | ||||
|                       } | ||||
|                       > p { | ||||
|                         display: block; | ||||
|                       } | ||||
|                     } | ||||
|                     > .partie, | ||||
|                     > #equipe { | ||||
|                       width: 100%; | ||||
| @@ -957,7 +988,9 @@ body{ | ||||
|                       } | ||||
|                       .partie-title, | ||||
|                       > .chiffres-cles, | ||||
|                       > .entretien { | ||||
|                       > .entretien, | ||||
|                       > .gallerie, | ||||
|                       &.related-ressources { | ||||
|                         > h3 { | ||||
|                           position: relative; | ||||
|                           display: inline-block; | ||||
| @@ -1094,6 +1127,99 @@ body{ | ||||
|                       > .videos iframe { | ||||
|                         margin: 2rem 0; | ||||
|                       } | ||||
|                       > .gallerie { | ||||
|                         > .intro { | ||||
|                           margin-bottom: 2rem; | ||||
|                         } | ||||
|                         > .images-grid { | ||||
|                           margin-top: 3rem; | ||||
|                           display: grid; | ||||
|                           grid-template-columns: 1fr 1fr; | ||||
|                           column-gap: 5rem; | ||||
|                           row-gap: 2rem; | ||||
|                           @media screen and (min-width: $desktop-min-width) { | ||||
|                             grid-template-columns: repeat(3, 1fr); | ||||
|                           } | ||||
|                         } | ||||
|                         figure { | ||||
|                           margin: 0; | ||||
|                           > img { | ||||
|                             cursor: pointer; | ||||
|                             transform: scale(1); | ||||
|                             transition: transform 0.2s ease-in-out; | ||||
|                             &:hover { | ||||
|                               transform: scale(1.02); | ||||
|                             } | ||||
|                             &.vertical { | ||||
|                               box-sizing: border-box; | ||||
|                               padding: 0 25%; | ||||
|                             } | ||||
|                           } | ||||
|                         } | ||||
|                       } | ||||
|                       > .document { | ||||
|                         .download { | ||||
|                           > a { | ||||
|                             color: $main-color; | ||||
|                             text-decoration: none; | ||||
|                             font-size: $sm-font-size-mobile; | ||||
|                             background-color: $brand-color; | ||||
|                             font-weight: bold; | ||||
|                             padding: 0.5rem 1rem; | ||||
|                             border-radius: 1rem; | ||||
|                             display: inline-block; | ||||
|                             transform: scale(1); | ||||
|                             transition: transform 0.2s ease-in-out; | ||||
|                             &:hover { | ||||
|                               transform: scale(1.02); | ||||
|                             } | ||||
|                             @media screen and (min-width: $desktop-min-width) { | ||||
|                               font-size: $sm-font-size-desktop; | ||||
|                             } | ||||
|                           } | ||||
|                         } | ||||
|                         > .document-grid { | ||||
|                           display: grid; | ||||
|                           grid-template-columns: 1fr 1fr; | ||||
|                           grid-template-rows: auto auto; | ||||
|                           column-gap: 2rem; | ||||
|                           > figure { | ||||
|                               grid-column: 1 / 2; | ||||
|                               grid-row: 1 / 3; | ||||
|                               margin: 0; | ||||
|                               background-color: var(--couleur); | ||||
|                               img { | ||||
|                                   padding: 0.6rem; | ||||
|                                   box-sizing: border-box; | ||||
|                                   width: 100%; | ||||
|                                   height: auto; | ||||
|                                   object-fit: cover; | ||||
|                                   transform: scale(1); | ||||
|                                   transition: transform 0.2s ease-in-out; | ||||
|                                   &:hover { | ||||
|                                       transform: scale(1.02); | ||||
|                                   } | ||||
|                                   @media screen and (min-width: $desktop-min-width) { | ||||
|                                     padding: 2rem; | ||||
|                                   } | ||||
|                               } | ||||
|                           } | ||||
|                           > .infos { | ||||
|                               > p { | ||||
|                                   margin: 0; | ||||
|                                   margin-bottom: 0.8rem; | ||||
|                                   &:first-of-type { | ||||
|                                       font-weight: bold; | ||||
|                                   } | ||||
|                               } | ||||
|                        | ||||
|                           } | ||||
|                           > .download { | ||||
|                               align-self: end; | ||||
|                               justify-self: end; | ||||
|                           } | ||||
|                         } | ||||
|                       } | ||||
|                     } | ||||
|                     .caption { | ||||
|                       font-size: $sm-font-size-mobile; | ||||
| @@ -1176,9 +1302,117 @@ body{ | ||||
|                         margin-top: 10vh; | ||||
|                       } | ||||
|                     } | ||||
|                     #centre-de-ressource { | ||||
|                         .filters { | ||||
|                           margin: 3rem 0; | ||||
|                           margin-top: 5rem; | ||||
|                           > .search-bar{ | ||||
|                             margin-right: 2rem; | ||||
|                             border: solid 1px var(--couleur); | ||||
|                             padding: 0.5rem 1rem; | ||||
|                             border-radius: 1rem; | ||||
|                             font-size: $sm-font-size-mobile; | ||||
|                             font-family: 'Marianne', sans-serif; | ||||
|                           } | ||||
|                           > select { | ||||
|                             max-width: 20%; | ||||
|                             margin-right: 2rem; | ||||
|                             appearance: none; | ||||
|                             border: solid 1px var(--couleur); | ||||
|                             padding: 0.5rem 1rem; | ||||
|                             font-size: $sm-font-size-mobile; | ||||
|                             font-family: 'Marianne', sans-serif; | ||||
|                             border-radius: 1rem; | ||||
|                             background-color: white; | ||||
|                             background: url("data:image/svg+xml,<svg height='10px' width='10px' viewBox='0 0 16 16' fill='rgba(128, 200, 191)' xmlns='http://www.w3.org/2000/svg'><path d='M7.247 11.14 2.451 5.658C1.885 5.013 2.345 4 3.204 4h9.592a1 1 0 0 1 .753 1.659l-4.796 5.48a1 1 0 0 1-1.506 0z'/></svg>") no-repeat; | ||||
|                             background-position: calc(100% - 1rem) center !important; | ||||
|                             padding-right: 2.5rem !important; | ||||
|                           } | ||||
|                       } | ||||
|                       > .intro { | ||||
|                         font-size: $sm-font-size-mobile; | ||||
|                         margin-bottom: 4rem; | ||||
|                         @media screen and (min-width: $desktop-min-width) { | ||||
|                           font-size: $sm-font-size-desktop; | ||||
|                           width: 66%; | ||||
|                         } | ||||
|                       } | ||||
|                       > .type-section { | ||||
|                         > h3 { | ||||
|                           display: inline-block; | ||||
|                           font-size: $l-font-size-mobile; | ||||
|                           font-family: 'Joost', sans-serif; | ||||
|                           margin: 0; | ||||
|                           margin-bottom: 2rem; | ||||
|                           z-index: 1; | ||||
|                           position: relative; | ||||
|                           padding: 0 0.5rem; | ||||
|                           background: linear-gradient(transparent 70%, $brand-color 70%); | ||||
|                           @media screen and (min-width: $desktop-min-width) { | ||||
|                             font-size: $l-font-size-desktop; | ||||
|                           }                         | ||||
|                         } | ||||
|                         > .button-container { | ||||
|                           display: flex; | ||||
|                           justify-content: center; | ||||
|                           > .ressource-button { | ||||
|                             display: inline; | ||||
|                             font-size: $sm-font-size-mobile; | ||||
|                             background-color: $brand-color; | ||||
|                             padding: 0.5rem 1.5rem; | ||||
|                             border-radius: 1rem; | ||||
|                             cursor: pointer; | ||||
|                             transition: transform 0.2s ease-in-out; | ||||
|                             transform: scale(1); | ||||
|                             &:hover { | ||||
|                               transform: scale(0.95); | ||||
|                             } | ||||
|                             @media screen and (min-width: $desktop-min-width) { | ||||
|                               font-size: $sm-font-size-desktop; | ||||
|                             } | ||||
|                           } | ||||
|                         } | ||||
|                       } | ||||
|                     } | ||||
|                     > #ressource-item-header { | ||||
|                       > .retour { | ||||
|                         margin-bottom: 1.5rem; | ||||
|                         > p { | ||||
|                           display: inline-block; | ||||
|                           cursor: pointer; | ||||
|                           font-size: $sm-font-size-mobile; | ||||
|                           @media screen and (min-width: $desktop-min-width) { | ||||
|                             font-size: $sm-font-size-desktop; | ||||
|                           } | ||||
|                         } | ||||
|                       } | ||||
|                       > .type, | ||||
|                       > .meta { | ||||
|                         text-align: center; | ||||
|                         font-size: $m-font-size-mobile; | ||||
|                         font-family: 'Joost', sans-serif; | ||||
|                         @media screen and (min-width: $desktop-min-width) { | ||||
|                           font-size: $m-font-size-desktop; | ||||
|                         } | ||||
|                       } | ||||
|                       > .title { | ||||
|                         width: 100%; | ||||
|                         text-align: center; | ||||
|                         margin-bottom: 1.5rem; | ||||
|                         > h2 { | ||||
|                           display: inline; | ||||
|                           margin-top: 1rem; | ||||
|                           font-size: $xl-font-size-mobile; | ||||
|                           font-family: 'Joost', sans-serif; | ||||
|                           @media screen and (min-width: $desktop-min-width) { | ||||
|                             font-size: $xl-font-size-desktop; | ||||
|                           } | ||||
|                         } | ||||
|                       } | ||||
|                     } | ||||
|                   } | ||||
|                   > .pieces-jointes { | ||||
|                     z-index: 1; | ||||
|                     z-index: 0; | ||||
|                     position: relative; | ||||
|                     padding: 0 $modale-x-padding; | ||||
|                     box-sizing: border-box; | ||||
| @@ -1232,7 +1466,11 @@ body{ | ||||
|                     } | ||||
|                   } | ||||
|                   > footer { | ||||
|                     z-index: 0; | ||||
|                     position: relative; | ||||
|                     z-index: -1; | ||||
|                     &.footer-ressource { | ||||
|                       overflow: hidden; | ||||
|                     } | ||||
|                     .pattern-bottom { | ||||
|                       mask-image: linear-gradient(to top, rgba(0,0,0,1), rgba(0,0,0,0)); | ||||
|                       height: $modale-bottom-padding; | ||||
| @@ -1243,15 +1481,33 @@ body{ | ||||
|                       background-size: 300px; | ||||
|                       background-size: repeat; | ||||
|                     } | ||||
|                     > div:has(.related-etape-label) { | ||||
|                       z-index: 1; | ||||
|                       position: relative; | ||||
|                       .related-etape-label { | ||||
|                         display: inline-block; | ||||
|                         padding: 0.5rem 1rem; | ||||
|                         padding-left: calc($modale-x-padding / 2); | ||||
|                         font-size: $sm-font-size-mobile; | ||||
|                         margin-bottom: 1.5rem; | ||||
|                         @media screen and (min-width: $desktop-min-width) { | ||||
|                           margin-bottom: 0; | ||||
|                           font-size: $sm-font-size-desktop; | ||||
|                         } | ||||
|                       } | ||||
|                     } | ||||
|                     .related-etape-links { | ||||
|                       position: absolute; | ||||
|                       //bottom: calc(($modale-bottom-padding / 2) * -1); | ||||
|                       width: 100%; | ||||
|                       box-sizing: border-box; | ||||
|                       padding: 0 calc($modale-x-padding / 2); | ||||
|                       display: grid; | ||||
|                       grid-template-rows: 1fr 1fr; | ||||
|                       grid-template-columns: 1fr; | ||||
|                       width: 75%; | ||||
|                       &:not(:has(.solo)) { | ||||
|                         width: 100%; | ||||
|                         position: absolute; | ||||
|                         grid-template-rows: 1fr 1fr; | ||||
|                       } | ||||
|                       @media screen and (min-width: $desktop-min-width) { | ||||
|                         grid-template-columns: 1fr 1fr; | ||||
|                         margin-top: 2.5rem; | ||||
| @@ -1287,24 +1543,12 @@ body{ | ||||
|                         } | ||||
|                         > .icon { | ||||
|                           z-index: 2; | ||||
|                           width: 10px; | ||||
|                           height: 30px; | ||||
|                           display: flex; | ||||
|                           flex-direction: column; | ||||
|                           justify-content: center; | ||||
|                           align-items: center; | ||||
|                           > div { | ||||
|                             display: block; | ||||
|                             width: 20px; | ||||
|                             height: 10px; | ||||
|                             &:first-of-type, &:last-of-type { | ||||
|                               height: 8px; | ||||
|                               clip-path: polygon(0 0, 100% 0, 50% 100%); | ||||
|                             } | ||||
|                             &:first-of-type { | ||||
|                               transform: rotate(180deg); | ||||
|                             } | ||||
|                           } | ||||
|                           width: 25px; | ||||
|                           height: 25px; | ||||
|                           mask-image: url("/themes/custom/caravane/assets/pictograms/hexagone.svg"); | ||||
|                           mask-size: contain; | ||||
|                           mask-repeat: no-repeat; | ||||
|                           margin-right: -8px; | ||||
|                         } | ||||
|                         > .card-content { | ||||
|                           z-index: 1; | ||||
| @@ -1318,21 +1562,21 @@ body{ | ||||
|                               padding: 1rem 0.5rem; | ||||
|                               font-weight: bold; | ||||
|                               font-family: 'Joost', sans-serif; | ||||
|                               font-size: $m-font-size-mobile; | ||||
|                               font-size: $labeur-font-size-mobile; | ||||
|                               @media screen and (min-width: $desktop-min-width) { | ||||
|                                 font-size: $m-font-size-desktop; | ||||
|                                 font-size: $labeur-font-size-desktop; | ||||
|                               } | ||||
|                               > span { | ||||
|                                 font-weight: lighter; | ||||
|                               } | ||||
|                             } | ||||
|                             > .date { | ||||
|                               font-size: $sm-font-size-mobile; | ||||
|                               font-size: $xsm-font-size-mobile; | ||||
|                               font-family: 'Marianne', sans-serif; | ||||
|                               font-weight: lighter; | ||||
|                               padding-bottom: 1rem; | ||||
|                               @media screen and (min-width: $desktop-min-width) { | ||||
|                                 font-size: $sm-font-size-desktop; | ||||
|                                 font-size: $xsm-font-size-desktop; | ||||
|                               } | ||||
|                             } | ||||
|                           } | ||||
| @@ -1353,6 +1597,84 @@ body{ | ||||
|                   } | ||||
|                 } | ||||
|               } | ||||
|               .ressource-list > div:not(.ressource-item), | ||||
|               .ressource-list.sm-ressource-list { | ||||
|                 display: grid; | ||||
|                 grid-template-columns: 1fr; | ||||
|                 align-items: start; | ||||
|                 gap: 2rem; | ||||
|                 margin-bottom: 2.5rem; | ||||
|                 @media screen and (min-width: $tablet-min-width) { | ||||
|                   grid-template-columns: repeat(2, 1fr); | ||||
|                 } | ||||
|                 @media screen and (min-width: $desktop-min-width) { | ||||
|                   grid-template-columns: repeat(4, 1fr); | ||||
|                 } | ||||
|                 &.sm-ressource-list { | ||||
|                   margin-top: 2rem; | ||||
|                   grid-template-columns: repeat(2, 1fr); | ||||
|                 } | ||||
|                 > .ressource-item > div { | ||||
|                   display: flex; | ||||
|                   gap: 0.8rem; | ||||
|  | ||||
|                   @media screen and (min-width: $desktop-min-width) { | ||||
|                     gap: 1.2rem; | ||||
|                   } | ||||
|                   cursor: pointer; | ||||
|                   transform: scale(1); | ||||
|                   transition: transform 0.2s ease-in-out; | ||||
|                   &.promoted { | ||||
|                     &::after { | ||||
|                         content: ''; | ||||
|                         position: absolute; | ||||
|                         top: -10px; | ||||
|                         left: -10px; | ||||
|                         width: 20px !important; | ||||
|                         height: 20px !important; | ||||
|                         background-color: $brand-color; | ||||
|                         mask-image: url("/themes/custom/caravane/assets/pictograms/hexagone.svg"); | ||||
|                         mask-size: contain; | ||||
|                         mask-repeat: no-repeat; | ||||
|                     } | ||||
|                   } | ||||
|                   &:hover { | ||||
|                     transform: scale(1.05); | ||||
|                   } | ||||
|                   > figure { | ||||
|                     width: 40%; | ||||
|                     margin: 0; | ||||
|                   } | ||||
|                   > div { | ||||
|                     width: 50%; | ||||
|                     > h4 { | ||||
|                       font-size: $m-font-size-mobile; | ||||
|                       font-family: 'Joost', sans-serif; | ||||
|                       margin: 0; | ||||
|                       margin-bottom: 0.5rem; | ||||
|                       @media screen and (min-width: $desktop-min-width) { | ||||
|                         font-size: $m-font-size-desktop; | ||||
|                       } | ||||
|                     } | ||||
|                     > p { | ||||
|                       margin: 0; | ||||
|                       font-size: $sm-font-size-mobile; | ||||
|                       @media screen and (min-width: $desktop-min-width) { | ||||
|                         font-size: $sm-font-size-desktop; | ||||
|                       } | ||||
|                     } | ||||
|                   } | ||||
|                 } | ||||
|               } | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|             } | ||||
|             > #animation-toggle { | ||||
|               transition: opacity 0.3s ease-out; | ||||
| @@ -1446,4 +1768,14 @@ body{ | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   &.toolbar-fixed { | ||||
|     #content-modale .content-wrapper { | ||||
|       top: 20vh !important; | ||||
|     } | ||||
|     &.toolbar-vertical.toolbar-tray-open { | ||||
|       #content-modale .content-wrapper { | ||||
|         left: 15vw !important; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -55,7 +55,4 @@ | ||||
|     {%- endfor %} | ||||
| </div> | ||||
| <div class="icone-arret"> | ||||
|     <div></div> | ||||
|     <div></div> | ||||
|     <div></div> | ||||
| </div> | ||||
		Reference in New Issue
	
	Block a user