diff --git a/mu-plugins/blocks/meetup-event-filter/README.md b/mu-plugins/blocks/meetup-event-filter/README.md
new file mode 100644
index 000000000..800c64288
--- /dev/null
+++ b/mu-plugins/blocks/meetup-event-filter/README.md
@@ -0,0 +1,33 @@
+# Event Filter
+
+This plugins creates the `wporg/event-filter` block, which displays a map and list of events that match a given filter during a given timeframe. Filters can be setup for anything, but some common examples are watch parties for WP20 and the State of the Word.
+
+It uses the `wporg/google-map` block to display a searchable list and/or map of the selected events.
+
+
+## Usage
+
+1. Setup the API key needed for the `wporg/google-maps` block.
+1. Add a new filter to `filter_potential_events()` if you're not using an existing one.
+1. Add the following to a pattern in your theme. `googleMapBlockAttributes` are the attributes that will be passed to the `wporg/google-map` block.
+
+ ```php
+ $filter_options = array(
+ 'filterSlug' => 'wp20',
+ 'startDate' => 'April 21, 2023',
+ 'endDate' => 'May 30, 2023',
+
+ // This has to be an object, see `WordPressdotorg\MU_Plugins\Meetup_Event_Filter\render()`.
+ 'googleMapBlockAttributes' => (object) array(
+ 'id' => 'wp20',
+ 'apiKey' => 'WORDCAMP_DEV_GOOGLE_MAPS_API_KEY',
+ ),
+ );
+
+ ?>
+
+
+ ```
+
+1. View the page where the block is used to create the cron job that updates the data.
+1. Run `wp cron event run prime_meetup_events_filter` to test the filtering. Look at each title, and add any false positives to `$false_positives` in `filter_potential_events()`. If any events that should be included were ignored, add a keyword from the title to `$keywords`. Run the command after those changes and make sure it's correct now.
diff --git a/mu-plugins/blocks/meetup-event-filter/meetup-event-filter.php b/mu-plugins/blocks/meetup-event-filter/meetup-event-filter.php
new file mode 100644
index 000000000..659203a9c
--- /dev/null
+++ b/mu-plugins/blocks/meetup-event-filter/meetup-event-filter.php
@@ -0,0 +1,229 @@
+ __NAMESPACE__ . '\render',
+ )
+ );
+}
+
+/**
+ * Render the block content.
+ */
+function render( array $attributes, string $content, WP_Block $block ): string {
+ $attributes['startDate'] = strtotime( $attributes['startDate'] );
+ $attributes['endDate'] = strtotime( $attributes['endDate'] );
+ $wrapper_attributes = get_block_wrapper_attributes( array( 'id' => 'wp-block-wporg-meetup-event-filter-' . $attributes['filterSlug'] ) );
+
+ // The REST API doesn't support associative arrays, so this had to be defined as an object in this block. It
+ // needs to be an array when passed to the Google Map block though.
+ // See `rest_is_array()`.
+ $map_options = (array) $attributes['googleMapBlockAttributes'];
+ $map_options['markers'] = get_events( $attributes['filterSlug'], $attributes['startDate'], $attributes['endDate'] );
+
+ // This has to be called in `render()` to know which slug/dates to use.
+ $cron_args = array( $attributes['filterSlug'], $attributes['startDate'], $attributes['endDate'], true );
+
+ if ( ! wp_next_scheduled( 'prime_meetup_events_filter', $cron_args ) ) {
+ wp_schedule_event(
+ time() + HOUR_IN_SECONDS,
+ 'hourly',
+ 'prime_meetup_events_filter',
+ $cron_args
+ );
+ }
+
+ ob_start();
+
+ ?>
+
+
>
+
+
+
+ prepare( '
+ SELECT
+ id, `type`, source_id, title, url, description, meetup, location, latitude, longitude, date_utc,
+ date_utc_offset AS tz_offset
+ FROM `wporg_events`
+ WHERE
+ status = "scheduled" AND
+ date_utc BETWEEN FROM_UNIXTIME( %d ) AND FROM_UNIXTIME( %d )
+ ORDER BY date_utc ASC
+ LIMIT 1000',
+ $start_timestamp,
+ $end_timestamp
+ );
+
+ if ( 'latin1' === DB_CHARSET ) {
+ $events = $wpdb->get_results( $query );
+ } else {
+ $events = get_latin1_results_with_prepared_query( $query );
+ }
+
+ foreach ( $events as $event ) {
+ // `capital_P_dangit()` won't work here because the current filter isn't `the_title` and there isn't a safelisted prefix before `$text`.
+ $event->title = str_replace( 'Wordpress', 'WordPress', $event->title );
+
+ // `date_utc` is a misnomer, the value is actually in the local timezone of the event. So, convert to a true Unix timestamp (UTC).
+ // Can't do this reliably in the query because MySQL converts it to the server timezone.
+ $event->timestamp = strtotime( $event->date_utc ) - $event->tz_offset;
+
+ unset( $event->date_utc );
+ }
+
+ return $events;
+}
+
+/**
+ * Query a table that's encoded with the `latin1` charset.
+ *
+ * wordpress.org uses a `DB_CHARSET` of `latin1` for legacy reasons, but wordcamp.org and others use `utf8mb4`.
+ * `wporg_events` uses `latin1`, so querying it with `utf8mb4` will produce Mojibake.
+ *
+ * @param string $prepared_query ⚠️ This must have already be ran through `$wpdb->prepare()` if needed.
+ *
+ * @return object|null
+ */
+function get_latin1_results_with_prepared_query( string $prepared_query ) {
+ global $wpdb;
+
+ // Local environments don't always use HyperDB, but production does.
+ $db_handle = is_a( $wpdb, 'hyperdb' ) ? $wpdb->db_connect( $prepared_query ) : $wpdb->dbh;
+ $wpdb->set_charset( $db_handle, 'latin1', 'latin1_swedish_ci' );
+
+ // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- This function doesn't have the context to prepare it, the caller must.
+ $results = $wpdb->get_results( $prepared_query );
+
+ // Revert to the default charset to avoid affecting other queries.
+ $wpdb->set_charset( $db_handle, DB_CHARSET, DB_COLLATE );
+
+ return $results;
+}
+
+/**
+ * Extract the desired events from an array of potential events.
+ */
+function filter_potential_events( string $filter_slug, array $potential_events ) : array {
+ $matched_events = array();
+ $other_events = array();
+
+ switch ( $filter_slug ) {
+ case 'sotw':
+ $false_positives = array();
+ $keywords = array(
+ 'sotw', 'state of the word',
+ );
+ break;
+
+ case 'wp20':
+ $false_positives = array( "292525625", "293437294" );
+ $keywords = array(
+ 'wp20', '20 year', '20 ano', '20 año', '20 candeline', '20 jaar', 'wordt 20', '20 yaşında',
+ 'anniversary', 'aniversário', 'aniversario', 'birthday', 'cumpleaños', 'celebrate',
+ 'Tanti auguri',
+ );
+ break;
+
+ default:
+ return array();
+ }
+
+ foreach ( $potential_events as $event ) {
+ $match = false;
+
+ // Have to use `source_id` because `id` is rotated by `REPLACE INTO` when table is updated.
+ if ( in_array( $event->source_id, $false_positives, true ) ) {
+ $other_events[] = $event;
+ continue;
+ }
+
+ foreach ( $keywords as $keyword ) {
+ if ( false !== stripos( $event->description, $keyword ) || false !== stripos( $event->title, $keyword ) ) {
+ // These are no longer needed, so remove it to save space in the database.
+ unset( $event->description );
+ unset( $event->source_id );
+ $matched_events[] = $event;
+ continue 2;
+ }
+ }
+
+ if ( ! $match ) {
+ $other_events[] = $event;
+ }
+ }
+
+ print_results( $matched_events, $other_events );
+
+ return $matched_events;
+}
+
+/**
+ * Print the matched/unmatched events for manual review.
+ *
+ * Run `wp cron event run prime_meetup_events_filter` to see this.
+ */
+function print_results( array $matched_events, array $other_events ) : void {
+ if ( 'cli' !== php_sapi_name() ) {
+ return;
+ }
+
+ $matched_names = wp_list_pluck( $matched_events, 'title' );
+ $other_names = wp_list_pluck( $other_events, 'title' );
+
+ sort( $matched_names );
+ sort( $other_names );
+
+ echo "\nIgnored these events. Double check for false-negatives.\n\n";
+ print_r( $other_names );
+
+ echo "\Included these events. Double check for false-positives.\n\n";
+ print_r( $matched_names );
+}
diff --git a/mu-plugins/blocks/meetup-event-filter/src/block.json b/mu-plugins/blocks/meetup-event-filter/src/block.json
new file mode 100644
index 000000000..83beb634a
--- /dev/null
+++ b/mu-plugins/blocks/meetup-event-filter/src/block.json
@@ -0,0 +1,33 @@
+{
+ "$schema": "https://schemas.wp.org/trunk/block.json",
+ "apiVersion": 2,
+ "name": "wporg/meetup-event-filter",
+ "title": "Meetup Event Filter",
+ "icon": "nametag",
+ "category": "widgets",
+ "description": "Displays a map and/or list of a subset of meetup events.",
+ "textdomain": "wporg",
+ "attributes": {
+ "filterSlug": {
+ "type": "string",
+ "default": ""
+ },
+ "startDate": {
+ "type": "string",
+ "default": ""
+ },
+ "endDate": {
+ "type": "string",
+ "default": ""
+ },
+ "googleMapBlockAttributes": {
+ "type": "object",
+ "default": []
+ }
+ },
+ "supports": {
+ "inserter": false
+ },
+ "editorScript": "file:./index.js",
+ "style": "file:./style.css"
+}
diff --git a/mu-plugins/blocks/meetup-event-filter/src/index.js b/mu-plugins/blocks/meetup-event-filter/src/index.js
new file mode 100644
index 000000000..c49831eaf
--- /dev/null
+++ b/mu-plugins/blocks/meetup-event-filter/src/index.js
@@ -0,0 +1,27 @@
+/**
+ * WordPress dependencies
+ */
+import { __ } from '@wordpress/i18n';
+import { registerBlockType } from '@wordpress/blocks';
+import { Placeholder } from '@wordpress/components';
+
+/**
+ * Internal dependencies
+ */
+import metadata from './block.json';
+
+function Edit() {
+ return (
+
+ );
+}
+
+registerBlockType( metadata.name, {
+ edit: Edit,
+} );
diff --git a/mu-plugins/loader.php b/mu-plugins/loader.php
index cf0cdaafb..73fda43da 100644
--- a/mu-plugins/loader.php
+++ b/mu-plugins/loader.php
@@ -34,6 +34,7 @@
require_once __DIR__ . '/blocks/local-navigation-bar/index.php';
require_once __DIR__ . '/blocks/latest-news/latest-news.php';
require_once __DIR__ . '/blocks/link-wrapper/index.php';
+require_once __DIR__ . '/blocks/meetup-event-filter/meetup-event-filter.php';
require_once __DIR__ . '/blocks/navigation/index.php';
require_once __DIR__ . '/blocks/notice/index.php';
require_once __DIR__ . '/blocks/query-filter/index.php';