├── .github └── workflows │ ├── build.yml │ ├── lint-frontend.yml │ └── phpunit.yml ├── .gitignore ├── .nvmrc ├── .phpcs.xml.dist ├── .wp-env.json ├── .wp-env └── wp-cli.yml ├── LICENSE ├── README.md ├── bin └── install-wp-tests.sh ├── composer.json ├── composer.lock ├── includes ├── ical-functions.php ├── ical-generator-functions.php ├── wporg-meeting-install.php └── wporg-meeting-posttype.php ├── package-lock.json ├── package.json ├── phpunit.xml.dist ├── plugin.php ├── readme.txt ├── src ├── edit.js ├── frontend │ ├── app │ │ └── index.js │ ├── calendar │ │ ├── cell.js │ │ ├── grid.js │ │ ├── header.js │ │ ├── index.js │ │ ├── modal.js │ │ └── utils.js │ ├── feed │ │ └── index.js │ ├── filter │ │ └── index.js │ ├── icons │ │ ├── arrow.js │ │ ├── calendar.js │ │ ├── collapse.js │ │ ├── expand.js │ │ ├── index.js │ │ └── list.js │ ├── index.js │ ├── list │ │ ├── index.js │ │ └── list-item.js │ ├── store │ │ ├── event-context.js │ │ ├── hooks │ │ │ └── use-window-size.js │ │ ├── utils.js │ │ └── view-context.js │ └── style.scss └── index.js ├── style.css └── tests ├── bootstrap.php ├── fixtures ├── events-with-cancel.ics └── events.ics ├── test-api.php ├── test-ical.php ├── test-meetingposttype.php └── test-shortcode.php /.github/workflows/build.yml: -------------------------------------------------------------------------------- 1 | name: Build new theme and push to `build` branch. 2 | 3 | on: 4 | push: 5 | branches: 6 | - master 7 | # Enable manually running action if necessary. 8 | workflow_dispatch: 9 | 10 | jobs: 11 | build: 12 | name: Build 13 | runs-on: ubuntu-latest 14 | steps: 15 | - name: Checkout repository 16 | uses: actions/checkout@v4 17 | 18 | - name: Install NodeJS 19 | uses: actions/setup-node@v4 20 | with: 21 | node-version-file: '.nvmrc' 22 | cache: 'npm' 23 | 24 | - name: Npm install and build 25 | run: | 26 | npm ci 27 | npm run build 28 | 29 | - name: Remove build artifacts 30 | run: | 31 | rm -rf node_modules 32 | 33 | - name: Ignore .gitignore 34 | run: | 35 | git add * --force 36 | 37 | - name: Commit and push 38 | # Using a specific hash here instead of a tagged version, for risk mitigation, since this action modifies our repo. 39 | uses: actions-js/push@a52398fac807b0c1e5f1492c969b477c8560a0ba # 1.3 40 | with: 41 | github_token: ${{ secrets.GITHUB_TOKEN }} 42 | branch: build 43 | force: true 44 | message: 'Build: ${{ github.sha }}' 45 | -------------------------------------------------------------------------------- /.github/workflows/lint-frontend.yml: -------------------------------------------------------------------------------- 1 | name: Static Analysis (Linting) 2 | 3 | # This workflow is triggered on pushes to trunk, and any PRs. 4 | on: 5 | push: 6 | branches: [master] 7 | pull_request: 8 | workflow_dispatch: 9 | 10 | jobs: 11 | check: 12 | name: Lint 13 | runs-on: ubuntu-latest 14 | steps: 15 | - uses: actions/checkout@v4 16 | 17 | - name: Install NodeJS 18 | uses: actions/setup-node@v4 19 | with: 20 | node-version-file: '.nvmrc' 21 | cache: 'npm' 22 | 23 | - name: npm install and build 24 | run: | 25 | npm ci 26 | npm run build 27 | 28 | - name: Lint Styles 29 | run: | 30 | npm run lint:css 31 | -------------------------------------------------------------------------------- /.github/workflows/phpunit.yml: -------------------------------------------------------------------------------- 1 | name: Unit Tests 2 | 3 | on: 4 | pull_request: 5 | paths: 6 | - '**.php' 7 | - '*.json' 8 | - '*.xml.dist' 9 | push: 10 | paths: 11 | - '**.php' 12 | - '*.json' 13 | - '*.xml.dist' 14 | branches: 15 | - master 16 | 17 | jobs: 18 | phpcs: 19 | name: PHP CodeSniffer 20 | 21 | runs-on: ubuntu-latest 22 | 23 | steps: 24 | - name: Checkout repository 25 | uses: actions/checkout@v4 26 | 27 | - name: Set up PHP 28 | uses: shivammathur/setup-php@v2 29 | with: 30 | php-version: '7.4' 31 | coverage: none 32 | tools: composer, cs2pr 33 | 34 | - name: Install Composer dependencies 35 | run: | 36 | composer install --prefer-dist --no-suggest --no-progress --no-ansi --no-interaction 37 | 38 | - name: Run PHPCS 39 | run: | 40 | vendor/bin/phpcs -q --report=checkstyle src | cs2pr 41 | 42 | unit-php: 43 | name: PHP Unit Tests 44 | 45 | runs-on: ubuntu-latest 46 | 47 | steps: 48 | - uses: actions/checkout@v4 49 | 50 | - name: Install NodeJS 51 | uses: actions/setup-node@v4 52 | with: 53 | node-version-file: '.nvmrc' 54 | cache: 'npm' 55 | 56 | - name: Npm install and build 57 | run: | 58 | npm ci 59 | npm run build 60 | 61 | - name: composer install 62 | run: | 63 | composer install 64 | 65 | - name: Install WordPress 66 | run: | 67 | chmod -R 767 ./ # TODO: Possibly integrate in wp-env 68 | npm run wp-env start 69 | 70 | - name: Run unit tests 71 | run: npm run test:unit-php 72 | if: ${{ success() || failure() }} 73 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | 3 | build/ 4 | node_modules/ 5 | vendor 6 | 7 | .phpunit.result.cache 8 | -------------------------------------------------------------------------------- /.nvmrc: -------------------------------------------------------------------------------- 1 | 20 2 | -------------------------------------------------------------------------------- /.phpcs.xml.dist: -------------------------------------------------------------------------------- 1 | 2 | 3 | Apply customized version of WordPress Coding Standards to WordPress.org projects. 4 | 5 | 16 | 17 | 18 | */vendor/* 19 | */node_modules/* 20 | 21 | */build/* 22 | 23 | */tests/* 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 | 67 | 68 | 69 | 70 | 71 | 72 | 73 | 74 | 75 | 76 | 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 | 130 | 131 | 132 | 133 | 134 | 135 | 136 | -------------------------------------------------------------------------------- /.wp-env.json: -------------------------------------------------------------------------------- 1 | { 2 | "core": "WordPress/WordPress", 3 | "plugins": [ 4 | "." 5 | ], 6 | "mappings": { 7 | "wp-cli.yml": "./.wp-env/wp-cli.yml" 8 | }, 9 | "phpVersion": "7.4" 10 | } 11 | -------------------------------------------------------------------------------- /.wp-env/wp-cli.yml: -------------------------------------------------------------------------------- 1 | apache_modules: 2 | - mod_rewrite 3 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | GNU GENERAL PUBLIC LICENSE 2 | Version 3, 29 June 2007 3 | 4 | Copyright (C) 2007 Free Software Foundation, Inc. 5 | Everyone is permitted to copy and distribute verbatim copies 6 | of this license document, but changing it is not allowed. 7 | 8 | Preamble 9 | 10 | The GNU General Public License is a free, copyleft license for 11 | software and other kinds of works. 12 | 13 | The licenses for most software and other practical works are designed 14 | to take away your freedom to share and change the works. By contrast, 15 | the GNU General Public License is intended to guarantee your freedom to 16 | share and change all versions of a program--to make sure it remains free 17 | software for all its users. We, the Free Software Foundation, use the 18 | GNU General Public License for most of our software; it applies also to 19 | any other work released this way by its authors. You can apply it to 20 | your programs, too. 21 | 22 | When we speak of free software, we are referring to freedom, not 23 | price. Our General Public Licenses are designed to make sure that you 24 | have the freedom to distribute copies of free software (and charge for 25 | them if you wish), that you receive source code or can get it if you 26 | want it, that you can change the software or use pieces of it in new 27 | free programs, and that you know you can do these things. 28 | 29 | To protect your rights, we need to prevent others from denying you 30 | these rights or asking you to surrender the rights. Therefore, you have 31 | certain responsibilities if you distribute copies of the software, or if 32 | you modify it: responsibilities to respect the freedom of others. 33 | 34 | For example, if you distribute copies of such a program, whether 35 | gratis or for a fee, you must pass on to the recipients the same 36 | freedoms that you received. You must make sure that they, too, receive 37 | or can get the source code. And you must show them these terms so they 38 | know their rights. 39 | 40 | Developers that use the GNU GPL protect your rights with two steps: 41 | (1) assert copyright on the software, and (2) offer you this License 42 | giving you legal permission to copy, distribute and/or modify it. 43 | 44 | For the developers' and authors' protection, the GPL clearly explains 45 | that there is no warranty for this free software. For both users' and 46 | authors' sake, the GPL requires that modified versions be marked as 47 | changed, so that their problems will not be attributed erroneously to 48 | authors of previous versions. 49 | 50 | Some devices are designed to deny users access to install or run 51 | modified versions of the software inside them, although the manufacturer 52 | can do so. This is fundamentally incompatible with the aim of 53 | protecting users' freedom to change the software. The systematic 54 | pattern of such abuse occurs in the area of products for individuals to 55 | use, which is precisely where it is most unacceptable. Therefore, we 56 | have designed this version of the GPL to prohibit the practice for those 57 | products. If such problems arise substantially in other domains, we 58 | stand ready to extend this provision to those domains in future versions 59 | of the GPL, as needed to protect the freedom of users. 60 | 61 | Finally, every program is threatened constantly by software patents. 62 | States should not allow patents to restrict development and use of 63 | software on general-purpose computers, but in those that do, we wish to 64 | avoid the special danger that patents applied to a free program could 65 | make it effectively proprietary. To prevent this, the GPL assures that 66 | patents cannot be used to render the program non-free. 67 | 68 | The precise terms and conditions for copying, distribution and 69 | modification follow. 70 | 71 | TERMS AND CONDITIONS 72 | 73 | 0. Definitions. 74 | 75 | "This License" refers to version 3 of the GNU General Public License. 76 | 77 | "Copyright" also means copyright-like laws that apply to other kinds of 78 | works, such as semiconductor masks. 79 | 80 | "The Program" refers to any copyrightable work licensed under this 81 | License. Each licensee is addressed as "you". "Licensees" and 82 | "recipients" may be individuals or organizations. 83 | 84 | To "modify" a work means to copy from or adapt all or part of the work 85 | in a fashion requiring copyright permission, other than the making of an 86 | exact copy. The resulting work is called a "modified version" of the 87 | earlier work or a work "based on" the earlier work. 88 | 89 | A "covered work" means either the unmodified Program or a work based 90 | on the Program. 91 | 92 | To "propagate" a work means to do anything with it that, without 93 | permission, would make you directly or secondarily liable for 94 | infringement under applicable copyright law, except executing it on a 95 | computer or modifying a private copy. Propagation includes copying, 96 | distribution (with or without modification), making available to the 97 | public, and in some countries other activities as well. 98 | 99 | To "convey" a work means any kind of propagation that enables other 100 | parties to make or receive copies. Mere interaction with a user through 101 | a computer network, with no transfer of a copy, is not conveying. 102 | 103 | An interactive user interface displays "Appropriate Legal Notices" 104 | to the extent that it includes a convenient and prominently visible 105 | feature that (1) displays an appropriate copyright notice, and (2) 106 | tells the user that there is no warranty for the work (except to the 107 | extent that warranties are provided), that licensees may convey the 108 | work under this License, and how to view a copy of this License. If 109 | the interface presents a list of user commands or options, such as a 110 | menu, a prominent item in the list meets this criterion. 111 | 112 | 1. Source Code. 113 | 114 | The "source code" for a work means the preferred form of the work 115 | for making modifications to it. "Object code" means any non-source 116 | form of a work. 117 | 118 | A "Standard Interface" means an interface that either is an official 119 | standard defined by a recognized standards body, or, in the case of 120 | interfaces specified for a particular programming language, one that 121 | is widely used among developers working in that language. 122 | 123 | The "System Libraries" of an executable work include anything, other 124 | than the work as a whole, that (a) is included in the normal form of 125 | packaging a Major Component, but which is not part of that Major 126 | Component, and (b) serves only to enable use of the work with that 127 | Major Component, or to implement a Standard Interface for which an 128 | implementation is available to the public in source code form. A 129 | "Major Component", in this context, means a major essential component 130 | (kernel, window system, and so on) of the specific operating system 131 | (if any) on which the executable work runs, or a compiler used to 132 | produce the work, or an object code interpreter used to run it. 133 | 134 | The "Corresponding Source" for a work in object code form means all 135 | the source code needed to generate, install, and (for an executable 136 | work) run the object code and to modify the work, including scripts to 137 | control those activities. However, it does not include the work's 138 | System Libraries, or general-purpose tools or generally available free 139 | programs which are used unmodified in performing those activities but 140 | which are not part of the work. For example, Corresponding Source 141 | includes interface definition files associated with source files for 142 | the work, and the source code for shared libraries and dynamically 143 | linked subprograms that the work is specifically designed to require, 144 | such as by intimate data communication or control flow between those 145 | subprograms and other parts of the work. 146 | 147 | The Corresponding Source need not include anything that users 148 | can regenerate automatically from other parts of the Corresponding 149 | Source. 150 | 151 | The Corresponding Source for a work in source code form is that 152 | same work. 153 | 154 | 2. Basic Permissions. 155 | 156 | All rights granted under this License are granted for the term of 157 | copyright on the Program, and are irrevocable provided the stated 158 | conditions are met. This License explicitly affirms your unlimited 159 | permission to run the unmodified Program. The output from running a 160 | covered work is covered by this License only if the output, given its 161 | content, constitutes a covered work. This License acknowledges your 162 | rights of fair use or other equivalent, as provided by copyright law. 163 | 164 | You may make, run and propagate covered works that you do not 165 | convey, without conditions so long as your license otherwise remains 166 | in force. You may convey covered works to others for the sole purpose 167 | of having them make modifications exclusively for you, or provide you 168 | with facilities for running those works, provided that you comply with 169 | the terms of this License in conveying all material for which you do 170 | not control copyright. Those thus making or running the covered works 171 | for you must do so exclusively on your behalf, under your direction 172 | and control, on terms that prohibit them from making any copies of 173 | your copyrighted material outside their relationship with you. 174 | 175 | Conveying under any other circumstances is permitted solely under 176 | the conditions stated below. Sublicensing is not allowed; section 10 177 | makes it unnecessary. 178 | 179 | 3. Protecting Users' Legal Rights From Anti-Circumvention Law. 180 | 181 | No covered work shall be deemed part of an effective technological 182 | measure under any applicable law fulfilling obligations under article 183 | 11 of the WIPO copyright treaty adopted on 20 December 1996, or 184 | similar laws prohibiting or restricting circumvention of such 185 | measures. 186 | 187 | When you convey a covered work, you waive any legal power to forbid 188 | circumvention of technological measures to the extent such circumvention 189 | is effected by exercising rights under this License with respect to 190 | the covered work, and you disclaim any intention to limit operation or 191 | modification of the work as a means of enforcing, against the work's 192 | users, your or third parties' legal rights to forbid circumvention of 193 | technological measures. 194 | 195 | 4. Conveying Verbatim Copies. 196 | 197 | You may convey verbatim copies of the Program's source code as you 198 | receive it, in any medium, provided that you conspicuously and 199 | appropriately publish on each copy an appropriate copyright notice; 200 | keep intact all notices stating that this License and any 201 | non-permissive terms added in accord with section 7 apply to the code; 202 | keep intact all notices of the absence of any warranty; and give all 203 | recipients a copy of this License along with the Program. 204 | 205 | You may charge any price or no price for each copy that you convey, 206 | and you may offer support or warranty protection for a fee. 207 | 208 | 5. Conveying Modified Source Versions. 209 | 210 | You may convey a work based on the Program, or the modifications to 211 | produce it from the Program, in the form of source code under the 212 | terms of section 4, provided that you also meet all of these conditions: 213 | 214 | a) The work must carry prominent notices stating that you modified 215 | it, and giving a relevant date. 216 | 217 | b) The work must carry prominent notices stating that it is 218 | released under this License and any conditions added under section 219 | 7. This requirement modifies the requirement in section 4 to 220 | "keep intact all notices". 221 | 222 | c) You must license the entire work, as a whole, under this 223 | License to anyone who comes into possession of a copy. This 224 | License will therefore apply, along with any applicable section 7 225 | additional terms, to the whole of the work, and all its parts, 226 | regardless of how they are packaged. This License gives no 227 | permission to license the work in any other way, but it does not 228 | invalidate such permission if you have separately received it. 229 | 230 | d) If the work has interactive user interfaces, each must display 231 | Appropriate Legal Notices; however, if the Program has interactive 232 | interfaces that do not display Appropriate Legal Notices, your 233 | work need not make them do so. 234 | 235 | A compilation of a covered work with other separate and independent 236 | works, which are not by their nature extensions of the covered work, 237 | and which are not combined with it such as to form a larger program, 238 | in or on a volume of a storage or distribution medium, is called an 239 | "aggregate" if the compilation and its resulting copyright are not 240 | used to limit the access or legal rights of the compilation's users 241 | beyond what the individual works permit. Inclusion of a covered work 242 | in an aggregate does not cause this License to apply to the other 243 | parts of the aggregate. 244 | 245 | 6. Conveying Non-Source Forms. 246 | 247 | You may convey a covered work in object code form under the terms 248 | of sections 4 and 5, provided that you also convey the 249 | machine-readable Corresponding Source under the terms of this License, 250 | in one of these ways: 251 | 252 | a) Convey the object code in, or embodied in, a physical product 253 | (including a physical distribution medium), accompanied by the 254 | Corresponding Source fixed on a durable physical medium 255 | customarily used for software interchange. 256 | 257 | b) Convey the object code in, or embodied in, a physical product 258 | (including a physical distribution medium), accompanied by a 259 | written offer, valid for at least three years and valid for as 260 | long as you offer spare parts or customer support for that product 261 | model, to give anyone who possesses the object code either (1) a 262 | copy of the Corresponding Source for all the software in the 263 | product that is covered by this License, on a durable physical 264 | medium customarily used for software interchange, for a price no 265 | more than your reasonable cost of physically performing this 266 | conveying of source, or (2) access to copy the 267 | Corresponding Source from a network server at no charge. 268 | 269 | c) Convey individual copies of the object code with a copy of the 270 | written offer to provide the Corresponding Source. This 271 | alternative is allowed only occasionally and noncommercially, and 272 | only if you received the object code with such an offer, in accord 273 | with subsection 6b. 274 | 275 | d) Convey the object code by offering access from a designated 276 | place (gratis or for a charge), and offer equivalent access to the 277 | Corresponding Source in the same way through the same place at no 278 | further charge. You need not require recipients to copy the 279 | Corresponding Source along with the object code. If the place to 280 | copy the object code is a network server, the Corresponding Source 281 | may be on a different server (operated by you or a third party) 282 | that supports equivalent copying facilities, provided you maintain 283 | clear directions next to the object code saying where to find the 284 | Corresponding Source. Regardless of what server hosts the 285 | Corresponding Source, you remain obligated to ensure that it is 286 | available for as long as needed to satisfy these requirements. 287 | 288 | e) Convey the object code using peer-to-peer transmission, provided 289 | you inform other peers where the object code and Corresponding 290 | Source of the work are being offered to the general public at no 291 | charge under subsection 6d. 292 | 293 | A separable portion of the object code, whose source code is excluded 294 | from the Corresponding Source as a System Library, need not be 295 | included in conveying the object code work. 296 | 297 | A "User Product" is either (1) a "consumer product", which means any 298 | tangible personal property which is normally used for personal, family, 299 | or household purposes, or (2) anything designed or sold for incorporation 300 | into a dwelling. In determining whether a product is a consumer product, 301 | doubtful cases shall be resolved in favor of coverage. For a particular 302 | product received by a particular user, "normally used" refers to a 303 | typical or common use of that class of product, regardless of the status 304 | of the particular user or of the way in which the particular user 305 | actually uses, or expects or is expected to use, the product. A product 306 | is a consumer product regardless of whether the product has substantial 307 | commercial, industrial or non-consumer uses, unless such uses represent 308 | the only significant mode of use of the product. 309 | 310 | "Installation Information" for a User Product means any methods, 311 | procedures, authorization keys, or other information required to install 312 | and execute modified versions of a covered work in that User Product from 313 | a modified version of its Corresponding Source. The information must 314 | suffice to ensure that the continued functioning of the modified object 315 | code is in no case prevented or interfered with solely because 316 | modification has been made. 317 | 318 | If you convey an object code work under this section in, or with, or 319 | specifically for use in, a User Product, and the conveying occurs as 320 | part of a transaction in which the right of possession and use of the 321 | User Product is transferred to the recipient in perpetuity or for a 322 | fixed term (regardless of how the transaction is characterized), the 323 | Corresponding Source conveyed under this section must be accompanied 324 | by the Installation Information. But this requirement does not apply 325 | if neither you nor any third party retains the ability to install 326 | modified object code on the User Product (for example, the work has 327 | been installed in ROM). 328 | 329 | The requirement to provide Installation Information does not include a 330 | requirement to continue to provide support service, warranty, or updates 331 | for a work that has been modified or installed by the recipient, or for 332 | the User Product in which it has been modified or installed. Access to a 333 | network may be denied when the modification itself materially and 334 | adversely affects the operation of the network or violates the rules and 335 | protocols for communication across the network. 336 | 337 | Corresponding Source conveyed, and Installation Information provided, 338 | in accord with this section must be in a format that is publicly 339 | documented (and with an implementation available to the public in 340 | source code form), and must require no special password or key for 341 | unpacking, reading or copying. 342 | 343 | 7. Additional Terms. 344 | 345 | "Additional permissions" are terms that supplement the terms of this 346 | License by making exceptions from one or more of its conditions. 347 | Additional permissions that are applicable to the entire Program shall 348 | be treated as though they were included in this License, to the extent 349 | that they are valid under applicable law. If additional permissions 350 | apply only to part of the Program, that part may be used separately 351 | under those permissions, but the entire Program remains governed by 352 | this License without regard to the additional permissions. 353 | 354 | When you convey a copy of a covered work, you may at your option 355 | remove any additional permissions from that copy, or from any part of 356 | it. (Additional permissions may be written to require their own 357 | removal in certain cases when you modify the work.) You may place 358 | additional permissions on material, added by you to a covered work, 359 | for which you have or can give appropriate copyright permission. 360 | 361 | Notwithstanding any other provision of this License, for material you 362 | add to a covered work, you may (if authorized by the copyright holders of 363 | that material) supplement the terms of this License with terms: 364 | 365 | a) Disclaiming warranty or limiting liability differently from the 366 | terms of sections 15 and 16 of this License; or 367 | 368 | b) Requiring preservation of specified reasonable legal notices or 369 | author attributions in that material or in the Appropriate Legal 370 | Notices displayed by works containing it; or 371 | 372 | c) Prohibiting misrepresentation of the origin of that material, or 373 | requiring that modified versions of such material be marked in 374 | reasonable ways as different from the original version; or 375 | 376 | d) Limiting the use for publicity purposes of names of licensors or 377 | authors of the material; or 378 | 379 | e) Declining to grant rights under trademark law for use of some 380 | trade names, trademarks, or service marks; or 381 | 382 | f) Requiring indemnification of licensors and authors of that 383 | material by anyone who conveys the material (or modified versions of 384 | it) with contractual assumptions of liability to the recipient, for 385 | any liability that these contractual assumptions directly impose on 386 | those licensors and authors. 387 | 388 | All other non-permissive additional terms are considered "further 389 | restrictions" within the meaning of section 10. If the Program as you 390 | received it, or any part of it, contains a notice stating that it is 391 | governed by this License along with a term that is a further 392 | restriction, you may remove that term. If a license document contains 393 | a further restriction but permits relicensing or conveying under this 394 | License, you may add to a covered work material governed by the terms 395 | of that license document, provided that the further restriction does 396 | not survive such relicensing or conveying. 397 | 398 | If you add terms to a covered work in accord with this section, you 399 | must place, in the relevant source files, a statement of the 400 | additional terms that apply to those files, or a notice indicating 401 | where to find the applicable terms. 402 | 403 | Additional terms, permissive or non-permissive, may be stated in the 404 | form of a separately written license, or stated as exceptions; 405 | the above requirements apply either way. 406 | 407 | 8. Termination. 408 | 409 | You may not propagate or modify a covered work except as expressly 410 | provided under this License. Any attempt otherwise to propagate or 411 | modify it is void, and will automatically terminate your rights under 412 | this License (including any patent licenses granted under the third 413 | paragraph of section 11). 414 | 415 | However, if you cease all violation of this License, then your 416 | license from a particular copyright holder is reinstated (a) 417 | provisionally, unless and until the copyright holder explicitly and 418 | finally terminates your license, and (b) permanently, if the copyright 419 | holder fails to notify you of the violation by some reasonable means 420 | prior to 60 days after the cessation. 421 | 422 | Moreover, your license from a particular copyright holder is 423 | reinstated permanently if the copyright holder notifies you of the 424 | violation by some reasonable means, this is the first time you have 425 | received notice of violation of this License (for any work) from that 426 | copyright holder, and you cure the violation prior to 30 days after 427 | your receipt of the notice. 428 | 429 | Termination of your rights under this section does not terminate the 430 | licenses of parties who have received copies or rights from you under 431 | this License. If your rights have been terminated and not permanently 432 | reinstated, you do not qualify to receive new licenses for the same 433 | material under section 10. 434 | 435 | 9. Acceptance Not Required for Having Copies. 436 | 437 | You are not required to accept this License in order to receive or 438 | run a copy of the Program. Ancillary propagation of a covered work 439 | occurring solely as a consequence of using peer-to-peer transmission 440 | to receive a copy likewise does not require acceptance. However, 441 | nothing other than this License grants you permission to propagate or 442 | modify any covered work. These actions infringe copyright if you do 443 | not accept this License. Therefore, by modifying or propagating a 444 | covered work, you indicate your acceptance of this License to do so. 445 | 446 | 10. Automatic Licensing of Downstream Recipients. 447 | 448 | Each time you convey a covered work, the recipient automatically 449 | receives a license from the original licensors, to run, modify and 450 | propagate that work, subject to this License. You are not responsible 451 | for enforcing compliance by third parties with this License. 452 | 453 | An "entity transaction" is a transaction transferring control of an 454 | organization, or substantially all assets of one, or subdividing an 455 | organization, or merging organizations. If propagation of a covered 456 | work results from an entity transaction, each party to that 457 | transaction who receives a copy of the work also receives whatever 458 | licenses to the work the party's predecessor in interest had or could 459 | give under the previous paragraph, plus a right to possession of the 460 | Corresponding Source of the work from the predecessor in interest, if 461 | the predecessor has it or can get it with reasonable efforts. 462 | 463 | You may not impose any further restrictions on the exercise of the 464 | rights granted or affirmed under this License. For example, you may 465 | not impose a license fee, royalty, or other charge for exercise of 466 | rights granted under this License, and you may not initiate litigation 467 | (including a cross-claim or counterclaim in a lawsuit) alleging that 468 | any patent claim is infringed by making, using, selling, offering for 469 | sale, or importing the Program or any portion of it. 470 | 471 | 11. Patents. 472 | 473 | A "contributor" is a copyright holder who authorizes use under this 474 | License of the Program or a work on which the Program is based. The 475 | work thus licensed is called the contributor's "contributor version". 476 | 477 | A contributor's "essential patent claims" are all patent claims 478 | owned or controlled by the contributor, whether already acquired or 479 | hereafter acquired, that would be infringed by some manner, permitted 480 | by this License, of making, using, or selling its contributor version, 481 | but do not include claims that would be infringed only as a 482 | consequence of further modification of the contributor version. For 483 | purposes of this definition, "control" includes the right to grant 484 | patent sublicenses in a manner consistent with the requirements of 485 | this License. 486 | 487 | Each contributor grants you a non-exclusive, worldwide, royalty-free 488 | patent license under the contributor's essential patent claims, to 489 | make, use, sell, offer for sale, import and otherwise run, modify and 490 | propagate the contents of its contributor version. 491 | 492 | In the following three paragraphs, a "patent license" is any express 493 | agreement or commitment, however denominated, not to enforce a patent 494 | (such as an express permission to practice a patent or covenant not to 495 | sue for patent infringement). To "grant" such a patent license to a 496 | party means to make such an agreement or commitment not to enforce a 497 | patent against the party. 498 | 499 | If you convey a covered work, knowingly relying on a patent license, 500 | and the Corresponding Source of the work is not available for anyone 501 | to copy, free of charge and under the terms of this License, through a 502 | publicly available network server or other readily accessible means, 503 | then you must either (1) cause the Corresponding Source to be so 504 | available, or (2) arrange to deprive yourself of the benefit of the 505 | patent license for this particular work, or (3) arrange, in a manner 506 | consistent with the requirements of this License, to extend the patent 507 | license to downstream recipients. "Knowingly relying" means you have 508 | actual knowledge that, but for the patent license, your conveying the 509 | covered work in a country, or your recipient's use of the covered work 510 | in a country, would infringe one or more identifiable patents in that 511 | country that you have reason to believe are valid. 512 | 513 | If, pursuant to or in connection with a single transaction or 514 | arrangement, you convey, or propagate by procuring conveyance of, a 515 | covered work, and grant a patent license to some of the parties 516 | receiving the covered work authorizing them to use, propagate, modify 517 | or convey a specific copy of the covered work, then the patent license 518 | you grant is automatically extended to all recipients of the covered 519 | work and works based on it. 520 | 521 | A patent license is "discriminatory" if it does not include within 522 | the scope of its coverage, prohibits the exercise of, or is 523 | conditioned on the non-exercise of one or more of the rights that are 524 | specifically granted under this License. You may not convey a covered 525 | work if you are a party to an arrangement with a third party that is 526 | in the business of distributing software, under which you make payment 527 | to the third party based on the extent of your activity of conveying 528 | the work, and under which the third party grants, to any of the 529 | parties who would receive the covered work from you, a discriminatory 530 | patent license (a) in connection with copies of the covered work 531 | conveyed by you (or copies made from those copies), or (b) primarily 532 | for and in connection with specific products or compilations that 533 | contain the covered work, unless you entered into that arrangement, 534 | or that patent license was granted, prior to 28 March 2007. 535 | 536 | Nothing in this License shall be construed as excluding or limiting 537 | any implied license or other defenses to infringement that may 538 | otherwise be available to you under applicable patent law. 539 | 540 | 12. No Surrender of Others' Freedom. 541 | 542 | If conditions are imposed on you (whether by court order, agreement or 543 | otherwise) that contradict the conditions of this License, they do not 544 | excuse you from the conditions of this License. If you cannot convey a 545 | covered work so as to satisfy simultaneously your obligations under this 546 | License and any other pertinent obligations, then as a consequence you may 547 | not convey it at all. For example, if you agree to terms that obligate you 548 | to collect a royalty for further conveying from those to whom you convey 549 | the Program, the only way you could satisfy both those terms and this 550 | License would be to refrain entirely from conveying the Program. 551 | 552 | 13. Use with the GNU Affero General Public License. 553 | 554 | Notwithstanding any other provision of this License, you have 555 | permission to link or combine any covered work with a work licensed 556 | under version 3 of the GNU Affero General Public License into a single 557 | combined work, and to convey the resulting work. The terms of this 558 | License will continue to apply to the part which is the covered work, 559 | but the special requirements of the GNU Affero General Public License, 560 | section 13, concerning interaction through a network will apply to the 561 | combination as such. 562 | 563 | 14. Revised Versions of this License. 564 | 565 | The Free Software Foundation may publish revised and/or new versions of 566 | the GNU General Public License from time to time. Such new versions will 567 | be similar in spirit to the present version, but may differ in detail to 568 | address new problems or concerns. 569 | 570 | Each version is given a distinguishing version number. If the 571 | Program specifies that a certain numbered version of the GNU General 572 | Public License "or any later version" applies to it, you have the 573 | option of following the terms and conditions either of that numbered 574 | version or of any later version published by the Free Software 575 | Foundation. If the Program does not specify a version number of the 576 | GNU General Public License, you may choose any version ever published 577 | by the Free Software Foundation. 578 | 579 | If the Program specifies that a proxy can decide which future 580 | versions of the GNU General Public License can be used, that proxy's 581 | public statement of acceptance of a version permanently authorizes you 582 | to choose that version for the Program. 583 | 584 | Later license versions may give you additional or different 585 | permissions. However, no additional obligations are imposed on any 586 | author or copyright holder as a result of your choosing to follow a 587 | later version. 588 | 589 | 15. Disclaimer of Warranty. 590 | 591 | THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY 592 | APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT 593 | HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY 594 | OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, 595 | THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 596 | PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM 597 | IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF 598 | ALL NECESSARY SERVICING, REPAIR OR CORRECTION. 599 | 600 | 16. Limitation of Liability. 601 | 602 | IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING 603 | WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS 604 | THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY 605 | GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE 606 | USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF 607 | DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD 608 | PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), 609 | EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF 610 | SUCH DAMAGES. 611 | 612 | 17. Interpretation of Sections 15 and 16. 613 | 614 | If the disclaimer of warranty and limitation of liability provided 615 | above cannot be given local legal effect according to their terms, 616 | reviewing courts shall apply local law that most closely approximates 617 | an absolute waiver of all civil liability in connection with the 618 | Program, unless a warranty or assumption of liability accompanies a 619 | copy of the Program in return for a fee. 620 | 621 | END OF TERMS AND CONDITIONS 622 | 623 | How to Apply These Terms to Your New Programs 624 | 625 | If you develop a new program, and you want it to be of the greatest 626 | possible use to the public, the best way to achieve this is to make it 627 | free software which everyone can redistribute and change under these terms. 628 | 629 | To do so, attach the following notices to the program. It is safest 630 | to attach them to the start of each source file to most effectively 631 | state the exclusion of warranty; and each file should have at least 632 | the "copyright" line and a pointer to where the full notice is found. 633 | 634 | 635 | Copyright (C) 636 | 637 | This program is free software: you can redistribute it and/or modify 638 | it under the terms of the GNU General Public License as published by 639 | the Free Software Foundation, either version 3 of the License, or 640 | (at your option) any later version. 641 | 642 | This program is distributed in the hope that it will be useful, 643 | but WITHOUT ANY WARRANTY; without even the implied warranty of 644 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 645 | GNU General Public License for more details. 646 | 647 | You should have received a copy of the GNU General Public License 648 | along with this program. If not, see . 649 | 650 | Also add information on how to contact you by electronic and paper mail. 651 | 652 | If the program does terminal interaction, make it output a short 653 | notice like this when it starts in an interactive mode: 654 | 655 | Copyright (C) 656 | This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. 657 | This is free software, and you are welcome to redistribute it 658 | under certain conditions; type `show c' for details. 659 | 660 | The hypothetical commands `show w' and `show c' should show the appropriate 661 | parts of the General Public License. Of course, your program's commands 662 | might be different; for a GUI interface, you would use an "about box". 663 | 664 | You should also get your employer (if you work as a programmer) or school, 665 | if any, to sign a "copyright disclaimer" for the program, if necessary. 666 | For more information on this, and how to apply and follow the GNU GPL, see 667 | . 668 | 669 | The GNU General Public License does not permit incorporating your program 670 | into proprietary programs. If your program is a subroutine library, you 671 | may consider it more useful to permit linking proprietary applications with 672 | the library. If this is what you want to do, use the GNU Lesser General 673 | Public License instead of this License. But first, please read 674 | . 675 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Meeting Calendar 2 | 3 | This Meeting Calendar provides a way of scheduling recurring meetings, and displaying a calendar or timetable. 4 | 5 | ## Getting Started 6 | 7 | 1. Make sure you have [`git`](https://git-scm.com/), [`node`](https://nodejs.org/), and [`npm`](https://www.npmjs.com/get-npm) installed. 8 | 2. Clone this repository into your `\plugins` folder. 9 | 3. Execute `npm install` from the root directory of the repository to install the dependencies. 10 | 4. Execute `npm start` for development mode (`npm run build` for a production build). 11 | 5. Activate the `Meeting Calendar` plugin in your WordPress plugin directory 12 | 6. Create some meetings 13 | 7. While editing your page/post, add in the `Meeting Calendar` block and publish! 14 | 15 | ## Development environment 16 | 17 | You can (optionally) use [`wp-env`](https://developer.wordpress.org/block-editor/packages/packages-env/) to set up a local environment. 18 | 19 | 1. Install the node dependencies `npm install` 20 | 2. Start the wp-env environment with `npm run wp-env start` 21 | 3. Visit your new local environment at `http://localhost:8888` 22 | 23 | ### Running PHPUnit Tests 24 | 25 | 1. Install the composer dependencies `composer install` 26 | 2. If you haven't yet, install the node dependencies `npm install` 27 | 3. Start the wp-env environment with `npm run wp-env start` 28 | 4. Run the tests with `npm run test:unit-php` 29 | 30 | ### Enable ICS links 31 | Calendars and individual events can be accessed through `.ics` links, for example http://localhost:8888/meetings.ics. For these links to work, the `permalink_structure` option must be set in the `wp_options` database table, and the appropriate rule must be present in `.htaccess`. 32 | 33 | You can set both with the following command: 34 | 35 | ```shell 36 | # The --hard flag updates .htaccess rules as well as rules in the database. 37 | # For more info see: 38 | # https://developer.wordpress.org/cli/commands/rewrite/structure/ 39 | 40 | wp-env run cli "wp rewrite structure --hard '/%postname%'" 41 | ``` 42 | 43 | ## License 44 | 45 | Meeting Calendar is licensed under [GNU General Public License v2 (or later)](./LICENSE.md). 46 | -------------------------------------------------------------------------------- /bin/install-wp-tests.sh: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | 3 | if [ $# -lt 3 ]; then 4 | echo "usage: $0 [db-host] [wp-version] [skip-database-creation]" 5 | exit 1 6 | fi 7 | 8 | DB_NAME=$1 9 | DB_USER=$2 10 | DB_PASS=$3 11 | DB_HOST=${4-localhost} 12 | WP_VERSION=${5-latest} 13 | SKIP_DB_CREATE=${6-false} 14 | 15 | TMPDIR=${TMPDIR-/tmp} 16 | TMPDIR=$(echo $TMPDIR | sed -e "s/\/$//") 17 | WP_TESTS_DIR=${WP_TESTS_DIR-$TMPDIR/wordpress-tests-lib} 18 | WP_CORE_DIR=${WP_CORE_DIR-$TMPDIR/wordpress/} 19 | 20 | download() { 21 | if [ `which curl` ]; then 22 | curl -s "$1" > "$2"; 23 | elif [ `which wget` ]; then 24 | wget -nv -O "$2" "$1" 25 | fi 26 | } 27 | 28 | if [[ $WP_VERSION =~ ^[0-9]+\.[0-9]+\-(beta|RC)[0-9]+$ ]]; then 29 | WP_BRANCH=${WP_VERSION%\-*} 30 | WP_TESTS_TAG="branches/$WP_BRANCH" 31 | 32 | elif [[ $WP_VERSION =~ ^[0-9]+\.[0-9]+$ ]]; then 33 | WP_TESTS_TAG="branches/$WP_VERSION" 34 | elif [[ $WP_VERSION =~ [0-9]+\.[0-9]+\.[0-9]+ ]]; then 35 | if [[ $WP_VERSION =~ [0-9]+\.[0-9]+\.[0] ]]; then 36 | # version x.x.0 means the first release of the major version, so strip off the .0 and download version x.x 37 | WP_TESTS_TAG="tags/${WP_VERSION%??}" 38 | else 39 | WP_TESTS_TAG="tags/$WP_VERSION" 40 | fi 41 | elif [[ $WP_VERSION == 'nightly' || $WP_VERSION == 'trunk' ]]; then 42 | WP_TESTS_TAG="trunk" 43 | else 44 | # http serves a single offer, whereas https serves multiple. we only want one 45 | download http://api.wordpress.org/core/version-check/1.7/ /tmp/wp-latest.json 46 | grep '[0-9]+\.[0-9]+(\.[0-9]+)?' /tmp/wp-latest.json 47 | LATEST_VERSION=$(grep -o '"version":"[^"]*' /tmp/wp-latest.json | sed 's/"version":"//') 48 | if [[ -z "$LATEST_VERSION" ]]; then 49 | echo "Latest WordPress version could not be found" 50 | exit 1 51 | fi 52 | WP_TESTS_TAG="tags/$LATEST_VERSION" 53 | fi 54 | set -ex 55 | 56 | install_wp() { 57 | 58 | if [ -d $WP_CORE_DIR ]; then 59 | return; 60 | fi 61 | 62 | mkdir -p $WP_CORE_DIR 63 | 64 | if [[ $WP_VERSION == 'nightly' || $WP_VERSION == 'trunk' ]]; then 65 | mkdir -p $TMPDIR/wordpress-nightly 66 | download https://wordpress.org/nightly-builds/wordpress-latest.zip $TMPDIR/wordpress-nightly/wordpress-nightly.zip 67 | unzip -q $TMPDIR/wordpress-nightly/wordpress-nightly.zip -d $TMPDIR/wordpress-nightly/ 68 | mv $TMPDIR/wordpress-nightly/wordpress/* $WP_CORE_DIR 69 | else 70 | if [ $WP_VERSION == 'latest' ]; then 71 | local ARCHIVE_NAME='latest' 72 | elif [[ $WP_VERSION =~ [0-9]+\.[0-9]+ ]]; then 73 | # https serves multiple offers, whereas http serves single. 74 | download https://api.wordpress.org/core/version-check/1.7/ $TMPDIR/wp-latest.json 75 | if [[ $WP_VERSION =~ [0-9]+\.[0-9]+\.[0] ]]; then 76 | # version x.x.0 means the first release of the major version, so strip off the .0 and download version x.x 77 | LATEST_VERSION=${WP_VERSION%??} 78 | else 79 | # otherwise, scan the releases and get the most up to date minor version of the major release 80 | local VERSION_ESCAPED=`echo $WP_VERSION | sed 's/\./\\\\./g'` 81 | LATEST_VERSION=$(grep -o '"version":"'$VERSION_ESCAPED'[^"]*' $TMPDIR/wp-latest.json | sed 's/"version":"//' | head -1) 82 | fi 83 | if [[ -z "$LATEST_VERSION" ]]; then 84 | local ARCHIVE_NAME="wordpress-$WP_VERSION" 85 | else 86 | local ARCHIVE_NAME="wordpress-$LATEST_VERSION" 87 | fi 88 | else 89 | local ARCHIVE_NAME="wordpress-$WP_VERSION" 90 | fi 91 | download https://wordpress.org/${ARCHIVE_NAME}.tar.gz $TMPDIR/wordpress.tar.gz 92 | tar --strip-components=1 -zxmf $TMPDIR/wordpress.tar.gz -C $WP_CORE_DIR 93 | fi 94 | 95 | download https://raw.github.com/markoheijnen/wp-mysqli/master/db.php $WP_CORE_DIR/wp-content/db.php 96 | } 97 | 98 | install_test_suite() { 99 | # portable in-place argument for both GNU sed and Mac OSX sed 100 | if [[ $(uname -s) == 'Darwin' ]]; then 101 | local ioption='-i.bak' 102 | else 103 | local ioption='-i' 104 | fi 105 | 106 | # set up testing suite if it doesn't yet exist 107 | if [ ! -d $WP_TESTS_DIR ]; then 108 | # set up testing suite 109 | mkdir -p $WP_TESTS_DIR 110 | svn co --quiet --ignore-externals https://develop.svn.wordpress.org/${WP_TESTS_TAG}/tests/phpunit/includes/ $WP_TESTS_DIR/includes 111 | svn co --quiet --ignore-externals https://develop.svn.wordpress.org/${WP_TESTS_TAG}/tests/phpunit/data/ $WP_TESTS_DIR/data 112 | fi 113 | 114 | if [ ! -f wp-tests-config.php ]; then 115 | download https://develop.svn.wordpress.org/${WP_TESTS_TAG}/wp-tests-config-sample.php "$WP_TESTS_DIR"/wp-tests-config.php 116 | # remove all forward slashes in the end 117 | WP_CORE_DIR=$(echo $WP_CORE_DIR | sed "s:/\+$::") 118 | sed $ioption "s:dirname( __FILE__ ) . '/src/':'$WP_CORE_DIR/':" "$WP_TESTS_DIR"/wp-tests-config.php 119 | sed $ioption "s/youremptytestdbnamehere/$DB_NAME/" "$WP_TESTS_DIR"/wp-tests-config.php 120 | sed $ioption "s/yourusernamehere/$DB_USER/" "$WP_TESTS_DIR"/wp-tests-config.php 121 | sed $ioption "s/yourpasswordhere/$DB_PASS/" "$WP_TESTS_DIR"/wp-tests-config.php 122 | sed $ioption "s|localhost|${DB_HOST}|" "$WP_TESTS_DIR"/wp-tests-config.php 123 | fi 124 | 125 | } 126 | 127 | install_db() { 128 | 129 | if [ ${SKIP_DB_CREATE} = "true" ]; then 130 | return 0 131 | fi 132 | 133 | # parse DB_HOST for port or socket references 134 | local PARTS=(${DB_HOST//\:/ }) 135 | local DB_HOSTNAME=${PARTS[0]}; 136 | local DB_SOCK_OR_PORT=${PARTS[1]}; 137 | local EXTRA="" 138 | 139 | if ! [ -z $DB_HOSTNAME ] ; then 140 | if [ $(echo $DB_SOCK_OR_PORT | grep -e '^[0-9]\{1,\}$') ]; then 141 | EXTRA=" --host=$DB_HOSTNAME --port=$DB_SOCK_OR_PORT --protocol=tcp" 142 | elif ! [ -z $DB_SOCK_OR_PORT ] ; then 143 | EXTRA=" --socket=$DB_SOCK_OR_PORT" 144 | elif ! [ -z $DB_HOSTNAME ] ; then 145 | EXTRA=" --host=$DB_HOSTNAME --protocol=tcp" 146 | fi 147 | fi 148 | 149 | # create database 150 | mysqladmin create $DB_NAME --user="$DB_USER" --password="$DB_PASS"$EXTRA 151 | } 152 | 153 | install_wp 154 | install_test_suite 155 | install_db 156 | -------------------------------------------------------------------------------- /composer.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "wporg/meeting-calendar", 3 | "type": "wordpress-plugin", 4 | "description": "", 5 | "homepage": "https://make.wordpress.org/meetings", 6 | "license": "GPL-2.0-or-later", 7 | "support": { 8 | "issues": "https://github.com/WordPress/meeting-calendar/issues" 9 | }, 10 | "config": { 11 | "platform": { 12 | "php": "7.4" 13 | }, 14 | "allow-plugins": { 15 | "dealerdirect/phpcodesniffer-composer-installer": true 16 | } 17 | }, 18 | "repositories": [ 19 | { 20 | "type": "composer", 21 | "url": "https://wpackagist.org/" 22 | } 23 | ], 24 | "require-dev": { 25 | "dealerdirect/phpcodesniffer-composer-installer": "^0.7.0", 26 | "wp-coding-standards/wpcs": "2.*", 27 | "phpcompatibility/phpcompatibility-wp": "*", 28 | "wp-phpunit/wp-phpunit": "^5.4", 29 | "phpunit/phpunit": "^9.5", 30 | "yoast/phpunit-polyfills": "^3.0" 31 | }, 32 | "scripts": { 33 | "format": "phpcbf -p", 34 | "lint": "phpcs" 35 | }, 36 | "dependencies": { 37 | "prettier": "^1.13.0", 38 | "typescript": "^2.8.4" 39 | } 40 | } 41 | -------------------------------------------------------------------------------- /includes/ical-functions.php: -------------------------------------------------------------------------------- 1 | 1, 25 | QUERY_TEAM_KEY => '$matches[1]', 26 | ), 27 | 'top' 28 | ); 29 | } 30 | add_action( 'init', __NAMESPACE__ . '\add_rewrite_rules' ); 31 | 32 | /** 33 | * Add the Query Var for the Rewrite Rules. 34 | */ 35 | function query_vars( $query_vars ) { 36 | $query_vars[] = QUERY_KEY; 37 | $query_vars[] = QUERY_TEAM_KEY; 38 | return $query_vars; 39 | } 40 | add_filter( 'query_vars', __NAMESPACE__ . '\query_vars' ); 41 | 42 | /** 43 | * Main handler for ICS output for matching requests. 44 | */ 45 | function parse_request( $request ) { 46 | if ( ! isset( $request->query_vars[ QUERY_KEY ] ) ) { 47 | return; 48 | } 49 | 50 | $team = strtolower( $request->query_vars[ QUERY_TEAM_KEY ] ); 51 | 52 | // Grab the meetings, optionally 53 | $posts = get_meeting_posts( $team ); 54 | 55 | // Output a 404 if there's no meetings, but still generate a ICS feed. 56 | if ( ! $posts ) { 57 | header( 58 | ( $_SERVER['SERVER_PROTOCOL'] ?? 'HTTP/1.0' ) . ' 404 No Meetings Found', 59 | true, 60 | 404 61 | ); 62 | } 63 | 64 | // If the calendar has a 'method' property, the 'Content-Type' header must also specify it 65 | header( 'Content-Type: text/calendar; charset=utf-8; method=publish' ); 66 | header( 'Content-Disposition: inline; filename=' . 'meetings' . ( $team ? "-{$team}" : '' ) . '.ics' ); 67 | 68 | // phpcs:ignore -- Direct output okay. 69 | echo Generator\generate( $posts, $team ); 70 | 71 | exit; 72 | } 73 | add_action( 'parse_request', __NAMESPACE__ . '\parse_request' ); 74 | 75 | /** 76 | * Get all meetings for a team. If the 'team' parameter is empty, all meetings are returned. 77 | * 78 | * @param string $team Name of the team to fetch meetings for. 79 | * @return array 80 | */ 81 | function get_meeting_posts( $team = '' ) { 82 | $meta_query = \Meeting_Post_Type::getInstance()->meeting_meta_query(); 83 | if ( $team ) { 84 | $meta_query = array( 85 | 'relation' => 'AND', 86 | array( 87 | 'key' => 'team', 88 | 'value' => $team, 89 | 'compare' => 'EQUALS', 90 | ), 91 | $meta_query, 92 | ); 93 | } 94 | 95 | $query = new \WP_Query( 96 | array( 97 | 'post_type' => 'meeting', 98 | 'post_status' => 'publish', 99 | 'nopaging' => true, 100 | 'meta_query' => $meta_query, 101 | // Avoid re-ordering the events. 102 | 'suppress_filters' => true, 103 | ) 104 | ); 105 | 106 | $posts = $query->get_posts(); 107 | // Update each post with next dates – needs to be called here because we `suppress_filters` above. 108 | \Meeting_Post_Type::getInstance()->meeting_set_next_meeting( $posts ); 109 | 110 | return $posts; 111 | } 112 | -------------------------------------------------------------------------------- /includes/ical-generator-functions.php: -------------------------------------------------------------------------------- 1 | ID; 45 | $title = $post->post_title; 46 | $location = $post->location; 47 | $link = $post->link; 48 | $wptv_url = $post->wptv_url; 49 | $team = $post->team; 50 | $recurring = $post->recurring; 51 | $sequence = empty( $post->sequence ) ? 0 : intval( $post->sequence ); 52 | 53 | $start_date = strftime( '%Y%m%d', strtotime( $post->start_date ) ); 54 | $start_time = strftime( '%H%M%S', strtotime( $post->time ) ); 55 | $start_date_time = "{$start_date}T{$start_time}Z"; 56 | 57 | $end_date = $start_date; 58 | $end_time = strftime( '%H%M%S', strtotime( "{$post->time} +1 hour" ) ); 59 | $end_date_time = "{$end_date}T{$end_time}Z"; 60 | 61 | $description = ''; 62 | $slack_channel = null; 63 | 64 | if ( $location && preg_match( '/^#([-\w]+)$/', trim( $location ), $match ) ) { 65 | $slack_channel = '#' . sanitize_title( $match[1] ); 66 | $location = "{$slack_channel} channel on Slack"; 67 | } 68 | 69 | if ( $wptv_url ) { 70 | $description .= "WordPress.tv link: {$wptv_url}\\n"; 71 | } 72 | 73 | if ( $link ) { 74 | if ( $slack_channel ) { 75 | $description .= "Slack channel link: https://wordpress.slack.com/messages/{$slack_channel}\\n"; 76 | } 77 | 78 | $description .= "For more information visit {$link}"; 79 | } 80 | 81 | $frequency = get_frequency( $recurring, $post->next_date, $post->occurrence ); 82 | 83 | $event = 'BEGIN:VEVENT' . NEWLINE; 84 | $event .= "UID:{$id}" . NEWLINE; 85 | 86 | $event .= "DTSTAMP:{$start_date_time}" . NEWLINE; 87 | $event .= "DTSTART:{$start_date_time}" . NEWLINE; 88 | $event .= "DTEND:{$end_date_time}" . NEWLINE; 89 | $event .= 'CATEGORIES:WordPress' . NEWLINE; 90 | // Some calendars require the organizer's name and email address 91 | $event .= "ORGANIZER;CN=WordPress {$team} Team:mailto:mail@example.com" . NEWLINE; 92 | $event .= "SUMMARY:{$team}: {$title}" . NEWLINE; 93 | // Incrementing the sequence number updates the specified event 94 | $event .= "SEQUENCE:{$sequence}" . NEWLINE; 95 | $event .= 'STATUS:CONFIRMED' . NEWLINE; 96 | $event .= 'TRANSP:OPAQUE' . NEWLINE; 97 | 98 | if ( ! empty( $location ) ) { 99 | $event .= "LOCATION:{$location}" . NEWLINE; 100 | } 101 | 102 | if ( ! empty( $description ) ) { 103 | $event .= "DESCRIPTION:{$description}" . NEWLINE; 104 | } 105 | 106 | if ( ! is_null( $frequency ) ) { 107 | $event .= "RRULE:FREQ={$frequency}" . NEWLINE; 108 | 109 | $cancelled = get_post_meta( $post->ID, 'meeting_cancelled', false ); 110 | if ( $cancelled ) { 111 | foreach ( $cancelled as $i => $cancelled_date ) { 112 | $exdate = strtotime( $cancelled_date ); 113 | // Only list cancelled dates that are valid and in the future or recent past 114 | if ( $exdate >= strtotime( 'yesterday' ) ) { 115 | $cancelled[ $i ] = strftime( '%Y%m%d', $exdate ); 116 | } 117 | } 118 | $event .= 'EXDATE:' . implode( ',', $cancelled ) . NEWLINE; 119 | } 120 | } 121 | 122 | $event .= 'END:VEVENT' . NEWLINE; 123 | 124 | return $event; 125 | } 126 | 127 | /** 128 | * Generate a frequency string in iCal format. 129 | * 130 | * See https://icalendar.org/iCalendar-RFC-5545/3-8-5-3-recurrence-rule.html 131 | * 132 | * @param string $recurrence One of 'weekly', 'biweekly', 'monthly', or 'occurance' (custom). 133 | * @param string $date The date of the first event (used for custom occurance). 134 | * @param int[] $occurrences Array of week numbers for repetition. 135 | * @return string 136 | */ 137 | function get_frequency( $recurrence, $date, $occurrences ) { 138 | switch ( $recurrence ) { 139 | case 'weekly': 140 | $frequency = 'WEEKLY'; 141 | break; 142 | case 'biweekly': 143 | $frequency = 'WEEKLY;INTERVAL=2'; 144 | break; 145 | case 'monthly': 146 | $frequency = 'MONTHLY'; 147 | break; 148 | case 'occurrence': 149 | $frequency = get_frequencies_by_day( $occurrences, $date ); 150 | break; 151 | default: 152 | $frequency = null; 153 | } 154 | 155 | return $frequency; 156 | } 157 | 158 | /** 159 | * Returns a comma separated list of days in which the event should repeat for the month. 160 | * 161 | * For example, given: 162 | * $occurrences = array( 1, 3 ) // 1st and 3rd week in the month 163 | * $date = '2019-09-15' // the day is Sunday 164 | * it will return: 'MONTHLY;BYDAY=1SU,3SU' 165 | * 166 | * @param int[] $occurrences Array of week numbers for repetition. 167 | * @param string $date The date of the first event. 168 | * @return string 169 | */ 170 | function get_frequencies_by_day( $occurrences, $date ) { 171 | // Get the first two letters of the day of the start date in uppercase letters 172 | $day = strtoupper( 173 | substr( strftime( '%a', strtotime( $date ) ), 0, 2 ) 174 | ); 175 | 176 | $by_days = array_reduce( 177 | array_keys( $occurrences ), 178 | function ( $carry, $key ) use ( $day, $occurrences ) { 179 | $carry .= $occurrences[ $key ] . $day; 180 | 181 | if ( $key < count( $occurrences ) - 1 ) { 182 | $carry .= ','; 183 | } 184 | 185 | return $carry; 186 | } 187 | ); 188 | 189 | return "MONTHLY;BYDAY={$by_days}"; 190 | } 191 | -------------------------------------------------------------------------------- /includes/wporg-meeting-install.php: -------------------------------------------------------------------------------- 1 | publish + wp_count_posts( 'meeting' )->draft <= 0 ) { 9 | // No posts of any status exist, so insert a few sample meetings. 10 | 11 | $meeting_ids = array(); 12 | 13 | $meeting_ids[] = wp_insert_post( 14 | array( 15 | 'post_title' => __( 'A weekly meeting', 'wporg-meeting-calendar' ), 16 | 'post_type' => 'meeting', 17 | 'post_status' => 'publish', 18 | 'meta_input' => array( 19 | 'team' => 'Team-A', 20 | 'start_date' => '2020-01-01', 21 | 'end_date' => '', 22 | 'time' => '14:00:00', 23 | 'recurring' => 'weekly', 24 | 'link' => 'wordpress.org', 25 | 'location' => '#meta', 26 | 'wptv_url' => 'https://wordpress.tv', 27 | ), 28 | ) 29 | ); 30 | 31 | $meeting_ids[] = wp_insert_post( 32 | array( 33 | 'post_title' => __( 'A monthly meeting', 'wporg-meeting-calendar' ), 34 | 'post_type' => 'meeting', 35 | 'post_status' => 'publish', 36 | 'meta_input' => array( 37 | 'team' => 'Team-B', 38 | 'start_date' => '2020-01-01', 39 | 'end_date' => '', 40 | 'time' => '15:00:00', 41 | 'recurring' => 'monthly', 42 | 'link' => 'wordpress.org', 43 | 'location' => '#meta', 44 | 'wptv_url' => 'https://wordpress.tv', 45 | ), 46 | ) 47 | ); 48 | 49 | $meeting_ids[] = wp_insert_post( 50 | array( 51 | 'post_title' => __( 'Third Wednesday of each month', 'wporg-meeting-calendar' ), 52 | 'post_type' => 'meeting', 53 | 'post_status' => 'publish', 54 | 'meta_input' => array( 55 | 'team' => 'Team-C', 56 | 'start_date' => '2020-01-01', 57 | 'end_date' => '', 58 | 'time' => '16:00:00', 59 | 'recurring' => 'occurrence', 60 | 'occurrence' => array( 3 ), 61 | 'link' => 'wordpress.org', 62 | 'location' => '#meta', 63 | 'wptv_url' => 'https://wordpress.tv', 64 | ), 65 | ) 66 | ); 67 | 68 | flush_rewrite_rules(); 69 | 70 | return $meeting_ids; 71 | } 72 | } 73 | -------------------------------------------------------------------------------- /includes/wporg-meeting-posttype.php: -------------------------------------------------------------------------------- 1 | 47 | 51 | __( 'Team', 'wporg-meeting-calendar' ) ) 57 | + array_slice( $columns, 1, -1, true ) 58 | + array( 'wptv_url' => __( 'WPTV URL', 'wporg-meeting-calendar' ) ) 59 | + array_slice( $columns, -1, null, true ); 60 | return $columns; 61 | } 62 | 63 | public function meeting_custom_columns( $column, $post_id ) { 64 | switch ( $column ) { 65 | case 'team': 66 | $team = get_post_meta( $post_id, 'team', true ); 67 | echo esc_html( ucwords( $team ) ); 68 | break; 69 | } 70 | switch ( $column ) { 71 | case 'wptv_url': 72 | $wptv_url = get_post_meta( $post_id, 'wptv_url', true ); 73 | echo esc_html( $wptv_url ?: '—' ); 74 | break; 75 | } 76 | } 77 | 78 | public function meeting_archive_page_query( $query ) { 79 | if ( is_admin() || ! $query->is_main_query() || ! $query->is_post_type_archive( 'meeting' ) ) { 80 | return; 81 | } 82 | // turn off paging on the archive page, to show all meetings in the table 83 | $query->set( 'nopaging', true ); 84 | 85 | // meta query to eliminate expired meetings from query 86 | $query->set( 'meta_query', $this->meeting_meta_query() ); 87 | } 88 | 89 | public function meeting_set_next_meeting( $posts ) { 90 | $is_meeting_list = array_reduce( $posts, function( $is_meeting, $post ) { 91 | return $is_meeting && 'meeting' === $post->post_type; 92 | }, true ); 93 | if ( ! $is_meeting_list ) { 94 | return $posts; 95 | } 96 | 97 | // for each entry, set a fake meta value to show the next date for recurring meetings 98 | array_walk( 99 | $posts, 100 | function ( &$post ) { 101 | if ( 'meeting' !== $post->post_type ) { 102 | return false; 103 | } 104 | $next = $this->get_next_occurrence( $post ); 105 | if ( $next ) { 106 | $post->next_date = $next; 107 | } else { 108 | // if the datetime is invalid, then set the post->next_date to the start date instead 109 | $post->next_date = $post->start_date; 110 | } 111 | } 112 | ); 113 | 114 | return $posts; 115 | } 116 | 117 | public function meeting_sort_upcoming_meetings( $posts, $query ) { 118 | // Avoid reordering posts in the admin. 119 | if ( is_admin() || ! is_array( $posts ) ) { 120 | return $posts; 121 | } 122 | 123 | $is_meeting_list = array_reduce( $posts, function( $is_meeting, $post ) { 124 | return $is_meeting && 'meeting' === $post->post_type; 125 | }, true ); 126 | if ( ! $is_meeting_list ) { 127 | return $posts; 128 | } 129 | 130 | // reorder the posts by next_date + time 131 | usort( 132 | $posts, 133 | function ( $a, $b ) { 134 | $adate = strtotime( $a->next_date . ' ' . $a->time ); 135 | $bdate = strtotime( $b->next_date . ' ' . $b->time ); 136 | if ( $adate == $bdate ) { 137 | return 0; 138 | } 139 | return ( $adate < $bdate ) ? -1 : 1; 140 | } 141 | ); 142 | 143 | return $posts; 144 | } 145 | 146 | /** 147 | * Returns the date of the next occurrence of the meeting. 148 | * 149 | * @param object $post A meeting post object. 150 | * @param string $after_datetime Find the next occurrence after this date. 151 | * 152 | * @return string|bool A date string representing the date of the next occurrence; false if there is no next meeting. 153 | */ 154 | public function get_next_occurrence( $post, $after_datetime = '-30 minutes' ) { 155 | if ( ! is_object( $post ) || 'meeting' !== $post->post_type ) { 156 | return false; 157 | } 158 | 159 | $next_date = false; 160 | 161 | if ( 'weekly' === $post->recurring || '1' === $post->recurring ) { 162 | try { 163 | // from the start date, advance the week until it's past now 164 | $start = new DateTime( sprintf( '%s %s GMT', $post->start_date, $post->time ) ); 165 | $next = $start; 166 | // minus 30 minutes to account for currently ongoing meetings 167 | $now = new DateTime( $after_datetime ); 168 | 169 | if ( $next <= $now ) { 170 | $interval = $start->diff( $now ); 171 | // add one to days to account for events that happened earlier today 172 | $weekdiff = ceil( ( $interval->days + 1 ) / 7 ); 173 | $next->modify( '+ ' . $weekdiff . ' weeks' ); 174 | } 175 | 176 | $next_date = $next->format( 'Y-m-d' ); 177 | } catch ( Exception $e ) { 178 | $next_date = false; 179 | } 180 | } elseif ( 'biweekly' === $post->recurring ) { 181 | try { 182 | // advance the start date 2 weeks at a time until it's past now 183 | $start = new DateTime( sprintf( '%s %s GMT', $post->start_date, $post->time ) ); 184 | $next = $start; 185 | // minus 30 minutes to account for currently ongoing meetings 186 | $now = new DateTime( $after_datetime ); 187 | 188 | while ( $next <= $now ) { 189 | $next->modify( '+2 weeks' ); 190 | } 191 | 192 | $next_date = $next->format( 'Y-m-d' ); 193 | } catch ( Exception $e ) { 194 | $next_date = false; 195 | } 196 | } elseif ( 'occurrence' === $post->recurring ) { 197 | try { 198 | // advance the occurrence day in the current month until it's past now 199 | $start = new DateTime( sprintf( '%s %s GMT', $post->start_date, $post->time ) ); 200 | $next = $start; 201 | // minus 30 minutes to account for currently ongoing meetings 202 | $now = new DateTime( $after_datetime ); 203 | 204 | $day_index = gmdate( 'w', strtotime( sprintf( '%s %s GMT', $post->start_date, $post->time ) ) ); 205 | $day_name = $GLOBALS['wp_locale']->get_weekday( $day_index ); 206 | $numerals = array( 'first', 'second', 'third', 'fourth' ); 207 | $months = array( 'last month', 'this month', 'next month', '+2 month' ); 208 | 209 | $next = clone $now; 210 | 211 | $limit = 12; 212 | do { 213 | $month_year = $next->format( 'F Y' ); 214 | foreach ( $post->occurrence as $index ) { 215 | $next = new DateTime( sprintf( '%s %s of %s %s GMT', $numerals[ $index - 1 ], $day_name, $month_year, $post->time ) ); 216 | if ( $next > $now ) { 217 | break 2; 218 | } 219 | } 220 | $next->modify( '+1 month' ); 221 | } while ( --$limit > 0 ); 222 | $next_date = $next->format( 'Y-m-d' ); 223 | } catch ( Exception $e ) { 224 | $next_date = false; 225 | } 226 | } elseif ( 'monthly' === $post->recurring ) { 227 | try { 228 | // advance the start date 1 month at a time until it's past now 229 | $start = new DateTime( sprintf( '%s %s GMT', $post->start_date, $post->time ) ); 230 | $next = $start; 231 | // minus 30 minutes to account for currently ongoing meetings 232 | $now = new DateTime( $after_datetime ); 233 | 234 | while ( $next <= $now ) { 235 | $next->modify( '+1 month' ); 236 | } 237 | 238 | $next_date = $next->format( 'Y-m-d' ); 239 | } catch ( Exception $e ) { 240 | $next_date = false; 241 | } 242 | } else { 243 | $next_date = $post->start_date; 244 | } 245 | 246 | return $next_date; 247 | } 248 | 249 | public function register_meeting_post_type() { 250 | $labels = array( 251 | 'name' => _x( 'Meetings', 'Post Type General Name', 'wporg-meeting-calendar' ), 252 | 'singular_name' => _x( 'Meeting', 'Post Type Singular Name', 'wporg-meeting-calendar' ), 253 | 'menu_name' => __( 'Meetings', 'wporg-meeting-calendar' ), 254 | 'name_admin_bar' => __( 'Meeting', 'wporg-meeting-calendar' ), 255 | 'parent_item_colon' => __( 'Parent Meeting:', 'wporg-meeting-calendar' ), 256 | 'all_items' => __( 'All Meetings', 'wporg-meeting-calendar' ), 257 | 'add_new_item' => __( 'Add New Meeting', 'wporg-meeting-calendar' ), 258 | 'add_new' => __( 'Add New', 'wporg-meeting-calendar' ), 259 | 'new_item' => __( 'New Meeting', 'wporg-meeting-calendar' ), 260 | 'edit_item' => __( 'Edit Meeting', 'wporg-meeting-calendar' ), 261 | 'update_item' => __( 'Update Meeting', 'wporg-meeting-calendar' ), 262 | 'view_item' => __( 'View Meeting', 'wporg-meeting-calendar' ), 263 | 'view_items' => __( 'View Meetings', 'wporg-meeting-calendar' ), 264 | 'search_items' => __( 'Search Meeting', 'wporg-meeting-calendar' ), 265 | 'not_found' => __( 'Not found', 'wporg-meeting-calendar' ), 266 | 'not_found_in_trash' => __( 'Not found in Trash', 'wporg-meeting-calendar' ), 267 | ); 268 | $args = array( 269 | 'label' => __( 'meeting', 'wporg-meeting-calendar' ), 270 | 'description' => __( 'Meeting', 'wporg-meeting-calendar' ), 271 | 'labels' => $labels, 272 | 'supports' => array( 'title', 'custom-fields' ), 273 | 'hierarchical' => false, 274 | 'public' => true, 275 | 'show_ui' => true, 276 | 'show_in_menu' => true, 277 | 'menu_position' => 20, 278 | 'menu_icon' => 'dashicons-calendar', 279 | 'show_in_admin_bar' => true, 280 | 'show_in_nav_menus' => false, 281 | 'show_in_rest' => true, 282 | 'can_export' => false, 283 | 'has_archive' => false, 284 | 'exclude_from_search' => true, 285 | 'publicly_queryable' => true, 286 | 'capability_type' => 'post', 287 | 'register_meta_box_cb' => array( $this, 'add_meta_boxes' ), 288 | 'rewrite' => false, 289 | ); 290 | register_post_type( 'meeting', $args ); 291 | } 292 | 293 | public function register_meta() { 294 | // Most are string types 295 | $meta_keys = array( 296 | 'team', 297 | 'start_date', 298 | 'end_date', 299 | 'time', 300 | 'recurring', 301 | 'link', 302 | 'location', 303 | 'wptv_url', 304 | ); 305 | foreach ( $meta_keys as $key ) { 306 | register_meta( 307 | 'post', 308 | $key, 309 | array( 310 | 'object_subtype' => 'meeting', 311 | 'type' => 'string', 312 | 'single' => true, 313 | 'show_in_rest' => true, 314 | ) 315 | ); 316 | } 317 | // 'occurrence' is an array of strings 318 | register_meta( 319 | 'post', 320 | 'occurrence', 321 | array( 322 | 'object_subtype' => 'meeting', 323 | 'type' => 'array', 324 | 'single' => true, 325 | 'show_in_rest' => array( 326 | 'schema' => array( 327 | 'type' => 'array', 328 | 'items' => array( 329 | 'type' => 'integer', 330 | ), 331 | ), 332 | ), 333 | ) 334 | ); 335 | } 336 | 337 | public function register_rest_field() { 338 | register_rest_field( 339 | 'meeting', 340 | 'future_occurrences', 341 | array( 342 | 'get_callback' => array( $this, 'get_future_occurrences' ), 343 | ) 344 | ); 345 | } 346 | 347 | public function is_meeting_cancelled( $meeting_id, $date ) { 348 | // Note: this assumes the meeting does occur on $date 349 | $cancellations = get_post_meta( $meeting_id, 'meeting_cancelled', false ); 350 | return in_array( $date, $cancellations, true ); 351 | } 352 | 353 | public function get_occurrences_for_period( $request ) { 354 | $meetings = get_posts( 355 | array( 356 | 'post_type' => 'meeting', 357 | 'post_status' => 'publish', 358 | 'numberposts' => -1, 359 | ) 360 | ); 361 | $out = array(); 362 | foreach ( $meetings as $meeting ) { 363 | $occurrences = $this->get_future_occurrences( $meeting, null, $request ); 364 | 365 | $frequency = ''; 366 | if ( ! empty( $meeting->recurring ) && ! empty( $occurrences ) ) { 367 | $frequency = get_frequency( $meeting->recurring, $occurrences[0], $meeting->occurrence ); 368 | } 369 | 370 | foreach ( $occurrences as $occurrence ) { 371 | $meeting->time = gmdate( 'H:i:s', strtotime( $meeting->time ) ); 372 | $out[] = array( 373 | 'meeting_id' => $meeting->ID, 374 | 'instance_id' => "{$meeting->ID}:{$occurrence}", 375 | 'date' => $occurrence, 376 | 'time' => $meeting->time, 377 | 'datetime' => "{$occurrence}T{$meeting->time}+00:00", 378 | 'team' => ucwords( $meeting->team ), 379 | 'link' => $meeting->link, 380 | 'title' => wp_specialchars_decode( $meeting->post_title, ENT_QUOTES ), 381 | 'location' => $meeting->location, 382 | 'wptv_url' => $meeting->wptv_url, 383 | 'recurring' => $meeting->recurring, 384 | 'occurrence' => $meeting->occurrence, 385 | 'status' => ( $this->is_meeting_cancelled( $meeting->ID, $occurrence ) ? 'cancelled' : 'active' ), 386 | 'rrule' => $frequency ? "RRULE:FREQ={$frequency}" : '', 387 | ); 388 | } 389 | } 390 | 391 | usort( 392 | $out, 393 | function( $a, $b ) { 394 | return $a['datetime'] <=> $b['datetime']; 395 | } 396 | ); 397 | return $out; 398 | } 399 | 400 | public function register_rest_routes() { 401 | register_rest_route( 402 | 'wp/v2/meetings', 403 | '/from/(?P\d\d\d\d-\d\d-\d\d)', 404 | array( 405 | 'methods' => 'GET', 406 | 'callback' => array( $this, 'get_occurrences_for_period' ), 407 | 'permission_callback' => '__return_true', 408 | ) 409 | ); 410 | register_rest_route( 411 | 'wp/v2/meetings', 412 | '/(?P\d+):(?P\d\d\d\d-\d\d-\d\d)', 413 | array( 414 | array( 415 | 'methods' => 'DELETE', 416 | 'callback' => array( $this, 'cancel_meeting' ), 417 | 'permission_callback' => array( $this, 'can_cancel_meeting' ), 418 | ), 419 | array( 420 | 'methods' => 'PUT', 421 | 'callback' => array( $this, 'uncancel_meeting' ), 422 | 'permission_callback' => array( $this, 'can_cancel_meeting' ), 423 | ), 424 | ) 425 | ); 426 | } 427 | 428 | public function cancel_meeting( $request ) { 429 | // TODO: should validate that the meeting does occur on the given date 430 | return add_post_meta( $request['meeting_id'], 'meeting_cancelled', $request['date'], false ); 431 | } 432 | 433 | public function uncancel_meeting( $request ) { 434 | return delete_post_meta( $request['meeting_id'], 'meeting_cancelled', $request['date'] ); 435 | } 436 | 437 | public function can_cancel_meeting( $request ) { 438 | return current_user_can( 'edit_post', $request['meeting_id'] ); 439 | } 440 | 441 | public function get_future_occurrences( $meeting, $attr, $request ) { 442 | if ( is_array( $meeting ) && ! empty( $meeting['id'] ) ) { 443 | // The register_rest_field callback passes a prepared array but we need the post object 444 | $meeting = get_post( $meeting['id'] ); 445 | } 446 | 447 | // The month the occurrences should be within. 448 | // Passed to the API endpoint as /meetings?month=2020-09-01 449 | $now = time(); 450 | if ( isset( $request['month'] ) ) { 451 | $now = strtotime( $request['month'] ); 452 | } 453 | 454 | $from = DateTime::createFromFormat( 'U', strtotime( '-30 minutes', $now ) ); 455 | $end = DateTime::createFromFormat( 'U', strtotime( '+2 month', $now ) ); 456 | if ( $meeting->end_date ) { 457 | $end = DateTime::createFromFormat( 'Y-m-d', $meeting->end_date ); 458 | } 459 | $max = 12; 460 | $occurrences = array(); 461 | do { 462 | $next = $this->get_next_occurrence( $meeting, $from->format( 'Y-m-d H:i:s P' ) ); 463 | if ( $next ) { 464 | $from = new DateTime( "{$next} {$meeting->time}" ); 465 | if ( $from <= $end ) { 466 | $occurrences[] = $next; 467 | } 468 | } 469 | } while ( --$max > 0 && $next && $from && $from < $end && $meeting->recurring ); 470 | 471 | return $occurrences; 472 | } 473 | 474 | public function add_meta_boxes() { 475 | add_meta_box( 476 | 'meeting-info', 477 | 'Meeting Info', 478 | array( $this, 'render_meta_boxes' ), 479 | 'meeting', 480 | 'normal', 481 | 'high' 482 | ); 483 | add_meta_box( 484 | 'upcoming-meetings', 485 | 'Upcoming Meetings', 486 | array( $this, 'render_meta_upcoming' ), 487 | 'meeting', 488 | 'normal', 489 | 'high' 490 | ); 491 | } 492 | 493 | public function render_meta_boxes( $post ) { 494 | wp_enqueue_script( 'jquery-ui-datepicker' ); 495 | wp_enqueue_style( 'jquery-ui-style', 'https://ajax.googleapis.com/ajax/libs/jqueryui/1.12.1/themes/smoothness/jquery-ui.css', true, '1.12.1' ); 496 | 497 | $meta = get_post_custom( $post->ID ); 498 | $team = isset( $meta['team'][0] ) ? $meta['team'][0] : ''; 499 | $start = isset( $meta['start_date'][0] ) ? $meta['start_date'][0] : ''; 500 | $end = isset( $meta['end_date'][0] ) ? $meta['end_date'][0] : ''; 501 | $time = isset( $meta['time'][0] ) ? $meta['time'][0] : ''; 502 | $recurring = isset( $meta['recurring'][0] ) ? $meta['recurring'][0] : ''; 503 | if ( '1' === $recurring ) { 504 | $recurring = 'weekly'; 505 | } 506 | $occurrence = isset( $meta['occurrence'][0] ) ? unserialize( $meta['occurrence'][0] ) : array(); 507 | $link = isset( $meta['link'][0] ) ? $meta['link'][0] : ''; 508 | $location = isset( $meta['location'][0] ) ? $meta['location'][0] : ''; 509 | $wptv_url = isset( $meta['wptv_url'][0] ) ? $meta['wptv_url'][0] : ''; 510 | wp_nonce_field( 'save_meeting_meta_' . $post->ID, 'meeting_nonce' ); 511 | ?> 512 | 513 |

514 | 518 |

519 |

520 | 524 |

525 |

526 | 530 |

531 |

532 |
533 |
537 |
541 | 542 |
546 | 547 | 551 | 555 | 559 | 563 |
567 | 568 | 572 |

573 | 574 |

575 | 579 |

580 |

581 | 584 |

585 |

586 | 589 |

590 |

591 | 592 | 599 |

600 | 601 | 621 | get_future_occurrences( $meeting, null, null ); 626 | if ( count( $occurrences ) ) { 627 | ?> 628 |
    629 | is_meeting_cancelled( $meeting->ID, $occurrence ); 633 | ?> 634 |
  • 635 | 646 |
  • 647 | 650 |
651 | 655 |

656 | 660 | 661 | 693 | post_type ) && 'revision' === $post->post_type ) { 711 | return $post_id; 712 | } 713 | 714 | // Check permissions 715 | if ( ! current_user_can( 'edit_post', $post->ID ) ) { 716 | return $post_id; 717 | } 718 | 719 | // Basic validation 720 | if ( empty( trim( $_POST['post_title'] ) ) ) { 721 | return false; 722 | } 723 | if ( empty( trim( $_POST['start_date'] ) ) ) { 724 | return false; 725 | } 726 | if ( false === strtotime( $_POST['start_date'] ) ) { 727 | return false; 728 | } 729 | if ( ! empty( trim( $_POST['end_date'] ) ) && false === strtotime( $_POST['end_date'] ) ) { 730 | return false; 731 | } 732 | if ( empty( trim( $_POST['time'] ) ) ) { 733 | return false; 734 | } 735 | if ( false === strtotime( $_POST['time'] ) ) { 736 | return false; 737 | } 738 | 739 | $meta['team'] = ( isset( $_POST['team'] ) ? esc_textarea( $_POST['team'] ) : '' ); 740 | $meta['start_date'] = ( isset( $_POST['start_date'] ) ? esc_textarea( $_POST['start_date'] ) : '' ); 741 | $meta['end_date'] = ( isset( $_POST['end_date'] ) ? esc_textarea( $_POST['end_date'] ) : '' ); 742 | $meta['time'] = ( isset( $_POST['time'] ) ? esc_textarea( $_POST['time'] ) : '' ); 743 | $meta['recurring'] = ( isset( $_POST['recurring'] ) 744 | && in_array( $_POST['recurring'], array( 'weekly', 'biweekly', 'occurrence', 'monthly' ) ) 745 | ? ( $_POST['recurring'] ) : '' ); 746 | $meta['occurrence'] = ( isset( $_POST['occurrence'] ) && 'occurrence' === $meta['recurring'] 747 | && is_array( $_POST['occurrence'] ) 748 | ? array_map( 'intval', $_POST['occurrence'] ) : array() ); 749 | $meta['link'] = ( isset( $_POST['link'] ) ? esc_url( $_POST['link'] ) : '' ); 750 | $meta['location'] = ( isset( $_POST['location'] ) ? esc_textarea( $_POST['location'] ) : '' ); 751 | $meta['wptv_url'] = ( isset( $_POST['wptv_url'] ) ? esc_url( $_POST['wptv_url'] ) : '' ); 752 | 753 | // Non-recurring events should not have a end_date set. 754 | if ( ! $meta['recurring'] && $meta['end_date'] ) { 755 | $meta['end_date'] = ''; 756 | } 757 | 758 | foreach ( $meta as $key => $value ) { 759 | update_post_meta( $post->ID, $key, $value ); 760 | } 761 | } 762 | 763 | /** 764 | * Adds "Edit Meetings" item after "Add New" menu. 765 | * 766 | * @param \WP_Admin_Bar $wp_admin_bar The admin bar instance. 767 | */ 768 | public function add_edit_meetings_item_to_admin_bar( $wp_admin_bar ) { 769 | if ( ! current_user_can( 'edit_posts' ) ) { 770 | return; 771 | } 772 | 773 | if ( is_admin() || ! is_post_type_archive( 'meeting' ) ) { 774 | return; 775 | } 776 | 777 | $wp_admin_bar->add_menu( 778 | array( 779 | 'id' => 'edit-meetings', 780 | 'title' => '' . __( 'Edit Meetings', 'wporg-meeting-calendar' ), 781 | 'href' => admin_url( 'edit.php?post_type=meeting' ), 782 | ) 783 | ); 784 | } 785 | 786 | /** 787 | * Adds icon for the "Edit Meetings" item. 788 | */ 789 | public function add_edit_meetings_icon_to_admin_bar() { 790 | if ( ! current_user_can( 'edit_posts' ) ) { 791 | return; 792 | } 793 | 794 | wp_add_inline_style( 795 | 'admin-bar', 796 | ' 797 | #wpadminbar #wp-admin-bar-edit-meetings .ab-icon:before { 798 | content: "\f145"; 799 | top: 2px; 800 | } 801 | ' 802 | ); 803 | } 804 | 805 | /** 806 | * Renders meeting information with the next meeting time based on user's local timezone. Used in Make homepage. 807 | */ 808 | public function meeting_time_shortcode( $attr, $content = '' ) { 809 | $attr = shortcode_atts( 810 | array( 811 | 'team' => null, 812 | 'limit' => 1, 813 | 'before' => __( 'Next meeting: ', 'wporg-meeting-calendar' ), 814 | 'titletag' => 'strong', 815 | 'more' => true, 816 | ), 817 | $attr 818 | ); 819 | 820 | if ( empty( $attr['team'] ) ) { 821 | return ''; 822 | } 823 | 824 | if ( 'Documentation' === $attr['team'] ) { 825 | $attr['team'] = 'Docs'; 826 | } 827 | 828 | if ( ! has_action( 'wp_footer', array( $this, 'time_conversion_script' ) ) ) { 829 | add_action( 'wp_footer', array( $this, 'time_conversion_script' ), 999 ); 830 | } 831 | 832 | // If we're on a network, assume the calendar exists on the main site 833 | if ( function_exists( 'switch_to_blog' ) ) { 834 | switch_to_blog( get_main_site_id() ); 835 | } 836 | 837 | $query = new WP_Query( 838 | array( 839 | 'post_type' => 'meeting', 840 | 'nopaging' => true, 841 | 'meta_query' => array( 842 | 'relation' => 'AND', 843 | array( 844 | 'key' => 'team', 845 | 'value' => $attr['team'], 846 | 'compare' => 'EQUALS', 847 | ), 848 | $this->meeting_meta_query(), 849 | ), 850 | ) 851 | ); 852 | 853 | $limit = $attr['limit'] > 0 ? $attr['limit'] : count( $query->posts ); 854 | 855 | $out = ''; 856 | foreach ( array_slice( $query->posts, 0, $limit ) as $post ) { 857 | $next_meeting_datestring = $post->next_date; 858 | $utc_time = strftime( '%H:%M:%S', strtotime( $post->time ) ); 859 | $next_meeting_iso = $next_meeting_datestring . 'T' . $utc_time . '+00:00'; 860 | $next_meeting_timestamp = strtotime( $next_meeting_datestring . ' ' . $utc_time ); 861 | $date_time = new DateTime( '@' . $next_meeting_timestamp ); 862 | $date_time->setTimezone( new DateTimeZone( 'UTC' ) ); 863 | $next_meeting_display = $date_time->format( 'D M d H:i:s Y T' ); 864 | 865 | $slack_channel = null; 866 | if ( $post->location && preg_match( '/^#([-\w]+)$/', trim( $post->location ), $match ) ) { 867 | $slack_channel = sanitize_title( $match[1] ); 868 | } 869 | 870 | $cancelled = $this->is_meeting_cancelled( $post->ID, $post->next_date ); 871 | 872 | $out .= '

'; 873 | $out .= ''; 874 | 875 | $out .= esc_html( $attr['before'] ); 876 | $out .= '' . esc_html( $post->post_title ) . ''; 877 | $display_more = $query->found_posts - intval( $limit ); 878 | if ( $display_more > 0 ) { 879 | $out .= ' ' . sprintf( __( '(+%s more)', 'wporg-meeting-calendar' ), $display_more ) . ''; 880 | } 881 | $out .= '
'; 882 | $out .= ' '; 883 | $out .= sprintf( esc_html__( '(%s from now)', 'wporg-meeting-calendar' ), human_time_diff( $next_meeting_timestamp, current_time( 'timestamp' ) ) ); 884 | if ( $post->location && $slack_channel ) { 885 | $out .= ' ' . sprintf( wp_kses( __( 'accessible via %2$s on Slack', 'wporg-meeting-calendar' ), array( 'a' => array( 'href' => array() ) ) ), 'https://wordpress.slack.com/messages/' . $slack_channel, $post->location ); 886 | } 887 | $out .= '
'; 888 | 889 | if ( $cancelled ) { 890 | $out .= '
'; 891 | $future_occurrences = $this->get_future_occurrences( $post, null, array() ); 892 | $next_meeting = null; 893 | foreach ( $future_occurrences as $occurrence ) { 894 | if ( ! $this->is_meeting_cancelled( $post->ID, $occurrence ) 895 | && $occurrence > $post->next_date ) { 896 | $next_meeting = $occurrence; 897 | break; 898 | } 899 | } 900 | if ( $next_meeting ) { 901 | $out .= '' . sprintf( esc_html__( 'This event is cancelled. The next meeting is scheduled for %s.', 'wporg-meeting-calendar' ), $next_meeting ) . ''; 902 | } else { 903 | $out .= '' . esc_html__( 'This event is cancelled.', 'wporg-meeting-calendar' ) . ''; 904 | } 905 | } 906 | $out .= '

'; 907 | } 908 | 909 | if ( function_exists( 'restore_current_blog' ) ) { 910 | restore_current_blog(); 911 | } 912 | 913 | return $out; 914 | } 915 | 916 | public function meeting_meta_query() { 917 | return array( 918 | 'relation' => 'OR', 919 | // not recurring AND start_date >= CURDATE() = one-time meeting today or still in future 920 | array( 921 | 'relation' => 'AND', 922 | array( 923 | 'key' => 'recurring', 924 | 'value' => array( 'weekly', 'biweekly', 'occurrence', 'monthly', '1' ), 925 | 'compare' => 'NOT IN', 926 | ), 927 | array( 928 | 'key' => 'start_date', 929 | 'type' => 'DATE', 930 | 'compare' => '>=', 931 | 'value' => gmdate( 'Y-m-d' ), 932 | ), 933 | ), 934 | // recurring = 1 AND ( end_date = '' OR end_date > CURDATE() ) = recurring meeting that has no end or has not ended yet 935 | array( 936 | 'relation' => 'AND', 937 | array( 938 | 'key' => 'recurring', 939 | 'value' => array( 'weekly', 'biweekly', 'occurrence', 'monthly', '1' ), 940 | 'compare' => 'IN', 941 | ), 942 | array( 943 | 'relation' => 'OR', 944 | array( 945 | 'key' => 'end_date', 946 | 'value' => '', 947 | 'compare' => '=', 948 | ), 949 | array( 950 | 'key' => 'end_date', 951 | 'type' => 'DATE', 952 | 'compare' => '>', 953 | 'value' => gmdate( 'Y-m-d' ), 954 | ), 955 | ), 956 | ), 957 | ); 958 | } 959 | 960 | public function time_conversion_script() { 961 | echo << 963 | 964 | var parse_date = function (text) { 965 | var m = /^([0-9]{4})-([0-9]{2})-([0-9]{2})T([0-9]{2}):([0-9]{2}):([0-9]{2})\+00:00$/.exec(text); 966 | var d = new Date(); 967 | d.setUTCFullYear(+m[1]); 968 | d.setUTCDate(+m[3]); 969 | d.setUTCMonth(+m[2]-1); 970 | d.setUTCHours(+m[4]); 971 | d.setUTCMinutes(+m[5]); 972 | d.setUTCSeconds(+m[6]); 973 | return d; 974 | } 975 | var format_time = function (d) { 976 | return d.toLocaleTimeString(navigator.language, {weekday: 'long', hour: '2-digit', minute: '2-digit', timeZoneName: 'short'}); 977 | } 978 | 979 | var nodes = document.getElementsByTagName('time'); 980 | for (var i=0; i 990 | EOF; 991 | } 992 | } 993 | 994 | // fire it up 995 | Meeting_Post_Type::init(); 996 | endif; 997 | 998 | 999 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "meeting-calendar", 3 | "version": "1.0.0", 4 | "description": "A meeting calendar for WordPress.org", 5 | "main": "index.js", 6 | "scripts": { 7 | "build": "wp-scripts build index=./src/index.js calendar=./src/frontend/index.js", 8 | "check-engines": "wp-scripts check-engines", 9 | "check-licenses": "wp-scripts check-licenses", 10 | "format:js": "wp-scripts format-js", 11 | "lint:css": "wp-scripts lint-style", 12 | "lint:js": "wp-scripts lint-js", 13 | "lint:md:docs": "wp-scripts lint-md-docs", 14 | "lint:md:js": "wp-scripts lint-md-js", 15 | "lint:pkg-json": "wp-scripts lint-pkg-json", 16 | "packages-update": "wp-scripts packages-update", 17 | "start": "wp-scripts start index=./src/index.js calendar=./src/frontend/index.js", 18 | "test:e2e": "wp-scripts test-e2e", 19 | "test:unit": "wp-scripts test-unit-js", 20 | "test:unit-php": "wp-env run tests-cli phpunit -c /var/www/html/wp-content/plugins/meeting-calendar/phpunit.xml.dist --verbose", 21 | "wp-env": "wp-env" 22 | }, 23 | "repository": { 24 | "type": "git", 25 | "url": "git+https://github.com/WordPress/meeting-calendar.git" 26 | }, 27 | "author": "", 28 | "license": "GPL-2.0-or-later", 29 | "bugs": { 30 | "url": "https://github.com/WordPress/meeting-calendar/issues" 31 | }, 32 | "homepage": "https://github.com/WordPress/meeting-calendar#readme", 33 | "devDependencies": { 34 | "@wordpress/env": "10.13.0", 35 | "@wordpress/scripts": "30.6.0", 36 | "@wordpress/stylelint-config": "23.5.0" 37 | } 38 | } 39 | -------------------------------------------------------------------------------- /phpunit.xml.dist: -------------------------------------------------------------------------------- 1 | 2 | 10 | 11 | 12 | ./tests/ 13 | ./tests/test-sample.php 14 | 15 | 16 | 17 | -------------------------------------------------------------------------------- /plugin.php: -------------------------------------------------------------------------------- 1 | set_query_params( array( 'per_page' => $per_page ) ); 25 | $response = rest_do_request( $request ); 26 | $server = rest_get_server(); 27 | $data = $server->response_to_data( $response, false ); 28 | return wp_json_encode( $data ); 29 | } 30 | 31 | /** 32 | * Render the block content (html) on the frontend of the site. 33 | * 34 | * @param array $attributes 35 | * @param string $content 36 | * @return string HTML output used by the calendar JS. 37 | */ 38 | function render_callback( $attributes, $content ) { 39 | $meetings = get_meeting_data( 12 ); 40 | return sprintf( 41 | '
Loading Calendar ...
', 42 | 'wporg-meeting-calendar-js', 43 | htmlspecialchars( $meetings, ENT_QUOTES ) 44 | ); 45 | } 46 | 47 | /** 48 | * Register scripts, styles, and block. 49 | */ 50 | function register_assets() { 51 | $block_deps_path = __DIR__ . '/build/index.asset.php'; 52 | $frontend_deps_path = __DIR__ . '/build/calendar.asset.php'; 53 | if ( ! file_exists( $block_deps_path ) || ! file_exists( $frontend_deps_path ) ) { 54 | return; 55 | } 56 | 57 | $block_info = require $block_deps_path; 58 | $frontend_info = require $frontend_deps_path; 59 | 60 | // Register our block script with WordPress. 61 | wp_register_script( 62 | 'wporg-calendar-block-script', 63 | plugins_url( 'build/index.js', __FILE__ ), 64 | $block_info['dependencies'], 65 | $block_info['version'], 66 | false 67 | ); 68 | 69 | // Add translation support. 70 | wp_set_script_translations( 'wporg-calendar-block-script', 'wporg-meeting-calendar' ); 71 | 72 | // Register our block's base CSS. 73 | wp_register_style( 74 | 'wporg-calendar-block-style', 75 | plugins_url( 'style.css', __FILE__ ), 76 | array(), 77 | $block_info['version'] 78 | ); 79 | 80 | // No frontend scripts in the editor 81 | if ( ! is_admin() ) { 82 | wp_register_script( 83 | 'wporg-calendar-script', 84 | plugin_dir_url( __FILE__ ) . 'build/calendar.js', 85 | $frontend_info['dependencies'], 86 | $frontend_info['version'], 87 | false 88 | ); 89 | 90 | // Add translation support. 91 | wp_set_script_translations( 'wporg-calendar-script', 'wporg-meeting-calendar' ); 92 | 93 | wp_register_style( 94 | 'wporg-calendar-style', 95 | plugin_dir_url( __FILE__ ) . 'build/style-calendar.css', 96 | array( 'wp-components' ), 97 | $frontend_info['version'] 98 | ); 99 | } 100 | 101 | // Enqueue the script in the editor. 102 | register_block_type( 103 | 'wporg-meeting-calendar/main', 104 | array( 105 | 'editor_script' => 'wporg-calendar-block-script', 106 | 'editor_style' => 'wporg-calendar-block-style', 107 | 'script' => 'wporg-calendar-script', 108 | 'style' => 'wporg-calendar-style', 109 | 'render_callback' => __NAMESPACE__ . '\render_callback', 110 | ) 111 | ); 112 | } 113 | add_action( 'init', __NAMESPACE__ . '\register_assets' ); 114 | 115 | /** 116 | * Conditionally remove the Script/Style assets added through `register_block_type()`. 117 | */ 118 | function conditionally_load_assets() { 119 | if ( ! is_singular() || ! has_block( 'wporg-meeting-calendar/main' ) ) { 120 | wp_dequeue_script( 'wporg-calendar-script' ); 121 | wp_dequeue_style( 'wporg-calendar-style' ); 122 | } 123 | } 124 | add_action( 'enqueue_block_assets', __NAMESPACE__ . '\conditionally_load_assets' ); 125 | 126 | /** 127 | * Set up the Meetings post type. 128 | */ 129 | function init() { 130 | require_once __DIR__ . '/includes/wporg-meeting-posttype.php'; 131 | new \Meeting_Post_Type(); 132 | } 133 | add_action( 'plugins_loaded', __NAMESPACE__ . '\init' ); 134 | 135 | /** 136 | * Set up the ICS support. 137 | */ 138 | function ics_init() { 139 | require __DIR__ . '/includes/ical-functions.php'; 140 | require __DIR__ . '/includes/ical-generator-functions.php'; 141 | } 142 | add_action( 'plugins_loaded', __NAMESPACE__ . '\ics_init' ); 143 | 144 | /** 145 | * First-Install activation hook. 146 | * 147 | * Creates some sample data, and sets up the Rewrite rules. 148 | */ 149 | function install( $is_network_wide ) { 150 | if ( ! $is_network_wide ) { 151 | // We need the CPT to be registered to install. 152 | init(); 153 | $meeting_post_type = new \Meeting_Post_Type(); 154 | $meeting_post_type->getInstance()->register_meeting_post_type(); 155 | 156 | require_once __DIR__ . '/includes/wporg-meeting-install.php'; 157 | wporg_meeting_install(); 158 | 159 | ics_init(); 160 | ICS\on_activate(); 161 | } 162 | } 163 | register_activation_hook( __FILE__, __NAMESPACE__ . '\install' ); 164 | -------------------------------------------------------------------------------- /readme.txt: -------------------------------------------------------------------------------- 1 | === Meeting Calendar === 2 | Contributors: tellyworth,dufresnesteven,dd32,ryelle 3 | Author URI: https://make.wordpress.org/meta/ 4 | Tags: meeting,calendar 5 | Requires at least: 4.5 6 | Tested up to: 5.2.1 7 | Stable tag: 0.0.0 8 | License: GPLv2 or later 9 | License URI: https://www.gnu.org/licenses/gpl-2.0.html 10 | 11 | Meeting calendar as used on Make.WordPress.org 12 | 13 | == Description == 14 | 15 | This provides a way of scheduling recurring meetings, and displaying a calendar or timetable. 16 | 17 | == Installation == 18 | 19 | 1. Activate the `Meeting Calendar` plugin in your WordPress plugin directory 20 | 2. Create some meetings 21 | 3. While editing your page/post, add in the `Meeting Calendar` block and publish! 22 | 23 | == Frequently Asked Questions == 24 | 25 | == Screenshots == 26 | 27 | == Changelog == 28 | 29 | = 0.1 = 30 | * Initial public release. 31 | 32 | -------------------------------------------------------------------------------- /src/edit.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { __ } from '@wordpress/i18n'; 5 | import { Placeholder } from '@wordpress/components'; 6 | 7 | const EditView = () => ( 8 | 12 | ); 13 | export default EditView; 14 | -------------------------------------------------------------------------------- /src/frontend/app/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Internal dependencies 3 | */ 4 | import { ViewProvider } from '../store/view-context'; 5 | import { EventsProvider } from '../store/event-context'; 6 | import useWindowSize from '../store/hooks/use-window-size'; 7 | import Calendar from '../calendar'; 8 | 9 | function App( { events } ) { 10 | const { isSmall } = useWindowSize(); 11 | 12 | return ( 13 | 14 | 15 | 16 | 17 | 18 | ); 19 | } 20 | 21 | export default App; 22 | -------------------------------------------------------------------------------- /src/frontend/calendar/cell.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { _n, sprintf } from '@wordpress/i18n'; 5 | import { Button, Dropdown, MenuGroup, MenuItem } from '@wordpress/components'; 6 | import { format } from '@wordpress/date'; 7 | 8 | /** 9 | * Internal dependencies 10 | */ 11 | import { getTeamClass, isToday, isCancelled, isUpcoming } from './utils'; 12 | 13 | function CalendarCell({ 14 | blank = false, 15 | year, 16 | month, 17 | day, 18 | events, 19 | onEventClick, 20 | }) { 21 | const MAX_EVENTS = 25; 22 | if (blank) { 23 | return ; 24 | } 25 | 26 | const date = new Date(year, month, day); 27 | const key = format('Y-m-d', date); 28 | const dayEvents = events[key] || []; 29 | const restOfEvents = dayEvents.slice(MAX_EVENTS); 30 | 31 | return ( 32 | 35 | 36 | 37 | {format('F j', date)}{' '} 38 | {sprintf( 39 | // translators: %d: Count of all events, ie: 4. 40 | _n( 41 | '%d event', 42 | '%d events', 43 | dayEvents.length, 44 | 'wporg-meeting-calendar' 45 | ), 46 | dayEvents.length 47 | )} 48 | 49 | 50 | {day} 51 | 52 | 53 | {dayEvents.slice(0, MAX_EVENTS).map((event) => { 54 | return ( 55 | 68 | ); 69 | })} 70 | 71 | {!!restOfEvents.length && ( 72 | ( 76 | 92 | )} 93 | renderContent={() => ( 94 | 95 | {restOfEvents.map((event) => { 96 | return ( 97 | void onEventClick(event)} 101 | className={ 102 | 'wporg-meeting-calendar__cell-event ' + 103 | getTeamClass(event.team) + 104 | (isCancelled(event.status) 105 | ? ' is-cancelled' 106 | : '') 107 | } 108 | > 109 | {format('g:i a: ', event.datetime)} 110 | {event.title} 111 | 112 | ); 113 | })} 114 | 115 | )} 116 | /> 117 | )} 118 | 119 | ); 120 | } 121 | 122 | export default CalendarCell; 123 | -------------------------------------------------------------------------------- /src/frontend/calendar/grid.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { Fragment, useState } from '@wordpress/element'; 5 | 6 | /** 7 | * Internal dependencies 8 | */ 9 | import CalendarCell from './cell'; 10 | import CalendarHeader from './header'; 11 | import EventModal from './modal'; 12 | import { getRows } from './utils'; 13 | import { useEvents } from '../store/event-context'; 14 | 15 | function CalendarGrid( { month, year } ) { 16 | const [ activeEvent, setActiveEvent ] = useState( null ); 17 | const rows = getRows( year, month ); 18 | const { events } = useEvents(); 19 | 20 | return ( 21 | 22 | 23 | 24 | 25 | { rows.map( ( row, i ) => ( 26 | 27 | { row.map( ( day, index ) => ( 28 | 34 | ) ) } 35 | 36 | ) ) } 37 | 38 |
39 | { activeEvent && ( 40 | void setActiveEvent( null ) } 43 | /> 44 | ) } 45 |
46 | ); 47 | } 48 | 49 | export default CalendarGrid; 50 | -------------------------------------------------------------------------------- /src/frontend/calendar/header.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { __experimentalGetSettings } from '@wordpress/date'; 5 | 6 | function CalendarHeader() { 7 | const { l10n } = __experimentalGetSettings(); 8 | return ( 9 | 10 | 11 | { l10n.weekdaysShort.map( ( day, i ) => { 12 | return ( 13 | 14 | 15 | { l10n.weekdays[ i ] } 16 | 17 | { day } 18 | 19 | ); 20 | } ) } 21 | 22 | 23 | ); 24 | } 25 | 26 | export default CalendarHeader; 27 | -------------------------------------------------------------------------------- /src/frontend/calendar/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { __ } from '@wordpress/i18n'; 5 | import { Button } from '@wordpress/components'; 6 | import { date } from '@wordpress/date'; 7 | import { Fragment, useState } from '@wordpress/element'; 8 | import { speak } from '@wordpress/a11y'; 9 | 10 | /** 11 | * Internal dependencies 12 | */ 13 | import CalendarGrid from './grid'; 14 | import List from '../list'; 15 | import Filter from '../filter'; 16 | import Feed from '../feed'; 17 | import { useViews } from '../store/view-context'; 18 | import { 19 | list as ListIcon, 20 | calendar as CalendarIcon, 21 | arrow as ArrowIcon, 22 | } from '../icons'; 23 | 24 | function Calendar() { 25 | const today = new Date(); 26 | const currentMonth = today.getMonth(); 27 | const currentYear = today.getFullYear(); 28 | const currentMonthYear = { 29 | month: currentMonth, 30 | year: currentYear, 31 | }; 32 | const [{ month, year }, setDate] = useState(currentMonthYear); 33 | const { 34 | isCalendarView, 35 | isListView, 36 | setCalendarView, 37 | setListView, 38 | shouldForceListView, 39 | } = useViews(); 40 | 41 | if (shouldForceListView && !isListView()) { 42 | setListView(); 43 | } 44 | 45 | return ( 46 | 47 |
48 | 72 |
73 |

74 | {date('F Y', new Date(year, month, 2))} 75 |

76 |
77 | 121 |
122 | 123 | {isCalendarView() && } 124 | {isListView() && } 125 | 126 |
127 | ); 128 | } 129 | 130 | export default Calendar; 131 | -------------------------------------------------------------------------------- /src/frontend/calendar/modal.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { __ } from '@wordpress/i18n'; 5 | import { addQueryArgs } from '@wordpress/url'; 6 | import { format, gmdate } from '@wordpress/date'; 7 | import { Modal, Notice } from '@wordpress/components'; 8 | 9 | /** 10 | * Internal dependencies 11 | */ 12 | import { getFrequencyLabel, getSlackLink, isCancelled } from './utils'; 13 | 14 | function EventModal( { event, onRequestClose } ) { 15 | const start = gmdate( 'Ymd\\THis\\Z', event.datetime ); 16 | const endTimestamp = Number( gmdate( 'U', event.datetime ) ) + 3600; 17 | const end = gmdate( 'Ymd\\THis\\Z', endTimestamp * 1000 ); 18 | 19 | const channel = event.location.replace( '#', '' ); 20 | let googleCalLink = addQueryArgs( 21 | 'https://calendar.google.com/calendar/render', 22 | { 23 | action: 'TEMPLATE', 24 | text: event.title, 25 | dates: `${ start }/${ end }`, 26 | details: `Location: #${ channel } on Slack - https://wordpress.slack.com/app_redirect?channel=${ channel } `, 27 | } 28 | ); 29 | if ( event.rrule ) { 30 | googleCalLink = addQueryArgs( googleCalLink, { recur: event.rrule } ); 31 | } 32 | 33 | return ( 34 | 40 | { ! isCancelled( event.status ) ? ( 41 |

42 | 43 | { format( 44 | 'l, F j, Y, g:i a (\\U\\T\\CP)', 45 | event.datetime 46 | ) } 47 | 48 |

49 | ) : ( 50 | 55 | { __( 56 | 'This meeting has been cancelled', 57 | 'wporg-meeting-calendar' 58 | ) } 59 | 60 | ) } 61 | 62 | { !! event.location && ( 63 |

Location: { getSlackLink( event.location ) }

64 | ) } 65 |

Meets: { getFrequencyLabel( event ) }

66 | { !! event.link && ( 67 |

68 | { event.title } 69 |

70 | ) } 71 |

72 | 73 | { __( 'Add to Google Calendar', 'wporg-meeting-calendar' ) } 74 | 75 |

76 | { !! event.wptv_url && ( 77 |

78 | 79 | { __( 'View Recording', 'wporg-meeting-calendar' ) } 80 | 81 |

82 | ) } 83 |
84 | ); 85 | } 86 | 87 | export default EventModal; 88 | -------------------------------------------------------------------------------- /src/frontend/calendar/utils.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { __, sprintf } from '@wordpress/i18n'; 5 | import { format } from '@wordpress/date'; 6 | 7 | // Default value for days not in this month. 8 | const emptyDate = { 9 | blank: true, 10 | }; 11 | 12 | /** 13 | * Get days in the given month in a 2-dimensional array of [week][day]. 14 | * 15 | * @param {number} year 16 | * @param {number} month 17 | */ 18 | export function getRows(year, month) { 19 | const daysInWeek = 7; 20 | const firstOffset = new Date(year, month, 1).getDay(); // Get day of the week. 21 | const monthLength = new Date(year, month + 1, 0).getDate(); // 0 gets the last day of the next month. 22 | const days = []; 23 | 24 | for (let i = 0; i < firstOffset; i++) { 25 | days.push(emptyDate); 26 | } 27 | for (let i = 1; i <= monthLength; i++) { 28 | days.push({ 29 | month, 30 | year, 31 | day: i, 32 | }); 33 | } 34 | 35 | const rows = []; 36 | for (let i = 0; i < Math.ceil(days.length / daysInWeek); i++) { 37 | const start = i * daysInWeek; 38 | let row = days.slice(start, start + daysInWeek); 39 | if (row.length !== daysInWeek) { 40 | row = [ 41 | ...row, 42 | ...Array( daysInWeek - row.length ).fill( emptyDate ), 43 | ]; 44 | } 45 | rows.push(row); 46 | } 47 | 48 | return rows; 49 | } 50 | 51 | /** 52 | * Get human-friendly reccurance string. 53 | * 54 | * @param {Object} event 55 | */ 56 | export function getFrequencyLabel(event) { 57 | const occurrences = { 58 | 1: __('1st', 'wporg-meeting-calendar'), 59 | 2: __('2nd', 'wporg-meeting-calendar'), 60 | 3: __('3rd', 'wporg-meeting-calendar'), 61 | 4: __('4th', 'wporg-meeting-calendar'), 62 | }; 63 | const dayOfWeek = format('l', event.datetime); 64 | 65 | switch (event.recurring) { 66 | case 'weekly': 67 | return sprintf( __( 'Every week on %s', 'wporg-meeting-calendar' ), dayOfWeek ); 68 | 69 | case 'biweekly': 70 | return sprintf( 71 | __('Every other week on %s', 'wporg-meeting-calendar'), 72 | dayOfWeek 73 | ); 74 | 75 | case 'monthly': 76 | return __('Every month', 'wporg-meeting-calendar'); 77 | 78 | case 'occurrence': 79 | if (event.occurrence.length) { 80 | return sprintf( 81 | __('Every month on the %s %s', 'wporg-meeting-calendar'), 82 | event.occurrence 83 | .map( ( o ) => occurrences[ o ] ) 84 | .join( ', ' ), 85 | dayOfWeek 86 | ); 87 | } 88 | return ''; 89 | 90 | default: 91 | return __('Does not repeat', 'wporg-meeting-calendar'); 92 | } 93 | } 94 | 95 | /** 96 | * Get link to the slack channel. 97 | * 98 | * @param {string} location 99 | */ 100 | export function getSlackLink(location) { 101 | if (location[0] === '#') { 102 | location = location.slice(1); 103 | } 104 | 105 | return ( 106 | 109 | #{location} 110 | 111 | ); 112 | } 113 | 114 | /** 115 | * Get a classname based on team name 116 | * 117 | * @param {string} team 118 | */ 119 | export function getTeamClass(team) { 120 | return ( 121 | 'wporg-meeting-calendar__team-' + 122 | team.replace([' ', '.'], '-').toLowerCase() 123 | ); 124 | } 125 | 126 | /** 127 | * Checks whether a date is today 128 | * 129 | * @param {Object} date 130 | */ 131 | export function isToday(date) { 132 | const today = new Date(); 133 | return ( 134 | date.getDate() == today.getDate() && 135 | date.getMonth() == today.getMonth() && 136 | date.getFullYear() == today.getFullYear() 137 | ); 138 | } 139 | 140 | /** 141 | * Returns whether the event is cancelled 142 | * 143 | * @param {string} status Status of the event Ie: active, cancelled 144 | */ 145 | export function isCancelled(status) { 146 | return 'cancelled' === status; 147 | } 148 | 149 | /** 150 | * Checks whether a date is upcoming 151 | * 152 | * @param {string} eventDatetime The date and time of the event in UTC 153 | */ 154 | export function isUpcoming(eventDatetime) { 155 | return new Date(eventDatetime) > new Date(); 156 | } 157 | -------------------------------------------------------------------------------- /src/frontend/feed/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { __ } from '@wordpress/i18n'; 5 | import { addQueryArgs } from '@wordpress/url'; 6 | 7 | /** 8 | * Internal dependencies 9 | */ 10 | import { useEvents } from '../store/event-context'; 11 | 12 | const Feed = () => { 13 | const { teams, team } = useEvents(); 14 | const selected = teams.find((option) => team === option.value); 15 | 16 | const getCalendarUrl = () => { 17 | const baseUrl = window.location.origin; 18 | if (!selected) { 19 | return `${baseUrl}/meetings.ics`; 20 | } 21 | const urlSafeTeam = encodeURIComponent( selected.value ); 22 | return `${baseUrl}/meetings-${urlSafeTeam}.ics`; 23 | }; 24 | 25 | const getGoogleCalendarUrl = () => { 26 | const calendarUrl = getCalendarUrl().replace('https://', 'webcal://'); 27 | return addQueryArgs('https://www.google.com/calendar/render', { 28 | cid: calendarUrl, 29 | }); 30 | }; 31 | 32 | const getRSSUrl = () => { 33 | const baseUrl = window.location.origin; 34 | 35 | return `${baseUrl}/feed/?post_type=meeting`; 36 | }; 37 | 38 | return ( 39 |
40 |

41 | {__( 42 | 'Events are shown in your local time zone.', 43 | 'wporg-meeting-calendar' 44 | )} 45 |

46 |

47 | {__('Subscribe to this calendar:', 'wporg-meeting-calendar')}{' '} 48 | 49 | {__('Google Calendar ↗', 'wporg-meeting-calendar')} 50 | {' '} 51 | ·{' '} 52 | 53 | {__('ICS', 'wporg-meeting-calendar')} 54 | {' '} 55 | ·{' '} 56 | {__('RSS', 'wporg-meeting-calendar')} 57 |

58 |
59 | ); 60 | }; 61 | 62 | export default Feed; 63 | -------------------------------------------------------------------------------- /src/frontend/filter/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { __, sprintf } from '@wordpress/i18n'; 5 | import { Button, SelectControl } from '@wordpress/components'; 6 | import { speak } from '@wordpress/a11y'; 7 | import { useRef } from '@wordpress/element'; 8 | 9 | /** 10 | * Internal dependencies 11 | */ 12 | import { useEvents } from '../store/event-context'; 13 | 14 | const Filter = () => { 15 | const { teams, team, setTeam } = useEvents(); 16 | const filterLabel = useRef(null); 17 | 18 | if (teams.length < 2) { 19 | return null; 20 | } 21 | 22 | // Sort the teams alphabetically. 23 | teams.sort( (a, b) => a.label.localeCompare( b.label ) ); 24 | 25 | const dropdownId = 'wporg-meeting-calendar__filter-dropdown'; 26 | const selected = teams.find((option) => team === option.value); 27 | 28 | return ( 29 |
30 | 37 | { 49 | setTeam(value); 50 | const newSelected = teams.find( 51 | (option) => value === option.value 52 | ); 53 | speak( 54 | sprintf( 55 | // translators: %s is the team name 56 | __( 57 | 'Showing meetings for %s', 58 | 'wporg-meeting-calendar' 59 | ), 60 | newSelected.label 61 | ), 62 | 'assertive' 63 | ); 64 | }} 65 | /> 66 | {'' !== team && ( 67 | <> 68 |

69 | {sprintf( 70 | // translators: %s is the team name 71 | __( 72 | 'Showing meetings for %s', 73 | 'wporg-meeting-calendar' 74 | ), 75 | selected.label 76 | )} 77 |

78 | 99 | 100 | )} 101 |
102 | ); 103 | }; 104 | 105 | export default Filter; 106 | -------------------------------------------------------------------------------- /src/frontend/icons/arrow.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { SVG, Path } from '@wordpress/primitives'; 5 | 6 | export default function () { 7 | return ( 8 | 15 | 20 | 21 | ); 22 | } 23 | -------------------------------------------------------------------------------- /src/frontend/icons/calendar.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { SVG, Path } from '@wordpress/primitives'; 5 | 6 | export default function () { 7 | return ( 8 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 26 | 27 | ); 28 | } 29 | -------------------------------------------------------------------------------- /src/frontend/icons/collapse.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { SVG, Path } from '@wordpress/primitives'; 5 | 6 | export default function () { 7 | return ( 8 | 14 | 15 | 16 | 17 | ); 18 | } 19 | -------------------------------------------------------------------------------- /src/frontend/icons/expand.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { SVG, Path } from '@wordpress/primitives'; 5 | 6 | export default function () { 7 | return ( 8 | 14 | 15 | 16 | 17 | ); 18 | } 19 | -------------------------------------------------------------------------------- /src/frontend/icons/index.js: -------------------------------------------------------------------------------- 1 | export { default as arrow } from './arrow'; 2 | export { default as calendar } from './calendar'; 3 | export { default as collapse } from './collapse'; 4 | export { default as expand } from './expand'; 5 | export { default as list } from './list'; 6 | -------------------------------------------------------------------------------- /src/frontend/icons/list.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { SVG, Path } from '@wordpress/primitives'; 5 | 6 | export default function () { 7 | return ( 8 | 15 | 20 | 21 | ); 22 | } 23 | -------------------------------------------------------------------------------- /src/frontend/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { createElement, render } from '@wordpress/element'; 5 | 6 | /** 7 | * Internal dependencies 8 | */ 9 | import App from './app'; 10 | import './style.scss'; 11 | 12 | const getMeetings = ( calendarEl ) => { 13 | return JSON.parse( calendarEl.getAttribute( 'data-meetings' ) ); 14 | }; 15 | 16 | const initCalendar = () => { 17 | const calendarEl = document.getElementById( 'wporg-meeting-calendar-js' ); 18 | if ( ! calendarEl ) { 19 | return; 20 | } 21 | const events = getMeetings( calendarEl ); 22 | render( createElement( App, { events } ), calendarEl ); 23 | }; 24 | 25 | document.addEventListener( 'DOMContentLoaded', initCalendar ); 26 | -------------------------------------------------------------------------------- /src/frontend/list/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * External dependencies 3 | */ 4 | import { flatten } from 'lodash'; 5 | 6 | /** 7 | * WordPress dependencies 8 | */ 9 | import { __ } from '@wordpress/i18n'; 10 | import { Button } from '@wordpress/components'; 11 | import { format } from '@wordpress/date'; 12 | import { useState } from '@wordpress/element'; 13 | 14 | /** 15 | * Internal dependencies 16 | */ 17 | import { collapse, expand } from '../icons'; 18 | import { getRows } from '../calendar/utils'; 19 | import { useEvents } from '../store/event-context'; 20 | import ListItem from './list-item'; 21 | 22 | function List({ month, year }) { 23 | const [showPast, setShowPast] = useState(false); 24 | const rows = getRows(year, month); 25 | const { events } = useEvents(); 26 | const allDays = flatten(rows).filter((i) => !i.blank); 27 | const cutoffDate = new Date(); 28 | // Reset the time to 11:59pm the night before, so that we show all meetings today. 29 | cutoffDate.setHours(-1, 59, 0, 0); 30 | // If today is later than the 1st of the month, there will be past meetings. 31 | const hasPast = new Date(year, month) < cutoffDate; 32 | 33 | const days = allDays 34 | .map((row, i) => { 35 | const date = new Date(row.year, row.month, row.day); 36 | const key = format('Y-m-d', date); 37 | const dayEvents = events[key] || []; 38 | 39 | // If we want to hide past events, skip over things before yesterday. 40 | if (!showPast && !(date > cutoffDate)) { 41 | return null; 42 | } 43 | 44 | if (!dayEvents.length) { 45 | return null; 46 | } 47 | 48 | return ; 49 | }) 50 | .filter((i) => !!i); 51 | 52 | return ( 53 | <> 54 | {hasPast && 55 | (!showPast ? ( 56 |

57 | 60 |

61 | ) : ( 62 |

63 | 69 |

70 | ))} 71 | 72 | {!days.length ? ( 73 |

74 | {__('No Events Scheduled', 'wporg-meeting-calendar')} 75 |

76 | ) : ( 77 |
    {days}
78 | )} 79 | 80 | ); 81 | } 82 | 83 | export default List; 84 | -------------------------------------------------------------------------------- /src/frontend/list/list-item.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { __, sprintf } from '@wordpress/i18n'; 5 | import { format } from '@wordpress/date'; 6 | import { speak } from '@wordpress/a11y'; 7 | 8 | /** 9 | * Internal dependencies 10 | */ 11 | import { 12 | getTeamClass, 13 | getSlackLink, 14 | getFrequencyLabel, 15 | isCancelled, 16 | } from '../calendar/utils'; 17 | import { useEvents } from '../store/event-context'; 18 | 19 | function ListItem({ date, events }) { 20 | const { setTeam } = useEvents(); 21 | 22 | return ( 23 |
  • 24 |

    25 | {format('l - F j, Y', date)} 26 |

    27 | 28 | {events.map((event) => { 29 | const onTeamClick = (clickEvent) => { 30 | clickEvent.preventDefault(); 31 | setTeam(event.team); 32 | speak( 33 | sprintf( 34 | // translators: %s is the team name 35 | __( 36 | 'Showing meetings for %s', 37 | 'wporg-meeting-calendar' 38 | ), 39 | event.team 40 | ) 41 | ); 42 | }; 43 | return ( 44 |
    50 | {event.team && ( 51 | 71 | )} 72 |
    73 |

    74 | {!!event.link ? ( 75 | 76 | 77 | 78 | ) : ( 79 | 80 | )} 81 |

    82 |
    83 | {format('g:i a ', event.datetime)} 84 | {format('(\\U\\T\\CP)', date)} 85 |
    86 |
    87 |
    88 |

    89 | {__('Meets: ', 'wporg-meeting-calendar')} 90 | {getFrequencyLabel(event)} 91 |

    92 |

    93 | {__('Location: ', 'wporg-meeting-calendar')} 94 | {getSlackLink(event.location)} 95 |

    96 |
    97 | {!!event.wptv_url && ( 98 | 114 | )} 115 |
    116 | ); 117 | })} 118 |
  • 119 | ); 120 | } 121 | 122 | function EventTitle({ event }) { 123 | return ( 124 | <> 125 | {event.title} 126 | {isCancelled(event.status) && ( 127 | 128 | {__(' Meeting is cancelled', 'wporg-meeting-calendar')} 129 | 130 | )} 131 | 132 | ); 133 | } 134 | 135 | export default ListItem; 136 | -------------------------------------------------------------------------------- /src/frontend/store/event-context.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { createContext, useContext, useState } from '@wordpress/element'; 5 | import { uniqBy } from 'lodash'; 6 | 7 | /** 8 | * Internal dependencies 9 | */ 10 | import { getSortedEvents } from './utils'; 11 | 12 | /** 13 | * Gets the team name if present in url. 14 | */ 15 | function getTeamOnLoad() { 16 | const { location } = window; 17 | const matches = location.href.match( /#(.+)/ ); 18 | return matches ? decodeURI( matches[ 1 ] ) : ''; 19 | } 20 | 21 | /** 22 | * Add the team to the current URL, pushing to history for browser navigation support. 23 | * 24 | * @param {string} team 25 | */ 26 | function setTeamEffect( team = '' ) { 27 | if ( '' === team ) { 28 | window.history.pushState( 29 | team, 30 | document.title, 31 | window.location.pathname 32 | ); 33 | } else { 34 | window.history.pushState( team, document.title, '#' + team ); 35 | } 36 | } 37 | 38 | const StateContext = createContext(); 39 | 40 | export function EventsProvider( { children, value } ) { 41 | const [ team, setTeam ] = useState( getTeamOnLoad() ); 42 | 43 | let eventsToDisplay = value; 44 | 45 | // Get a list of all teams available. 46 | const teams = uniqBy( 47 | value 48 | .map( ( e ) => ( { 49 | label: e.team, 50 | value: e.team.toLowerCase(), 51 | } ) ) 52 | .filter( ( { value } ) => !! value ), 53 | 'value' 54 | ); 55 | 56 | // Validate the team is valid. 57 | if ( team && team.trim().length ) { 58 | const teamExists = teams.find( 59 | ( t ) => t.value.toLowerCase() === team.toLowerCase() 60 | ); 61 | if ( ! teamExists ) { 62 | team = ''; 63 | setTeam( '' ); 64 | } 65 | } 66 | 67 | // Filter the initial events list. 68 | if ( team && team.trim().length ) { 69 | eventsToDisplay = value.filter( 70 | ( e ) => e.team.toLowerCase() === team.toLowerCase() 71 | ); 72 | } 73 | 74 | const initialState = { 75 | events: getSortedEvents( eventsToDisplay ), 76 | team, 77 | teams, 78 | setTeam: ( newTeam ) => { 79 | newTeam = newTeam.toLowerCase(); 80 | setTeam( newTeam ); 81 | setTeamEffect( newTeam ); 82 | }, 83 | }; 84 | 85 | return ( 86 | 87 | { children } 88 | 89 | ); 90 | } 91 | 92 | export function useEvents() { 93 | const context = useContext( StateContext ); 94 | if ( context === undefined ) { 95 | throw new Error( 'useEvents must be used within a Provider' ); 96 | } 97 | return context; 98 | } 99 | -------------------------------------------------------------------------------- /src/frontend/store/hooks/use-window-size.js: -------------------------------------------------------------------------------- 1 | /** 2 | * External dependencies 3 | */ 4 | import { useState, useEffect } from 'react'; 5 | 6 | const breakpoints = { 7 | small: 600, 8 | }; 9 | 10 | const useWindowSize = () => { 11 | const isClient = typeof window === 'object'; 12 | 13 | function getSize() { 14 | return { 15 | isSmall: isClient 16 | ? window.innerWidth < breakpoints.small 17 | : undefined, 18 | }; 19 | } 20 | 21 | const [ windowSize, setWindowSize ] = useState( getSize ); 22 | 23 | useEffect( () => { 24 | if ( ! isClient ) { 25 | return false; 26 | } 27 | 28 | const handleResize = () => { 29 | setWindowSize( getSize() ); 30 | }; 31 | 32 | window.addEventListener( 'resize', handleResize ); 33 | return () => window.removeEventListener( 'resize', handleResize ); 34 | }, [] ); 35 | 36 | return windowSize; 37 | }; 38 | 39 | export default useWindowSize; 40 | -------------------------------------------------------------------------------- /src/frontend/store/utils.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { format } from '@wordpress/date'; 5 | 6 | /** 7 | * Get the list of events in day-buckets. 8 | * 9 | * @param {Array} events 10 | */ 11 | export function getSortedEvents( events ) { 12 | const sortedEvents = {}; 13 | events.forEach( ( event ) => { 14 | const d = new Date( event.datetime ); 15 | const key = format( 'Y-m-d', d ); 16 | if ( sortedEvents.hasOwnProperty( key ) ) { 17 | sortedEvents[ key ].push( event ); 18 | } else { 19 | sortedEvents[ key ] = [ event ]; 20 | } 21 | } ); 22 | return sortedEvents; 23 | } 24 | -------------------------------------------------------------------------------- /src/frontend/store/view-context.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { createContext, useContext, useState } from '@wordpress/element'; 5 | 6 | const StateContext = createContext(); 7 | 8 | const CALENDAR_VIEW = 'calendar_view'; 9 | const LIST_VIEW = 'list_view'; 10 | 11 | export function ViewProvider( { children, isSmallViewport } ) { 12 | return ( 13 | 19 | { children } 20 | 21 | ); 22 | } 23 | 24 | export function useViews() { 25 | const context = useContext( StateContext ); 26 | const isView = ( toMatch ) => currentView === toMatch; 27 | 28 | if ( context === undefined ) { 29 | throw new Error( 'useViews must be used within a Provider' ); 30 | } 31 | 32 | const [ currentView, setView ] = useState( context.defaultState ); 33 | 34 | return { 35 | isCalendarView: () => isView( CALENDAR_VIEW ), 36 | isListView: () => isView( LIST_VIEW ), 37 | setCalendarView: () => setView( CALENDAR_VIEW ), 38 | setListView: () => setView( LIST_VIEW ), 39 | shouldForceListView: context.isSmallViewport, 40 | }; 41 | } 42 | -------------------------------------------------------------------------------- /src/frontend/style.scss: -------------------------------------------------------------------------------- 1 | /* stylelint-disable selector-class-pattern */ 2 | @use "sass:color"; 3 | 4 | .wporg-block-meeting-calendar { 5 | margin: 0 auto 32px; 6 | font-size: var(--wp--preset--font-size--normal, 16px); 7 | } 8 | 9 | .wporg-meeting-calendar__filter { 10 | display: flex; 11 | flex-wrap: wrap; 12 | align-items: center; 13 | margin: var(--wp--preset--spacing--20, 20px) 0; 14 | 15 | .components-base-control__field { 16 | margin-bottom: unset; 17 | } 18 | } 19 | 20 | .wporg-meeting-calendar__filter-dropdown { 21 | max-width: none; 22 | 23 | /** Overwriting a default rule on the dropdown container */ 24 | > div { 25 | margin: 0 auto 0 !important; 26 | } 27 | } 28 | 29 | .wporg-meeting-calendar__filter-remove.components-button.is-link { 30 | text-decoration: none; 31 | 32 | span:not(.dashicons) { 33 | text-decoration: underline; 34 | } 35 | } 36 | 37 | @media ( max-width: 782px ) { 38 | 39 | .wporg-meeting-calendar__filter { 40 | display: block; 41 | } 42 | 43 | .wporg-meeting-calendar__filter p { 44 | margin: 1em 0 0.5em; 45 | } 46 | 47 | .wporg-meeting-calendar__filter-dropdown { 48 | max-width: 200px; 49 | } 50 | } 51 | 52 | .wporg-meeting-calendar__filter-label { 53 | padding-right: 6px; 54 | margin-bottom: 0; 55 | } 56 | 57 | .wporg-meeting-calendar__filter-applied { 58 | font-size: var(--wp--preset--font-size--normal, 16px); 59 | margin: 0 1em; 60 | } 61 | 62 | .wporg-meeting-calendar__filter-label, 63 | .wporg-meeting-calendar__filter-applied { 64 | font-size: var(--wp--preset--font-size--small, 14px); 65 | } 66 | 67 | .wporg-meeting-calendar__header { 68 | display: flex; 69 | align-items: center; 70 | margin-bottom: 18px; 71 | } 72 | 73 | .wporg-meeting-calendar__header div { 74 | flex: 1; 75 | padding: 16px 0; 76 | } 77 | 78 | @media ( min-width: 550px ) { 79 | 80 | .wporg-meeting-calendar__header { 81 | flex-direction: row; 82 | } 83 | 84 | .wporg-meeting-calendar__header div { 85 | padding: 0; 86 | } 87 | } 88 | 89 | .wporg-meeting-calendar__header div:last-child { 90 | text-align: right; 91 | } 92 | 93 | .wporg-meeting-calendar__header h2, 94 | .wp-site-blocks .wporg-meeting-calendar__header h2:not([class*="-font-size"], [style*="font-size"]) { 95 | margin: 0 0 0 var(--wp--preset--spacing--10, 10px); 96 | font-size: var(--wp--preset--font-size--heading-4, 20px); 97 | } 98 | 99 | .wporg-meeting-calendar__header h2::before { 100 | display: none; 101 | } 102 | 103 | .wporg-meeting-calendar__btn-group { 104 | display: flex; 105 | gap: 4px; 106 | 107 | .components-button { 108 | width: 40px; 109 | height: 40px; 110 | 111 | &:disabled { 112 | background-color: var(--wp--preset--color--light-grey-2, #f6f6f6); 113 | box-shadow: none !important; 114 | color: var(--wp--preset--color--charcoal-1, #1e1e1e); 115 | 116 | svg { 117 | opacity: 0.5; 118 | } 119 | } 120 | 121 | &:last-child svg { 122 | rotate: 180deg; 123 | } 124 | } 125 | } 126 | 127 | .wporg-meeting-calendar__header .components-button-group { 128 | display: flex; 129 | 130 | .components-button { 131 | display: flex; 132 | align-items: center; 133 | justify-content: center; 134 | background-color: unset; 135 | width: 40px; 136 | height: 40px; 137 | padding: unset; 138 | position: relative; 139 | border-radius: 2px; 140 | margin-left: unset; 141 | 142 | &:not(:focus) { 143 | box-shadow: unset; 144 | } 145 | 146 | &:disabled { 147 | display: none; 148 | } 149 | 150 | &.is-primary { 151 | background-color: var(--wp--preset--color--charcoal-1, #1e1e1e); 152 | } 153 | 154 | &.is-secondary { 155 | color: var(--wp--preset--color--charcoal-1, #1e1e1e); 156 | } 157 | } 158 | } 159 | 160 | .wporg-meeting-calendar__cell-event.components-button { 161 | display: block; 162 | position: relative; 163 | padding: 8px; 164 | margin: 8px 0; 165 | font-size: var(--wp--preset--font-size--small, 14px); 166 | border-radius: 2px; 167 | width: 100%; 168 | 169 | &.is-link { 170 | border: 1px solid var(--wp-components-color-accent, var(--wp-admin-theme-color, #3858e9)); 171 | background: var(--wp--preset--color--blueberry-4, #eff2ff); 172 | text-decoration: none; 173 | } 174 | } 175 | 176 | .wporg-meeting-calendar__cell-event-title { 177 | display: block; 178 | overflow: hidden; 179 | } 180 | 181 | .wporg-meeting-calendar__cell-event.is-cancelled, 182 | .wporg-meeting-calendar__cell-event.is-cancelled .wporg-meeting-calendar__cell-event-title, 183 | .wporg-meeting-calendar__dropdown.is-cancelled { 184 | text-decoration: line-through !important; 185 | } 186 | 187 | .wporg-meeting-calendar__cell-event:focus { 188 | text-decoration: underline; 189 | box-shadow: 0 0 0 1px var(--wp-components-color-accent, var(--wp-admin-theme-color, #3858e9)); 190 | } 191 | 192 | .wporg-meeting-calendar__dropdown .components-menu-group { 193 | padding: 7px; 194 | box-sizing: border-box; 195 | } 196 | 197 | .wporg-meeting-calendar__dropdown .components-menu-item__button { 198 | white-space: nowrap; 199 | overflow: hidden; 200 | } 201 | 202 | .wporg-meeting-calendar__cell-event-time { 203 | font-weight: 700; 204 | margin-bottom: 4px; 205 | } 206 | 207 | .wporg-meeting-calendar__list-expand { 208 | margin: var(--wp--preset--spacing--20, 20px) 0 0; 209 | text-align: center; 210 | } 211 | 212 | .wporg-meeting-calendar__list { 213 | padding: 0; 214 | margin: 0; 215 | list-style: none; 216 | } 217 | 218 | .wporg-meeting-calendar__list-title { 219 | display: block; 220 | font-size: var(--wp--preset--font-size--medium, 18px); 221 | font-weight: 600; 222 | margin: var(--wp--preset--spacing--30, 30px) 0 var(--wp--preset--spacing--20, 20px); 223 | } 224 | 225 | .wporg-meeting-calendar__list li { 226 | list-style: none; 227 | margin: 0; 228 | 229 | &:first-child .wporg-meeting-calendar__list-title { 230 | margin-top: 0; 231 | } 232 | } 233 | 234 | .wporg-meeting-calendar__list-event { 235 | position: relative; 236 | display: flex; 237 | flex-direction: column; 238 | gap: var(--wp--preset--spacing--20, 20px); 239 | background: var(--wp--preset--color--white, #fff); 240 | border: 1px solid var(--wp--preset--color--light-grey-1, #d9d9d9); 241 | padding: var(--wp--preset--spacing--20, 20px); 242 | 243 | &:first-of-type { 244 | border-top-left-radius: 2px; 245 | border-top-right-radius: 2px; 246 | } 247 | 248 | &:last-of-type { 249 | border-bottom-left-radius: 2px; 250 | border-bottom-right-radius: 2px; 251 | } 252 | 253 | + .wporg-meeting-calendar__list-event { 254 | border-top: unset; 255 | } 256 | 257 | @media (min-width: 600px) { 258 | align-items: center; 259 | flex-direction: row; 260 | } 261 | } 262 | 263 | .wporg-meeting-calendar__list-event-time { 264 | font-size: var(--wp--preset--font-size--small, 14px); 265 | color: var(--wp--preset--color--charcoal-4, #656a71); 266 | } 267 | 268 | .wporg-meeting-calendar__list-event-team-wrapper { 269 | flex-basis: 15%; 270 | } 271 | 272 | .wporg-meeting-calendar__list-event-team { 273 | display: inline-block; 274 | font-size: var(--wp--preset--font-size--small, 14px); 275 | text-align: center; 276 | padding: 7px 12px; 277 | margin-bottom: 8px; 278 | border-radius: 2px; 279 | text-decoration: none !important; 280 | background: var(--wp--preset--color--blueberry-4, #eff2ff); 281 | color: var(--wp--preset--color--blueberry-1, #3858e9) !important; 282 | line-height: var(--wp--custom--heading--level-1--typography--line-height, 1.3); 283 | } 284 | 285 | .wporg-meeting-calendar__list-event-team:hover, 286 | .wporg-meeting-calendar__list-event-team:active, 287 | .wporg-meeting-calendar__list-event-team:focus { 288 | text-decoration: underline !important; 289 | } 290 | 291 | .wporg-meeting-calendar__list-event-title { 292 | font-size: var(--wp--preset--font-size--normal, 16px); 293 | font-weight: 400; 294 | margin: unset !important; 295 | padding-bottom: unset; 296 | border-bottom: unset; 297 | 298 | a { 299 | color: var(--wp--preset--color--charcoal-1, #1e1e1e); 300 | text-decoration: none !important; 301 | 302 | &:hover { 303 | text-decoration: underline !important; 304 | } 305 | } 306 | } 307 | 308 | .wporg-meeting-calendar__list-event-copy { 309 | margin-top: 0 !important; 310 | margin-bottom: 4px; 311 | 312 | &:last-child { 313 | margin-bottom: 0; 314 | } 315 | } 316 | 317 | .wporg-meeting-calendar__list-event.is-cancelled .wporg-meeting-calendar__list-event-title a > span:first-child, 318 | .wporg-meeting-calendar__list-event.is-cancelled .wporg-meeting-calendar__list-event-copy { 319 | text-decoration: line-through; 320 | } 321 | 322 | .wporg-meeting-calendar__list-event.is-cancelled .wporg-meeting-calendar__list-event-title a > span:last-child { 323 | padding-left: 8px; 324 | font-size: 16px; 325 | font-weight: 400; 326 | vertical-align: bottom; 327 | } 328 | 329 | .wporg-meeting-calendar__list-event-header { 330 | flex: 1; 331 | } 332 | 333 | .wporg-meeting-calendar__list-event-details { 334 | flex-basis: 35%; 335 | } 336 | 337 | .wporg-meeting-calendar__header, 338 | .wporg-meeting-calendar__list { 339 | padding: 12px 0; 340 | } 341 | 342 | @media (min-width: 960px) { 343 | 344 | .wporg-meeting-calendar__header, 345 | .wporg-meeting-calendar__list { 346 | padding: 0; 347 | } 348 | } 349 | 350 | .wporg-block-meeting-calendar table { 351 | table-layout: fixed; 352 | margin-top: 0; /* There is a global style that is adding a bottom margin to tables */ 353 | margin-bottom: 0; /* There is a global style that is adding a bottom margin to tables */ 354 | width: 100%; 355 | border-collapse: collapse; 356 | background: #fff; 357 | } 358 | 359 | .wporg-block-meeting-calendar table th { 360 | text-align: center; 361 | font-weight: 400; 362 | font-size: var(--wp--preset--font-size--normal, 16px); 363 | padding: 8px; 364 | border: 1px solid var(--wp--preset--color--light-grey-1, #d9d9d9); 365 | } 366 | 367 | .wporg-meeting-calendar__cell-day { 368 | display: inline-block; 369 | width: 30px; 370 | height: 100%; 371 | } 372 | 373 | .wporg-meeting-calendar__cell { 374 | border: 1px solid var(--wp--preset--color--light-grey-1, #d9d9d9); 375 | background-color: var(--wp--preset--color--light-grey-2, #f6f6f6); 376 | padding: var(--wp--preset--spacing--10, 10px); 377 | vertical-align: top; 378 | height: 180px; /* height acts as min-height on table cells */ 379 | 380 | strong { 381 | display: block; 382 | height: 30px; 383 | line-height: 30px; 384 | text-align: center; 385 | font-weight: 400; 386 | font-size: var(--wp--preset--font-size--normal, 16px); 387 | color: var(--wp--preset--color--charcoal-4, #656a71); 388 | } 389 | 390 | &.is-today, 391 | &.is-upcoming { 392 | background-color: var(--wp--preset--color--white, #fff); 393 | } 394 | 395 | &.is-today .wporg-meeting-calendar__cell-day { 396 | background-color: var(--wp-components-color-accent, var(--wp-admin-theme-color, #3858e9)); 397 | border-radius: 50%; 398 | color: var(--wp--preset--color--white, #fff); 399 | } 400 | 401 | &.is-upcoming strong { 402 | color: var(--wp--preset--color--charcoal-1, #1e1e1e); 403 | } 404 | } 405 | 406 | /** Colors for each team */ 407 | .wporg-meeting-calendar__team-core { 408 | border-color: #cd0000 !important; 409 | background-color: color.adjust(#cd0000, $lightness: 55%) !important; 410 | color: #cd0000 !important; 411 | } 412 | 413 | .wporg-meeting-calendar__team-design { 414 | border-color: #eec26a !important; 415 | background-color: color.adjust(#eec26a, $lightness: 30%) !important; 416 | color: color.adjust(#eec26a, $lightness: -40%) !important; 417 | } 418 | 419 | .wporg-meeting-calendar__team-mobile { 420 | border-color: #fba16c !important; 421 | background-color: color.adjust(#fba16c, $lightness: 30%) !important; 422 | color: #fba16c !important; 423 | } 424 | 425 | .wporg-meeting-calendar__team-polyglots { 426 | border-color: #c32283 !important; 427 | background-color: color.adjust(#c32283, $lightness: 50%) !important; 428 | color: color.adjust(#c32283, $lightness: -10%) !important; 429 | } 430 | 431 | .wporg-meeting-calendar__team-support { 432 | border-color: #33b4ce !important; 433 | background-color: color.adjust(#33b4ce, $lightness: 40%) !important; 434 | color: color.adjust(#33b4ce, $lightness: -20%) !important; 435 | } 436 | 437 | .wporg-meeting-calendar__team-docs, 438 | .wporg-meeting-calendar__team-documentation { 439 | border-color: #3b7236 !important; 440 | background-color: color.adjust(#3b7236, $lightness: 55%) !important; 441 | color: #3b7236 !important; 442 | } 443 | 444 | .wporg-meeting-calendar__team-themes { 445 | border-color: #4e3288 !important; 446 | background-color: color.adjust(#4e3288, $lightness: 55%) !important; 447 | color: #4e3288 !important; 448 | } 449 | 450 | .wporg-meeting-calendar__team-plugins { 451 | border-color: #f06723 !important; 452 | background-color: color.adjust(#f06723, $lightness: 55%) !important; 453 | color: #f06723 !important; 454 | } 455 | 456 | .wporg-meeting-calendar__team-accessibility, 457 | .wporg-meeting-calendar__team-community { 458 | border-color: #11799d !important; 459 | background-color: color.adjust(#11799d, $lightness: 60%) !important; 460 | color: color.adjust(#11799d, $lightness: -10%) !important; 461 | } 462 | 463 | .wporg-meeting-calendar__team-meta { 464 | border-color: #aeadad !important; 465 | background-color: color.adjust(#aeadad, $lightness: 25%) !important; 466 | color: color.adjust(#aeadad, $lightness: -30%) !important; 467 | } 468 | 469 | .wporg-meeting-calendar__team-training, 470 | .wporg-meeting-calendar__team-openverse { 471 | border-color: #e9c02d !important; 472 | background-color: color.adjust(#e9c02d, $lightness: 40%) !important; 473 | color: color.adjust(#e9c02d, $lightness: -25%) !important; 474 | } 475 | 476 | .wporg-meeting-calendar__team-tv { 477 | border-color: #73ad30 !important; 478 | background-color: color.adjust(#73ad30, $lightness: 45%) !important; 479 | color: color.adjust(#73ad30, $lightness: -20%) !important; 480 | } 481 | 482 | .wporg-meeting-calendar__team-marketing { 483 | border-color: #47bea7 !important; 484 | background-color: color.adjust(#47bea7, $lightness: 55%) !important; 485 | color: #47bea7 !important; 486 | } 487 | 488 | .wporg-meeting-calendar__team-cli { 489 | border-color: #424242 !important; 490 | background-color: color.adjust(#424242, $lightness: 55%) !important; 491 | color: #424242 !important; 492 | } 493 | 494 | .wporg-meeting-calendar__team-hosting { 495 | border-color: #5358a6 !important; 496 | background-color: color.adjust(#5358a6, $lightness: 45%) !important; 497 | color: #5358a6 !important; 498 | } 499 | 500 | .wporg-meeting-calendar__team-tide { 501 | border-color: #1526ff !important; 502 | background-color: color.adjust(#1526ff, $lightness: 40%) !important; 503 | color: #1526ff !important; 504 | } 505 | 506 | .wporg-meeting-calendar__team-bbpress, 507 | .wporg-meeting-calendar__team-sustainability { 508 | border-color: #2d8e42 !important; 509 | background-color: color.adjust(#2d8e42, $lightness: 55%) !important; 510 | color: color.adjust(#2d8e42, $lightness: -10%) !important; 511 | } 512 | 513 | .wporg-meeting-calendar__team-buddypress, 514 | .wporg-meeting-calendar__team-test { 515 | border-color: #d84800 !important; 516 | background-color: color.adjust(#d84800, $lightness: 50%) !important; 517 | color: color.adjust(#d84800, $lightness: -10%) !important; 518 | } 519 | 520 | .wporg-meeting-calendar__cell-event:not(.is-upcoming) { 521 | opacity: 0.58; 522 | color: #000 !important; 523 | } 524 | 525 | .wporg-meeting-calendar__modal { 526 | border-radius: unset; 527 | box-shadow: unset; 528 | 529 | .components-modal__content { 530 | padding: 0 var(--wp--preset--spacing--30, 30px) var(--wp--preset--spacing--30, 30px); 531 | } 532 | 533 | .components-modal__header .components-modal__header-heading { 534 | font-family: var(--wp--preset--font-family--inter, sans-serif); 535 | line-height: var(--wp--custom--heading--level-1--typography--line-height, 1.3); 536 | } 537 | } 538 | 539 | .wporg-meeting-calendar__modal h1 { 540 | font-size: 16px !important; 541 | } 542 | 543 | .wporg-meeting-calendar__modal h1::before { 544 | display: none !important; 545 | } 546 | 547 | .wporg-meeting-calendar__modal-overlay { 548 | font-size: var(--wp--preset--font-size--small, 14px); 549 | z-index: 1000001; /* popover z-index + 1 */ 550 | } 551 | 552 | .wporg-meeting-calendar__modal-notice { 553 | margin: 0 0 16px 0; 554 | } 555 | 556 | .wporg-meeting-calendar__modal-export-links { 557 | margin-top: 1em; 558 | } 559 | 560 | .meeting-cancelled .wporg-meeting-detail { 561 | text-decoration: line-through; 562 | } 563 | 564 | .wporg-meeting-calendar__feed { 565 | display: flex; 566 | justify-content: space-between; 567 | font-size: var(--wp--preset--font-size--small, 14px); 568 | } 569 | -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * WordPress dependencies 3 | */ 4 | import { registerBlockType } from '@wordpress/blocks'; 5 | 6 | /** 7 | * Internal dependencies 8 | */ 9 | import edit from './edit'; 10 | 11 | registerBlockType( 'wporg-meeting-calendar/main', { 12 | title: 'Meeting Calendar', 13 | icon: 'calendar', 14 | category: 'widgets', 15 | attributes: {}, 16 | edit, 17 | save: () => null, 18 | } ); 19 | -------------------------------------------------------------------------------- /style.css: -------------------------------------------------------------------------------- 1 | /* stylelint-disable no-empty-source */ 2 | -------------------------------------------------------------------------------- /tests/bootstrap.php: -------------------------------------------------------------------------------- 1 | server = $wp_rest_server = new \WP_REST_Server(); 22 | do_action( 'rest_api_init' ); 23 | 24 | // Install test data 25 | $this->meeting_ids = Meeting_Calendar\wporg_meeting_install(); 26 | 27 | // Make sure the meta keys are registered - setUp/tearDown nukes these 28 | Meeting_Post_Type::getInstance()->register_meta(); 29 | Meeting_Post_Type::getInstance()->register_rest_routes(); 30 | } 31 | 32 | 33 | /** 34 | * A single example test. 35 | */ 36 | public function test_sample() { 37 | // Replace this with some actual testing code. 38 | $this->assertTrue( true ); 39 | } 40 | 41 | 42 | public function test_register_route() { 43 | $routes = $this->server->get_routes(); 44 | // Standard route for the CPT 45 | $this->assertArrayHasKey( '/wp/v2/meeting', $routes ); 46 | // Main endpoint for listing events 47 | $this->assertArrayHasKey( '/wp/v2/meetings/from/(?P\d\d\d\d-\d\d-\d\d)', $routes ); 48 | // Endpoint for cancelling 49 | $this->assertArrayHasKey( '/wp/v2/meetings/(?P\d+):(?P\d\d\d\d-\d\d-\d\d)', $routes ); 50 | } 51 | 52 | public function test_get_meetings() { 53 | $request = new WP_REST_Request( 'GET', '/wp/v2/meeting' ); 54 | $response = $this->server->dispatch( $request ); 55 | $this->assertEquals( 200, $response->get_status() ); 56 | $this->assertEquals( 3, count( $response->get_data() ) ); 57 | } 58 | 59 | public function test_meeting_weekly() { 60 | $request = new WP_REST_Request( 'GET', '/wp/v2/meeting/' . $this->meeting_ids[0] ); 61 | $response = $this->server->dispatch( $request ); 62 | $this->assertEquals( 200, $response->get_status() ); 63 | $meeting = $response->get_data(); 64 | 65 | // Make sure the postmeta is all present 66 | $this->assertEquals( 'meeting', $meeting['type'] ); 67 | $this->assertEquals( 'Team-A', $meeting['meta']['team'] ); 68 | $this->assertEquals( '2020-01-01', $meeting['meta']['start_date'] ); 69 | $this->assertEquals( '', $meeting['meta']['end_date'] ); 70 | $this->assertEquals( '14:00:00', $meeting['meta']['time'] ); 71 | $this->assertEquals( 'weekly', $meeting['meta']['recurring'] ); 72 | $this->assertEquals( 'wordpress.org', $meeting['meta']['link'] ); 73 | $this->assertEquals( 'https://wordpress.tv', $meeting['meta']['wptv_url'] ); 74 | $this->assertEquals( array(), $meeting['meta']['occurrence'] ); 75 | 76 | $this->assertTrue( is_array( $meeting['future_occurrences'] ) ); 77 | $this->assertGreaterThanOrEqual( 4, count( $meeting['future_occurrences'] ) ); 78 | $this->assertLessThanOrEqual( 10, count( $meeting['future_occurrences'] ) ); 79 | // There should be no duplicates 80 | $this->assertEquals( $meeting['future_occurrences'], array_unique( $meeting['future_occurrences'] ) ); 81 | $last = false; 82 | foreach ( $meeting['future_occurrences'] as $future_date ) { 83 | // Make sure it's in the expected date format 84 | $this->assertEquals( 1, preg_match( '/^(\d\d\d\d)-(\d\d)-(\d\d)$/', $future_date, $matches ) ); 85 | // And it's a valid date 86 | $this->assertTrue( checkdate( $matches[2], $matches[3], $matches[1] ) ); 87 | 88 | $dt = new DateTime( $future_date ); 89 | // It should be in the future 90 | $this->assertGreaterThanOrEqual( new DateTime( 'yesterday' ), $dt ); 91 | // It should be within 2 months (the default range) 92 | $this->assertLessThanOrEqual( new DateTime( '+2 months' ), $dt ); 93 | // Day of week should be Wednesday, same as the original 94 | $this->assertEquals( 3, $dt->format( 'N' ) ); 95 | 96 | if ( $last ) { 97 | $interval = $last->diff( $dt ); 98 | // Should be exactly 7 days after the prior date 99 | $this->assertEquals( '+7 days', $interval->format( '%R%a days' ) ); 100 | } 101 | 102 | $last = $dt; 103 | } 104 | } 105 | 106 | public function test_meeting_monthly() { 107 | $request = new WP_REST_Request( 'GET', '/wp/v2/meeting/' . $this->meeting_ids[1] ); 108 | $response = $this->server->dispatch( $request ); 109 | $this->assertEquals( 200, $response->get_status() ); 110 | $meeting = $response->get_data(); 111 | 112 | // Make sure the postmeta is all present 113 | $this->assertEquals( 'meeting', $meeting['type'] ); 114 | $this->assertEquals( 'Team-B', $meeting['meta']['team'] ); 115 | $this->assertEquals( '2020-01-01', $meeting['meta']['start_date'] ); 116 | $this->assertEquals( '', $meeting['meta']['end_date'] ); 117 | $this->assertEquals( '15:00:00', $meeting['meta']['time'] ); 118 | $this->assertEquals( 'monthly', $meeting['meta']['recurring'] ); 119 | $this->assertEquals( 'wordpress.org', $meeting['meta']['link'] ); 120 | $this->assertEquals( 'https://wordpress.tv', $meeting['meta']['wptv_url'] ); 121 | $this->assertEquals( array(), $meeting['meta']['occurrence'] ); 122 | 123 | $this->assertTrue( is_array( $meeting['future_occurrences'] ) ); 124 | $this->assertEquals( 2, count( $meeting['future_occurrences'] ) ); 125 | // There should be no duplicates 126 | $this->assertEquals( $meeting['future_occurrences'], array_unique( $meeting['future_occurrences'] ) ); 127 | $last = false; 128 | foreach ( $meeting['future_occurrences'] as $future_date ) { 129 | // Make sure it's in the expected date format 130 | $this->assertEquals( 1, preg_match( '/^(\d\d\d\d)-(\d\d)-(\d\d)$/', $future_date, $matches ) ); 131 | // And it's a valid date 132 | $this->assertTrue( checkdate( $matches[2], $matches[3], $matches[1] ) ); 133 | 134 | $dt = new DateTime( $future_date ); 135 | // It should be in the future 136 | $this->assertGreaterThanOrEqual( new DateTime(), $dt ); 137 | // Day of week should be the first of the month, same as the original 138 | $this->assertEquals( 1, $dt->format( 'd' ) ); 139 | 140 | if ( $last ) { 141 | $interval = $last->diff( $dt ); 142 | // Should be exactly 7 days after the prior date 143 | $this->assertEquals( '+1 month 0 days', $interval->format( '%R%m month %d days' ) ); 144 | } 145 | 146 | $last = $dt; 147 | } 148 | } 149 | 150 | public function test_meeting_third_wednesday() { 151 | $request = new WP_REST_Request( 'GET', '/wp/v2/meeting/' . $this->meeting_ids[2] ); 152 | $response = $this->server->dispatch( $request ); 153 | $this->assertEquals( 200, $response->get_status() ); 154 | $meeting = $response->get_data(); 155 | 156 | // Make sure the postmeta is all present 157 | $this->assertEquals( 'meeting', $meeting['type'] ); 158 | $this->assertEquals( 'Team-C', $meeting['meta']['team'] ); 159 | $this->assertEquals( '2020-01-01', $meeting['meta']['start_date'] ); 160 | $this->assertEquals( '', $meeting['meta']['end_date'] ); 161 | $this->assertEquals( '16:00:00', $meeting['meta']['time'] ); 162 | $this->assertEquals( 'occurrence', $meeting['meta']['recurring'] ); 163 | $this->assertEquals( 'wordpress.org', $meeting['meta']['link'] ); 164 | $this->assertEquals( 'https://wordpress.tv', $meeting['meta']['wptv_url'] ); 165 | $this->assertEquals( array( 3 ), $meeting['meta']['occurrence'] ); 166 | 167 | $this->assertTrue( is_array( $meeting['future_occurrences'] ) ); 168 | $this->assertEquals( 2, count( $meeting['future_occurrences'] ) ); 169 | // There should be no duplicates 170 | $this->assertEquals( $meeting['future_occurrences'], array_unique( $meeting['future_occurrences'] ) ); 171 | $last = false; 172 | foreach ( $meeting['future_occurrences'] as $future_date ) { 173 | // Make sure it's in the expected date format 174 | $this->assertEquals( 1, preg_match( '/^(\d\d\d\d)-(\d\d)-(\d\d)$/', $future_date, $matches ) ); 175 | // And it's a valid date 176 | $this->assertTrue( checkdate( $matches[2], $matches[3], $matches[1] ) ); 177 | 178 | $dt = new DateTime( $future_date . ' ' . $meeting['meta']['time'] ); 179 | // It should be in the future 180 | $this->assertGreaterThanOrEqual( new DateTime(), $dt ); 181 | 182 | // Day of week should be Wednesday, same as the original 183 | $this->assertEquals( 3, $dt->format( 'N' ) ); 184 | 185 | if ( $last ) { 186 | $interval = $last->diff( $dt ); 187 | // Should be between 28 and 36 days since the last meeting 188 | $this->assertGreaterThanOrEqual( 28, $interval->format( '%R%a' ) ); 189 | $this->assertLessThanOrEqual( 36, $interval->format( '%R%a' ) ); 190 | } 191 | 192 | $last = $dt; 193 | } 194 | } 195 | 196 | public function _january_meetings() { 197 | return array( 198 | 0 => 199 | array( 200 | 'meeting_id' => $this->meeting_ids[0], 201 | 'instance_id' => $this->meeting_ids[0] . ':2020-01-01', 202 | 'date' => '2020-01-01', 203 | 'time' => '14:00:00', 204 | 'datetime' => '2020-01-01T14:00:00+00:00', 205 | 'team' => 'Team-A', 206 | 'link' => 'wordpress.org', 207 | 'wptv_url' => 'https://wordpress.tv', 208 | 'title' => 'A weekly meeting', 209 | 'location' => '#meta', 210 | 'recurring' => 'weekly', 211 | 'occurrence' => '', 212 | 'status' => 'active', 213 | 'rrule' => 'RRULE:FREQ=WEEKLY', 214 | ), 215 | 1 => 216 | array( 217 | 'meeting_id' => $this->meeting_ids[1], 218 | 'instance_id' => $this->meeting_ids[1] . ':2020-01-01', 219 | 'date' => '2020-01-01', 220 | 'time' => '15:00:00', 221 | 'datetime' => '2020-01-01T15:00:00+00:00', 222 | 'team' => 'Team-B', 223 | 'link' => 'wordpress.org', 224 | 'wptv_url' => 'https://wordpress.tv', 225 | 'title' => 'A monthly meeting', 226 | 'location' => '#meta', 227 | 'recurring' => 'monthly', 228 | 'occurrence' => '', 229 | 'status' => 'active', 230 | 'rrule' => 'RRULE:FREQ=MONTHLY', 231 | ), 232 | 2 => 233 | array( 234 | 'meeting_id' => $this->meeting_ids[0], 235 | 'instance_id' => $this->meeting_ids[0] . ':2020-01-08', 236 | 'date' => '2020-01-08', 237 | 'time' => '14:00:00', 238 | 'datetime' => '2020-01-08T14:00:00+00:00', 239 | 'team' => 'Team-A', 240 | 'link' => 'wordpress.org', 241 | 'wptv_url' => 'https://wordpress.tv', 242 | 'title' => 'A weekly meeting', 243 | 'location' => '#meta', 244 | 'recurring' => 'weekly', 245 | 'occurrence' => '', 246 | 'status' => 'active', 247 | 'rrule' => 'RRULE:FREQ=WEEKLY', 248 | ), 249 | 3 => 250 | array( 251 | 'meeting_id' => $this->meeting_ids[0], 252 | 'instance_id' => $this->meeting_ids[0] . ':2020-01-15', 253 | 'date' => '2020-01-15', 254 | 'time' => '14:00:00', 255 | 'datetime' => '2020-01-15T14:00:00+00:00', 256 | 'team' => 'Team-A', 257 | 'link' => 'wordpress.org', 258 | 'wptv_url' => 'https://wordpress.tv', 259 | 'title' => 'A weekly meeting', 260 | 'location' => '#meta', 261 | 'recurring' => 'weekly', 262 | 'occurrence' => '', 263 | 'status' => 'active', 264 | 'rrule' => 'RRULE:FREQ=WEEKLY', 265 | ), 266 | 4 => 267 | array( 268 | 'meeting_id' => $this->meeting_ids[2], 269 | 'instance_id' => $this->meeting_ids[2] . ':2020-01-15', 270 | 'date' => '2020-01-15', 271 | 'time' => '16:00:00', 272 | 'datetime' => '2020-01-15T16:00:00+00:00', 273 | 'team' => 'Team-C', 274 | 'link' => 'wordpress.org', 275 | 'wptv_url' => 'https://wordpress.tv', 276 | 'title' => 'Third Wednesday of each month', 277 | 'location' => '#meta', 278 | 'recurring' => 'occurrence', 279 | 'occurrence' => 280 | array( 281 | 0 => 3, 282 | ), 283 | 'status' => 'active', 284 | 'rrule' => 'RRULE:FREQ=MONTHLY;BYDAY=3WE', 285 | ), 286 | 5 => 287 | array( 288 | 'meeting_id' => $this->meeting_ids[0], 289 | 'instance_id' => $this->meeting_ids[0] . ':2020-01-22', 290 | 'date' => '2020-01-22', 291 | 'time' => '14:00:00', 292 | 'datetime' => '2020-01-22T14:00:00+00:00', 293 | 'team' => 'Team-A', 294 | 'link' => 'wordpress.org', 295 | 'wptv_url' => 'https://wordpress.tv', 296 | 'title' => 'A weekly meeting', 297 | 'location' => '#meta', 298 | 'recurring' => 'weekly', 299 | 'occurrence' => '', 300 | 'status' => 'active', 301 | 'rrule' => 'RRULE:FREQ=WEEKLY', 302 | ), 303 | 6 => 304 | array( 305 | 'meeting_id' => $this->meeting_ids[0], 306 | 'instance_id' => $this->meeting_ids[0] . ':2020-01-29', 307 | 'date' => '2020-01-29', 308 | 'time' => '14:00:00', 309 | 'datetime' => '2020-01-29T14:00:00+00:00', 310 | 'team' => 'Team-A', 311 | 'link' => 'wordpress.org', 312 | 'wptv_url' => 'https://wordpress.tv', 313 | 'title' => 'A weekly meeting', 314 | 'location' => '#meta', 315 | 'recurring' => 'weekly', 316 | 'occurrence' => '', 317 | 'status' => 'active', 318 | 'rrule' => 'RRULE:FREQ=WEEKLY', 319 | ), 320 | 7 => 321 | array( 322 | 'meeting_id' => $this->meeting_ids[1], 323 | 'instance_id' => $this->meeting_ids[1] . ':2020-02-01', 324 | 'date' => '2020-02-01', 325 | 'time' => '15:00:00', 326 | 'datetime' => '2020-02-01T15:00:00+00:00', 327 | 'team' => 'Team-B', 328 | 'link' => 'wordpress.org', 329 | 'wptv_url' => 'https://wordpress.tv', 330 | 'title' => 'A monthly meeting', 331 | 'location' => '#meta', 332 | 'recurring' => 'monthly', 333 | 'occurrence' => '', 334 | 'status' => 'active', 335 | 'rrule' => 'RRULE:FREQ=MONTHLY', 336 | ), 337 | 8 => 338 | array( 339 | 'meeting_id' => $this->meeting_ids[0], 340 | 'instance_id' => $this->meeting_ids[0] . ':2020-02-05', 341 | 'date' => '2020-02-05', 342 | 'time' => '14:00:00', 343 | 'datetime' => '2020-02-05T14:00:00+00:00', 344 | 'team' => 'Team-A', 345 | 'link' => 'wordpress.org', 346 | 'wptv_url' => 'https://wordpress.tv', 347 | 'title' => 'A weekly meeting', 348 | 'location' => '#meta', 349 | 'recurring' => 'weekly', 350 | 'occurrence' => '', 351 | 'status' => 'active', 352 | 'rrule' => 'RRULE:FREQ=WEEKLY', 353 | ), 354 | 9 => 355 | array( 356 | 'meeting_id' => $this->meeting_ids[0], 357 | 'instance_id' => $this->meeting_ids[0] . ':2020-02-12', 358 | 'date' => '2020-02-12', 359 | 'time' => '14:00:00', 360 | 'datetime' => '2020-02-12T14:00:00+00:00', 361 | 'team' => 'Team-A', 362 | 'link' => 'wordpress.org', 363 | 'wptv_url' => 'https://wordpress.tv', 364 | 'title' => 'A weekly meeting', 365 | 'location' => '#meta', 366 | 'recurring' => 'weekly', 367 | 'occurrence' => '', 368 | 'status' => 'active', 369 | 'rrule' => 'RRULE:FREQ=WEEKLY', 370 | ), 371 | 10 => 372 | array( 373 | 'meeting_id' => $this->meeting_ids[0], 374 | 'instance_id' => $this->meeting_ids[0] . ':2020-02-19', 375 | 'date' => '2020-02-19', 376 | 'time' => '14:00:00', 377 | 'datetime' => '2020-02-19T14:00:00+00:00', 378 | 'team' => 'Team-A', 379 | 'link' => 'wordpress.org', 380 | 'wptv_url' => 'https://wordpress.tv', 381 | 'title' => 'A weekly meeting', 382 | 'location' => '#meta', 383 | 'recurring' => 'weekly', 384 | 'occurrence' => '', 385 | 'status' => 'active', 386 | 'rrule' => 'RRULE:FREQ=WEEKLY', 387 | ), 388 | 11 => 389 | array( 390 | 'meeting_id' => $this->meeting_ids[2], 391 | 'instance_id' => $this->meeting_ids[2] . ':2020-02-19', 392 | 'date' => '2020-02-19', 393 | 'time' => '16:00:00', 394 | 'datetime' => '2020-02-19T16:00:00+00:00', 395 | 'team' => 'Team-C', 396 | 'link' => 'wordpress.org', 397 | 'wptv_url' => 'https://wordpress.tv', 398 | 'title' => 'Third Wednesday of each month', 399 | 'location' => '#meta', 400 | 'recurring' => 'occurrence', 401 | 'occurrence' => 402 | array( 403 | 0 => 3, 404 | ), 405 | 'status' => 'active', 406 | 'rrule' => 'RRULE:FREQ=MONTHLY;BYDAY=3WE', 407 | ), 408 | 12 => 409 | array( 410 | 'meeting_id' => $this->meeting_ids[0], 411 | 'instance_id' => $this->meeting_ids[0] . ':2020-02-26', 412 | 'date' => '2020-02-26', 413 | 'time' => '14:00:00', 414 | 'datetime' => '2020-02-26T14:00:00+00:00', 415 | 'team' => 'Team-A', 416 | 'link' => 'wordpress.org', 417 | 'wptv_url' => 'https://wordpress.tv', 418 | 'title' => 'A weekly meeting', 419 | 'location' => '#meta', 420 | 'recurring' => 'weekly', 421 | 'occurrence' => '', 422 | 'status' => 'active', 423 | 'rrule' => 'RRULE:FREQ=WEEKLY', 424 | ), 425 | ); 426 | } 427 | 428 | public function test_meetings_january_2020() { 429 | $request = new WP_REST_Request( 'GET', '/wp/v2/meetings/from/2020-01-01' ); 430 | $response = $this->server->dispatch( $request ); 431 | $this->assertEquals( 200, $response->get_status() ); 432 | $meetings = $response->get_data(); 433 | 434 | $this->assertEquals( $this->_january_meetings(), $meetings ); 435 | } 436 | 437 | public function test_meetings_february_2020() { 438 | $request = new WP_REST_Request( 'GET', '/wp/v2/meetings/from/2020-02-01' ); 439 | $response = $this->server->dispatch( $request ); 440 | $this->assertEquals( 200, $response->get_status() ); 441 | $meetings = $response->get_data(); 442 | 443 | $expected_datetimes = array( 444 | 0 => '2020-02-01T15:00:00+00:00', 445 | 1 => '2020-02-05T14:00:00+00:00', 446 | 2 => '2020-02-12T14:00:00+00:00', 447 | 3 => '2020-02-19T14:00:00+00:00', 448 | 4 => '2020-02-19T16:00:00+00:00', 449 | 5 => '2020-02-26T14:00:00+00:00', 450 | 6 => '2020-03-01T15:00:00+00:00', 451 | 7 => '2020-03-04T14:00:00+00:00', 452 | 8 => '2020-03-11T14:00:00+00:00', 453 | 9 => '2020-03-18T14:00:00+00:00', 454 | 10 => '2020-03-18T16:00:00+00:00', 455 | 11 => '2020-03-25T14:00:00+00:00', 456 | ); 457 | $this->assertEquals( $expected_datetimes, wp_list_pluck( $meetings, 'datetime' ) ); 458 | 459 | $expected_teams = array( 460 | 0 => 'Team-B', 461 | 1 => 'Team-A', 462 | 2 => 'Team-A', 463 | 3 => 'Team-A', 464 | 4 => 'Team-C', 465 | 5 => 'Team-A', 466 | 6 => 'Team-B', 467 | 7 => 'Team-A', 468 | 8 => 'Team-A', 469 | 9 => 'Team-A', 470 | 10 => 'Team-C', 471 | 11 => 'Team-A', 472 | ); 473 | $this->assertEquals( $expected_teams, wp_list_pluck( $meetings, 'team' ) ); 474 | } 475 | 476 | public function test_cancel_meeting_permissions_noauth() { 477 | $request = new WP_REST_Request( 'DELETE', '/wp/v2/meetings/' . $this->meeting_ids[0] . ':2020-01-15' ); 478 | $response = $this->server->dispatch( $request ); 479 | $this->assertEquals( 401, $response->get_status() ); 480 | 481 | $request = new WP_REST_Request( 'PUT', '/wp/v2/meetings/' . $this->meeting_ids[0] . ':2020-01-15' ); 482 | $response = $this->server->dispatch( $request ); 483 | $this->assertEquals( 401, $response->get_status() ); 484 | } 485 | 486 | public function test_cancel_meeting_permissions_valid() { 487 | $user_id = $this->factory->user->create( array( 'role' => 'editor' ) ); 488 | wp_set_current_user( $user_id ); 489 | 490 | $request = new WP_REST_Request( 'DELETE', '/wp/v2/meetings/' . $this->meeting_ids[0] . ':2020-01-15' ); 491 | $response = $this->server->dispatch( $request ); 492 | $this->assertEquals( 200, $response->get_status() ); 493 | 494 | $request = new WP_REST_Request( 'PUT', '/wp/v2/meetings/' . $this->meeting_ids[0] . ':2020-01-15' ); 495 | $response = $this->server->dispatch( $request ); 496 | $this->assertEquals( 200, $response->get_status() ); 497 | } 498 | 499 | public function test_cancel_meeting_output() { 500 | $user_id = $this->factory->user->create( array( 'role' => 'editor' ) ); 501 | wp_set_current_user( $user_id ); 502 | 503 | // Cancel the meeting on 2020-01-15 504 | $request = new WP_REST_Request( 'DELETE', '/wp/v2/meetings/' . $this->meeting_ids[0] . ':2020-01-15' ); 505 | $response = $this->server->dispatch( $request ); 506 | $this->assertEquals( 200, $response->get_status() ); 507 | 508 | // The feed should show that one meeting as cancelled 509 | $request = new WP_REST_Request( 'GET', '/wp/v2/meetings/from/2020-01-01' ); 510 | $response = $this->server->dispatch( $request ); 511 | $this->assertEquals( 200, $response->get_status() ); 512 | $meetings = $response->get_data(); 513 | 514 | $january_meetings = $this->_january_meetings(); 515 | $january_meetings[3]['status'] = 'cancelled'; 516 | $this->assertEquals( $january_meetings, $meetings ); 517 | 518 | // Now uncancel it 519 | $request = new WP_REST_Request( 'PUT', '/wp/v2/meetings/' . $this->meeting_ids[0] . ':2020-01-15' ); 520 | $response = $this->server->dispatch( $request ); 521 | $this->assertEquals( 200, $response->get_status() ); 522 | 523 | // The feed should be as before 524 | $request = new WP_REST_Request( 'GET', '/wp/v2/meetings/from/2020-01-01' ); 525 | $response = $this->server->dispatch( $request ); 526 | $this->assertEquals( 200, $response->get_status() ); 527 | $meetings = $response->get_data(); 528 | 529 | $january_meetings[3]['status'] = 'active'; 530 | $this->assertEquals( $january_meetings, $meetings ); 531 | } 532 | 533 | 534 | } 535 | -------------------------------------------------------------------------------- /tests/test-ical.php: -------------------------------------------------------------------------------- 1 | server = $wp_rest_server = new \WP_REST_Server(); 25 | do_action( 'rest_api_init' ); 26 | 27 | // Install test data 28 | $this->meeting_ids = Meeting_Calendar\wporg_meeting_install(); 29 | 30 | // Make sure the meta keys are registered - setUp/tearDown nukes these 31 | Meeting_Post_Type::getInstance()->register_meta(); 32 | } 33 | 34 | public function test_get_recurring_strings() { 35 | // 2020-01-01 is a Sunday. 36 | $freq = get_frequencies_by_day( array( 1, 3 ), '2019-09-15' ); 37 | $this->assertEquals( 'MONTHLY;BYDAY=1SU,3SU', $freq ); 38 | 39 | // 2020-01-01 is a Wednesday. 40 | $freq = get_frequencies_by_day( array( 3 ), '2020-01-01' ); 41 | $this->assertEquals( 'MONTHLY;BYDAY=3WE', $freq ); 42 | 43 | // 2025-03-14 is a Friday. 44 | $freq = get_frequencies_by_day( array( 4 ), '2025-03-14' ); 45 | $this->assertEquals( 'MONTHLY;BYDAY=4FR', $freq ); 46 | } 47 | 48 | public function test_get_ical() { 49 | $posts = get_meeting_posts(); 50 | $ical_feed = generate( $posts, '' ); 51 | $events_ics = file_get_contents( __DIR__ . '/fixtures/events.ics' ); 52 | $events_ics = str_replace( '%ID1%', str_replace( '-', '', $posts[0]->ID ), $events_ics ); 53 | $events_ics = str_replace( '%ID2%', str_replace( '-', '', $posts[1]->ID ), $events_ics ); 54 | $events_ics = str_replace( '%ID3%', str_replace( '-', '', $posts[2]->ID ), $events_ics ); 55 | 56 | $this->assertEquals( 57 | preg_split( '/\r\n|\r|\n/', $events_ics ), 58 | preg_split( '/\r\n|\r|\n/', $ical_feed ) 59 | ); 60 | } 61 | 62 | public function test_get_ical_with_cancellation() { 63 | $posts = get_meeting_posts( 'Team-A' ); 64 | // Cancel the second occurrence of the weekly meeting 65 | $occurrences = Meeting_Post_Type::getInstance()->get_future_occurrences( get_post( $posts[0]->ID ), null, null ); 66 | $this->assertGreaterThan( 67 | 0, 68 | Meeting_Post_Type::getInstance()->cancel_meeting( 69 | array( 70 | 'meeting_id' => $posts[0]->ID, 71 | 'date' => $occurrences[1], 72 | ) 73 | ) 74 | ); 75 | 76 | $ical_feed = generate( $posts, '' ); 77 | $events_ics = file_get_contents( __DIR__ . '/fixtures/events-with-cancel.ics' ); 78 | $events_ics = str_replace( '%ID%', str_replace( '-', '', $posts[0]->ID ), $events_ics ); 79 | $events_ics = str_replace( '%EXDATE%', str_replace( '-', '', $occurrences[1] ), $events_ics ); 80 | 81 | $this->assertEquals( 82 | preg_split( '/\r\n|\r|\n/', $events_ics ), 83 | preg_split( '/\r\n|\r|\n/', $ical_feed ) 84 | ); 85 | } 86 | } 87 | -------------------------------------------------------------------------------- /tests/test-meetingposttype.php: -------------------------------------------------------------------------------- 1 | meeting_ids = Meeting_Calendar\wporg_meeting_install(); 22 | 23 | $this->mpt = Meeting_Post_Type::getInstance(); 24 | } 25 | 26 | function test_single_meeting() { 27 | // See https://github.com/WordPress/meeting-calendar/issues/34 28 | 29 | $meeting_id = $this->factory->post->create( 30 | array( 31 | 'post_title' => __( 'A single meeting', 'wporg-meeting-calendar' ), 32 | 'post_type' => 'meeting', 33 | 'post_status' => 'publish', 34 | 'meta_input' => array( 35 | 'team' => 'Team-A', 36 | 'start_date' => strftime( '%Y-%m-%d', strtotime( 'tomorrow' ) ), 37 | 'end_date' => '', 38 | 'time' => '01:00', 39 | 'recurring' => '', 40 | 'link' => 'wordpress.org', 41 | 'wptv_url' => 'https://wordpress.tv', 42 | 'location' => '#meta', 43 | ), 44 | ) 45 | ); 46 | 47 | $meetings = $this->mpt->get_occurrences_for_period( null ); 48 | $found = 0; 49 | foreach ( $meetings as $meeting ) { 50 | if ( $meeting['meeting_id'] === $meeting_id ) { 51 | ++ $found; 52 | } 53 | } 54 | 55 | $this->assertEquals( 1, $found, 'There should be exactly one instance of a single meeting.' ); 56 | } 57 | 58 | function test_invalid_time() { 59 | $meeting_id = $this->factory->post->create( 60 | array( 61 | 'post_title' => __( 'A meeting with an invalid time', 'wporg-meeting-calendar' ), 62 | 'post_type' => 'meeting', 63 | 'post_status' => 'publish', 64 | 'meta_input' => array( 65 | 'team' => 'Team-A', 66 | 'start_date' => strftime( '%Y-%m-%d', strtotime( 'tomorrow' ) ), 67 | 'end_date' => '', 68 | 'time' => '0100 UTC', // Some production data is formatted like this 69 | 'recurring' => '', 70 | 'link' => 'wordpress.org', 71 | 'wptv_url' => 'https://wordpress.tv', 72 | 'location' => '#meta', 73 | ), 74 | ) 75 | ); 76 | 77 | $meetings = $this->mpt->get_occurrences_for_period( null ); 78 | 79 | $found = 0; 80 | foreach ( $meetings as $meeting ) { 81 | if ( $meeting['meeting_id'] === $meeting_id ) { 82 | ++$found; 83 | $this->assertEquals( '01:00:00', $meeting['time'] ); 84 | $this->assertEquals( "{$meeting['date']}T01:00:00+00:00", $meeting['datetime'] ); 85 | } 86 | } 87 | $this->assertGreaterThan( 0, $found, 'Found no meeting to test' ); 88 | } 89 | 90 | function test_encoding() { 91 | $meeting_id = $this->factory->post->create( 92 | array( 93 | 'post_title' => __( 'A & B meeting', 'wporg-meeting-calendar' ), 94 | 'post_type' => 'meeting', 95 | 'post_status' => 'publish', 96 | 'meta_input' => array( 97 | 'team' => 'Team-A&B', 98 | 'start_date' => strftime( '%Y-%m-%d', strtotime( 'tomorrow' ) ), 99 | 'end_date' => '', 100 | 'time' => '01:00', 101 | 'recurring' => '', 102 | 'link' => '&wordpress.org', 103 | 'wptv_url' => '&https://wordpress.tv', 104 | 'location' => '&meta', 105 | ), 106 | ) 107 | ); 108 | 109 | $meetings = $this->mpt->get_occurrences_for_period( null ); 110 | 111 | $found = 0; 112 | foreach ( $meetings as $meeting ) { 113 | if ( $meeting['meeting_id'] === $meeting_id ) { 114 | ++$found; 115 | $this->assertEquals( 'Team-A&B', $meeting['team'] ); 116 | $this->assertEquals( '&wordpress.org', $meeting['link'] ); 117 | $this->assertEquals( '&https://wordpress.tv', $meeting['wptv_url'] ); 118 | $this->assertEquals( '&meta', $meeting['location'] ); 119 | $this->assertEquals( 'A & B meeting', $meeting['title'] ); 120 | } 121 | } 122 | $this->assertGreaterThan( 0, $found, 'Found no meeting to test' ); 123 | } 124 | 125 | /* 126 | * There was a bug with the meeting_set_next_meeting() filter, where it was interfering with the sorting of other post types. 127 | * See https://github.com/WordPress/meeting-calendar/issues/98 128 | */ 129 | function test_sort_filter_bug() { 130 | $category = $this->factory->category->create( array( 131 | 'slug' => 'test_sort_filter_bug' 132 | ) ); 133 | 134 | $this->factory->post->create( array( 135 | 'post_title' => 'older post', 136 | 'post_date' => '2020-04-01 17:00:00', 137 | 'post_category' => array( $category ), 138 | 'meta_input' => array( 139 | 'time' => '17:00' 140 | ) 141 | ) ); 142 | 143 | $this->factory->post->create( array( 144 | 'post_title' => 'newer post', 145 | 'post_date' => '2020-04-02 18:00:00', 146 | 'post_category' => array( $category ), 147 | 'meta_input' => array( 148 | 'time' => '18:00' 149 | ) 150 | ) ); 151 | 152 | $posts = get_posts( array( 153 | 'suppress_filters' => false, 154 | 'category' => $category, 155 | ) ); 156 | 157 | $this->assertTrue( is_array( $posts ) ); 158 | $this->assertEquals( 2, count( $posts ) ); 159 | 160 | // The newer post should come first. 161 | // The bug meanth that the filter sorted all post types that happened to have a `time` postmeta key, not just `meeting` CPTs. 162 | $this->assertEquals( '2020-04-02 18:00:00', $posts[0]->post_date ); 163 | $this->assertEquals( 'newer post', $posts[0]->post_title ); 164 | $this->assertEquals( '2020-04-01 17:00:00', $posts[1]->post_date ); 165 | $this->assertEquals( 'older post', $posts[1]->post_title ); 166 | } 167 | 168 | } 169 | -------------------------------------------------------------------------------- /tests/test-shortcode.php: -------------------------------------------------------------------------------- 1 | meeting_ids = Meeting_Calendar\wporg_meeting_install(); 22 | 23 | $this->mpt = Meeting_Post_Type::getInstance(); 24 | } 25 | 26 | function test_shortcode_simple() { 27 | 28 | // A one-off meeting tomorrow 29 | $this->factory->post->create( 30 | array( 31 | 'post_title' => __( 'Meeting One', 'wporg-meeting-calendar' ), 32 | 'post_type' => 'meeting', 33 | 'post_status' => 'publish', 34 | 'meta_input' => array( 35 | 'team' => 'Team-F', 36 | 'start_date' => strftime( '%Y-%m-%d', strtotime( 'tomorrow' ) ), 37 | 'end_date' => '', 38 | 'time' => '01:00', 39 | 'recurring' => '', 40 | 'link' => 'wordpress.org', 41 | 'wptv_url' => 'https://wordpress.tv', 42 | 'location' => '#meta', 43 | ), 44 | ) 45 | ); 46 | 47 | // A recurring weekly meeting, starting yesterday 48 | $this->factory->post->create( 49 | array( 50 | 'post_title' => __( 'Meeting Two', 'wporg-meeting-calendar' ), 51 | 'post_type' => 'meeting', 52 | 'post_status' => 'publish', 53 | 'meta_input' => array( 54 | 'team' => 'Team-F', 55 | 'start_date' => strftime( '%Y-%m-%d', strtotime( 'yesterday' ) ), 56 | 'end_date' => '', 57 | 'time' => '02:00', 58 | 'recurring' => 'weekly', 59 | 'link' => 'wordpress.org', 60 | 'wptv_url' => 'https://wordpress.tv', 61 | 'location' => '#meta', 62 | ), 63 | ) 64 | ); 65 | 66 | $actual = do_shortcode( '[meeting_time team="Team-F" before="" more=0 limit=-1 /]' ); 67 | 68 | $expected = strftime( 'Meeting One
    ', strtotime( 'tomorrow' ) ); 69 | $substr = substr( $actual, strpos( $actual, 'Meeting One' ), strlen( $expected ) ); 70 | $this->assertEquals( $substr, $expected ); 71 | 72 | $expected = strftime( 'Meeting Two
    ', strtotime( 'yesterday +7 days' ) ); 73 | $substr = substr( $actual, strpos( $actual, 'Meeting Two' ), strlen( $expected ) ); 74 | $this->assertEquals( $substr, $expected ); 75 | } 76 | 77 | function test_shortcode_cancelled() { 78 | 79 | // A recurring weekly meeting, starting yesterday 80 | $meeting_1 = $this->factory->post->create( 81 | array( 82 | 'post_title' => __( 'Meeting One', 'wporg-meeting-calendar' ), 83 | 'post_type' => 'meeting', 84 | 'post_status' => 'publish', 85 | 'meta_input' => array( 86 | 'team' => 'Team-F', 87 | 'start_date' => strftime( '%Y-%m-%d', strtotime( 'yesterday' ) ), 88 | 'end_date' => '', 89 | 'time' => '01:00', 90 | 'recurring' => 'weekly', 91 | 'link' => 'wordpress.org', 92 | 'wptv_url' => 'https://wordpress.tv', 93 | 'location' => '#meta', 94 | ), 95 | ) 96 | ); 97 | 98 | // Cancel the meeting that's in 6 days 99 | $response = $this->mpt->cancel_meeting( 100 | array( 101 | 'meeting_id' => $meeting_1, 102 | 'date' => strftime( 103 | '%Y-%m-%d', 104 | strtotime( 'yesterday +7 days' ) 105 | ), 106 | ) 107 | ); 108 | $this->assertGreaterThan( 0, $response ); 109 | 110 | $actual = do_shortcode( '[meeting_time team="Team-F" before="" more=0 limit=-1 /]' ); 111 | 112 | // The shortcode should show the next meeting is in 7 days 113 | $expected = strftime( 'Meeting One
    ', strtotime( 'yesterday +7 days' ) ); 114 | $substr = substr( $actual, strpos( $actual, 'Meeting One' ), strlen( $expected ) ); 115 | $this->assertEquals( $substr, $expected ); 116 | 117 | // It should be listed as cancelled 118 | $this->assertGreaterThanOrEqual( 0, strpos( $actual, '

    assertGreaterThan( 0, strpos( $actual, strftime( 'This event is cancelled. The next meeting is scheduled for %Y-%m-%d.', strtotime( 'yesterday +14 days' ) ) ) ); 122 | } 123 | } 124 | --------------------------------------------------------------------------------