3 <html lang=
"en" class=
"no-js">
7 <meta name=
"viewport" content=
"width=device-width,initial-scale=1">
11 <link rel=
"canonical" href=
"https://docs.woltlab.com/6.0/tutorial/series/part_2/">
13 <link rel=
"icon" href=
"../../../assets/default.favicon.ico">
14 <meta name=
"generator" content=
"mkdocs-1.4.0, mkdocs-material-8.5.4">
18 <title>Part
2 - WoltLab Suite Documentation
</title>
22 <link rel=
"stylesheet" href=
"../../../assets/stylesheets/main.80dcb947.min.css">
25 <link rel=
"stylesheet" href=
"../../../assets/stylesheets/palette.cbb835fc.min.css">
29 <meta name=
"theme-color" content=
"#009485">
40 <link rel=
"stylesheet" href=
"../../../stylesheets/extra.css">
42 <script>__md_scope=new URL(
"../../..",location),__md_hash=e=
>[...e].reduce((e,_)=
>(e<
<5)-e+_.charCodeAt(
0),
0),__md_get=(e,_=localStorage,t=__md_scope)=
>JSON.parse(_.getItem(t.pathname+
"."+e)),__md_set=(e,_,t=localStorage,a=__md_scope)=
>{try{t.setItem(a.pathname+
"."+e,JSON.stringify(_))}catch(e){}}
</script>
57 <body dir=
"ltr" data-md-color-scheme=
"" data-md-color-primary=
"teal" data-md-color-accent=
"">
61 <input class=
"md-toggle" data-md-toggle=
"drawer" type=
"checkbox" id=
"__drawer" autocomplete=
"off">
62 <input class=
"md-toggle" data-md-toggle=
"search" type=
"checkbox" id=
"__search" autocomplete=
"off">
63 <label class=
"md-overlay" for=
"__drawer"></label>
64 <div data-md-component=
"skip">
67 <a href=
"#part-2-event-and-template-listeners" class=
"md-skip">
72 <div data-md-component=
"announce">
74 <aside class=
"md-banner">
75 <div class=
"md-banner__inner md-grid md-typeset">
78 <a href=
"https://www.woltlab.com">Back to
<strong>woltlab.com
</strong></a>
86 <div data-md-component=
"outdated" hidden
>
93 <header class=
"md-header" data-md-component=
"header">
94 <nav class=
"md-header__inner md-grid" aria-label=
"Header">
95 <a href=
"../../.." title=
"WoltLab Suite Documentation" class=
"md-header__button md-logo" aria-label=
"WoltLab Suite Documentation" data-md-component=
"logo">
97 <img src=
"../../../assets/logo.png" alt=
"logo">
100 <label class=
"md-header__button md-icon" for=
"__drawer">
101 <svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M3 6h18v2H3V6m0 5h18v2H3v-2m0 5h18v2H3v-2Z"/></svg>
103 <div class=
"md-header__title" data-md-component=
"header-title">
104 <div class=
"md-header__ellipsis">
105 <div class=
"md-header__topic">
106 <span class=
"md-ellipsis">
107 WoltLab Suite Documentation
110 <div class=
"md-header__topic" data-md-component=
"header-topic">
111 <span class=
"md-ellipsis">
122 <label class=
"md-header__button md-icon" for=
"__search">
123 <svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M9.5 3A6.5 6.5 0 0 1 16 9.5c0 1.61-.59 3.09-1.56 4.23l.27.27h.79l5 5-1.5 1.5-5-5v-.79l-.27-.27A6.516 6.516 0 0 1 9.5 16 6.5 6.5 0 0 1 3 9.5 6.5 6.5 0 0 1 9.5 3m0 2C7 5 5 7 5 9.5S7 14 9.5 14 14 12 14 9.5 12 5 9.5 5Z"/></svg>
125 <div class=
"md-search" data-md-component=
"search" role=
"dialog">
126 <label class=
"md-search__overlay" for=
"__search"></label>
127 <div class=
"md-search__inner" role=
"search">
128 <form class=
"md-search__form" name=
"search">
129 <input type=
"text" class=
"md-search__input" name=
"query" aria-label=
"Search" placeholder=
"Search" autocapitalize=
"off" autocorrect=
"off" autocomplete=
"off" spellcheck=
"false" data-md-component=
"search-query" required
>
130 <label class=
"md-search__icon md-icon" for=
"__search">
131 <svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M9.5 3A6.5 6.5 0 0 1 16 9.5c0 1.61-.59 3.09-1.56 4.23l.27.27h.79l5 5-1.5 1.5-5-5v-.79l-.27-.27A6.516 6.516 0 0 1 9.5 16 6.5 6.5 0 0 1 3 9.5 6.5 6.5 0 0 1 9.5 3m0 2C7 5 5 7 5 9.5S7 14 9.5 14 14 12 14 9.5 12 5 9.5 5Z"/></svg>
132 <svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M20 11v2H8l5.5 5.5-1.42 1.42L4.16 12l7.92-7.92L13.5 5.5 8 11h12Z"/></svg>
134 <nav class=
"md-search__options" aria-label=
"Search">
136 <button type=
"reset" class=
"md-search__icon md-icon" title=
"Clear" aria-label=
"Clear" tabindex=
"-1">
137 <svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M19 6.41 17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12 19 6.41Z"/></svg>
142 <div class=
"md-search__output">
143 <div class=
"md-search__scrollwrap" data-md-scrollfix
>
144 <div class=
"md-search-result" data-md-component=
"search-result">
145 <div class=
"md-search-result__meta">
148 <ol class=
"md-search-result__list"></ol>
156 <div class=
"md-header__source">
157 <a href=
"https://github.com/WoltLab/docs.woltlab.com/" title=
"Go to repository" class=
"md-source" data-md-component=
"source">
158 <div class=
"md-source__icon md-icon">
160 <svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 448 512"><!--! Font Awesome Free 6.2.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License) Copyright 2022 Fonticons, Inc.--><path d=
"M439.55 236.05 244 40.45a28.87 28.87 0 0 0-40.81 0l-40.66 40.63 51.52 51.52c27.06-9.14 52.68 16.77 43.39 43.68l49.66 49.66c34.23-11.8 61.18 31 35.47 56.69-26.49 26.49-70.21-2.87-56-37.34L240.22 199v121.85c25.3 12.54 22.26 41.85 9.08 55a34.34 34.34 0 0 1-48.55 0c-17.57-17.6-11.07-46.91 11.25-56v-123c-20.8-8.51-24.6-30.74-18.64-45L142.57 101 8.45 235.14a28.86 28.86 0 0 0 0 40.81l195.61 195.6a28.86 28.86 0 0 0 40.8 0l194.69-194.69a28.86 28.86 0 0 0 0-40.81z"/></svg>
162 <div class=
"md-source__repository">
172 <div class=
"md-container" data-md-component=
"container">
179 <main class=
"md-main" data-md-component=
"main">
180 <div class=
"md-main__inner md-grid">
184 <div class=
"md-sidebar md-sidebar--primary" data-md-component=
"sidebar" data-md-type=
"navigation" >
185 <div class=
"md-sidebar__scrollwrap">
186 <div class=
"md-sidebar__inner">
190 <nav class=
"md-nav md-nav--primary" aria-label=
"Navigation" data-md-level=
"0">
191 <label class=
"md-nav__title" for=
"__drawer">
192 <a href=
"../../.." title=
"WoltLab Suite Documentation" class=
"md-nav__button md-logo" aria-label=
"WoltLab Suite Documentation" data-md-component=
"logo">
194 <img src=
"../../../assets/logo.png" alt=
"logo">
197 WoltLab Suite Documentation
200 <div class=
"md-nav__source">
201 <a href=
"https://github.com/WoltLab/docs.woltlab.com/" title=
"Go to repository" class=
"md-source" data-md-component=
"source">
202 <div class=
"md-source__icon md-icon">
204 <svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 448 512"><!--! Font Awesome Free 6.2.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License) Copyright 2022 Fonticons, Inc.--><path d=
"M439.55 236.05 244 40.45a28.87 28.87 0 0 0-40.81 0l-40.66 40.63 51.52 51.52c27.06-9.14 52.68 16.77 43.39 43.68l49.66 49.66c34.23-11.8 61.18 31 35.47 56.69-26.49 26.49-70.21-2.87-56-37.34L240.22 199v121.85c25.3 12.54 22.26 41.85 9.08 55a34.34 34.34 0 0 1-48.55 0c-17.57-17.6-11.07-46.91 11.25-56v-123c-20.8-8.51-24.6-30.74-18.64-45L142.57 101 8.45 235.14a28.86 28.86 0 0 0 0 40.81l195.61 195.6a28.86 28.86 0 0 0 40.8 0l194.69-194.69a28.86 28.86 0 0 0 0-40.81z"/></svg>
206 <div class=
"md-source__repository">
212 <ul class=
"md-nav__list" data-md-scrollfix
>
221 <li class=
"md-nav__item">
222 <a href=
"../../../getting-started/" class=
"md-nav__link">
237 <li class=
"md-nav__item md-nav__item--nested">
240 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_2" type=
"checkbox" id=
"__nav_2" >
245 <label class=
"md-nav__link" for=
"__nav_2">
247 <span class=
"md-nav__icon md-icon"></span>
250 <nav class=
"md-nav" aria-label=
"PHP API" data-md-level=
"1">
251 <label class=
"md-nav__title" for=
"__nav_2">
252 <span class=
"md-nav__icon md-icon"></span>
255 <ul class=
"md-nav__list" data-md-scrollfix
>
262 <li class=
"md-nav__item">
263 <a href=
"../../../php/pages/" class=
"md-nav__link">
276 <li class=
"md-nav__item">
277 <a href=
"../../../php/database-objects/" class=
"md-nav__link">
290 <li class=
"md-nav__item">
291 <a href=
"../../../php/database-access/" class=
"md-nav__link">
304 <li class=
"md-nav__item">
305 <a href=
"../../../php/exceptions/" class=
"md-nav__link">
319 <li class=
"md-nav__item md-nav__item--nested">
322 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_2_5" type=
"checkbox" id=
"__nav_2_5" >
327 <label class=
"md-nav__link" for=
"__nav_2_5">
329 <span class=
"md-nav__icon md-icon"></span>
332 <nav class=
"md-nav" aria-label=
"API" data-md-level=
"2">
333 <label class=
"md-nav__title" for=
"__nav_2_5">
334 <span class=
"md-nav__icon md-icon"></span>
337 <ul class=
"md-nav__list" data-md-scrollfix
>
345 <li class=
"md-nav__item md-nav__item--nested">
348 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_2_5_1" type=
"checkbox" id=
"__nav_2_5_1" >
353 <label class=
"md-nav__link" for=
"__nav_2_5_1">
355 <span class=
"md-nav__icon md-icon"></span>
358 <nav class=
"md-nav" aria-label=
"Caches" data-md-level=
"3">
359 <label class=
"md-nav__title" for=
"__nav_2_5_1">
360 <span class=
"md-nav__icon md-icon"></span>
363 <ul class=
"md-nav__list" data-md-scrollfix
>
370 <li class=
"md-nav__item">
371 <a href=
"../../../php/api/caches/" class=
"md-nav__link">
384 <li class=
"md-nav__item">
385 <a href=
"../../../php/api/caches_persistent-caches/" class=
"md-nav__link">
398 <li class=
"md-nav__item">
399 <a href=
"../../../php/api/caches_runtime-caches/" class=
"md-nav__link">
419 <li class=
"md-nav__item">
420 <a href=
"../../../php/api/comments/" class=
"md-nav__link">
433 <li class=
"md-nav__item">
434 <a href=
"../../../php/api/cronjobs/" class=
"md-nav__link">
447 <li class=
"md-nav__item">
448 <a href=
"../../../php/api/events/" class=
"md-nav__link">
462 <li class=
"md-nav__item md-nav__item--nested">
465 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_2_5_5" type=
"checkbox" id=
"__nav_2_5_5" >
470 <label class=
"md-nav__link" for=
"__nav_2_5_5">
472 <span class=
"md-nav__icon md-icon"></span>
475 <nav class=
"md-nav" aria-label=
"Form Builder" data-md-level=
"3">
476 <label class=
"md-nav__title" for=
"__nav_2_5_5">
477 <span class=
"md-nav__icon md-icon"></span>
480 <ul class=
"md-nav__list" data-md-scrollfix
>
487 <li class=
"md-nav__item">
488 <a href=
"../../../php/api/form_builder/overview/" class=
"md-nav__link">
501 <li class=
"md-nav__item">
502 <a href=
"../../../php/api/form_builder/structure/" class=
"md-nav__link">
515 <li class=
"md-nav__item">
516 <a href=
"../../../php/api/form_builder/form_fields/" class=
"md-nav__link">
529 <li class=
"md-nav__item">
530 <a href=
"../../../php/api/form_builder/validation_data/" class=
"md-nav__link">
543 <li class=
"md-nav__item">
544 <a href=
"../../../php/api/form_builder/dependencies/" class=
"md-nav__link">
564 <li class=
"md-nav__item">
565 <a href=
"../../../php/api/package_installation_plugins/" class=
"md-nav__link">
566 Package Installation Plugins
578 <li class=
"md-nav__item">
579 <a href=
"../../../php/api/user_activity_points/" class=
"md-nav__link">
592 <li class=
"md-nav__item">
593 <a href=
"../../../php/api/user_notifications/" class=
"md-nav__link">
606 <li class=
"md-nav__item">
607 <a href=
"../../../php/api/sitemaps/" class=
"md-nav__link">
627 <li class=
"md-nav__item">
628 <a href=
"../../../php/code-style/" class=
"md-nav__link">
641 <li class=
"md-nav__item">
642 <a href=
"../../../php/apps/" class=
"md-nav__link">
655 <li class=
"md-nav__item">
656 <a href=
"../../../php/gdpr/" class=
"md-nav__link">
678 <li class=
"md-nav__item md-nav__item--nested">
681 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_3" type=
"checkbox" id=
"__nav_3" >
686 <label class=
"md-nav__link" for=
"__nav_3">
687 Languages, Templates & CSS
688 <span class=
"md-nav__icon md-icon"></span>
691 <nav class=
"md-nav" aria-label=
"Languages, Templates & CSS" data-md-level=
"1">
692 <label class=
"md-nav__title" for=
"__nav_3">
693 <span class=
"md-nav__icon md-icon"></span>
694 Languages, Templates & CSS
696 <ul class=
"md-nav__list" data-md-scrollfix
>
703 <li class=
"md-nav__item">
704 <a href=
"../../../view/languages/" class=
"md-nav__link">
717 <li class=
"md-nav__item">
718 <a href=
"../../../view/templates/" class=
"md-nav__link">
731 <li class=
"md-nav__item">
732 <a href=
"../../../view/template-plugins/" class=
"md-nav__link">
745 <li class=
"md-nav__item">
746 <a href=
"../../../view/css/" class=
"md-nav__link">
768 <li class=
"md-nav__item md-nav__item--nested">
771 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_4" type=
"checkbox" id=
"__nav_4" >
776 <label class=
"md-nav__link" for=
"__nav_4">
777 TypeScript and JavaScript API
778 <span class=
"md-nav__icon md-icon"></span>
781 <nav class=
"md-nav" aria-label=
"TypeScript and JavaScript API" data-md-level=
"1">
782 <label class=
"md-nav__title" for=
"__nav_4">
783 <span class=
"md-nav__icon md-icon"></span>
784 TypeScript and JavaScript API
786 <ul class=
"md-nav__list" data-md-scrollfix
>
793 <li class=
"md-nav__item">
794 <a href=
"../../../javascript/general-usage/" class=
"md-nav__link">
807 <li class=
"md-nav__item">
808 <a href=
"../../../javascript/typescript/" class=
"md-nav__link">
822 <li class=
"md-nav__item md-nav__item--nested">
825 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_4_3" type=
"checkbox" id=
"__nav_4_3" >
830 <label class=
"md-nav__link" for=
"__nav_4_3">
832 <span class=
"md-nav__icon md-icon"></span>
835 <nav class=
"md-nav" aria-label=
"Components" data-md-level=
"2">
836 <label class=
"md-nav__title" for=
"__nav_4_3">
837 <span class=
"md-nav__icon md-icon"></span>
840 <ul class=
"md-nav__list" data-md-scrollfix
>
847 <li class=
"md-nav__item">
848 <a href=
"../../../javascript/components_confirmation/" class=
"md-nav__link">
861 <li class=
"md-nav__item">
862 <a href=
"../../../javascript/components_dialog/" class=
"md-nav__link">
883 <li class=
"md-nav__item md-nav__item--nested">
886 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_4_4" type=
"checkbox" id=
"__nav_4_4" >
891 <label class=
"md-nav__link" for=
"__nav_4_4">
893 <span class=
"md-nav__icon md-icon"></span>
896 <nav class=
"md-nav" aria-label=
"New API" data-md-level=
"2">
897 <label class=
"md-nav__title" for=
"__nav_4_4">
898 <span class=
"md-nav__icon md-icon"></span>
901 <ul class=
"md-nav__list" data-md-scrollfix
>
908 <li class=
"md-nav__item">
909 <a href=
"../../../javascript/new-api_writing-a-module/" class=
"md-nav__link">
922 <li class=
"md-nav__item">
923 <a href=
"../../../javascript/new-api_data-structures/" class=
"md-nav__link">
936 <li class=
"md-nav__item">
937 <a href=
"../../../javascript/new-api_core/" class=
"md-nav__link">
950 <li class=
"md-nav__item">
951 <a href=
"../../../javascript/new-api_dom/" class=
"md-nav__link">
964 <li class=
"md-nav__item">
965 <a href=
"../../../javascript/new-api_events/" class=
"md-nav__link">
978 <li class=
"md-nav__item">
979 <a href=
"../../../javascript/new-api_ajax/" class=
"md-nav__link">
992 <li class=
"md-nav__item">
993 <a href=
"../../../javascript/new-api_dialogs/" class=
"md-nav__link">
1006 <li class=
"md-nav__item">
1007 <a href=
"../../../javascript/new-api_browser/" class=
"md-nav__link">
1008 Browser and Screen Sizes
1020 <li class=
"md-nav__item">
1021 <a href=
"../../../javascript/new-api_ui/" class=
"md-nav__link">
1041 <li class=
"md-nav__item">
1042 <a href=
"../../../javascript/legacy-api/" class=
"md-nav__link">
1055 <li class=
"md-nav__item">
1056 <a href=
"../../../javascript/helper-functions/" class=
"md-nav__link">
1069 <li class=
"md-nav__item">
1070 <a href=
"../../../javascript/code-snippets/" class=
"md-nav__link">
1092 <li class=
"md-nav__item md-nav__item--nested">
1095 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_5" type=
"checkbox" id=
"__nav_5" >
1100 <label class=
"md-nav__link" for=
"__nav_5">
1102 <span class=
"md-nav__icon md-icon"></span>
1105 <nav class=
"md-nav" aria-label=
"Package Components" data-md-level=
"1">
1106 <label class=
"md-nav__title" for=
"__nav_5">
1107 <span class=
"md-nav__icon md-icon"></span>
1110 <ul class=
"md-nav__list" data-md-scrollfix
>
1117 <li class=
"md-nav__item">
1118 <a href=
"../../../package/package-xml/" class=
"md-nav__link">
1132 <li class=
"md-nav__item md-nav__item--nested">
1135 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_5_2" type=
"checkbox" id=
"__nav_5_2" >
1140 <label class=
"md-nav__link" for=
"__nav_5_2">
1142 <span class=
"md-nav__icon md-icon"></span>
1145 <nav class=
"md-nav" aria-label=
"PIPs" data-md-level=
"2">
1146 <label class=
"md-nav__title" for=
"__nav_5_2">
1147 <span class=
"md-nav__icon md-icon"></span>
1150 <ul class=
"md-nav__list" data-md-scrollfix
>
1157 <li class=
"md-nav__item">
1158 <a href=
"../../../package/pip/" class=
"md-nav__link">
1171 <li class=
"md-nav__item">
1172 <a href=
"../../../package/pip/acl-option/" class=
"md-nav__link">
1185 <li class=
"md-nav__item">
1186 <a href=
"../../../package/pip/acp-menu/" class=
"md-nav__link">
1199 <li class=
"md-nav__item">
1200 <a href=
"../../../package/pip/acp-search-provider/" class=
"md-nav__link">
1213 <li class=
"md-nav__item">
1214 <a href=
"../../../package/pip/acp-template/" class=
"md-nav__link">
1227 <li class=
"md-nav__item">
1228 <a href=
"../../../package/pip/acp-template-delete/" class=
"md-nav__link">
1241 <li class=
"md-nav__item">
1242 <a href=
"../../../package/pip/bbcode/" class=
"md-nav__link">
1255 <li class=
"md-nav__item">
1256 <a href=
"../../../package/pip/box/" class=
"md-nav__link">
1269 <li class=
"md-nav__item">
1270 <a href=
"../../../package/pip/clipboard-action/" class=
"md-nav__link">
1283 <li class=
"md-nav__item">
1284 <a href=
"../../../package/pip/core-object/" class=
"md-nav__link">
1297 <li class=
"md-nav__item">
1298 <a href=
"../../../package/pip/cronjob/" class=
"md-nav__link">
1311 <li class=
"md-nav__item">
1312 <a href=
"../../../package/pip/database/" class=
"md-nav__link">
1325 <li class=
"md-nav__item">
1326 <a href=
"../../../package/pip/event-listener/" class=
"md-nav__link">
1339 <li class=
"md-nav__item">
1340 <a href=
"../../../package/pip/file/" class=
"md-nav__link">
1353 <li class=
"md-nav__item">
1354 <a href=
"../../../package/pip/file-delete/" class=
"md-nav__link">
1367 <li class=
"md-nav__item">
1368 <a href=
"../../../package/pip/language/" class=
"md-nav__link">
1381 <li class=
"md-nav__item">
1382 <a href=
"../../../package/pip/media-provider/" class=
"md-nav__link">
1395 <li class=
"md-nav__item">
1396 <a href=
"../../../package/pip/menu/" class=
"md-nav__link">
1409 <li class=
"md-nav__item">
1410 <a href=
"../../../package/pip/menu-item/" class=
"md-nav__link">
1423 <li class=
"md-nav__item">
1424 <a href=
"../../../package/pip/object-type/" class=
"md-nav__link">
1437 <li class=
"md-nav__item">
1438 <a href=
"../../../package/pip/object-type-definition/" class=
"md-nav__link">
1439 objectTypeDefinition
1451 <li class=
"md-nav__item">
1452 <a href=
"../../../package/pip/option/" class=
"md-nav__link">
1465 <li class=
"md-nav__item">
1466 <a href=
"../../../package/pip/page/" class=
"md-nav__link">
1479 <li class=
"md-nav__item">
1480 <a href=
"../../../package/pip/pip/" class=
"md-nav__link">
1493 <li class=
"md-nav__item">
1494 <a href=
"../../../package/pip/script/" class=
"md-nav__link">
1507 <li class=
"md-nav__item">
1508 <a href=
"../../../package/pip/smiley/" class=
"md-nav__link">
1521 <li class=
"md-nav__item">
1522 <a href=
"../../../package/pip/sql/" class=
"md-nav__link">
1535 <li class=
"md-nav__item">
1536 <a href=
"../../../package/pip/style/" class=
"md-nav__link">
1549 <li class=
"md-nav__item">
1550 <a href=
"../../../package/pip/template/" class=
"md-nav__link">
1563 <li class=
"md-nav__item">
1564 <a href=
"../../../package/pip/template-delete/" class=
"md-nav__link">
1577 <li class=
"md-nav__item">
1578 <a href=
"../../../package/pip/template-listener/" class=
"md-nav__link">
1591 <li class=
"md-nav__item">
1592 <a href=
"../../../package/pip/user-group-option/" class=
"md-nav__link">
1605 <li class=
"md-nav__item">
1606 <a href=
"../../../package/pip/user-menu/" class=
"md-nav__link">
1619 <li class=
"md-nav__item">
1620 <a href=
"../../../package/pip/user-notification-event/" class=
"md-nav__link">
1621 userNotificationEvent
1633 <li class=
"md-nav__item">
1634 <a href=
"../../../package/pip/user-option/" class=
"md-nav__link">
1647 <li class=
"md-nav__item">
1648 <a href=
"../../../package/pip/user-profile-menu/" class=
"md-nav__link">
1668 <li class=
"md-nav__item">
1669 <a href=
"../../../package/database-php-api/" class=
"md-nav__link">
1691 <li class=
"md-nav__item md-nav__item--nested">
1694 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_6" type=
"checkbox" id=
"__nav_6" >
1699 <label class=
"md-nav__link" for=
"__nav_6">
1701 <span class=
"md-nav__icon md-icon"></span>
1704 <nav class=
"md-nav" aria-label=
"Migration" data-md-level=
"1">
1705 <label class=
"md-nav__title" for=
"__nav_6">
1706 <span class=
"md-nav__icon md-icon"></span>
1709 <ul class=
"md-nav__list" data-md-scrollfix
>
1717 <li class=
"md-nav__item md-nav__item--nested">
1720 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_6_1" type=
"checkbox" id=
"__nav_6_1" >
1725 <label class=
"md-nav__link" for=
"__nav_6_1">
1726 From WoltLab Suite
5.5
1727 <span class=
"md-nav__icon md-icon"></span>
1730 <nav class=
"md-nav" aria-label=
"From WoltLab Suite 5.5" data-md-level=
"2">
1731 <label class=
"md-nav__title" for=
"__nav_6_1">
1732 <span class=
"md-nav__icon md-icon"></span>
1733 From WoltLab Suite
5.5
1735 <ul class=
"md-nav__list" data-md-scrollfix
>
1742 <li class=
"md-nav__item">
1743 <a href=
"../../../migration/wsc55/php/" class=
"md-nav__link">
1756 <li class=
"md-nav__item">
1757 <a href=
"../../../migration/wsc55/javascript/" class=
"md-nav__link">
1758 TypeScript and JavaScript
1770 <li class=
"md-nav__item">
1771 <a href=
"../../../migration/wsc55/templates/" class=
"md-nav__link">
1784 <li class=
"md-nav__item">
1785 <a href=
"../../../migration/wsc55/icons/" class=
"md-nav__link">
1798 <li class=
"md-nav__item">
1799 <a href=
"../../../migration/wsc55/libraries/" class=
"md-nav__link">
1800 Third Party Libraries
1812 <li class=
"md-nav__item">
1813 <a href=
"../../../migration/wsc55/deprecations_removals/" class=
"md-nav__link">
1814 Deprecations and Removals
1834 <li class=
"md-nav__item md-nav__item--nested">
1837 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_6_2" type=
"checkbox" id=
"__nav_6_2" >
1842 <label class=
"md-nav__link" for=
"__nav_6_2">
1843 From WoltLab Suite
5.4
1844 <span class=
"md-nav__icon md-icon"></span>
1847 <nav class=
"md-nav" aria-label=
"From WoltLab Suite 5.4" data-md-level=
"2">
1848 <label class=
"md-nav__title" for=
"__nav_6_2">
1849 <span class=
"md-nav__icon md-icon"></span>
1850 From WoltLab Suite
5.4
1852 <ul class=
"md-nav__list" data-md-scrollfix
>
1859 <li class=
"md-nav__item">
1860 <a href=
"../../../migration/wsc54/php/" class=
"md-nav__link">
1873 <li class=
"md-nav__item">
1874 <a href=
"../../../migration/wsc54/javascript/" class=
"md-nav__link">
1875 TypeScript and JavaScript
1887 <li class=
"md-nav__item">
1888 <a href=
"../../../migration/wsc54/templates/" class=
"md-nav__link">
1901 <li class=
"md-nav__item">
1902 <a href=
"../../../migration/wsc54/libraries/" class=
"md-nav__link">
1903 Third Party Libraries
1915 <li class=
"md-nav__item">
1916 <a href=
"../../../migration/wsc54/deprecations_removals/" class=
"md-nav__link">
1917 Deprecations and Removals
1937 <li class=
"md-nav__item md-nav__item--nested">
1940 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_6_3" type=
"checkbox" id=
"__nav_6_3" >
1945 <label class=
"md-nav__link" for=
"__nav_6_3">
1946 From WoltLab Suite
5.3
1947 <span class=
"md-nav__icon md-icon"></span>
1950 <nav class=
"md-nav" aria-label=
"From WoltLab Suite 5.3" data-md-level=
"2">
1951 <label class=
"md-nav__title" for=
"__nav_6_3">
1952 <span class=
"md-nav__icon md-icon"></span>
1953 From WoltLab Suite
5.3
1955 <ul class=
"md-nav__list" data-md-scrollfix
>
1962 <li class=
"md-nav__item">
1963 <a href=
"../../../migration/wsc53/php/" class=
"md-nav__link">
1976 <li class=
"md-nav__item">
1977 <a href=
"../../../migration/wsc53/session/" class=
"md-nav__link">
1978 Session Handling and Authentication
1990 <li class=
"md-nav__item">
1991 <a href=
"../../../migration/wsc53/javascript/" class=
"md-nav__link">
1992 TypeScript and JavaScript
2004 <li class=
"md-nav__item">
2005 <a href=
"../../../migration/wsc53/templates/" class=
"md-nav__link">
2018 <li class=
"md-nav__item">
2019 <a href=
"../../../migration/wsc53/libraries/" class=
"md-nav__link">
2020 Third Party Libraries
2040 <li class=
"md-nav__item md-nav__item--nested">
2043 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_6_4" type=
"checkbox" id=
"__nav_6_4" >
2048 <label class=
"md-nav__link" for=
"__nav_6_4">
2049 From WoltLab Suite
5.2
2050 <span class=
"md-nav__icon md-icon"></span>
2053 <nav class=
"md-nav" aria-label=
"From WoltLab Suite 5.2" data-md-level=
"2">
2054 <label class=
"md-nav__title" for=
"__nav_6_4">
2055 <span class=
"md-nav__icon md-icon"></span>
2056 From WoltLab Suite
5.2
2058 <ul class=
"md-nav__list" data-md-scrollfix
>
2065 <li class=
"md-nav__item">
2066 <a href=
"../../../migration/wsc52/php/" class=
"md-nav__link">
2079 <li class=
"md-nav__item">
2080 <a href=
"../../../migration/wsc52/templates/" class=
"md-nav__link">
2081 Templates and Languages
2093 <li class=
"md-nav__item">
2094 <a href=
"../../../migration/wsc52/libraries/" class=
"md-nav__link">
2095 Third Party Libraries
2115 <li class=
"md-nav__item md-nav__item--nested">
2118 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_6_5" type=
"checkbox" id=
"__nav_6_5" >
2123 <label class=
"md-nav__link" for=
"__nav_6_5">
2124 From WoltLab Suite
3.1
2125 <span class=
"md-nav__icon md-icon"></span>
2128 <nav class=
"md-nav" aria-label=
"From WoltLab Suite 3.1" data-md-level=
"2">
2129 <label class=
"md-nav__title" for=
"__nav_6_5">
2130 <span class=
"md-nav__icon md-icon"></span>
2131 From WoltLab Suite
3.1
2133 <ul class=
"md-nav__list" data-md-scrollfix
>
2140 <li class=
"md-nav__item">
2141 <a href=
"../../../migration/wsc31/php/" class=
"md-nav__link">
2162 <li class=
"md-nav__item md-nav__item--nested">
2165 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_6_6" type=
"checkbox" id=
"__nav_6_6" >
2170 <label class=
"md-nav__link" for=
"__nav_6_6">
2171 From WoltLab Suite
3.0
2172 <span class=
"md-nav__icon md-icon"></span>
2175 <nav class=
"md-nav" aria-label=
"From WoltLab Suite 3.0" data-md-level=
"2">
2176 <label class=
"md-nav__title" for=
"__nav_6_6">
2177 <span class=
"md-nav__icon md-icon"></span>
2178 From WoltLab Suite
3.0
2180 <ul class=
"md-nav__list" data-md-scrollfix
>
2187 <li class=
"md-nav__item">
2188 <a href=
"../../../migration/wsc30/php/" class=
"md-nav__link">
2201 <li class=
"md-nav__item">
2202 <a href=
"../../../migration/wsc30/javascript/" class=
"md-nav__link">
2215 <li class=
"md-nav__item">
2216 <a href=
"../../../migration/wsc30/templates/" class=
"md-nav__link">
2229 <li class=
"md-nav__item">
2230 <a href=
"../../../migration/wsc30/css/" class=
"md-nav__link">
2243 <li class=
"md-nav__item">
2244 <a href=
"../../../migration/wsc30/package/" class=
"md-nav__link">
2265 <li class=
"md-nav__item md-nav__item--nested">
2268 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_6_7" type=
"checkbox" id=
"__nav_6_7" >
2273 <label class=
"md-nav__link" for=
"__nav_6_7">
2275 <span class=
"md-nav__icon md-icon"></span>
2278 <nav class=
"md-nav" aria-label=
"From WCF 2.1" data-md-level=
"2">
2279 <label class=
"md-nav__title" for=
"__nav_6_7">
2280 <span class=
"md-nav__icon md-icon"></span>
2283 <ul class=
"md-nav__list" data-md-scrollfix
>
2290 <li class=
"md-nav__item">
2291 <a href=
"../../../migration/wcf21/php/" class=
"md-nav__link">
2304 <li class=
"md-nav__item">
2305 <a href=
"../../../migration/wcf21/templates/" class=
"md-nav__link">
2318 <li class=
"md-nav__item">
2319 <a href=
"../../../migration/wcf21/css/" class=
"md-nav__link">
2332 <li class=
"md-nav__item">
2333 <a href=
"../../../migration/wcf21/package/" class=
"md-nav__link">
2364 <li class=
"md-nav__item md-nav__item--active md-nav__item--nested">
2367 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_7" type=
"checkbox" id=
"__nav_7" checked
>
2372 <label class=
"md-nav__link" for=
"__nav_7">
2374 <span class=
"md-nav__icon md-icon"></span>
2377 <nav class=
"md-nav" aria-label=
"Tutorials" data-md-level=
"1">
2378 <label class=
"md-nav__title" for=
"__nav_7">
2379 <span class=
"md-nav__icon md-icon"></span>
2382 <ul class=
"md-nav__list" data-md-scrollfix
>
2392 <li class=
"md-nav__item md-nav__item--active md-nav__item--nested">
2395 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"__nav_7_1" type=
"checkbox" id=
"__nav_7_1" checked
>
2400 <label class=
"md-nav__link" for=
"__nav_7_1">
2402 <span class=
"md-nav__icon md-icon"></span>
2405 <nav class=
"md-nav" aria-label=
"Tutorial Series" data-md-level=
"2">
2406 <label class=
"md-nav__title" for=
"__nav_7_1">
2407 <span class=
"md-nav__icon md-icon"></span>
2410 <ul class=
"md-nav__list" data-md-scrollfix
>
2417 <li class=
"md-nav__item">
2418 <a href=
"../overview/" class=
"md-nav__link">
2431 <li class=
"md-nav__item">
2432 <a href=
"../part_1/" class=
"md-nav__link">
2447 <li class=
"md-nav__item md-nav__item--active">
2449 <input class=
"md-nav__toggle md-toggle" data-md-toggle=
"toc" type=
"checkbox" id=
"__toc">
2455 <label class=
"md-nav__link md-nav__link--active" for=
"__toc">
2457 <span class=
"md-nav__icon md-icon"></span>
2460 <a href=
"./" class=
"md-nav__link md-nav__link--active">
2466 <nav class=
"md-nav md-nav--secondary" aria-label=
"Table of contents">
2473 <label class=
"md-nav__title" for=
"__toc">
2474 <span class=
"md-nav__icon md-icon"></span>
2477 <ul class=
"md-nav__list" data-md-component=
"toc" data-md-scrollfix
>
2479 <li class=
"md-nav__item">
2480 <a href=
"#package-functionality" class=
"md-nav__link">
2481 Package Functionality
2486 <li class=
"md-nav__item">
2487 <a href=
"#used-components" class=
"md-nav__link">
2493 <li class=
"md-nav__item">
2494 <a href=
"#package-structure" class=
"md-nav__link">
2500 <li class=
"md-nav__item">
2501 <a href=
"#extending-person-model" class=
"md-nav__link">
2502 Extending Person Model
2507 <li class=
"md-nav__item">
2508 <a href=
"#setting-birthday-in-acp" class=
"md-nav__link">
2509 Setting Birthday in ACP
2514 <li class=
"md-nav__item">
2515 <a href=
"#adding-birthday-table-column-in-acp" class=
"md-nav__link">
2516 Adding Birthday Table Column in ACP
2521 <li class=
"md-nav__item">
2522 <a href=
"#adding-birthday-in-front-end" class=
"md-nav__link">
2523 Adding Birthday in Front End
2528 <li class=
"md-nav__item">
2529 <a href=
"#templatelistenerxml" class=
"md-nav__link">
2530 templateListener.xml
2535 <li class=
"md-nav__item">
2536 <a href=
"#eventlistenerxml" class=
"md-nav__link">
2542 <li class=
"md-nav__item">
2543 <a href=
"#packagexml" class=
"md-nav__link">
2563 <li class=
"md-nav__item">
2564 <a href=
"../part_3/" class=
"md-nav__link">
2577 <li class=
"md-nav__item">
2578 <a href=
"../part_4/" class=
"md-nav__link">
2591 <li class=
"md-nav__item">
2592 <a href=
"../part_5/" class=
"md-nav__link">
2605 <li class=
"md-nav__item">
2606 <a href=
"../part_6/" class=
"md-nav__link">
2635 <div class=
"md-sidebar md-sidebar--secondary" data-md-component=
"sidebar" data-md-type=
"toc" >
2636 <div class=
"md-sidebar__scrollwrap">
2637 <div class=
"md-sidebar__inner">
2640 <nav class=
"md-nav md-nav--secondary" aria-label=
"Table of contents">
2647 <label class=
"md-nav__title" for=
"__toc">
2648 <span class=
"md-nav__icon md-icon"></span>
2651 <ul class=
"md-nav__list" data-md-component=
"toc" data-md-scrollfix
>
2653 <li class=
"md-nav__item">
2654 <a href=
"#package-functionality" class=
"md-nav__link">
2655 Package Functionality
2660 <li class=
"md-nav__item">
2661 <a href=
"#used-components" class=
"md-nav__link">
2667 <li class=
"md-nav__item">
2668 <a href=
"#package-structure" class=
"md-nav__link">
2674 <li class=
"md-nav__item">
2675 <a href=
"#extending-person-model" class=
"md-nav__link">
2676 Extending Person Model
2681 <li class=
"md-nav__item">
2682 <a href=
"#setting-birthday-in-acp" class=
"md-nav__link">
2683 Setting Birthday in ACP
2688 <li class=
"md-nav__item">
2689 <a href=
"#adding-birthday-table-column-in-acp" class=
"md-nav__link">
2690 Adding Birthday Table Column in ACP
2695 <li class=
"md-nav__item">
2696 <a href=
"#adding-birthday-in-front-end" class=
"md-nav__link">
2697 Adding Birthday in Front End
2702 <li class=
"md-nav__item">
2703 <a href=
"#templatelistenerxml" class=
"md-nav__link">
2704 templateListener.xml
2709 <li class=
"md-nav__item">
2710 <a href=
"#eventlistenerxml" class=
"md-nav__link">
2716 <li class=
"md-nav__item">
2717 <a href=
"#packagexml" class=
"md-nav__link">
2732 <div class=
"md-content" data-md-component=
"content">
2733 <article class=
"md-content__inner md-typeset">
2737 <a href=
"https://github.com/WoltLab/docs.woltlab.com/edit/6.0/docs/tutorial/series/part_2.md" title=
"Edit this page" class=
"md-content__button md-icon">
2739 <svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M20.71 7.04c.39-.39.39-1.04 0-1.41l-2.34-2.34c-.37-.39-1.02-.39-1.41 0l-1.84 1.83 3.75 3.75M3 17.25V21h3.75L17.81 9.93l-3.75-3.75L3 17.25Z"/></svg>
2743 <h1 id=
"part-2-event-and-template-listeners">Part
2: Event and Template Listeners
<a class=
"headerlink" href=
"#part-2-event-and-template-listeners" title=
"Permanent link">#
</a></h1>
2744 <p>In the
<a href=
"../part_1/">first part
</a> of this tutorial series, we have created the base structure of our people management package.
2745 In further parts, we will use the package of the first part as a basis to directly add new features.
2746 In order to explain how event listeners and template works, however, we will not directly adding a new feature to the package by altering it in this part, but we will assume that somebody else created the package and that we want to extend it the “correct” way by creating a plugin.
</p>
2747 <p>The goal of the small plugin that will be created in this part is to add the birthday of the managed people.
2748 As in the first part, we will not bother with careful validation of the entered date but just make sure that it is a valid date.
</p>
2749 <h2 id=
"package-functionality">Package Functionality
<a class=
"headerlink" href=
"#package-functionality" title=
"Permanent link">#
</a></h2>
2750 <p>The package should provide the following possibilities/functions:
</p>
2752 <li>List person’s birthday (if set) in people list in the ACP
</li>
2753 <li>Sort people list by birthday in the ACP
</li>
2754 <li>Add or remove birthday when adding or editing person
</li>
2755 <li>List person’s birthday (if set) in people list in the front end
</li>
2756 <li>Sort people list by birthday in the front end
</li>
2758 <h2 id=
"used-components">Used Components
<a class=
"headerlink" href=
"#used-components" title=
"Permanent link">#
</a></h2>
2759 <p>We will use the following package installation plugins:
</p>
2761 <li><a href=
"../../../package/pip/database/">database package installation plugin
</a>,
</li>
2762 <li><a href=
"../../../package/pip/event-listener/">eventListener package installation plugin
</a>,
</li>
2763 <li><a href=
"../../../package/pip/file/">file package installation plugin
</a>,
</li>
2764 <li><a href=
"../../../package/pip/language/">language package installation plugin
</a>,
</li>
2765 <li><a href=
"../../../package/pip/template/">template package installation plugin
</a>,
</li>
2766 <li><a href=
"../../../package/pip/template-listener/">templateListener package installation plugin
</a>.
</li>
2768 <p>For more information about the event system, please refer to the
<a href=
"../../../php/api/events/">dedicated page on events
</a>.
</p>
2769 <h2 id=
"package-structure">Package Structure
<a class=
"headerlink" href=
"#package-structure" title=
"Permanent link">#
</a></h2>
2770 <p>The package will have the following file structure:
</p>
2771 <div class=
"highlight"><table class=
"highlighttable"><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal"> 1</span>
2772 <span class=
"normal"> 2</span>
2773 <span class=
"normal"> 3</span>
2774 <span class=
"normal"> 4</span>
2775 <span class=
"normal"> 5</span>
2776 <span class=
"normal"> 6</span>
2777 <span class=
"normal"> 7</span>
2778 <span class=
"normal"> 8</span>
2779 <span class=
"normal"> 9</span>
2780 <span class=
"normal">10</span>
2781 <span class=
"normal">11</span>
2782 <span class=
"normal">12</span>
2783 <span class=
"normal">13</span>
2784 <span class=
"normal">14</span>
2785 <span class=
"normal">15</span>
2786 <span class=
"normal">16</span>
2787 <span class=
"normal">17</span>
2788 <span class=
"normal">18</span>
2789 <span class=
"normal">19</span></pre></div></td><td class=
"code"><div><pre><span></span><code>├── eventListener.xml
2793 │ │ └── install_com.woltlab.wcf.people.birthday.php
2798 │ ├── BirthdayPersonAddFormListener.class.php
2799 │ └── BirthdaySortFieldPersonListPageListener.class.php
2804 ├── templateListener.xml
2806 ├── __personListBirthday.tpl
2807 └── __personListBirthdaySortField.tpl
2808 </code></pre></div></td></tr></table></div>
2809 <h2 id=
"extending-person-model">Extending Person Model
<a class=
"headerlink" href=
"#extending-person-model" title=
"Permanent link">#
</a></h2>
2810 <p>The existing model of a person only contains the person’s first name and their last name (in additional to the id used to identify created people).
2811 To add the birthday to the model, we need to create an additional database table column using the
<a href=
"../../../package/pip/database/"><code>database
</code> package installation plugin
</a>:
</p>
2812 <div class=
"highlight"><table class=
"highlighttable"><tr><th colspan=
"2" class=
"filename"><span class=
"filename">files/acp/database/install_com.woltlab.wcf.people.birthday.php
<a class=
"codeBoxTitleGitHubLink" href=
"https://github.com/WoltLab/docs.woltlab.com/tree/6.0/snippets/tutorial/tutorial-series/part-2/files/acp/database/install_com.woltlab.wcf.people.birthday.php" title=
"View on GitHub"><span class=
"twemoji"><svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7a5 5 0 0 0-5 5 5 5 0 0 0 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1M8 13h8v-2H8v2m9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1 0 1.71-1.39 3.1-3.1 3.1h-4V17h4a5 5 0 0 0 5-5 5 5 0 0 0-5-5Z"/></svg></span></a></span></th></tr><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal"> 1</span>
2813 <span class=
"normal"> 2</span>
2814 <span class=
"normal"> 3</span>
2815 <span class=
"normal"> 4</span>
2816 <span class=
"normal"> 5</span>
2817 <span class=
"normal"> 6</span>
2818 <span class=
"normal"> 7</span>
2819 <span class=
"normal"> 8</span>
2820 <span class=
"normal"> 9</span>
2821 <span class=
"normal">10</span>
2822 <span class=
"normal">11</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"o"><?
</span><span class=
"n">php
</span><span class=
"w"></span>
2824 <span class=
"n">use
</span><span class=
"w"> </span><span class=
"n">wcf
</span><span class=
"err">\
</span><span class=
"k">system
</span><span class=
"err">\
</span><span class=
"k">database
</span><span class=
"err">\
</span><span class=
"k">table
</span><span class=
"err">\
</span><span class=
"k">column
</span><span class=
"err">\
</span><span class=
"n">DateDatabaseTableColumn
</span><span class=
"p">;
</span><span class=
"w"></span>
2825 <span class=
"n">use
</span><span class=
"w"> </span><span class=
"n">wcf
</span><span class=
"err">\
</span><span class=
"k">system
</span><span class=
"err">\
</span><span class=
"k">database
</span><span class=
"err">\
</span><span class=
"k">table
</span><span class=
"err">\
</span><span class=
"n">PartialDatabaseTable
</span><span class=
"p">;
</span><span class=
"w"></span>
2827 <span class=
"k">return
</span><span class=
"w"> </span><span class=
"p">[
</span><span class=
"w"></span>
2828 <span class=
"w"> </span><span class=
"n">PartialDatabaseTable
</span><span class=
"p">::
</span><span class=
"k">create
</span><span class=
"p">(
</span><span class=
"s1">'wcf1_person
'</span><span class=
"p">)
</span><span class=
"w"></span>
2829 <span class=
"w"> </span><span class=
"o">-
></span><span class=
"n">columns
</span><span class=
"p">([
</span><span class=
"w"></span>
2830 <span class=
"w"> </span><span class=
"n">DateDatabaseTableColumn
</span><span class=
"p">::
</span><span class=
"k">create
</span><span class=
"p">(
</span><span class=
"s1">'birthday
'</span><span class=
"p">),
</span><span class=
"w"></span>
2831 <span class=
"w"> </span><span class=
"p">]),
</span><span class=
"w"></span>
2832 <span class=
"p">];
</span><span class=
"w"></span>
2833 </code></pre></div></td></tr></table></div>
2834 <p>If we have a
<a href=
"../part_1/#person"><code>Person
</code> object
</a>, this new property can be accessed the same way as the
<code>personID
</code> property, the
<code>firstName
</code> property, or the
<code>lastName
</code> property from the base package:
<code>$person-
>birthday
</code>.
</p>
2835 <h2 id=
"setting-birthday-in-acp">Setting Birthday in ACP
<a class=
"headerlink" href=
"#setting-birthday-in-acp" title=
"Permanent link">#
</a></h2>
2836 <p>To set the birthday of a person, we only have to add another form field with an event listener:
</p>
2837 <div class=
"highlight"><table class=
"highlighttable"><tr><th colspan=
"2" class=
"filename"><span class=
"filename">files/lib/system/event/listener/BirthdayPersonAddFormListener.class.php
<a class=
"codeBoxTitleGitHubLink" href=
"https://github.com/WoltLab/docs.woltlab.com/tree/6.0/snippets/tutorial/tutorial-series/part-2/files/lib/system/event/listener/BirthdayPersonAddFormListener.class.php" title=
"View on GitHub"><span class=
"twemoji"><svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7a5 5 0 0 0-5 5 5 5 0 0 0 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1M8 13h8v-2H8v2m9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1 0 1.71-1.39 3.1-3.1 3.1h-4V17h4a5 5 0 0 0 5-5 5 5 0 0 0-5-5Z"/></svg></span></a></span></th></tr><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal"> 1</span>
2838 <span class=
"normal"> 2</span>
2839 <span class=
"normal"> 3</span>
2840 <span class=
"normal"> 4</span>
2841 <span class=
"normal"> 5</span>
2842 <span class=
"normal"> 6</span>
2843 <span class=
"normal"> 7</span>
2844 <span class=
"normal"> 8</span>
2845 <span class=
"normal"> 9</span>
2846 <span class=
"normal">10</span>
2847 <span class=
"normal">11</span>
2848 <span class=
"normal">12</span>
2849 <span class=
"normal">13</span>
2850 <span class=
"normal">14</span>
2851 <span class=
"normal">15</span>
2852 <span class=
"normal">16</span>
2853 <span class=
"normal">17</span>
2854 <span class=
"normal">18</span>
2855 <span class=
"normal">19</span>
2856 <span class=
"normal">20</span>
2857 <span class=
"normal">21</span>
2858 <span class=
"normal">22</span>
2859 <span class=
"normal">23</span>
2860 <span class=
"normal">24</span>
2861 <span class=
"normal">25</span>
2862 <span class=
"normal">26</span>
2863 <span class=
"normal">27</span>
2864 <span class=
"normal">28</span>
2865 <span class=
"normal">29</span>
2866 <span class=
"normal">30</span>
2867 <span class=
"normal">31</span>
2868 <span class=
"normal">32</span>
2869 <span class=
"normal">33</span>
2870 <span class=
"normal">34</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"o"><?
</span><span class=
"nx">php
</span>
2872 <span class=
"k">namespace
</span> <span class=
"nx">wcf\system\event\listener
</span><span class=
"p">;
</span>
2874 <span class=
"k">use
</span> <span class=
"nx">wcf\acp\form\PersonAddForm
</span><span class=
"p">;
</span>
2875 <span class=
"k">use
</span> <span class=
"nx">wcf\form\AbstractFormBuilderForm
</span><span class=
"p">;
</span>
2876 <span class=
"k">use
</span> <span class=
"nx">wcf\system\form\builder\container\FormContainer
</span><span class=
"p">;
</span>
2877 <span class=
"k">use
</span> <span class=
"nx">wcf\system\form\builder\field\DateFormField
</span><span class=
"p">;
</span>
2879 <span class=
"sd">/**
</span>
2880 <span class=
"sd"> * Handles setting the birthday when adding and editing people.
</span>
2881 <span class=
"sd"> *
</span>
2882 <span class=
"sd"> * @author Matthias Schmidt
</span>
2883 <span class=
"sd"> * @copyright
2001-
2021 WoltLab GmbH
</span>
2884 <span class=
"sd"> * @license GNU Lesser General Public License
<http://opensource.org/licenses/lgpl-license.php
></span>
2885 <span class=
"sd"> * @package WoltLabSuite\Core\System\Event\Listener
</span>
2886 <span class=
"sd"> */
</span>
2887 <span class=
"k">class
</span> <span class=
"nc">BirthdayPersonAddFormListener
</span> <span class=
"k">extends
</span> <span class=
"nx">AbstractEventListener
</span>
2888 <span class=
"p">{
</span>
2889 <span class=
"sd">/**
</span>
2890 <span class=
"sd"> * @see AbstractFormBuilderForm::createForm()
</span>
2891 <span class=
"sd"> */
</span>
2892 <span class=
"k">protected
</span> <span class=
"k">function
</span> <span class=
"nf">onCreateForm
</span><span class=
"p">(
</span><span class=
"nx">PersonAddForm
</span> <span class=
"nv">$form
</span><span class=
"p">)
</span><span class=
"o">:
</span> <span class=
"nx">void
</span>
2893 <span class=
"p">{
</span>
2894 <span class=
"sd">/** @var FormContainer $dataContainer */
</span>
2895 <span class=
"nv">$dataContainer
</span> <span class=
"o">=
</span> <span class=
"nv">$form
</span><span class=
"o">-
></span><span class=
"na">form
</span><span class=
"o">-
></span><span class=
"na">getNodeById
</span><span class=
"p">(
</span><span class=
"s1">'data
'</span><span class=
"p">);
</span>
2896 <span class=
"nv">$dataContainer
</span><span class=
"o">-
></span><span class=
"na">appendChild
</span><span class=
"p">(
</span>
2897 <span class=
"nx">DateFormField
</span><span class=
"o">::
</span><span class=
"na">create
</span><span class=
"p">(
</span><span class=
"s1">'birthday
'</span><span class=
"p">)
</span>
2898 <span class=
"o">-
></span><span class=
"na">label
</span><span class=
"p">(
</span><span class=
"s1">'wcf.person.birthday
'</span><span class=
"p">)
</span>
2899 <span class=
"o">-
></span><span class=
"na">saveValueFormat
</span><span class=
"p">(
</span><span class=
"s1">'Y-m-d
'</span><span class=
"p">)
</span>
2900 <span class=
"o">-
></span><span class=
"na">nullable
</span><span class=
"p">()
</span>
2901 <span class=
"p">);
</span>
2902 <span class=
"p">}
</span>
2903 <span class=
"p">}
</span>
2904 </code></pre></div></td></tr></table></div>
2905 <p>registered via
</p>
2906 <div class=
"highlight"><table class=
"highlighttable"><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal">1</span>
2907 <span class=
"normal">2</span>
2908 <span class=
"normal">3</span>
2909 <span class=
"normal">4</span>
2910 <span class=
"normal">5</span>
2911 <span class=
"normal">6</span>
2912 <span class=
"normal">7</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"nt"><eventlistener
</span> <span class=
"na">name=
</span><span class=
"s">"createForm@wcf\acp\form\PersonAddForm
"</span><span class=
"nt">></span>
2913 <span class=
"nt"><environment
></span>admin
<span class=
"nt"></environment
></span>
2914 <span class=
"nt"><eventclassname
></span>wcf\acp\form\PersonAddForm
<span class=
"nt"></eventclassname
></span>
2915 <span class=
"nt"><eventname
></span>createForm
<span class=
"nt"></eventname
></span>
2916 <span class=
"nt"><listenerclassname
></span>wcf\system\event\listener\BirthdayPersonAddFormListener
<span class=
"nt"></listenerclassname
></span>
2917 <span class=
"nt"><inherit
></span>1<span class=
"nt"></inherit
></span>
2918 <span class=
"nt"></eventlistener
></span>
2919 </code></pre></div></td></tr></table></div>
2920 <p>in
<code>eventListener.xml
</code>,
<a href=
"#eventlistenerxml">see below
</a>.
</p>
2921 <p>As
<code>BirthdayPersonAddFormListener
</code> extends
<code>AbstractEventListener
</code> and as the name of relevant event is
<code>createForm
</code>,
<code>AbstractEventListener
</code> internally automatically calls
<code>onCreateForm()
</code> with the event object as the parameter.
2922 It is important to set
<code><inherit
>1</inherit
></code> so that the event listener is also executed for
<code>PersonEditForm
</code>, which extends
<code>PersonAddForm
</code>.
</p>
2923 <p>The language item
<code>wcf.person.birthday
</code> used in the label is the only new one for this package:
</p>
2924 <div class=
"highlight"><table class=
"highlighttable"><tr><th colspan=
"2" class=
"filename"><span class=
"filename">language/de.xml
<a class=
"codeBoxTitleGitHubLink" href=
"https://github.com/WoltLab/docs.woltlab.com/tree/6.0/snippets/tutorial/tutorial-series/part-2/language/de.xml" title=
"View on GitHub"><span class=
"twemoji"><svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7a5 5 0 0 0-5 5 5 5 0 0 0 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1M8 13h8v-2H8v2m9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1 0 1.71-1.39 3.1-3.1 3.1h-4V17h4a5 5 0 0 0 5-5 5 5 0 0 0-5-5Z"/></svg></span></a></span></th></tr><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal">1</span>
2925 <span class=
"normal">2</span>
2926 <span class=
"normal">3</span>
2927 <span class=
"normal">4</span>
2928 <span class=
"normal">5</span>
2929 <span class=
"normal">6</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"o"><?
</span><span class=
"n">xml
</span><span class=
"w"> </span><span class=
"k">version
</span><span class=
"o">=
</span><span class=
"ss">"1.0"</span><span class=
"w"> </span><span class=
"k">encoding
</span><span class=
"o">=
</span><span class=
"ss">"UTF-
8"</span><span class=
"o">?
></span><span class=
"w"></span>
2930 <span class=
"o"><</span><span class=
"k">language
</span><span class=
"w"> </span><span class=
"n">xmlns
</span><span class=
"o">=
</span><span class=
"ss">"http://www.woltlab.com
"</span><span class=
"w"> </span><span class=
"n">xmlns
</span><span class=
"p">:
</span><span class=
"n">xsi
</span><span class=
"o">=
</span><span class=
"ss">"http://www.w3.org/
2001/XMLSchema-instance
"</span><span class=
"w"> </span><span class=
"n">xsi
</span><span class=
"p">:
</span><span class=
"n">schemaLocation
</span><span class=
"o">=
</span><span class=
"ss">"http://www.woltlab.com http://www.woltlab.com/XSD/
5.4/language.xsd
"</span><span class=
"w"> </span><span class=
"n">languagecode
</span><span class=
"o">=
</span><span class=
"ss">"de
"</span><span class=
"o">></span><span class=
"w"></span>
2931 <span class=
"w"> </span><span class=
"o"><</span><span class=
"n">category
</span><span class=
"w"> </span><span class=
"n">name
</span><span class=
"o">=
</span><span class=
"ss">"wcf.person
"</span><span class=
"o">></span><span class=
"w"></span>
2932 <span class=
"w"> </span><span class=
"o"><</span><span class=
"n">item
</span><span class=
"w"> </span><span class=
"n">name
</span><span class=
"o">=
</span><span class=
"ss">"wcf.person.birthday
"</span><span class=
"o">><!
</span><span class=
"p">[
</span><span class=
"n">CDATA
</span><span class=
"p">[
</span><span class=
"n">Geburtstag
</span><span class=
"p">]]
</span><span class=
"o">></
</span><span class=
"n">item
</span><span class=
"o">></span><span class=
"w"></span>
2933 <span class=
"w"> </span><span class=
"o"></
</span><span class=
"n">category
</span><span class=
"o">></span><span class=
"w"></span>
2934 <span class=
"o"></
</span><span class=
"k">language
</span><span class=
"o">></span><span class=
"w"></span>
2935 </code></pre></div></td></tr></table></div>
2936 <div class=
"highlight"><table class=
"highlighttable"><tr><th colspan=
"2" class=
"filename"><span class=
"filename">language/en.xml
<a class=
"codeBoxTitleGitHubLink" href=
"https://github.com/WoltLab/docs.woltlab.com/tree/6.0/snippets/tutorial/tutorial-series/part-2/language/en.xml" title=
"View on GitHub"><span class=
"twemoji"><svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7a5 5 0 0 0-5 5 5 5 0 0 0 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1M8 13h8v-2H8v2m9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1 0 1.71-1.39 3.1-3.1 3.1h-4V17h4a5 5 0 0 0 5-5 5 5 0 0 0-5-5Z"/></svg></span></a></span></th></tr><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal">1</span>
2937 <span class=
"normal">2</span>
2938 <span class=
"normal">3</span>
2939 <span class=
"normal">4</span>
2940 <span class=
"normal">5</span>
2941 <span class=
"normal">6</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"o"><?
</span><span class=
"n">xml
</span><span class=
"w"> </span><span class=
"k">version
</span><span class=
"o">=
</span><span class=
"ss">"1.0"</span><span class=
"w"> </span><span class=
"k">encoding
</span><span class=
"o">=
</span><span class=
"ss">"UTF-
8"</span><span class=
"o">?
></span><span class=
"w"></span>
2942 <span class=
"o"><</span><span class=
"k">language
</span><span class=
"w"> </span><span class=
"n">xmlns
</span><span class=
"o">=
</span><span class=
"ss">"http://www.woltlab.com
"</span><span class=
"w"> </span><span class=
"n">xmlns
</span><span class=
"p">:
</span><span class=
"n">xsi
</span><span class=
"o">=
</span><span class=
"ss">"http://www.w3.org/
2001/XMLSchema-instance
"</span><span class=
"w"> </span><span class=
"n">xsi
</span><span class=
"p">:
</span><span class=
"n">schemaLocation
</span><span class=
"o">=
</span><span class=
"ss">"http://www.woltlab.com http://www.woltlab.com/XSD/
5.4/language.xsd
"</span><span class=
"w"> </span><span class=
"n">languagecode
</span><span class=
"o">=
</span><span class=
"ss">"en
"</span><span class=
"o">></span><span class=
"w"></span>
2943 <span class=
"w"> </span><span class=
"o"><</span><span class=
"n">category
</span><span class=
"w"> </span><span class=
"n">name
</span><span class=
"o">=
</span><span class=
"ss">"wcf.person
"</span><span class=
"o">></span><span class=
"w"></span>
2944 <span class=
"w"> </span><span class=
"o"><</span><span class=
"n">item
</span><span class=
"w"> </span><span class=
"n">name
</span><span class=
"o">=
</span><span class=
"ss">"wcf.person.birthday
"</span><span class=
"o">><!
</span><span class=
"p">[
</span><span class=
"n">CDATA
</span><span class=
"p">[
</span><span class=
"n">Birthday
</span><span class=
"p">]]
</span><span class=
"o">></
</span><span class=
"n">item
</span><span class=
"o">></span><span class=
"w"></span>
2945 <span class=
"w"> </span><span class=
"o"></
</span><span class=
"n">category
</span><span class=
"o">></span><span class=
"w"></span>
2946 <span class=
"o"></
</span><span class=
"k">language
</span><span class=
"o">></span><span class=
"w"></span>
2947 </code></pre></div></td></tr></table></div>
2948 <h2 id=
"adding-birthday-table-column-in-acp">Adding Birthday Table Column in ACP
<a class=
"headerlink" href=
"#adding-birthday-table-column-in-acp" title=
"Permanent link">#
</a></h2>
2949 <p>To add a birthday column to the person list page in the ACP, we need three parts:
</p>
2951 <li>an event listener that makes the
<code>birthday
</code> database table column a valid sort field,
</li>
2952 <li>a template listener that adds the birthday column to the table’s head, and
</li>
2953 <li>a template listener that adds the birthday column to the table’s rows.
</li>
2955 <p>The first part is a very simple class:
</p>
2956 <div class=
"highlight"><table class=
"highlighttable"><tr><th colspan=
"2" class=
"filename"><span class=
"filename">files/lib/system/event/listener/BirthdaySortFieldPersonListPageListener.class.php
<a class=
"codeBoxTitleGitHubLink" href=
"https://github.com/WoltLab/docs.woltlab.com/tree/6.0/snippets/tutorial/tutorial-series/part-2/files/lib/system/event/listener/BirthdaySortFieldPersonListPageListener.class.php" title=
"View on GitHub"><span class=
"twemoji"><svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7a5 5 0 0 0-5 5 5 5 0 0 0 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1M8 13h8v-2H8v2m9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1 0 1.71-1.39 3.1-3.1 3.1h-4V17h4a5 5 0 0 0 5-5 5 5 0 0 0-5-5Z"/></svg></span></a></span></th></tr><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal"> 1</span>
2957 <span class=
"normal"> 2</span>
2958 <span class=
"normal"> 3</span>
2959 <span class=
"normal"> 4</span>
2960 <span class=
"normal"> 5</span>
2961 <span class=
"normal"> 6</span>
2962 <span class=
"normal"> 7</span>
2963 <span class=
"normal"> 8</span>
2964 <span class=
"normal"> 9</span>
2965 <span class=
"normal">10</span>
2966 <span class=
"normal">11</span>
2967 <span class=
"normal">12</span>
2968 <span class=
"normal">13</span>
2969 <span class=
"normal">14</span>
2970 <span class=
"normal">15</span>
2971 <span class=
"normal">16</span>
2972 <span class=
"normal">17</span>
2973 <span class=
"normal">18</span>
2974 <span class=
"normal">19</span>
2975 <span class=
"normal">20</span>
2976 <span class=
"normal">21</span>
2977 <span class=
"normal">22</span>
2978 <span class=
"normal">23</span>
2979 <span class=
"normal">24</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"o"><?
</span><span class=
"nx">php
</span>
2981 <span class=
"k">namespace
</span> <span class=
"nx">wcf\system\event\listener
</span><span class=
"p">;
</span>
2983 <span class=
"k">use
</span> <span class=
"nx">wcf\page\SortablePage
</span><span class=
"p">;
</span>
2985 <span class=
"sd">/**
</span>
2986 <span class=
"sd"> * Makes people
's birthday a valid sort field in the ACP and the front end.
</span>
2987 <span class=
"sd"> *
</span>
2988 <span class=
"sd"> * @author Matthias Schmidt
</span>
2989 <span class=
"sd"> * @copyright
2001-
2021 WoltLab GmbH
</span>
2990 <span class=
"sd"> * @license GNU Lesser General Public License
<http://opensource.org/licenses/lgpl-license.php
></span>
2991 <span class=
"sd"> * @package WoltLabSuite\Core\System\Event\Listener
</span>
2992 <span class=
"sd"> */
</span>
2993 <span class=
"k">class
</span> <span class=
"nc">BirthdaySortFieldPersonListPageListener
</span> <span class=
"k">extends
</span> <span class=
"nx">AbstractEventListener
</span>
2994 <span class=
"p">{
</span>
2995 <span class=
"sd">/**
</span>
2996 <span class=
"sd"> * @see SortablePage::validateSortField()
</span>
2997 <span class=
"sd"> */
</span>
2998 <span class=
"k">public
</span> <span class=
"k">function
</span> <span class=
"nf">onValidateSortField
</span><span class=
"p">(
</span><span class=
"nx">SortablePage
</span> <span class=
"nv">$page
</span><span class=
"p">)
</span><span class=
"o">:
</span> <span class=
"nx">void
</span>
2999 <span class=
"p">{
</span>
3000 <span class=
"nv">$page
</span><span class=
"o">-
></span><span class=
"na">validSortFields
</span><span class=
"p">[]
</span> <span class=
"o">=
</span> <span class=
"s1">'birthday
'</span><span class=
"p">;
</span>
3001 <span class=
"p">}
</span>
3002 <span class=
"p">}
</span>
3003 </code></pre></div></td></tr></table></div>
3004 <div class=
"admonition info">
3005 <p class=
"admonition-title">We use
<code>SortablePage
</code> as a type hint instead of
<code>wcf\acp\page\PersonListPage
</code> because we will be using the same event listener class in the front end to also allow sorting that list by birthday.
</p>
3007 <p>As the relevant template codes are only one line each, we will simply put them directly in the
<code>templateListener.xml
</code> file that will be shown
<a href=
"#templatelistenerxml">later on
</a>.
3008 The code for the table head is similar to the other
<code>th
</code> elements:
</p>
3009 <div class=
"highlight"><table class=
"highlighttable"><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal">1</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"x"><th class=
"columnDate columnBirthday
</span><span class=
"cp">{
</span><span class=
"nf">if
</span> <span class=
"nv">$sortField
</span> <span class=
"o">==
</span> <span class=
"s1">'birthday
'</span><span class=
"cp">}
</span><span class=
"x"> active
</span><span class=
"cp">{
</span><span class=
"o">@
</span><span class=
"nv">$sortOrder
</span><span class=
"cp">}{
</span><span class=
"nf">/if
</span><span class=
"cp">}
</span><span class=
"x">"><a href=
"</span><span class=
"cp">{
</span><span class=
"nf">link
</span> <span class=
"na">controller
</span><span class=
"o">=
</span><span class=
"s1">'PersonList
'</span><span class=
"cp">}
</span><span class=
"x">pageNo=
</span><span class=
"cp">{
</span><span class=
"o">@
</span><span class=
"nv">$pageNo
</span><span class=
"cp">}
</span><span class=
"x">&sortField=birthday
&sortOrder=
</span><span class=
"cp">{
</span><span class=
"nf">if
</span> <span class=
"nv">$sortField
</span> <span class=
"o">==
</span> <span class=
"s1">'birthday
'</span> <span class=
"o">&&</span> <span class=
"nv">$sortOrder
</span> <span class=
"o">==
</span> <span class=
"s1">'ASC
'</span><span class=
"cp">}
</span><span class=
"x">DESC
</span><span class=
"cp">{
</span><span class=
"nf">else
</span><span class=
"cp">}
</span><span class=
"x">ASC
</span><span class=
"cp">{
</span><span class=
"nf">/if
</span><span class=
"cp">}{
</span><span class=
"nf">/link
</span><span class=
"cp">}
</span><span class=
"x">"></span><span class=
"cp">{
</span><span class=
"nf">lang
</span><span class=
"cp">}
</span><span class=
"x">wcf.person.birthday
</span><span class=
"cp">{
</span><span class=
"nf">/lang
</span><span class=
"cp">}
</span><span class=
"x"></a
></th
></span>
3010 </code></pre></div></td></tr></table></div>
3011 <p>For the table body’s column, we need to make sure that the birthday is only show if it is actually set:
</p>
3012 <div class=
"highlight"><table class=
"highlighttable"><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal">1</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"x"><td class=
"columnDate columnBirthday
"></span><span class=
"cp">{
</span><span class=
"nf">if
</span> <span class=
"nv">$person
</span><span class=
"o">-
></span><span class=
"na">birthday
</span><span class=
"cp">}{
</span><span class=
"o">@
</span><span class=
"nv">$person
</span><span class=
"o">-
></span><span class=
"na">birthday
</span><span class=
"o">|
</span><span class=
"na">strtotime
</span><span class=
"o">|
</span><span class=
"na">date
</span><span class=
"cp">}{
</span><span class=
"nf">/if
</span><span class=
"cp">}
</span><span class=
"x"></td
></span>
3013 </code></pre></div></td></tr></table></div>
3014 <h2 id=
"adding-birthday-in-front-end">Adding Birthday in Front End
<a class=
"headerlink" href=
"#adding-birthday-in-front-end" title=
"Permanent link">#
</a></h2>
3015 <p>In the front end, we also want to make the list sortable by birthday and show the birthday as part of each person’s “statistics”.
</p>
3016 <p>To add the birthday as a valid sort field, we use
<code>BirthdaySortFieldPersonListPageListener
</code> just as in the ACP.
3017 In the front end, we will now use a template (
<code>__personListBirthdaySortField.tpl
</code>) instead of a directly putting the template code in the
<code>templateListener.xml
</code> file:
</p>
3018 <div class=
"highlight"><table class=
"highlighttable"><tr><th colspan=
"2" class=
"filename"><span class=
"filename">templates/__personListBirthdaySortField.tpl
<a class=
"codeBoxTitleGitHubLink" href=
"https://github.com/WoltLab/docs.woltlab.com/tree/6.0/snippets/tutorial/tutorial-series/part-2/templates/__personListBirthdaySortField.tpl" title=
"View on GitHub"><span class=
"twemoji"><svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7a5 5 0 0 0-5 5 5 5 0 0 0 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1M8 13h8v-2H8v2m9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1 0 1.71-1.39 3.1-3.1 3.1h-4V17h4a5 5 0 0 0 5-5 5 5 0 0 0-5-5Z"/></svg></span></a></span></th></tr><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal">1</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"x"><option value=
"birthday
"</span><span class=
"cp">{
</span><span class=
"nf">if
</span> <span class=
"nv">$sortField
</span> <span class=
"o">==
</span> <span class=
"s1">'birthday
'</span><span class=
"cp">}
</span><span class=
"x"> selected
</span><span class=
"cp">{
</span><span class=
"nf">/if
</span><span class=
"cp">}
</span><span class=
"x">></span><span class=
"cp">{
</span><span class=
"nf">lang
</span><span class=
"cp">}
</span><span class=
"x">wcf.person.birthday
</span><span class=
"cp">{
</span><span class=
"nf">/lang
</span><span class=
"cp">}
</span><span class=
"x"></option
></span>
3019 </code></pre></div></td></tr></table></div>
3020 <div class=
"admonition info">
3021 <p class=
"admonition-title">You might have noticed the two underscores at the beginning of the template file. For templates that are included via template listeners, this is the naming convention we use.
</p>
3023 <p>Putting the template code into a file has the advantage that in the administrator is able to edit the code directly via a custom template group, even though in this case this might not be very probable.
</p>
3024 <p>To show the birthday, we use the following template code for the
<code>personStatistics
</code> template event, which again makes sure that the birthday is only shown if it is actually set:
</p>
3025 <div class=
"highlight"><table class=
"highlighttable"><tr><th colspan=
"2" class=
"filename"><span class=
"filename">templates/__personListBirthday.tpl
<a class=
"codeBoxTitleGitHubLink" href=
"https://github.com/WoltLab/docs.woltlab.com/tree/6.0/snippets/tutorial/tutorial-series/part-2/templates/__personListBirthday.tpl" title=
"View on GitHub"><span class=
"twemoji"><svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7a5 5 0 0 0-5 5 5 5 0 0 0 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1M8 13h8v-2H8v2m9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1 0 1.71-1.39 3.1-3.1 3.1h-4V17h4a5 5 0 0 0 5-5 5 5 0 0 0-5-5Z"/></svg></span></a></span></th></tr><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal">1</span>
3026 <span class=
"normal">2</span>
3027 <span class=
"normal">3</span>
3028 <span class=
"normal">4</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"cp">{
</span><span class=
"nf">if
</span> <span class=
"nv">$person
</span><span class=
"o">-
></span><span class=
"na">birthday
</span><span class=
"cp">}
</span><span class=
"x"></span>
3029 <span class=
"x"> <dt
></span><span class=
"cp">{
</span><span class=
"nf">lang
</span><span class=
"cp">}
</span><span class=
"x">wcf.person.birthday
</span><span class=
"cp">{
</span><span class=
"nf">/lang
</span><span class=
"cp">}
</span><span class=
"x"></dt
></span>
3030 <span class=
"x"> <dd
></span><span class=
"cp">{
</span><span class=
"o">@
</span><span class=
"nv">$person
</span><span class=
"o">-
></span><span class=
"na">birthday
</span><span class=
"o">|
</span><span class=
"na">strtotime
</span><span class=
"o">|
</span><span class=
"na">date
</span><span class=
"cp">}
</span><span class=
"x"></dd
></span>
3031 <span class=
"cp">{
</span><span class=
"nf">/if
</span><span class=
"cp">}
</span><span class=
"x"></span>
3032 </code></pre></div></td></tr></table></div>
3033 <h2 id=
"templatelistenerxml"><code>templateListener.xml
</code><a class=
"headerlink" href=
"#templatelistenerxml" title=
"Permanent link">#
</a></h2>
3034 <p>The following code shows the
<code>templateListener.xml
</code> file used to install all mentioned template listeners:
</p>
3035 <div class=
"highlight"><table class=
"highlighttable"><tr><th colspan=
"2" class=
"filename"><span class=
"filename">templateListener.xml
<a class=
"codeBoxTitleGitHubLink" href=
"https://github.com/WoltLab/docs.woltlab.com/tree/6.0/snippets/tutorial/tutorial-series/part-2/templateListener.xml" title=
"View on GitHub"><span class=
"twemoji"><svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7a5 5 0 0 0-5 5 5 5 0 0 0 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1M8 13h8v-2H8v2m9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1 0 1.71-1.39 3.1-3.1 3.1h-4V17h4a5 5 0 0 0 5-5 5 5 0 0 0-5-5Z"/></svg></span></a></span></th></tr><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal"> 1</span>
3036 <span class=
"normal"> 2</span>
3037 <span class=
"normal"> 3</span>
3038 <span class=
"normal"> 4</span>
3039 <span class=
"normal"> 5</span>
3040 <span class=
"normal"> 6</span>
3041 <span class=
"normal"> 7</span>
3042 <span class=
"normal"> 8</span>
3043 <span class=
"normal"> 9</span>
3044 <span class=
"normal">10</span>
3045 <span class=
"normal">11</span>
3046 <span class=
"normal">12</span>
3047 <span class=
"normal">13</span>
3048 <span class=
"normal">14</span>
3049 <span class=
"normal">15</span>
3050 <span class=
"normal">16</span>
3051 <span class=
"normal">17</span>
3052 <span class=
"normal">18</span>
3053 <span class=
"normal">19</span>
3054 <span class=
"normal">20</span>
3055 <span class=
"normal">21</span>
3056 <span class=
"normal">22</span>
3057 <span class=
"normal">23</span>
3058 <span class=
"normal">24</span>
3059 <span class=
"normal">25</span>
3060 <span class=
"normal">26</span>
3061 <span class=
"normal">27</span>
3062 <span class=
"normal">28</span>
3063 <span class=
"normal">29</span>
3064 <span class=
"normal">30</span>
3065 <span class=
"normal">31</span>
3066 <span class=
"normal">32</span>
3067 <span class=
"normal">33</span>
3068 <span class=
"normal">34</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"cp"><?xml version=
"1.0" encoding=
"UTF-
8"?
></span>
3069 <span class=
"nt"><data
</span> <span class=
"na">xmlns=
</span><span class=
"s">"http://www.woltlab.com
"</span> <span class=
"na">xmlns:xsi=
</span><span class=
"s">"http://www.w3.org/
2001/XMLSchema-instance
"</span> <span class=
"na">xsi:schemaLocation=
</span><span class=
"s">"http://www.woltlab.com http://www.woltlab.com/tornado/
5.4/templateListener.xsd
"</span><span class=
"nt">></span>
3070 <span class=
"nt"><import
></span>
3071 <span class=
"cm"><!-- admin --
></span>
3072 <span class=
"nt"><templatelistener
</span> <span class=
"na">name=
</span><span class=
"s">"personListBirthdayColumnHead
"</span><span class=
"nt">></span>
3073 <span class=
"nt"><eventname
></span>columnHeads
<span class=
"nt"></eventname
></span>
3074 <span class=
"nt"><environment
></span>admin
<span class=
"nt"></environment
></span>
3075 <span class=
"nt"><templatecode
></span><span class=
"cp"><![CDATA[
<th class=
"columnDate columnBirthday{if $sortField ==
'birthday
'} active {@$sortOrder}{/if}
"><a href=
"{link controller=
'PersonList
'}pageNo={@$pageNo}
&sortField=birthday
&sortOrder={if $sortField ==
'birthday
' && $sortOrder ==
'ASC
'}DESC{else}ASC{/if}{/link}
">{lang}wcf.person.birthday{/lang}
</a
></th
>]]
></span><span class=
"nt"></templatecode
></span>
3076 <span class=
"nt"><templatename
></span>personList
<span class=
"nt"></templatename
></span>
3077 <span class=
"nt"></templatelistener
></span>
3078 <span class=
"nt"><templatelistener
</span> <span class=
"na">name=
</span><span class=
"s">"personListBirthdayColumn
"</span><span class=
"nt">></span>
3079 <span class=
"nt"><eventname
></span>columns
<span class=
"nt"></eventname
></span>
3080 <span class=
"nt"><environment
></span>admin
<span class=
"nt"></environment
></span>
3081 <span class=
"nt"><templatecode
></span><span class=
"cp"><![CDATA[
<td class=
"columnDate columnBirthday
">{if $person-
>birthday}{@$person-
>birthday|strtotime|date}{/if}
</td
>]]
></span><span class=
"nt"></templatecode
></span>
3082 <span class=
"nt"><templatename
></span>personList
<span class=
"nt"></templatename
></span>
3083 <span class=
"nt"></templatelistener
></span>
3084 <span class=
"cm"><!-- /admin --
></span>
3086 <span class=
"cm"><!-- user --
></span>
3087 <span class=
"nt"><templatelistener
</span> <span class=
"na">name=
</span><span class=
"s">"personListBirthday
"</span><span class=
"nt">></span>
3088 <span class=
"nt"><eventname
></span>personStatistics
<span class=
"nt"></eventname
></span>
3089 <span class=
"nt"><environment
></span>user
<span class=
"nt"></environment
></span>
3090 <span class=
"nt"><templatecode
></span><span class=
"cp"><![CDATA[{include file=
'__personListBirthday
'}]]
></span><span class=
"nt"></templatecode
></span>
3091 <span class=
"nt"><templatename
></span>personList
<span class=
"nt"></templatename
></span>
3092 <span class=
"nt"></templatelistener
></span>
3093 <span class=
"nt"><templatelistener
</span> <span class=
"na">name=
</span><span class=
"s">"personListBirthdaySortField
"</span><span class=
"nt">></span>
3094 <span class=
"nt"><eventname
></span>sortField
<span class=
"nt"></eventname
></span>
3095 <span class=
"nt"><environment
></span>user
<span class=
"nt"></environment
></span>
3096 <span class=
"nt"><templatecode
></span><span class=
"cp"><![CDATA[{include file=
'__personListBirthdaySortField
'}]]
></span><span class=
"nt"></templatecode
></span>
3097 <span class=
"nt"><templatename
></span>personList
<span class=
"nt"></templatename
></span>
3098 <span class=
"nt"></templatelistener
></span>
3099 <span class=
"cm"><!-- /user --
></span>
3100 <span class=
"nt"></import
></span>
3101 <span class=
"nt"></data
></span>
3102 </code></pre></div></td></tr></table></div>
3103 <p>In cases where a template is used, we simply use the
<code>include
</code> syntax to load the template.
</p>
3104 <h2 id=
"eventlistenerxml"><code>eventListener.xml
</code><a class=
"headerlink" href=
"#eventlistenerxml" title=
"Permanent link">#
</a></h2>
3105 <p>There are two event listeners that make
<code>birthday
</code> a valid sort field in the ACP and the front end, respectively, and the third event listener takes care of setting the birthday.
</p>
3106 <div class=
"highlight"><table class=
"highlighttable"><tr><th colspan=
"2" class=
"filename"><span class=
"filename">eventListener.xml
<a class=
"codeBoxTitleGitHubLink" href=
"https://github.com/WoltLab/docs.woltlab.com/tree/6.0/snippets/tutorial/tutorial-series/part-2/eventListener.xml" title=
"View on GitHub"><span class=
"twemoji"><svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7a5 5 0 0 0-5 5 5 5 0 0 0 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1M8 13h8v-2H8v2m9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1 0 1.71-1.39 3.1-3.1 3.1h-4V17h4a5 5 0 0 0 5-5 5 5 0 0 0-5-5Z"/></svg></span></a></span></th></tr><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal"> 1</span>
3107 <span class=
"normal"> 2</span>
3108 <span class=
"normal"> 3</span>
3109 <span class=
"normal"> 4</span>
3110 <span class=
"normal"> 5</span>
3111 <span class=
"normal"> 6</span>
3112 <span class=
"normal"> 7</span>
3113 <span class=
"normal"> 8</span>
3114 <span class=
"normal"> 9</span>
3115 <span class=
"normal">10</span>
3116 <span class=
"normal">11</span>
3117 <span class=
"normal">12</span>
3118 <span class=
"normal">13</span>
3119 <span class=
"normal">14</span>
3120 <span class=
"normal">15</span>
3121 <span class=
"normal">16</span>
3122 <span class=
"normal">17</span>
3123 <span class=
"normal">18</span>
3124 <span class=
"normal">19</span>
3125 <span class=
"normal">20</span>
3126 <span class=
"normal">21</span>
3127 <span class=
"normal">22</span>
3128 <span class=
"normal">23</span>
3129 <span class=
"normal">24</span>
3130 <span class=
"normal">25</span>
3131 <span class=
"normal">26</span>
3132 <span class=
"normal">27</span>
3133 <span class=
"normal">28</span>
3134 <span class=
"normal">29</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"cp"><?xml version=
"1.0" encoding=
"UTF-
8"?
></span>
3135 <span class=
"nt"><data
</span> <span class=
"na">xmlns=
</span><span class=
"s">"http://www.woltlab.com
"</span> <span class=
"na">xmlns:xsi=
</span><span class=
"s">"http://www.w3.org/
2001/XMLSchema-instance
"</span> <span class=
"na">xsi:schemaLocation=
</span><span class=
"s">"http://www.woltlab.com http://www.woltlab.com/XSD/
5.4/eventListener.xsd
"</span><span class=
"nt">></span>
3136 <span class=
"nt"><import
></span>
3137 <span class=
"cm"><!-- admin --
></span>
3138 <span class=
"nt"><eventlistener
</span> <span class=
"na">name=
</span><span class=
"s">"validateSortField@wcf\acp\page\PersonListPage
"</span><span class=
"nt">></span>
3139 <span class=
"nt"><environment
></span>admin
<span class=
"nt"></environment
></span>
3140 <span class=
"nt"><eventclassname
></span>wcf\acp\page\PersonListPage
<span class=
"nt"></eventclassname
></span>
3141 <span class=
"nt"><eventname
></span>validateSortField
<span class=
"nt"></eventname
></span>
3142 <span class=
"nt"><listenerclassname
></span>wcf\system\event\listener\BirthdaySortFieldPersonListPageListener
<span class=
"nt"></listenerclassname
></span>
3143 <span class=
"nt"></eventlistener
></span>
3144 <span class=
"nt"><eventlistener
</span> <span class=
"na">name=
</span><span class=
"s">"createForm@wcf\acp\form\PersonAddForm
"</span><span class=
"nt">></span>
3145 <span class=
"nt"><environment
></span>admin
<span class=
"nt"></environment
></span>
3146 <span class=
"nt"><eventclassname
></span>wcf\acp\form\PersonAddForm
<span class=
"nt"></eventclassname
></span>
3147 <span class=
"nt"><eventname
></span>createForm
<span class=
"nt"></eventname
></span>
3148 <span class=
"nt"><listenerclassname
></span>wcf\system\event\listener\BirthdayPersonAddFormListener
<span class=
"nt"></listenerclassname
></span>
3149 <span class=
"nt"><inherit
></span>1<span class=
"nt"></inherit
></span>
3150 <span class=
"nt"></eventlistener
></span>
3151 <span class=
"cm"><!-- /admin --
></span>
3153 <span class=
"cm"><!-- user --
></span>
3154 <span class=
"nt"><eventlistener
</span> <span class=
"na">name=
</span><span class=
"s">"validateSortField@wcf\page\PersonListPage
"</span><span class=
"nt">></span>
3155 <span class=
"nt"><environment
></span>user
<span class=
"nt"></environment
></span>
3156 <span class=
"nt"><eventclassname
></span>wcf\page\PersonListPage
<span class=
"nt"></eventclassname
></span>
3157 <span class=
"nt"><eventname
></span>validateSortField
<span class=
"nt"></eventname
></span>
3158 <span class=
"nt"><listenerclassname
></span>wcf\system\event\listener\BirthdaySortFieldPersonListPageListener
<span class=
"nt"></listenerclassname
></span>
3159 <span class=
"nt"></eventlistener
></span>
3160 <span class=
"cm"><!-- /user --
></span>
3161 <span class=
"nt"></import
></span>
3162 <span class=
"nt"></data
></span>
3163 </code></pre></div></td></tr></table></div>
3164 <h2 id=
"packagexml"><code>package.xml
</code><a class=
"headerlink" href=
"#packagexml" title=
"Permanent link">#
</a></h2>
3165 <p>The only relevant difference between the
<code>package.xml
</code> file of the base page from part
1 and the
<code>package.xml
</code> file of this package is that this package requires the base package
<code>com.woltlab.wcf.people
</code> (see
<code><requiredpackages
></code>):
</p>
3166 <div class=
"highlight"><table class=
"highlighttable"><tr><th colspan=
"2" class=
"filename"><span class=
"filename">package.xml
<a class=
"codeBoxTitleGitHubLink" href=
"https://github.com/WoltLab/docs.woltlab.com/tree/6.0/snippets/tutorial/tutorial-series/part-2/package.xml" title=
"View on GitHub"><span class=
"twemoji"><svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7a5 5 0 0 0-5 5 5 5 0 0 0 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1M8 13h8v-2H8v2m9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1 0 1.71-1.39 3.1-3.1 3.1h-4V17h4a5 5 0 0 0 5-5 5 5 0 0 0-5-5Z"/></svg></span></a></span></th></tr><tr><td class=
"linenos"><div class=
"linenodiv"><pre><span></span><span class=
"normal"> 1</span>
3167 <span class=
"normal"> 2</span>
3168 <span class=
"normal"> 3</span>
3169 <span class=
"normal"> 4</span>
3170 <span class=
"normal"> 5</span>
3171 <span class=
"normal"> 6</span>
3172 <span class=
"normal"> 7</span>
3173 <span class=
"normal"> 8</span>
3174 <span class=
"normal"> 9</span>
3175 <span class=
"normal">10</span>
3176 <span class=
"normal">11</span>
3177 <span class=
"normal">12</span>
3178 <span class=
"normal">13</span>
3179 <span class=
"normal">14</span>
3180 <span class=
"normal">15</span>
3181 <span class=
"normal">16</span>
3182 <span class=
"normal">17</span>
3183 <span class=
"normal">18</span>
3184 <span class=
"normal">19</span>
3185 <span class=
"normal">20</span>
3186 <span class=
"normal">21</span>
3187 <span class=
"normal">22</span>
3188 <span class=
"normal">23</span>
3189 <span class=
"normal">24</span>
3190 <span class=
"normal">25</span>
3191 <span class=
"normal">26</span>
3192 <span class=
"normal">27</span>
3193 <span class=
"normal">28</span>
3194 <span class=
"normal">29</span>
3195 <span class=
"normal">30</span>
3196 <span class=
"normal">31</span>
3197 <span class=
"normal">32</span>
3198 <span class=
"normal">33</span></pre></div></td><td class=
"code"><div><pre><span></span><code><span class=
"cp"><?xml version=
"1.0" encoding=
"UTF-
8"?
></span>
3199 <span class=
"nt"><package
</span> <span class=
"na">name=
</span><span class=
"s">"com.woltlab.wcf.people.birthday
"</span> <span class=
"na">xmlns=
</span><span class=
"s">"http://www.woltlab.com
"</span> <span class=
"na">xmlns:xsi=
</span><span class=
"s">"http://www.w3.org/
2001/XMLSchema-instance
"</span> <span class=
"na">xsi:schemaLocation=
</span><span class=
"s">"http://www.woltlab.com http://www.woltlab.com/XSD/
5.4/package.xsd
"</span><span class=
"nt">></span>
3200 <span class=
"nt"><packageinformation
></span>
3201 <span class=
"nt"><packagename
></span>WoltLab Suite Core Tutorial: People (Birthday)
<span class=
"nt"></packagename
></span>
3202 <span class=
"nt"><packagedescription
></span>Adds a birthday field to the people management system as part of a tutorial to create packages.
<span class=
"nt"></packagedescription
></span>
3203 <span class=
"nt"><version
></span>5.4.0<span class=
"nt"></version
></span>
3204 <span class=
"nt"><date
></span>2022-
01-
17<span class=
"nt"></date
></span>
3205 <span class=
"nt"></packageinformation
></span>
3207 <span class=
"nt"><authorinformation
></span>
3208 <span class=
"nt"><author
></span>WoltLab GmbH
<span class=
"nt"></author
></span>
3209 <span class=
"nt"><authorurl
></span>http://www.woltlab.com
<span class=
"nt"></authorurl
></span>
3210 <span class=
"nt"></authorinformation
></span>
3212 <span class=
"nt"><requiredpackages
></span>
3213 <span class=
"nt"><requiredpackage
</span> <span class=
"na">minversion=
</span><span class=
"s">"5.4.10"</span><span class=
"nt">></span>com.woltlab.wcf
<span class=
"nt"></requiredpackage
></span>
3214 <span class=
"nt"><requiredpackage
</span> <span class=
"na">minversion=
</span><span class=
"s">"5.4.0"</span><span class=
"nt">></span>com.woltlab.wcf.people
<span class=
"nt"></requiredpackage
></span>
3215 <span class=
"nt"></requiredpackages
></span>
3217 <span class=
"nt"><excludedpackages
></span>
3218 <span class=
"nt"><excludedpackage
</span> <span class=
"na">version=
</span><span class=
"s">"6.0.0 Alpha
1"</span><span class=
"nt">></span>com.woltlab.wcf
<span class=
"nt"></excludedpackage
></span>
3219 <span class=
"nt"></excludedpackages
></span>
3221 <span class=
"nt"><instructions
</span> <span class=
"na">type=
</span><span class=
"s">"install
"</span><span class=
"nt">></span>
3222 <span class=
"nt"><instruction
</span> <span class=
"na">type=
</span><span class=
"s">"file
"</span> <span class=
"nt">/
></span>
3223 <span class=
"nt"><instruction
</span> <span class=
"na">type=
</span><span class=
"s">"database
"</span><span class=
"nt">></span>acp/database/install_com.woltlab.wcf.people.birthday.php
<span class=
"nt"></instruction
></span>
3224 <span class=
"nt"><instruction
</span> <span class=
"na">type=
</span><span class=
"s">"template
"</span> <span class=
"nt">/
></span>
3225 <span class=
"nt"><instruction
</span> <span class=
"na">type=
</span><span class=
"s">"language
"</span> <span class=
"nt">/
></span>
3227 <span class=
"nt"><instruction
</span> <span class=
"na">type=
</span><span class=
"s">"eventListener
"</span> <span class=
"nt">/
></span>
3228 <span class=
"nt"><instruction
</span> <span class=
"na">type=
</span><span class=
"s">"templateListener
"</span> <span class=
"nt">/
></span>
3229 <span class=
"nt"></instructions
></span>
3230 <span class=
"nt"></package
></span>
3231 </code></pre></div></td></tr></table></div>
3233 <p>This concludes the second part of our tutorial series after which you now have extended the base package using event listeners and template listeners that allow you to enter the birthday of the people.
</p>
3234 <p>The complete source code of this part can be found on
<a href=
"https://github.com/WoltLab/docs.woltlab.com/tree/6.0/snippets/tutorial/tutorial-series/part-2">GitHub
</a>.
</p>
3237 <div class=
"md-source-file">
3259 <footer class=
"md-footer">
3262 <nav class=
"md-footer__inner md-grid" aria-label=
"Footer" >
3265 <a href=
"../part_1/" class=
"md-footer__link md-footer__link--prev" aria-label=
"Previous: Part 1" rel=
"prev">
3266 <div class=
"md-footer__button md-icon">
3267 <svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M20 11v2H8l5.5 5.5-1.42 1.42L4.16 12l7.92-7.92L13.5 5.5 8 11h12Z"/></svg>
3269 <div class=
"md-footer__title">
3270 <div class=
"md-ellipsis">
3271 <span class=
"md-footer__direction">
3281 <a href=
"../part_3/" class=
"md-footer__link md-footer__link--next" aria-label=
"Next: Part 3" rel=
"next">
3282 <div class=
"md-footer__title">
3283 <div class=
"md-ellipsis">
3284 <span class=
"md-footer__direction">
3290 <div class=
"md-footer__button md-icon">
3291 <svg xmlns=
"http://www.w3.org/2000/svg" viewBox=
"0 0 24 24"><path d=
"M4 11v2h12l-5.5 5.5 1.42 1.42L19.84 12l-7.92-7.92L10.5 5.5 16 11H4Z"/></svg>
3297 <div class=
"md-footer-meta md-typeset">
3298 <div class=
"md-footer-meta__inner md-grid">
3299 <div class=
"md-copyright">
3301 <div class=
"md-copyright__highlight">
3302 Copyright ©
2020 WoltLab GmbH
3307 <a href=
"https://squidfunk.github.io/mkdocs-material/" target=
"_blank" rel=
"noopener">
3313 <div class=
"md-copyright">
3314 <a href=
"https://www.woltlab.com/legal-notice/">Legal Notice
</a>
3315 <a href=
"https://www.woltlab.com/privacy-policy/">Privacy Policy
</a>
3323 <div class=
"md-dialog" data-md-component=
"dialog">
3324 <div class=
"md-dialog__inner md-typeset"></div>
3327 <script id=
"__config" type=
"application/json">{
"base":
"../../..",
"features": [
"navigation.tracking"],
"search":
"../../../assets/javascripts/workers/search.5bf1dace.min.js",
"translations": {
"clipboard.copied":
"Copied to clipboard",
"clipboard.copy":
"Copy to clipboard",
"search.config.lang":
"en",
"search.config.pipeline":
"trimmer, stopWordFilter",
"search.config.separator":
"[\\s\\-]+",
"search.placeholder":
"Search",
"search.result.more.one":
"1 more on this page",
"search.result.more.other":
"# more on this page",
"search.result.none":
"No matching documents",
"search.result.one":
"1 matching document",
"search.result.other":
"# matching documents",
"search.result.placeholder":
"Type to start searching",
"search.result.term.missing":
"Missing",
"select.version.title":
"Select version"},
"version": {
"provider":
"mike"}}
</script>
3330 <script src=
"../../../assets/javascripts/bundle.078830c0.min.js"></script>