-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathdemo-standalone-self-contained.html
More file actions
1759 lines (1546 loc) · 88 KB
/
demo-standalone-self-contained.html
File metadata and controls
1759 lines (1546 loc) · 88 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>microMVC+ Demo 2026 — by Fabien Conéjéro</title>
<link rel="canonical" href="https://madjeek-web.github.io/microMVC__Demo_2026.html">
<!-- Open Graph -->
<meta name="author" content="Fabien Conéjéro" foaf:maker data-internal-note="Balise pour indiquer l'auteur du contenu. Cela aide à établir la responsabilité et l'attribution, ce qui est essentiel pour la transparence et la confiance des utilisateurs." />
<link rel="author" href="https://github.com/madjeek-web" data-internal-note="Balise pour spécifier un lien vers le profil de l'auteur. Cela permet aux utilisateurs de consulter plus d'informations sur l'auteur, améliorant ainsi l'engagement et la transparence." />
<meta property="og:title" content="microMVC+ Demo 2026 — by Fabien Conéjéro" data-internal-note="Balise pour spécifier le titre du contenu lorsque la page est partagée sur les réseaux sociaux. Un titre bien formulé attire l'attention des utilisateurs et donne un aperçu du contenu de la page, ce qui peut augmenter le taux de clics. Le site officiel des Open Graph (OG) est le suivant : https://ogp.me. Introduction aux Balises Open Graph : Les balises Open Graph sont des métadonnées intégrées dans le code HTML d'une page web. Elles permettent de spécifier comment le contenu doit être affiché sur les réseaux sociaux lorsqu'il est partagé. L'idée est de contrôler l'apparence des partages sur des plateformes comme Facebook, Twitter, LinkedIn, et d'autres réseaux sociaux. Balise meta property=og:title> Fonction : La balise meta property=og:title> définit le titre du contenu qui s'affiche lorsque la page est partagée sur les réseaux sociaux. Un titre bien formulé attire l'attention des utilisateurs et fournit un aperçu accrocheur du contenu. Impact : Un bon titre peut augmenter le taux de clics (CTR) en incitant les utilisateurs à cliquer sur le lien partagé. Il doit être concis tout en étant suffisamment descriptif pour donner une idée claire du contenu." />
<meta property="og:description" content="microMVC+ by Fabien Conéjéro — Zero-dependency JavaScript MVC micro-framework. 6 live demos: bank transfer, real-time notifications, task manager, profile editor, event log, async flows." data-internal-note="Balise pour fournir une description concise du contenu. Cette description apparaît sous le titre lors du partage sur les réseaux sociaux, offrant aux utilisateurs un aperçu clair de ce à quoi s'attendre. Une bonne description peut inciter davantage d'utilisateurs à cliquer sur le lien." />
<meta property="og:type" content="article">
<meta property="article:author" content="Fabien Conéjéro">
<meta property="article:published_time" content="2026-01-01">
<meta property="article:modified_time" content="2026-02-23">
<meta property="article:section" content="Technology">
<meta property="article:tag" content="JavaScript">
<meta property="article:tag" content="MVC">
<meta property="article:tag" content="Frontend">
<meta property="article:tag" content="Open Source">
<meta property="og:url" content="https://madjeek-web.github.io/microMVC__Demo_2026.html">
<meta property="og:site_name" content="madjeek-web — Fabien Conéjéro">
<!-- Twitter Card -->
<meta name="twitter:card" content="summary">
<meta name="twitter:title" content="microMVC+ Demo 2026 — by Fabien Conéjéro">
<meta name="twitter:description" content="microMVC+ by Fabien Conéjéro — Zero-dependency JavaScript MVC micro-framework. 6 live demos.">
<meta name="title" content="microMVC+ Demo 2026 — by Fabien Conéjéro" data-internal-note="Balise pour définir un titre alternatif pour le contenu. Bien que le titre soit déjà spécifié dans la balise title>, cette balise peut être utilisée par certains moteurs de recherche ou outils d'analyse pour des informations supplémentaires. Cela contribue à la cohérence et à l'accessibilité des données." />
<meta name="description" content="microMVC+ by Fabien Conéjéro — Zero-dependency JavaScript MVC micro-framework. 6 live demos: bank transfer, real-time notifications, task manager, profile editor, event log, async flows." data-internal-note="Balise pour fournir une description concise du contenu de la page. Une bonne description peut influencer le taux de clics dans les résultats de recherche, car elle donne aux utilisateurs un aperçu de ce à quoi s'attendre. Elle est également cruciale pour le référencement en aidant les moteurs de recherche à comprendre le sujet principal de la page." />
<meta name="abstract" content="microMVC+ by Fabien Conéjéro — Zero-dependency JavaScript MVC micro-framework. 6 live demos: bank transfer, real-time notifications, task manager, profile editor, event log, async flows." data-internal-note="Balise pour fournir un résumé du contenu de la page. Cela peut être utilisé pour donner un aperçu plus détaillé des sujets abordés. Bien que moins courante que la balise description, elle peut être utile dans certains contextes pour pourvoir des informations supplémentaires aux utilisateurs et aux moteurs de recherche." />
<meta name="keywords" content="zero, dependency, javaScript, mvc, micro, framework, interactive, demos, fabien, conéjéro, 2026, mit, license" datetime="1970-01-01" data-internal-note="Balise pour spécifier une liste de mots-clés associés au contenu de la page. Bien que moins influents qu'auparavant pour le référencement, ils peuvent servir à aider certains moteurs de recherche à comprendre les thèmes principaux du contenu. Il est essentiel de choisir des mots-clés pertinents et de ne pas surcharger cette balise. * La balise meta name=keywords> a été largement utilisée dans le passé pour indiquer aux moteurs de recherche les mots-clés associés au contenu d'une page web. Cependant, son utilisation a considérablement diminué pour plusieurs raisons : Date d'Arrêt d'Utilisation : Années 2000 : Dans les années 1990 et au début des années 2000, la balise keywords était populaire. Cependant, à partir des années 2000, les moteurs de recherche, notamment Google, ont commencé à dévaluer son importance en raison de l'abus qui en était fait. De nombreux webmasters surchargeaient cette balise avec des mots-clés non pertinents pour essayer d'améliorer leur classement. 2009 et au-delà : Google a officiellement annoncé qu'il n'utilisait plus la balise keywords pour le classement des pages à partir de 2009. D'autres moteurs de recherche ont également suivi cette tendance, réduisant ainsi l'importance de cette balise. Raison de la Dévaluation : Surcharge de Mots-Clés : De nombreux sites ont abusé de cette balise en ajoutant un grand nombre de mots-clés, souvent non pertinents, pour manipuler les classements. Amélioration des Algorithmes : Les moteurs de recherche ont amélioré leurs algorithmes pour évaluer le contenu d'une page de manière plus holistique, se basant sur des critères tels que la qualité du contenu, les liens entrants, et l'expérience utilisateur, plutôt que sur des balises meta. Bien que la balise meta name=keywords> puisse encore être utilisée pour des raisons de documentation ou de structuration interne, elle n'a plus d'impact significatif sur le référencement naturel. Les webmasters sont maintenant encouragés à se concentrer sur des pratiques SEO modernes, telles que la création de contenu de qualité et l'optimisation des balises title> et meta description>, ainsi que sur l'utilisation de balises de titres appropriées." />
<meta name="news_keywords" content="zero, dependency, javaScript, mvc, micro, framework, interactive, demos, fabien, conéjéro, 2026, mit, license" data-internal-note="Balise pour spécifier des mots-clés d'actualité, utilisés principalement dans le contexte des articles de presse ou des contenus d'actualité. Cela aide à mieux indexer le contenu lié à des événements récents ou à des tendances, permettant ainsi aux utilisateurs de trouver des informations pertinentes. * La balise meta name=news_keywords> a été introduite pour aider à indexer les articles d'actualité et à relier le contenu à des événements récents ou à des tendances. Voici quelques informations sur son utilisation et son déclin : Date d'Utilisation et Déclin : Années 2000 : La balise news_keywords était principalement utilisée dans le contexte du contenu d'actualité, offrant une façon de signaler des mots-clés pertinents pour les articles de presse. Elle visait à améliorer la visibilité des nouvelles dans les moteurs de recherche comme Google News. Début des Années 2010 : À mesure que les algorithmes de recherche se sont améliorés, la nécessité de cette balise a diminué. Google et d'autres moteurs de recherche ont commencé à se concentrer sur le contenu de qualité et les signaux contextuels, plutôt que sur des balises spécifiques pour déterminer la pertinence d'un contenu. Raison de la Dévaluation : Évolution des Algorithmes : Les moteurs de recherche utilisent des algorithmes sophistiqués qui prennent en compte des facteurs tels que la pertinence, la qualité du contenu, et les liens entrants, rendant les balises de mots-clés moins essentielles. Mauvaise Utilisation : Comme pour d'autres balises de mots-clés, certains sites ont abusé de cette balise en ajoutant des mots-clés non pertinents, ce qui a conduit à son déclin. Bien que la balise meta name=news_keywords> puisse encore exister dans certains contextes, son efficacité et sa pertinence ont largement diminué. Les éditeurs de contenu d'actualité se concentrent désormais sur des pratiques SEO modernes et sur la création de contenu de haute qualité pour améliorer leur visibilité. Pour indexer des articles d'actualité, il est plus efficace de se concentrer sur des techniques telles que l'optimisation des titres, des descriptions, et des balises de contenu sans dépendre des balises de mots-clés spécifiques." />
<meta name="generator" content="GitHub Page, visual studio code (https://code.visualstudio.com/)" data-internal-note="Balise pour indiquer le générateur du site, ici GiyHub Page. Cela informe les utilisateurs et les moteurs de recherche sur les technologies utilisées pour créer et gérer le site. Cela peut également être utile pour la gestion des mises à jour et des améliorations." />
<meta name="internal-note" content="★★★ CODE : 11 ★★★ Balises rares 2025 peuvent aider à affiner la compréhension du contenu par des moteurs de recherche, des IA ou des outils d'analyse, tout en permettant une meilleure segmentation et ciblage. Ces balises sont conçues pour répondre aux attentes des technologies avancées et pour optimiser le référencement dans un paysage numérique de plus en plus complexe. Elles permettent également de mieux contextualiser le contenu pour les utilisateurs finaux, améliorant ainsi l'expérience utilisateur et l'efficacité des stratégies de marketing. Directive générale pour tous les robots d'indexation respectant les standards. Les balises relatives aux IA (ChatGPT, Claude, Bard, etc.) sont particulièrement importantes aujourd'hui. Certains bots comme mediapartners-google sont spécifiques à AdSense. Ces balises meta sont cruciales pour optimiser le référencement et guider les crawlers sur la manière de traiter une page. Les valeurs courantes index, follow et archive déterminent si une page doit être indexée, archivée ou les deux. La balise bot est très générale et peut couvrir certains robots non spécifiquement listés. Cette liste complète couvre la plupart des robots importants du web, des moteurs de recherche traditionnels aux nouvelles IA et assistants intelligents. Ces balises meta> sont cruciales pour optimiser le référencement et guider les crawlers sur la manière de traiter une page. Les valeurs courantes index, follow et archive déterminent si une page doit être indexée, archivée ou les deux. Comment fonctionnent les moteurs de recherche IA ? Un moteur de recherche basé sur l'IA utilise des algorithmes d'apprentissage automatique (Machine Learning) pour comprendre et interpréter le contenu disponible sur le web. Ces algorithmes analysent des millions de pages en quelques secondes pour fournir des résultats pertinents et personnalisés. Grâce au traitement du langage naturel (NLP), ils interprètent les mots et les phrases, capturent les nuances, et comprennent les questions complexes, allant bien au-delà de la simple correspondance de mots-clés pour saisir le contexte et l'intention de l'utilisateur. Le monde de la recherche en ligne est en pleine mutation. Fini le temps où il suffisait de taper des mots-clés simples pour dégoter l'information. L'intelligence artificielle (IA) est en train de révolutionner la manière dont nous interagissons avec la technologie, transformant les moteurs de recherche en de véritables assistants intelligents. Comment fonctionnent les moteurs de recherche IA ? Un moteur de recherche basé sur l'IA utilise des algorithmes d'apprentissage automatique (Machine Learning) pour comprendre et interpréter le contenu disponible sur le web. Ces algorithmes analysent des millions de pages en quelques secondes pour fournir des résultats pertinents et personnalisés. Grâce au traitement du langage naturel (NLP), ils interprètent les mots et les phrases, capturent les nuances, et comprennent les questions complexes, allant bien au-delà de la simple correspondance de mots-clés pour saisir le contexte et l'intention de l'utilisateur. Les avantages des moteurs de recherche IA Les avantages sont clairs : - Une recherche plus précise et contextuelle - Une rapidité accrue - Une personnalisation poussée des résultats - Une réduction significative du « bruit inutile » (résultats peu pertinents, publicités intrusives pour certains) Conclusion : Les moteurs de recherche IA redéfinissent le paysage numérique. Comprendre leur fonctionnement est la clé pour rester compétitif et visible sur Internet ! L'avenir de votre visibilité : le GEO L'émergence de ces moteurs de recherche IA introduit une nouvelle branche du SEO (Search Engine Optimization) appelée Generative Engine Optimization (GEO). Désormais, l'objectif n'est plus seulement de figurer dans les résultats de Google, mais aussi d'apparaître dans les réponses générées par ces moteurs IA, réduisant le besoin de cliquer sur de multiples liens.">
<meta name="robots" content="follow, index" data-internal-note="Balise pour indiquer aux robots des moteurs de recherche de suivre les liens et d'indexer la page. Cela est essentiel pour le référencement, car cela permet d'augmenter la visibilité du contenu sur les moteurs de recherche et d'encourager les visites sur le site." />
<meta name="googlebot" content="index, follow" data-internal-note="Balise spécifique pour Googlebot, le crawler de Google. Ici, la directive index, follow informe Google que cette page peut être indexée et que ses liens peuvent être suivis, ce qui peut améliorer le classement du site dans les résultats de recherche de Google." />
<meta name="slurp" content="index, follow" data-internal-note="Rôle : Indique aux crawlers de Yahoo (Slurp) d'indexer la page et de suivre les liens présents. Objectif : Améliorer le référencement sur Yahoo. Remplissage : Rempli avec index, follow pour permettre l'indexation et le suivi." /><meta name="msnbot" content="index, follow" data-internal-note="Rôle : Indique aux crawlers de Bing (msnbot) d'indexer la page et de suivre les liens. Objectif : Optimiser le référencement sur Bing. Remplissage : Utilise également index, follow." /><meta name="teoma" content="index, follow" data-internal-note="Rôle : Indique au crawler Teoma d'indexer et de suivre. Objectif : Améliorer la visibilité du site sur des moteurs utilisant ce crawler." /><meta name="teoma" content="archive" data-internal-note="Rôle : Indique que la page doit être archivée par Teoma. Objectif : Conserver une version de la page pour consultation future. Remplissage : Rempli avec archive, indiquant que l'indexation n'est pas nécessaire à ce moment." /><meta name="robots" content="archive" data-internal-note="Rôle : Indique à tous les robots d'indexation de ne pas indexer, mais d'archiver la page. Objectif : Préserver le contenu sans le rendre visible dans les résultats de recherche." /><meta name="googlebot" content="archive" data-internal-note="Rôle : Indique à Google de conserver une archive de la page sans l'indexer. Objectif : Préserver le contenu pour référencement futur sans qu'il soit affiché dans les résultats." /><meta name="Slurp" content="archive" data-internal-note="Rôle : Indique à Yahoo (Slurp) d'archiver la page. Objectif : Conserver une version de la page dans l'historique d'indexation." /><meta name="msnbot" content="archive" data-internal-note="Rôle : Indique à Bing d'archiver la page. Objectif : Offrir une version historique sans indexation active." /><meta name="bingbot" content="archive" data-internal-note="Rôle : Indique à Bingbot d'archiver la page. Objectif : Préserver le contenu sans l'afficher dans les résultats de recherche." /><meta name="bingbot" content="index, follow" data-internal-note="Rôle : Indique à Bingbot d'indexer la page et de suivre les liens. Objectif : Améliorer le référencement et l'exploration des liens." /><meta name="bot" content="index, follow" data-internal-note="Rôle : Générique pour tous les bots, leur demandant d'indexer et de suivre. Objectif : Appliquer des règles d'indexation larges à tous les crawlers." /><meta name="yahoobot" content="index, follow" data-internal-note="Rôle : Indique à Yahoo d'indexer la page et de suivre les liens. Objectif : Optimiser la visibilité sur Yahoo." /><meta name="baiduspider" content="index, follow" data-internal-note="Rôle : Indique au crawler Baidu d'indexer et de suivre.Objectif : Améliorer le référencement sur le moteur de recherche Baidu. Robot de Baidu (principal moteur de recherche chinois)" /><meta name="duckduckbot" content="index, follow" data-internal-note="Robot de DuckDuckGo (moteur de recherche respectueux de la vie privée)" /><meta name="yandex" content="index, follow" data-internal-note="Rôle : Indique au crawler Yandex d'indexer et de suivre. Objectif : Améliorer le référencement sur Yandex. Robot de Yandex (principal moteur de recherche russe)" /><meta name="facebookexternalhit" content="index, follow" data-internal-note="Rôle : Indique au crawler de Facebook d'indexer et de suivre. Objectif : Optimiser l'affichage des liens partagés sur Facebook." /><meta name="twitterbot" content="index, follow" data-internal-note="Rôle : Indique au crawler de Twitter d'indexer et de suivre. Objectif : Améliorer la visibilité des liens partagés sur Twitter." /><meta name="linkedinbot" content="index, follow" data-internal-note="Rôle : Indique au crawler de LinkedIn d'indexer et de suivre. Objectif : Optimiser le référencement sur LinkedIn." /><meta name="mediapartners-google" content="index, follow" data-internal-note="Rôle : Indique à Google AdSense d'indexer et de suivre. Objectif : Optimiser le contenu pour les annonces Google." /><meta name="ia_archiver" content="index, follow" data-internal-note="Rôle : Indique à l'archiviste Internet d'indexer et de suivre. Objectif : Préserver une copie de la page pour l'Internet Archive." /><meta name="pinterest" content="index, follow" data-internal-note="Rôle : Indique au crawler de Pinterest d'indexer et de suivre. Objectif : Améliorer la visibilité des images sur Pinterest." /><meta name="whatsapp" content="index, follow" data-internal-note="Rôle : Indique au crawler de WhatsApp d'indexer et de suivre. Objectif : Optimiser le contenu partagé via WhatsApp." /><meta name="newsboat" content="index, follow" data-internal-note="Rôle : Indique au lecteur de flux RSS Newsboat d'indexer et de suivre. Objectif : Améliorer l'accès aux flux de contenu." /><meta name="googlebot-news" content="index, follow" data-internal-note="Robot spécialisé de Google pour l'indexation des actualités" /><meta name="googlebot-image" content="index, follow" data-internal-note="Robot spécialisé de Google pour l'indexation des images" /><meta name="googlebot-video" content="index, follow" data-internal-note="Robot spécialisé de Google pour l'indexation des vidéos" /><meta name="sogou" content="index, follow" data-internal-note="Robot de Sogou (moteur de recherche chinois)" /><meta name="qwantbot" content="index, follow" data-internal-note="Robot de Qwant (moteur de recherche européen respectueux de la vie privée)" /><meta name="seozoom" content="index, follow" data-internal-note="Robot de SEOZoom (outil d'analyse SEO)" /><meta name="mj12bot" content="index, follow" data-internal-note="Robot de Majestic SEO (outil d'analyse de backlinks)" /><meta name="ahrefsbot" content="index, follow" data-internal-note="Robot de Ahrefs (outil d'analyse SEO et de backlinks)" /><meta name="semrushbot" content="index, follow" data-internal-note="Robot de SEMrush (plateforme de marketing digital)" /><meta name="facebookcatalog" content="index, follow" data-internal-note="Robot de Facebook pour l'indexation des catalogues produits" /><meta name="discordbot" content="index, follow" data-internal-note="Robot de Discord pour prévisualiser les liens partagés" /><meta name="redditbot" content="index, follow" data-internal-note="Robot de Reddit pour analyser les liens partagés" /><meta name="telegrambot" content="index, follow" data-internal-note="Robot de Telegram pour prévisualiser les liens partagés" /><meta name="alexabot" content="index, follow" data-internal-note="Robot d'Amazon Alexa (assistant vocal et services web)" /><meta name="claudebot" content="index, follow" data-internal-note="Robot de Claude IA (Anthropic) pour l'indexation des connaissances" /><meta name="chatgpt-user" content="index, follow" data-internal-note="Robot d'OpenAI pour ChatGPT (exploration web pour l'IA)" /><meta name="gptbot" content="index, follow" data-internal-note="Nouveau robot officiel d'OpenAI pour l'exploration web" /><meta name="bingai" content="index, follow" data-internal-note="Robot de Microsoft Bing AI (assistant intelligent)" /><meta name="bard" content="index, follow" data-internal-note="Robot de Google Bard (assistant IA, maintenant Gemini)" /><meta name="perplexitybot" content="index, follow" data-internal-note="Robot de Perplexity AI (moteur de recherche assisté par IA)" /><meta name="youchatbot" content="index, follow" data-internal-note="Robot de You.com (moteur de recherche assisté par IA)" /><meta name="phindbot" content="index, follow" data-internal-note="Robot de Phind.com (moteur de recherche pour développeurs)" /><meta name="meta-ai" content="index, follow" data-internal-note="Robot de Meta AI (assistant intelligent de Facebook/Meta)" /><meta name="coherebot" content="index, follow" data-internal-note="Robot de Cohere (plateforme de langage IA)" /><meta name="huggingfacebot" content="index, follow" data-internal-note="Robot de Hugging Face (plateforme de modèles de ML)" /><meta name="archiveorg_bot" content="index, follow" data-internal-note="Robot officiel de Internet Archive (alternative à ia_archiver)" /><meta name="applebot" content="index, follow" data-internal-note="Robot d'Apple pour Siri et les services Apple" /><meta name="samsungbot" content="index, follow" data-internal-note="Robot de Samsung pour les services intelligents Samsung" /><meta name="zoomindexbot" content="index, follow" data-internal-note="Robot de Zoom pour l'indexation du contenu des réunions" />
<meta name="internal-note" content="Les 2 balises suivantes : Mistral AI - Plateforme française d'IA open source"><meta name="mistral-ai" content="index, follow" /><meta name="mistralbot" content="index, follow" />
<meta name="internal-note" content="Les 2 balises suivantes : Qwen - Modèle d'IA d'Alibaba"><meta name="qwen-ai" content="index, follow" /><meta name="alibaba-ai" content="index, follow" />
<meta name="internal-note" content="La balise suivante : SearchGPT - Moteur de recherche conversationnel d'OpenAI"><meta name="searchgpt" content="index, follow" />
<meta name="internal-note" content="Les 2 balises suivantes : Brave AI - Moteur de recherche respectueux de la vie privée"><meta name="brave-ai" content="index, follow" /><meta name="brave-search" content="index, follow" />
<meta name="internal-note" content="Les 2 balises suivantes : Andi Search - Moteur de recherche conversationnel innovant"><meta name="andisearch" content="index, follow" /><meta name="andi-ai" content="index, follow" />
<meta name="internal-note" content="Les 2 balises suivantes : Phind - Moteur de recherche technique pour développeurs"><meta name="phind" content="index, follow" /><meta name="phind-search" content="index, follow" />
<meta name="internal-note" content="Les 2 balises suivantes : You.com - Moteur de recherche personnalisable"><meta name="youcom" content="index, follow" /><meta name="you-search" content="index, follow" />
<meta name="internal-note" content="Les 2 balises suivantes : Microsoft Copilot - Assistant IA intégré à Bing"><meta name="copilot" content="index, follow" /><meta name="microsoft-copilot" content="index, follow" />
<meta name="internal-note" content="Les 2 balises suivantes : Komo Search - Moteur de recherche social et privé"><meta name="komo" content="index, follow" /><meta name="komo-search" content="index, follow" />
<meta name="internal-note" content="Les 2 balises suivantes : Google SGE (Search Generative Experience)"><meta name="google-sge" content="index, follow" /><meta name="search-generative-experience" content="index, follow" />
<meta name="internal-note" content="Les 2 balises suivantes : Waldo Search - Assistant de recherche professionnelle"><meta name="waldo-search" content="index, follow" /><meta name="waldosearch" content="index, follow" />
<meta name="internal-note" content="Les 9 balises suivantes : Modèles spécifiques supplémentaires"><meta name="claude-3-opus" content="index, follow" /><meta name="claude-3-sonnet" content="index, follow" /><meta name="claude-3-haiku" content="index, follow" /><meta name="gemini-ultra" content="index, follow" /><meta name="gemini-pro" content="index, follow" /><meta name="grok-2" content="index, follow" /><meta name="grok-3" content="index, follow" /><meta name="llama-3" content="index, follow" /><meta name="llama-4" content="index, follow" />
<meta name="internal-note" content="Les 6 balises suivantes : Plateformes de recherche IA émergentes"><meta name="exa-ai" content="index, follow" /><meta name="exabot" content="index, follow" /><meta name="nat-dev" content="index, follow" /><meta name="natbot" content="index, follow" /><meta name="lexii" content="index, follow" /><meta name="lexiibot" content="index, follow" />
<meta name="internal-note" content="★★★ Fin du Code : 11 ★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★">
<script type="application/ld+json">
{
"@context": "https://schema.org",
"@graph": [
{
"@type": "WebPage",
"@id": "https://madjeek-web.github.io/microMVC__Demo_2026.html#webpage",
"url": "https://madjeek-web.github.io/microMVC__Demo_2026.html",
"name": "microMVC+ Demo 2026 — Interactive JavaScript MVC Framework by Fabien Conéjéro",
"description": "microMVC+ by Fabien Conéjéro — Zero-dependency JavaScript MVC micro-framework. 6 live demos: bank transfer, real-time notifications, task manager, profile editor, event log, async flows.",
"inLanguage": "en",
"datePublished": "2026-01-01",
"dateModified": "2026-02-23",
"isPartOf": { "@id": "https://madjeek-web.github.io/#website" },
"about": { "@id": "https://madjeek-web.github.io/microMVC__Demo_2026.html#software" },
"author": { "@id": "https://madjeek-web.github.io/#person" },
"breadcrumb": { "@id": "https://madjeek-web.github.io/microMVC__Demo_2026.html#breadcrumb" },
"mainEntity": { "@id": "https://madjeek-web.github.io/microMVC__Demo_2026.html#software" }
},
{
"@type": "WebSite",
"@id": "https://madjeek-web.github.io/#website",
"url": "https://madjeek-web.github.io/",
"name": "madjeek-web — Fabien Conéjéro GitHub Pages",
"description": "Personal GitHub Pages of Fabien Conéjéro — open-source projects including microMVC+ (JavaScript MVC framework), Popup Blocker Pro (browser extension), Taskflow to-do list, symfony-mailjet-bundle (Symfony 7 / PHP 8.3), a GitHub guide, and a Creative Commons license template.",
"inLanguage": "en",
"publisher": { "@id": "https://madjeek-web.github.io/#person" },
"author": { "@id": "https://madjeek-web.github.io/#person" },
"copyrightHolder": { "@id": "https://madjeek-web.github.io/#person" },
"copyrightYear": "2026"
},
{
"@type": "Person",
"@id": "https://madjeek-web.github.io/#person",
"name": "Fabien Conéjéro",
"givenName": "Fabien",
"familyName": "Conéjéro",
"jobTitle": "Frontend Developer",
"description": "Fabien Conéjéro is a Frontend Developer and open-source author. He created microMVC+ (zero-dependency JavaScript MVC framework), Popup Blocker Pro (browser extension, Chrome MV3 / Firefox), Taskflow (web to-do list app), symfony-mailjet-bundle (Symfony 7 / PHP 8.3 async email bundle), a GitHub guide and a Creative Commons license template.",
"url": "https://madjeek-web.github.io/",
"sameAs": [
"https://github.com/madjeek-web"
],
"knowsAbout": [
"JavaScript",
"MVC Architecture",
"Browser Extensions",
"Frontend Development",
"Open Source Software",
"Manifest V3",
"Chrome Extensions",
"Firefox Add-ons",
"PHP",
"Symfony",
"Git",
"GitHub",
"Creative Commons Licensing",
"Task Management Applications"
],
"mainEntityOfPage": { "@id": "https://madjeek-web.github.io/#website" },
"owns": [
{ "@id": "#project-micromvc" },
{ "@id": "#project-popup-blocker" },
{ "@id": "#project-taskflow" },
{ "@id": "#project-symfony-mailjet" },
{ "@id": "#project-github-guide" },
{ "@id": "#project-licence-cc" }
]
},
{
"@type": "ItemList",
"@id": "https://madjeek-web.github.io/#projects",
"name": "Open-source projects by Fabien Conéjéro",
"description": "All public GitHub repositories created and maintained by Fabien Conéjéro.",
"numberOfItems": 6,
"itemListElement": [
{
"@type": "ListItem",
"position": 1,
"item": {
"@type": "SoftwareSourceCode",
"@id": "#project-micromvc",
"name": "micromvc-plus",
"description": "A modern, zero-dependency JavaScript MVC micro-framework. Built from scratch for 2026.",
"programmingLanguage": "JavaScript",
"codeRepository": "https://github.com/madjeek-web/micromvc-plus",
"url": "https://madjeek-web.github.io/microMVC__Demo_2026.html",
"license": "https://opensource.org/licenses/MIT",
"author": { "@id": "https://madjeek-web.github.io/#person" }
}
},
{
"@type": "ListItem",
"position": 2,
"item": {
"@type": "SoftwareApplication",
"@id": "#project-popup-blocker",
"name": "Popup Blocker Pro",
"description": "An open source, free, secure and powerful browser extension that blocks attempts to open tabs in the background without user consent. By Fabien Conéjéro.",
"applicationCategory": "BrowserApplication",
"operatingSystem": "Chrome, Firefox, Edge, Brave, Opera",
"programmingLanguage": "JavaScript",
"codeRepository": "https://github.com/madjeek-web/Popup-Blocker-Pro",
"url": "https://github.com/madjeek-web/Popup-Blocker-Pro",
"license": "https://opensource.org/licenses/MIT",
"author": { "@id": "https://madjeek-web.github.io/#person" },
"offers": {
"@type": "Offer",
"price": "0",
"priceCurrency": "EUR",
"availability": "https://schema.org/InStock"
}
}
},
{
"@type": "ListItem",
"position": 3,
"item": {
"@type": "SoftwareApplication",
"@id": "#project-taskflow",
"name": "Taskflow to-do list",
"description": "A free, simple, and powerful web app to manage your daily tasks. Works directly in your browser, with no complicated installation.",
"applicationCategory": "WebApplication",
"operatingSystem": "Any (browser-based)",
"programmingLanguage": "JavaScript",
"codeRepository": "https://github.com/madjeek-web/Taskflow_to-do-list",
"url": "https://github.com/madjeek-web/Taskflow_to-do-list",
"license": "https://opensource.org/licenses/MIT",
"author": { "@id": "https://madjeek-web.github.io/#person" },
"offers": {
"@type": "Offer",
"price": "0",
"priceCurrency": "EUR",
"availability": "https://schema.org/InStock"
}
}
},
{
"@type": "ListItem",
"position": 4,
"item": {
"@type": "SoftwareSourceCode",
"@id": "#project-symfony-mailjet",
"name": "symfony-mailjet-bundle",
"description": "A modern, async-ready Symfony 7 bundle for Mailjet — PHP 8.3, Messenger, HttpClient, Webhooks and full test coverage.",
"programmingLanguage": "PHP",
"runtimePlatform": "Symfony 7 / PHP 8.3",
"codeRepository": "https://github.com/madjeek-web/symfony-mailjet-bundle",
"url": "https://github.com/madjeek-web/symfony-mailjet-bundle",
"license": "https://opensource.org/licenses/MIT",
"author": { "@id": "https://madjeek-web.github.io/#person" }
}
},
{
"@type": "ListItem",
"position": 5,
"item": {
"@type": "Book",
"@id": "#project-github-guide",
"name": "GITHUB-GUIDE",
"description": "Features and craftsmanship with GitHub — Guide / Helper / Git Cheat Sheet. A practical reference for developers working with GitHub.",
"genre": "Technical Guide",
"codeRepository": "https://github.com/madjeek-web/GITHUB-GUIDE",
"url": "https://github.com/madjeek-web/GITHUB-GUIDE",
"author": { "@id": "https://madjeek-web.github.io/#person" }
}
},
{
"@type": "ListItem",
"position": 6,
"item": {
"@type": "CreativeWork",
"@id": "#project-licence-cc",
"name": "LICENCE-CC-BY-NC-ND-4.0",
"description": "Contrat Type / Modèle / LICENCE CC BY-NC-ND 4.0 — licence Creative Commons (CC). A ready-to-use template explaining how to sell commercial usage of CC-licensed works.",
"license": "https://creativecommons.org/licenses/by-nc-nd/4.0/",
"codeRepository": "https://github.com/madjeek-web/LICENCE-CC-BY-NC-ND-4.0",
"url": "https://github.com/madjeek-web/LICENCE-CC-BY-NC-ND-4.0",
"author": { "@id": "https://madjeek-web.github.io/#person" }
}
}
]
},
{
"@type": "SoftwareApplication",
"@id": "https://madjeek-web.github.io/microMVC__Demo_2026.html#software",
"name": "microMVC+",
"alternateName": "microMVC Plus",
"version": "1.0.0",
"description": "microMVC+ by Fabien Conéjéro — Zero-dependency JavaScript MVC micro-framework compatible ES5+. Provides Model/View/Controller separation, publish/subscribe event system, setState/getState API, once() listeners, publishAsync for Promise-based flows, and a built-in event logger.",
"applicationCategory": "DeveloperApplication",
"applicationSubCategory": "JavaScript Framework",
"operatingSystem": "Any (browser-based)",
"url": "https://madjeek-web.github.io/microMVC__Demo_2026.html",
"downloadUrl": "https://github.com/madjeek-web/micromvc-plus",
"softwareVersion": "1.0.0",
"releaseNotes": "First stable release. New features: Event log (Demo 5), publishAsync Promise-based flow (Demo 6), setState/getState API, once() listener support.",
"license": "https://opensource.org/licenses/MIT",
"programmingLanguage": { "@type": "ComputerLanguage", "name": "JavaScript" },
"featureList": [
"Zero external dependencies",
"ES5+ compatible",
"Model / View / Controller architecture",
"Publish / Subscribe event system",
"setState and getState API",
"once() one-time event listener",
"publishAsync Promise-based events",
"Built-in event log / debugger",
"MIT License"
],
"author": { "@id": "https://madjeek-web.github.io/#person" },
"offers": {
"@type": "Offer",
"price": "0",
"priceCurrency": "EUR",
"availability": "https://schema.org/InStock"
}
},
{
"@type": "BreadcrumbList",
"@id": "https://madjeek-web.github.io/microMVC__Demo_2026.html#breadcrumb",
"itemListElement": [
{
"@type": "ListItem",
"position": 1,
"name": "Fabien Conéjéro — GitHub Pages",
"item": "https://madjeek-web.github.io/"
},
{
"@type": "ListItem",
"position": 2,
"name": "microMVC+ Demo 2026",
"item": "https://madjeek-web.github.io/microMVC__Demo_2026.html"
}
]
},
{
"@type": "ItemList",
"@id": "https://madjeek-web.github.io/microMVC__Demo_2026.html#demos",
"name": "microMVC+ Interactive Demos by Fabien Conéjéro",
"description": "6 live interactive demos illustrating the main features of the microMVC+ framework, created by Fabien Conéjéro.",
"numberOfItems": 6,
"itemListElement": [
{
"@type": "ListItem",
"position": 1,
"name": "Demo 1 — Instant bank transfer",
"description": "The model holds the balance. The view listens. When the controller validates the transfer, the view updates instantly without any page reload.",
"url": "https://madjeek-web.github.io/microMVC__Demo_2026.html#demo1"
},
{
"@type": "ListItem",
"position": 2,
"name": "Demo 2 — Real-time notifications",
"description": "One model, three views updated simultaneously with a single publish call. Facebook-style scenario: bell icon, tab title, and notification list all in sync.",
"url": "https://madjeek-web.github.io/microMVC__Demo_2026.html#demo2"
},
{
"@type": "ListItem",
"position": 3,
"name": "Demo 3 — Task manager with state",
"description": "Demonstrates the setState/getState API on models. The view reacts to any state change automatically via the state.changed event.",
"url": "https://madjeek-web.github.io/microMVC__Demo_2026.html#demo3"
},
{
"@type": "ListItem",
"position": 4,
"name": "Demo 4 — Live profile editor",
"description": "Type in the fields and watch the profile card update instantly. Demonstrates once(): the welcome message fires only the first time you save.",
"url": "https://madjeek-web.github.io/microMVC__Demo_2026.html#demo4"
},
{
"@type": "ListItem",
"position": 5,
"name": "Demo 5 — Event log (new in v1.0.0)",
"description": "Every publish call is logged automatically. Use this during development to trace the exact sequence of events across all components.",
"url": "https://madjeek-web.github.io/microMVC__Demo_2026.html#demo5"
},
{
"@type": "ListItem",
"position": 6,
"name": "Demo 6 — publishAsync (Promise-based)",
"description": "Fire an event asynchronously and chain logic after all subscribers have been notified. Useful for loading states and sequential async flows.",
"url": "https://madjeek-web.github.io/microMVC__Demo_2026.html#demo6"
}
]
},
{
"@type": "FAQPage",
"@id": "https://madjeek-web.github.io/microMVC__Demo_2026.html#faq",
"mainEntity": [
{
"@type": "Question",
"name": "What is microMVC+ ?",
"acceptedAnswer": {
"@type": "Answer",
"text": "microMVC+ is a zero-dependency JavaScript MVC micro-framework created by Fabien Conéjéro. ES5+ compatible, it features a publish/subscribe event system, setState/getState API, once() listeners, publishAsync for Promise-based flows, and a built-in event logger. Released under MIT License."
}
},
{
"@type": "Question",
"name": "Who created microMVC+ ?",
"acceptedAnswer": {
"@type": "Answer",
"text": "microMVC+ was created by Fabien Conéjéro, a Frontend Developer who also authored Popup Blocker Pro (browser extension), Taskflow to-do list, symfony-mailjet-bundle (Symfony 7 / PHP 8.3), a GitHub guide and a Creative Commons license template. All projects are published as open-source on GitHub."
}
},
{
"@type": "Question",
"name": "What other projects has Fabien Conéjéro published ?",
"acceptedAnswer": {
"@type": "Answer",
"text": "Fabien Conéjéro has published 6 open-source projects on GitHub: microMVC+ (JavaScript MVC framework), Popup Blocker Pro (browser extension for Chrome and Firefox), Taskflow to-do list (browser web app), symfony-mailjet-bundle (Symfony 7 / PHP 8.3 async email bundle), GITHUB-GUIDE (Git cheat sheet and helper), and LICENCE-CC-BY-NC-ND-4.0 (Creative Commons license template)."
}
},
{
"@type": "Question",
"name": "Does microMVC+ require any external dependencies ?",
"acceptedAnswer": {
"@type": "Answer",
"text": "No. microMVC+ has zero external dependencies. It works in any modern browser and is fully compatible with ES5+ environments."
}
},
{
"@type": "Question",
"name": "Is microMVC+ free to use ?",
"acceptedAnswer": {
"@type": "Answer",
"text": "Yes. microMVC+ by Fabien Conéjéro is released under the MIT License, free for personal and commercial use."
}
},
{
"@type": "Question",
"name": "What is publishAsync in microMVC+ ?",
"acceptedAnswer": {
"@type": "Answer",
"text": "publishAsync is a Promise-based event publishing method in microMVC+. It fires an event asynchronously and allows chaining logic after all subscribers have been notified — ideal for loading states and sequential flows."
}
}
]
}
]
}
</script>
<style>
*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }
:root {
--bg : #f4f6f9;
--card : #ffffff;
--primary : #2563eb;
--accent : #10b981;
--danger : #ef4444;
--text : #1e293b;
--muted : #64748b;
--border : #e2e8f0;
--radius : 10px;
--shadow : 0 2px 12px rgba(0,0,0,0.08);
}
body {
font-family : system-ui, -apple-system, sans-serif;
background : var(--bg);
color : var(--text);
min-height : 100vh;
padding : 2rem 1rem;
}
header {
text-align : center;
margin-bottom : 2.5rem;
}
header h1 { font-size : 2rem; font-weight : 800; color : var(--primary); }
header p { color : var(--muted); margin-top : 0.4rem; font-size : 1rem; }
.badge-row {
display : flex;
flex-wrap : wrap;
justify-content : center;
gap : 0.5rem;
margin-top : 0.8rem;
}
.badge {
background : var(--primary);
color : #fff;
font-size : 0.72rem;
font-weight : 600;
padding : 3px 10px;
border-radius : 20px;
letter-spacing: 0.02em;
}
.badge.green { background : var(--accent); }
.badge.gray { background : var(--muted); }
.demos {
display : grid;
grid-template-columns : repeat(auto-fit, minmax(320px, 1fr));
gap : 1.5rem;
max-width : 1100px;
margin : 0 auto;
}
.card {
background : var(--card);
border-radius : var(--radius);
box-shadow : var(--shadow);
padding : 1.5rem;
border : 1px solid var(--border);
}
.card h2 {
font-size : 1.05rem;
font-weight : 700;
margin-bottom : 0.3rem;
color : var(--primary);
}
.card p.desc {
font-size : 0.85rem;
color : var(--muted);
margin-bottom : 1rem;
}
label {
display : block;
font-size : 0.82rem;
font-weight : 600;
margin-bottom : 0.3rem;
color : var(--muted);
}
input, select {
width : 100%;
padding : 0.5rem 0.75rem;
border : 1px solid var(--border);
border-radius : 6px;
font-size : 0.9rem;
margin-bottom : 0.8rem;
color : var(--text);
background : #fff;
}
button {
padding : 0.5rem 1.2rem;
border-radius : 6px;
border : none;
cursor : pointer;
font-weight : 600;
font-size : 0.88rem;
transition : opacity 0.15s;
}
button:hover { opacity : 0.85; }
button:disabled { opacity : 0.4; cursor : not-allowed; }
.btn-primary { background : var(--primary); color : #fff; }
.btn-accent { background : var(--accent); color : #fff; }
.btn-danger { background : var(--danger); color : #fff; }
.btn-outline { background : transparent; border : 1px solid var(--border); color : var(--text); }
.btn-row {
display : flex;
gap : 0.5rem;
flex-wrap: wrap;
}
.result-box {
margin-top : 1rem;
padding : 0.9rem 1rem;
background : var(--bg);
border-radius : 8px;
border : 1px solid var(--border);
font-size : 0.88rem;
min-height : 50px;
}
.balance {
font-size : 2rem;
font-weight : 800;
color : var(--accent);
display : block;
margin : 0.5rem 0;
}
.balance.negative { color : var(--danger); }
.notification-bell {
position : relative;
display : inline-block;
font-size : 1.8rem;
cursor : pointer;
user-select : none;
}
.notif-badge {
position : absolute;
top : -4px;
right : -8px;
background : var(--danger);
color : #fff;
font-size : 0.65rem;
font-weight : 800;
border-radius : 20px;
padding : 1px 5px;
min-width : 18px;
text-align : center;
}
.notif-list {
margin-top : 0.8rem;
font-size : 0.83rem;
}
.notif-item {
padding : 0.4rem 0.6rem;
border-radius : 6px;
margin-bottom : 0.3rem;
background : #eff6ff;
color : var(--primary);
border-left : 3px solid var(--primary);
}
.task-list {
list-style : none;
margin : 0.8rem 0;
}
.task-item {
display : flex;
align-items : center;
gap : 0.5rem;
padding : 0.4rem 0;
border-bottom : 1px solid var(--border);
font-size : 0.88rem;
}
.task-item.done span { text-decoration : line-through; color : var(--muted); }
.task-item button {
margin-left : auto;
padding : 2px 8px;
font-size : 0.75rem;
border-radius : 4px;
}
.profile-zone {
display : flex;
align-items : center;
gap : 1rem;
margin : 0.8rem 0;
}
.avatar {
width : 64px;
height : 64px;
border-radius : 50%;
background : var(--primary);
display : flex;
align-items : center;
justify-content : center;
font-size : 1.8rem;
color : #fff;
font-weight : 800;
flex-shrink : 0;
overflow : hidden;
}
.avatar img { width : 100%; height : 100%; object-fit : cover; }
.profile-info { font-size : 0.9rem; }
.profile-info strong { display : block; font-size : 1rem; }
.profile-info span { color : var(--muted); font-size : 0.82rem; }
.log-box {
background : #0f172a;
color : #94a3b8;
border-radius : 8px;
padding : 0.8rem 1rem;
font-family : monospace;
font-size : 0.78rem;
max-height : 160px;
overflow-y : auto;
margin-top : 1rem;
}
.log-box p { margin-bottom : 0.2rem; }
.log-box .ts { color : #475569; margin-right : 0.5rem; }
.log-box .ctx { color : #38bdf8; }
.log-box .val { color : #a3e635; }
.status-pill {
display : inline-block;
padding : 2px 10px;
border-radius : 20px;
font-size : 0.75rem;
font-weight : 700;
}
.status-pill.ok { background : #d1fae5; color : #065f46; }
.status-pill.err { background : #fee2e2; color : #991b1b; }
.status-pill.wait { background : #fef3c7; color : #92400e; }
.progress-bar {
height : 8px;
background : var(--border);
border-radius : 4px;
margin : 0.6rem 0;
overflow : hidden;
}
.progress-fill {
height : 100%;
background : var(--accent);
transition : width 0.4s ease;
border-radius : 4px;
}
footer {
text-align : center;
color : var(--muted);
font-size : 0.8rem;
margin-top : 3rem;
}
</style>
</head>
<body>
<header>
<h1>microMVC+</h1>
<p>Zero-dependency JavaScript MVC micro-framework. Interactive demos for 2026.</p>
<div class="badge-row">
<span class="badge">v1.0.0</span>
<span class="badge green">Zero dependencies</span>
<span class="badge gray">MIT License</span>
<span class="badge">ES5+ compatible</span>
</div>
</header>
<div class="demos">
<!-- DEMO 1 : Bank transfer -->
<div class="card">
<h2>Demo 1 : Instant bank transfer</h2>
<p class="desc">
The model holds your balance. The view listens. When the controller validates
the transfer, the view updates instantly without any page reload.
</p>
<label>Your current balance</label>
<span class="balance" id="balance-display">2 500.00 EUR</span>
<div class="progress-bar">
<div class="progress-fill" id="balance-bar" style="width: 100%"></div>
</div>
<label>Transfer amount (EUR)</label>
<input type="number" id="transfer-amount" value="200" min="1" />
<label>Recipient</label>
<input type="text" id="transfer-recipient" value="Alice Martin" />
<div class="btn-row">
<button class="btn-primary" id="btn-transfer">Send transfer</button>
<button class="btn-outline" id="btn-reset-bank">Reset</button>
</div>
<div class="result-box" id="bank-log">Waiting for a transfer...</div>
</div>
<!-- DEMO 2 : Notification center -->
<div class="card">
<h2>Demo 2 : Real-time notifications</h2>
<p class="desc">
One model, three views updated at the same time with a single publish call.
This is the exact Facebook-style scenario : bell, tab title, and list all in sync.
</p>
<div class="profile-zone">
<div>
<div class="notification-bell">
<span id="notif-bell-icon">bell</span>
<span class="notif-badge" id="notif-count-bell">0</span>
</div>
</div>
<div style="flex: 1; font-size: 0.85rem; color: var(--muted);">
Tab title counter : <strong id="notif-count-tab">0 notifications</strong><br />
Bell counter : <strong id="notif-count-header">0</strong>
</div>
</div>
<div class="btn-row">
<button class="btn-primary" id="btn-add-notif">Receive a notification</button>
<button class="btn-outline" id="btn-clear-notif">Mark all read</button>
</div>
<div class="notif-list" id="notif-list"></div>
</div>
<!-- DEMO 3 : Task manager with state -->
<div class="card">
<h2>Demo 3 : Task manager with state</h2>
<p class="desc">
Demonstrates the new setState/getState API on models. The view reacts to any
state change automatically via the "state.changed" event.
</p>
<label>New task</label>
<input type="text" id="task-input" placeholder="e.g. Review the pull request" />
<div class="btn-row">
<button class="btn-accent" id="btn-add-task">Add task</button>
<button class="btn-outline" id="btn-clear-tasks">Clear all</button>
</div>
<ul class="task-list" id="task-list"></ul>
<div class="result-box" id="task-stats">No tasks yet.</div>
</div>
<!-- DEMO 4 : Profile editor -->
<div class="card">
<h2>Demo 4 : Live profile editor</h2>
<p class="desc">
Type in the fields and watch the profile card update instantly.
Demonstrates once() : the welcome message fires only the first time you save.
</p>
<div class="profile-zone">
<div class="avatar" id="profile-avatar">F</div>
<div class="profile-info">
<strong id="profile-name">Fabien Conéjéro</strong>
<span id="profile-role">Frontend Developer</span>
</div>
</div>
<label>Full name</label>
<input type="text" id="input-name" value="Fabien Conéjéro" />
<label>Role</label>
<input type="text" id="input-role" value="Frontend Developer" />
<div class="btn-row">
<button class="btn-primary" id="btn-save-profile">Save profile</button>
</div>
<div class="result-box" id="profile-status">
<span class="status-pill ok">Saved</span> Profile ready.
</div>
</div>
<!-- DEMO 5 : Event log / debugger -->
<div class="card">
<h2>Demo 5 : Event log (new in v1.0.0)</h2>
<p class="desc">
Every publish call is logged automatically. Use this during development
to trace the exact sequence of events across your components.
</p>
<div class="btn-row">
<button class="btn-primary" id="btn-show-log">Show log</button>
<button class="btn-outline" id="btn-clear-log">Clear log</button>
</div>
<div class="log-box" id="event-log">
<p>Interact with the demos above, then click "Show log".</p>
</div>
</div>
<!-- DEMO 6 : publishAsync -->
<div class="card">
<h2>Demo 6 : publishAsync (Promise-based)</h2>
<p class="desc">
Fire an event asynchronously and chain logic after all subscribers
have been notified. Useful for loading states and sequential flows.
</p>
<div class="btn-row">
<button class="btn-primary" id="btn-async-start">Run async flow</button>
</div>
<div class="result-box" id="async-log">Click the button to start.</div>
</div>
</div>
<footer>
<p>microMVC+ v1.0.0 - Author : Fabien Conéjéro (2026) - MIT License</p>
</footer>
<script>
/**
* microMVC+
* A modern, zero-dependency JavaScript MVC micro-framework.
*
* Author : Fabien Conéjéro
* Version : 1.0.0
* Date : February 2026
* License : MIT
* Repository : https://github.com/madjeek-web
*/
(function (root, factory) {
if (typeof define === 'function' && define.amd) {
define([], factory);
} else if (typeof module === 'object' && module.exports) {
module.exports = factory();
} else {
root.microMVC = factory();
}
}(typeof self !== 'undefined' ? self : this, function () {
'use strict';
/**
* ELEMENT
*
* The base object that every model, view, and controller element inherits from.
* It knows its own id and which component it belongs to.
* It can publish events upward to its component.
*
* --- For a 14-year-old ---
* Think of an Element like a player in a team. Each player has a name (id),
* knows which team they play for (component), and can shout things to the team.
*
* --- For a junior dev ---
* Element is the base prototype for all registered MVC pieces.
* It holds a reference to its parent Component and exposes publish shortcuts.
*
* --- For teachers and trainers ---
* Element acts as the base mediator object. It is instantiated indirectly via
* Component.add(). The _constructor hook replaces the native constructor to
* allow deferred initialization after prototype assembly.
*/
var Element = function () {};
Element.prototype = {
/**
* _constructor
* Called automatically after the element is built and registered.
* Override this in your element definition to run setup code.
*/
_constructor : function () {},
/**
* publish
* Fires a namespaced event scoped to this element within its component.
* The full context becomes : elementId.yourContext
*
* @param {string} context - Event name (e.g. "onLogin")
* @param {object} event - Data payload to send to subscribers
*/
publish : function (context, event) {
this._component.publish(this._id + '.' + context, event);
},
/**
* publishAsync
* Same as publish but deferred to the next event loop tick.
* Returns a Promise so callers can chain or await.
*
* --- For a junior dev ---
* Useful when you need to ensure the DOM has updated before notifying,
* or when you want non-blocking fire-and-forget notifications.
*
* @param {string} context - Event name
* @param {object} event - Data payload
* @returns {Promise}
*/
publishAsync : function (context, event) {
return this._component.publishAsync(this._id + '.' + context, event);
},
/**
* destroy
* Removes this element from its component.
*/
destroy : function () {
this._component.remove(this._id);
}
};
/**
* COMPONENT
*
* A container that holds elements (models, views, or controllers).
* It manages the publish/subscribe messaging bus that allows all pieces to talk.
*
* --- For a 14-year-old ---
* Think of a Component like a WhatsApp group chat. Anyone in the group
* can send a message (publish) and anyone who subscribed will receive it.
* You can also leave the group at any time (unsubscribe).
*
* --- For a junior dev ---
* Component is a mediator. It holds a namespaced subscriber tree and fires
* callbacks when matching contexts are published. It also serves as the
* registry for all named elements.
*
* --- For teachers and trainers ---
* The subscriber namespace is a plain recursive object tree, traversed by a
* regex walker. Each node stores a _subscribers array. The publish method
* walks the tree and invokes callbacks at each matching depth level.
* This design allows hierarchical event bubbling with dot-notation scoping.
*/
var Component = function (app) {
this.__app__ = app;
this.subscribers = {};
this._log = [];
};
Component.prototype = {
/**
* add
* Registers a new element inside this component.
*
* @param {string} id - Unique name for the element (e.g. "userModel")
* @param {object} obj - Object with the element's methods and properties
* @param {function} ext - Optional base class to extend (default : Element)
* @returns {object} The created element instance
*/
add : function (id, obj, ext) {
if (this.has(id)) {
console.warn('microMVC : element "' + id + '" already exists. Returning existing instance.');
return this[id];
}
ext = ext || Element;
var element = function () {};