3 Commits

Author SHA1 Message Date
b2d4c240ad Split ICS subscribe actions by platform
Some checks failed
Release Plugin Zip / release (push) Has been cancelled
2026-04-01 18:25:47 -05:00
6837907aa9 Bump plugin version to 0.1.7 2026-04-01 18:22:46 -05:00
bfc74fcab6 Refactor schedule exports into feed builders 2026-04-01 18:20:56 -05:00
6 changed files with 1994 additions and 536 deletions

View File

@@ -10,256 +10,8 @@ if ( ! defined( 'ABSPATH' ) ) {
}
/**
* Register the SportsPress calendar CSV feed endpoint.
*
* @return void
* SportsPress event CSV importer tools.
*/
function tse_sp_register_calendar_csv_feed() {
add_feed( 'sp-csv', 'tse_sp_render_calendar_csv_feed' );
}
add_action( 'init', 'tse_sp_register_calendar_csv_feed', 20 );
/**
* Replace the stock SportsPress calendar feeds metabox with one that includes CSV.
*
* @return void
*/
function tse_sp_replace_calendar_feeds_metabox() {
remove_meta_box( 'sp_feedsdiv', 'sp_calendar', 'side' );
add_meta_box(
'sp_feedsdiv',
esc_attr__( 'Feeds', 'sportspress' ),
'tse_sp_render_calendar_feeds_metabox',
'sp_calendar',
'side',
'default'
);
}
add_action( 'add_meta_boxes_sp_calendar', 'tse_sp_replace_calendar_feeds_metabox', 40 );
/**
* Return the CSV feed format definition used in the metabox.
*
* @return array
*/
function tse_sp_get_calendar_csv_feed_formats() {
return array(
'download' => array(
'name' => __( 'CSV Download', 'tonys-sportspress-enhancements' ),
),
);
}
/**
* Render the calendar feeds metabox with CSV support.
*
* @param WP_Post $post Current calendar post.
* @return void
*/
function tse_sp_render_calendar_feeds_metabox( $post ) {
$feeds = new SP_Feeds();
$calendar_feeds = is_array( $feeds->calendar ) ? $feeds->calendar : array();
$calendar_feeds['csv'] = tse_sp_get_calendar_csv_feed_formats();
?>
<div>
<?php foreach ( $calendar_feeds as $slug => $formats ) : ?>
<?php
if ( 'csv' === $slug ) {
$link = tse_sp_get_calendar_csv_url( $post->ID );
} else {
$link = add_query_arg( 'feed', 'sp-' . $slug, untrailingslashit( get_post_permalink( $post ) ) );
}
?>
<?php foreach ( $formats as $format ) : ?>
<?php
if ( 'csv' === $slug ) {
$feed = $link;
} else {
$protocol = sp_array_value( $format, 'protocol' );
if ( $protocol ) {
$feed = str_replace( array( 'http:', 'https:' ), 'webcal:', $link );
} else {
$feed = $link;
}
$prefix = sp_array_value( $format, 'prefix' );
if ( $prefix ) {
$feed = $prefix . urlencode( $feed );
}
}
?>
<p>
<strong><?php echo esc_html( sp_array_value( $format, 'name' ) ); ?></strong>
<a class="sp-link" href="<?php echo esc_url( $feed ); ?>" target="_blank" title="<?php esc_attr_e( 'Link', 'sportspress' ); ?>"></a>
</p>
<p>
<input type="text" value="<?php echo esc_attr( $feed ); ?>" readonly="readonly" class="code widefat">
</p>
<?php if ( 'csv' === $slug ) : ?>
<p class="description">
<?php esc_html_e( 'Optional team filter: add &team_id=123 to only include games for that team.', 'tonys-sportspress-enhancements' ); ?>
</p>
<?php endif; ?>
<?php endforeach; ?>
<?php endforeach; ?>
</div>
<?php
}
/**
* Build the CSV feed URL for a SportsPress calendar.
*
* @param int $calendar_id SportsPress calendar post ID.
* @param int $team_id Optional team ID filter.
* @return string
*/
function tse_sp_get_calendar_csv_url( $calendar_id, $team_id = 0 ) {
$calendar_id = absint( $calendar_id );
$team_id = absint( $team_id );
if ( ! $calendar_id || 'sp_calendar' !== get_post_type( $calendar_id ) ) {
return '';
}
$url = add_query_arg( 'feed', 'sp-csv', get_post_permalink( $calendar_id ) );
if ( $team_id ) {
$url = add_query_arg( 'team_id', $team_id, $url );
}
return $url;
}
/**
* Get the queried SportsPress calendar post for the CSV feed.
*
* @return WP_Post|null
*/
function tse_sp_get_calendar_csv_post() {
$post = get_post();
if ( $post instanceof WP_Post && 'sp_calendar' === $post->post_type ) {
return $post;
}
$queried_object = get_queried_object();
if ( $queried_object instanceof WP_Post && 'sp_calendar' === $queried_object->post_type ) {
return $queried_object;
}
$calendar_id = get_queried_object_id();
if ( $calendar_id && 'sp_calendar' === get_post_type( $calendar_id ) ) {
return get_post( $calendar_id );
}
return null;
}
/**
* Return the home and away teams for an event in stored order.
*
* @param int $event_id SportsPress event ID.
* @return array
*/
function tse_sp_get_event_home_away_teams( $event_id ) {
$teams = array_values( array_filter( array_map( 'absint', get_post_meta( $event_id, 'sp_team', false ) ) ) );
return array(
'home' => isset( $teams[0] ) ? get_the_title( $teams[0] ) : '',
'away' => isset( $teams[1] ) ? get_the_title( $teams[1] ) : '',
);
}
/**
* Return the field name(s) for an event.
*
* @param int $event_id SportsPress event ID.
* @return string
*/
function tse_sp_get_event_field_name( $event_id ) {
$venues = get_the_terms( $event_id, 'sp_venue' );
if ( empty( $venues ) || is_wp_error( $venues ) ) {
return '';
}
return implode( ', ', wp_list_pluck( $venues, 'name' ) );
}
/**
* Render the SportsPress calendar CSV feed.
*
* @return void
*/
function tse_sp_render_calendar_csv_feed() {
if ( ! class_exists( 'SP_Calendar' ) ) {
wp_die( esc_html__( 'ERROR: SportsPress is required for this feed.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 500 ) );
}
$calendar = tse_sp_get_calendar_csv_post();
if ( ! $calendar ) {
wp_die( esc_html__( 'ERROR: This is not a valid calendar feed.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 404 ) );
}
$team_id = isset( $_GET['team_id'] ) ? absint( wp_unslash( $_GET['team_id'] ) ) : 0;
$calendar_data = new SP_Calendar( $calendar );
if ( $team_id ) {
$calendar_data->team = $team_id;
}
$events = (array) $calendar_data->data();
$filename = sanitize_title( $calendar->post_name ? $calendar->post_name : $calendar->post_title );
if ( '' === $filename ) {
$filename = 'schedule';
}
if ( $team_id ) {
$filename .= '-team-' . $team_id;
}
header( 'Content-Type: text/csv; charset=utf-8' );
header( 'Content-Disposition: inline; filename=' . $filename . '.csv' );
$output = fopen( 'php://output', 'w' );
if ( false === $output ) {
wp_die( esc_html__( 'ERROR: Unable to generate the CSV feed.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 500 ) );
}
// Excel expects a BOM for UTF-8 CSV files.
fwrite( $output, "\xEF\xBB\xBF" );
fputcsv(
$output,
array(
'Date',
'Time',
'Away Team',
'Home Team',
'Field Name',
)
);
foreach ( $events as $event ) {
$teams = tse_sp_get_event_home_away_teams( $event->ID );
fputcsv(
$output,
array(
sp_get_date( $event ),
sp_get_time( $event ),
$teams['away'],
$teams['home'],
tse_sp_get_event_field_name( $event->ID ),
)
);
}
fclose( $output );
exit;
}
/**
* CSV headers recognized by this importer.

View File

@@ -0,0 +1,922 @@
<?php
/**
* Standalone SportsPress schedule feeds.
*
* @package Tonys_Sportspress_Enhancements
*/
if ( ! defined( 'ABSPATH' ) ) {
exit;
}
/**
* Register the standalone SportsPress CSV feed endpoint.
*
* @return void
*/
function tse_sp_event_export_register_feed() {
add_feed( 'sp-csv', 'tse_sp_event_export_render_feed' );
add_feed( 'sp-ics', 'tse_sp_event_export_render_ical_feed' );
}
add_action( 'init', 'tse_sp_event_export_register_feed', 20 );
/**
* Get available export formats.
*
* @return array
*/
function tse_sp_event_export_get_formats() {
return array(
'matchup' => array(
'label' => __( 'Matchup', 'tonys-sportspress-enhancements' ),
'description' => __( 'Date, time, away team, home team, and field columns.', 'tonys-sportspress-enhancements' ),
),
'team' => array(
'label' => __( 'Team', 'tonys-sportspress-enhancements' ),
'description' => __( 'Team-centric schedule rows with opponent and home/away columns.', 'tonys-sportspress-enhancements' ),
),
);
}
/**
* Get available export columns grouped by format.
*
* @return array
*/
function tse_sp_event_export_get_column_definitions() {
return array(
'matchup' => array(
'date' => __( 'Date', 'tonys-sportspress-enhancements' ),
'time' => __( 'Time', 'tonys-sportspress-enhancements' ),
'season' => __( 'Season', 'tonys-sportspress-enhancements' ),
'league' => __( 'League', 'tonys-sportspress-enhancements' ),
'away_team' => __( 'Away Team', 'tonys-sportspress-enhancements' ),
'home_team' => __( 'Home Team', 'tonys-sportspress-enhancements' ),
'field_name' => __( 'Field Name', 'tonys-sportspress-enhancements' ),
'officials' => __( 'Officials', 'tonys-sportspress-enhancements' ),
),
'team' => array(
'label' => __( 'Extra Label', 'tonys-sportspress-enhancements' ),
'date' => __( 'Date', 'tonys-sportspress-enhancements' ),
'time' => __( 'Time', 'tonys-sportspress-enhancements' ),
'season' => __( 'Season', 'tonys-sportspress-enhancements' ),
'league' => __( 'League', 'tonys-sportspress-enhancements' ),
'team_name' => __( 'Team', 'tonys-sportspress-enhancements' ),
'opponent_name' => __( 'Opponent', 'tonys-sportspress-enhancements' ),
'location_flag' => __( 'Home/Away', 'tonys-sportspress-enhancements' ),
'field_name' => __( 'Field Name', 'tonys-sportspress-enhancements' ),
'field_abbreviation' => __( 'Field Abbreviation', 'tonys-sportspress-enhancements' ),
'field_short_name' => __( 'Field Short Name', 'tonys-sportspress-enhancements' ),
'officials' => __( 'Officials', 'tonys-sportspress-enhancements' ),
'home_team' => __( 'Home Team', 'tonys-sportspress-enhancements' ),
'away_team' => __( 'Away Team', 'tonys-sportspress-enhancements' ),
),
);
}
/**
* Get default columns for an export format.
*
* @param string $format Export format.
* @return array
*/
function tse_sp_event_export_get_default_columns( $format ) {
$defaults = array(
'matchup' => array( 'date', 'time', 'season', 'league', 'away_team', 'home_team', 'field_name' ),
'team' => array( 'label', 'date', 'time', 'season', 'league', 'opponent_name', 'location_flag', 'field_name' ),
);
return isset( $defaults[ $format ] ) ? $defaults[ $format ] : $defaults['matchup'];
}
/**
* Sanitize an export format.
*
* @param string $format Raw format.
* @return string
*/
function tse_sp_event_export_sanitize_format( $format ) {
$format = sanitize_key( (string) $format );
$formats = tse_sp_event_export_get_formats();
return isset( $formats[ $format ] ) ? $format : 'matchup';
}
/**
* Sanitize requested columns for an export format.
*
* @param string $format Export format.
* @param string|array $columns Requested columns.
* @return array
*/
function tse_sp_event_export_sanitize_columns( $format, $columns ) {
$definitions = tse_sp_event_export_get_column_definitions();
$available = isset( $definitions[ $format ] ) ? $definitions[ $format ] : array();
if ( ! is_array( $columns ) ) {
$columns = explode( ',', (string) $columns );
}
$sanitized = array();
foreach ( $columns as $column ) {
$key = sanitize_key( (string) $column );
if ( '' === $key || ! isset( $available[ $key ] ) || in_array( $key, $sanitized, true ) ) {
continue;
}
$sanitized[] = $key;
}
if ( empty( $sanitized ) ) {
return tse_sp_event_export_get_default_columns( $format );
}
return $sanitized;
}
/**
* Normalize one or more numeric IDs from a request value.
*
* Accepts scalars, arrays, and comma-delimited strings.
*
* @param mixed $value Raw request value.
* @return int[]
*/
function tse_sp_event_export_parse_id_list( $value ) {
if ( is_array( $value ) ) {
$raw_values = $value;
} else {
$raw_values = explode( ',', (string) $value );
}
$ids = array();
foreach ( $raw_values as $raw_value ) {
$id = absint( trim( (string) $raw_value ) );
if ( $id > 0 ) {
$ids[] = $id;
}
}
$ids = array_values( array_unique( $ids ) );
return $ids;
}
/**
* Normalize export request arguments.
*
* @param array|null $source Optional input source.
* @return array
*/
function tse_sp_event_export_normalize_request_args( $source = null ) {
$source = is_array( $source ) ? $source : $_GET;
$format = isset( $source['format'] ) ? tse_sp_event_export_sanitize_format( wp_unslash( $source['format'] ) ) : 'matchup';
$team_ids = isset( $source['team_id'] ) ? tse_sp_event_export_parse_id_list( wp_unslash( $source['team_id'] ) ) : array();
$season_ids = isset( $source['season_id'] ) ? tse_sp_event_export_parse_id_list( wp_unslash( $source['season_id'] ) ) : array();
$league_ids = isset( $source['league_id'] ) ? tse_sp_event_export_parse_id_list( wp_unslash( $source['league_id'] ) ) : array();
$field_ids = isset( $source['field_id'] ) ? tse_sp_event_export_parse_id_list( wp_unslash( $source['field_id'] ) ) : array();
return array(
'team_id' => isset( $team_ids[0] ) ? $team_ids[0] : 0,
'team_ids' => $team_ids,
'season_id' => isset( $season_ids[0] ) ? $season_ids[0] : 0,
'season_ids'=> $season_ids,
'league_id' => isset( $league_ids[0] ) ? $league_ids[0] : 0,
'league_ids'=> $league_ids,
'field_id' => isset( $field_ids[0] ) ? $field_ids[0] : 0,
'field_ids' => $field_ids,
'format' => $format,
'columns' => isset( $source['columns'] ) ? tse_sp_event_export_sanitize_columns( $format, wp_unslash( $source['columns'] ) ) : tse_sp_event_export_get_default_columns( $format ),
);
}
/**
* Validate export filters for the requested format.
*
* @param array $filters Export filters.
* @return void
*/
function tse_sp_event_export_validate_filters( $filters ) {
$format = tse_sp_event_export_sanitize_format( isset( $filters['format'] ) ? $filters['format'] : 'matchup' );
$team_ids = isset( $filters['team_ids'] ) && is_array( $filters['team_ids'] ) ? $filters['team_ids'] : array();
if ( 'team' !== $format ) {
return;
}
if ( empty( $team_ids ) ) {
wp_die( esc_html__( 'Team format requires a team filter.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 400 ) );
}
if ( count( $team_ids ) > 1 ) {
wp_die( esc_html__( 'Team format does not support multiple teams.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 400 ) );
}
}
/**
* Query matching event posts for export.
*
* @param array $filters Export filters.
* @return WP_Post[]
*/
function tse_sp_event_export_query_posts( $filters ) {
$team_ids = isset( $filters['team_ids'] ) && is_array( $filters['team_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['team_ids'] ) ) ) : array();
$season_ids = isset( $filters['season_ids'] ) && is_array( $filters['season_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['season_ids'] ) ) ) : array();
$league_ids = isset( $filters['league_ids'] ) && is_array( $filters['league_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['league_ids'] ) ) ) : array();
$field_ids = isset( $filters['field_ids'] ) && is_array( $filters['field_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['field_ids'] ) ) ) : array();
$args = array(
'post_type' => 'sp_event',
'post_status' => array( 'publish', 'future' ),
'posts_per_page' => -1,
'orderby' => 'date',
'order' => 'ASC',
'no_found_rows' => true,
);
if ( ! empty( $team_ids ) ) {
$args['meta_query'] = array(
array(
'key' => 'sp_team',
'value' => array_map( 'strval', $team_ids ),
'compare' => 'IN',
),
);
}
$tax_query = array();
if ( ! empty( $season_ids ) ) {
$tax_query[] = array(
'taxonomy' => 'sp_season',
'field' => 'term_id',
'terms' => $season_ids,
);
}
if ( ! empty( $league_ids ) ) {
$tax_query[] = array(
'taxonomy' => 'sp_league',
'field' => 'term_id',
'terms' => $league_ids,
);
}
if ( ! empty( $field_ids ) ) {
$tax_query[] = array(
'taxonomy' => 'sp_venue',
'field' => 'term_id',
'terms' => $field_ids,
);
}
if ( ! empty( $tax_query ) ) {
if ( count( $tax_query ) > 1 ) {
$tax_query['relation'] = 'AND';
}
$args['tax_query'] = $tax_query;
}
$query = new WP_Query( $args );
return is_array( $query->posts ) ? $query->posts : array();
}
/**
* Query matching schedule events for export.
*
* @param array $filters Export filters.
* @return array
*/
function tse_sp_event_export_get_events( $filters ) {
$team_id = isset( $filters['team_id'] ) ? absint( $filters['team_id'] ) : 0;
$selected_ids = isset( $filters['team_ids'] ) && is_array( $filters['team_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['team_ids'] ) ) ) : array();
$query_posts = tse_sp_event_export_query_posts( $filters );
$events = array();
$team_name = $team_id > 0 ? get_the_title( $team_id ) : '';
foreach ( $query_posts as $event ) {
$event_id = $event instanceof WP_Post ? (int) $event->ID : 0;
if ( $event_id <= 0 ) {
continue;
}
$teams = array_values( array_unique( array_map( 'intval', get_post_meta( $event_id, 'sp_team', false ) ) ) );
if ( ! empty( $selected_ids ) && empty( array_intersect( $selected_ids, $teams ) ) ) {
continue;
}
$home_id = isset( $teams[0] ) ? (int) $teams[0] : 0;
$away_id = isset( $teams[1] ) ? (int) $teams[1] : 0;
$venue = tse_sp_event_export_get_primary_field( $event_id );
if ( $team_id > 0 ) {
$location_flag = $home_id === $team_id ? 'Home' : 'Away';
$opponent_id = $home_id === $team_id ? $away_id : $home_id;
} else {
$location_flag = '';
$opponent_id = 0;
}
$events[] = array(
'event_id' => $event_id,
'label' => '',
'date' => get_post_time( 'm/d/Y', false, $event_id, true ),
'time' => strtoupper( (string) ( function_exists( 'sp_get_time' ) ? sp_get_time( $event_id ) : get_post_time( get_option( 'time_format' ), false, $event_id, true ) ) ),
'team_name' => is_string( $team_name ) ? $team_name : '',
'opponent_name' => $opponent_id > 0 ? get_the_title( $opponent_id ) : '',
'location_flag' => $location_flag,
'home_team' => $home_id > 0 ? get_the_title( $home_id ) : '',
'away_team' => $away_id > 0 ? get_the_title( $away_id ) : '',
'field_name' => isset( $venue['name'] ) ? $venue['name'] : '',
'field_abbreviation' => isset( $venue['abbreviation'] ) ? $venue['abbreviation'] : '',
'field_short_name' => isset( $venue['short_name'] ) ? $venue['short_name'] : '',
'season' => tse_sp_event_export_get_event_term_names( $event_id, 'sp_season' ),
'league' => tse_sp_event_export_get_event_term_names( $event_id, 'sp_league' ),
'officials' => tse_sp_event_export_get_officials_value( $event_id ),
);
}
foreach ( $events as $index => $event ) {
$events[ $index ]['label'] = sprintf( 'G#%02d', $index + 1 );
}
wp_reset_postdata();
return $events;
}
/**
* Get event term names as a semicolon-delimited string.
*
* @param int $event_id Event ID.
* @param string $taxonomy Taxonomy name.
* @return string
*/
function tse_sp_event_export_get_event_term_names( $event_id, $taxonomy ) {
$terms = get_the_terms( $event_id, $taxonomy );
if ( ! is_array( $terms ) || empty( $terms ) ) {
return '';
}
$names = array_values( array_filter( array_map( 'strval', wp_list_pluck( $terms, 'name' ) ) ) );
return implode( '; ', array_unique( $names ) );
}
/**
* Get primary field metadata for an event.
*
* @param int $event_id Event ID.
* @return array
*/
function tse_sp_event_export_get_primary_field( $event_id ) {
$venues = get_the_terms( $event_id, 'sp_venue' );
if ( ! is_array( $venues ) || ! isset( $venues[0] ) || ! $venues[0] instanceof WP_Term ) {
return array(
'name' => '',
'abbreviation' => '',
'short_name' => '',
);
}
$venue = $venues[0];
return array(
'name' => isset( $venue->name ) ? (string) $venue->name : '',
'abbreviation' => trim( (string) get_term_meta( $venue->term_id, 'tse_abbreviation', true ) ),
'short_name' => trim( (string) get_term_meta( $venue->term_id, 'tse_short_name', true ) ),
);
}
/**
* Get event officials as a semicolon-delimited string.
*
* @param int $event_id Event ID.
* @return string
*/
function tse_sp_event_export_get_officials_value( $event_id ) {
$official_groups = get_post_meta( $event_id, 'sp_officials', true );
if ( ! is_array( $official_groups ) || empty( $official_groups ) ) {
return '';
}
$official_names = array();
foreach ( $official_groups as $official_ids ) {
if ( ! is_array( $official_ids ) ) {
continue;
}
foreach ( $official_ids as $official_id ) {
$official_id = absint( $official_id );
if ( $official_id <= 0 || 'sp_official' !== get_post_type( $official_id ) ) {
continue;
}
$name = get_the_title( $official_id );
if ( '' === $name ) {
continue;
}
$official_names[ $official_id ] = $name;
}
}
if ( empty( $official_names ) ) {
return '';
}
return implode( '; ', array_values( $official_names ) );
}
/**
* Escape iCalendar text content.
*
* @param string $value Raw value.
* @return string
*/
function tse_sp_event_export_escape_ical_text( $value ) {
$value = html_entity_decode( wp_strip_all_tags( (string) $value ), ENT_QUOTES, get_bloginfo( 'charset' ) );
$value = str_replace( array( '\\', "\r\n", "\r", "\n", ',', ';' ), array( '\\\\', '\n', '\n', '\n', '\,', '\;' ), $value );
return $value;
}
/**
* Fold an iCalendar content line.
*
* @param string $line Raw line.
* @return string
*/
function tse_sp_event_export_fold_ical_line( $line ) {
return wordwrap( (string) $line, 60, "\r\n\t", true );
}
/**
* Get the ICS summary for an event.
*
* For matchup format, this mirrors the SportsPress result/title behavior.
* For team format, this becomes a team-centric opponent summary.
*
* @param WP_Post $event Event post.
* @param array $filters Export filters.
* @return string
*/
function tse_sp_event_export_get_ical_summary( $event, $filters ) {
$format = tse_sp_event_export_sanitize_format( isset( $filters['format'] ) ? $filters['format'] : 'matchup' );
$team_ids = isset( $filters['team_ids'] ) && is_array( $filters['team_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['team_ids'] ) ) ) : array();
$team_id = isset( $team_ids[0] ) ? $team_ids[0] : 0;
if ( 'team' === $format && $team_id > 0 ) {
$teams = array_values( array_unique( array_map( 'intval', get_post_meta( $event->ID, 'sp_team', false ) ) ) );
$home_id = isset( $teams[0] ) ? (int) $teams[0] : 0;
$away_id = isset( $teams[1] ) ? (int) $teams[1] : 0;
if ( in_array( $team_id, $teams, true ) ) {
$is_home = $home_id === $team_id;
$opponent_id = $is_home ? $away_id : $home_id;
$opponent = $opponent_id > 0 ? get_the_title( $opponent_id ) : __( 'TBD', 'tonys-sportspress-enhancements' );
$summary = sprintf(
/* translators: 1: preposition, 2: opponent name. */
__( '%1$s %2$s', 'tonys-sportspress-enhancements' ),
$is_home ? 'vs' : 'at',
$opponent
);
return apply_filters( 'sportspress_ical_feed_summary', $summary, $event );
}
}
$main_result = get_option( 'sportspress_primary_result', null );
$results = array();
$teams = (array) get_post_meta( $event->ID, 'sp_team', false );
$teams = array_filter( array_unique( $teams ) );
if ( ! empty( $teams ) ) {
$event_results = get_post_meta( $event->ID, 'sp_results', true );
foreach ( $teams as $team_id ) {
$team_id = absint( $team_id );
if ( $team_id <= 0 ) {
continue;
}
$team = get_post( $team_id );
if ( ! $team instanceof WP_Post ) {
continue;
}
$team_results = is_array( $event_results ) && isset( $event_results[ $team_id ] ) ? $event_results[ $team_id ] : null;
if ( $main_result ) {
$team_result = is_array( $team_results ) && isset( $team_results[ $main_result ] ) ? $team_results[ $main_result ] : null;
} else {
if ( is_array( $team_results ) ) {
end( $team_results );
$team_result = prev( $team_results );
} else {
$team_result = null;
}
}
if ( null !== $team_result && '' !== (string) $team_result ) {
$results[] = get_the_title( $team_id ) . ' ' . $team_result;
}
}
}
$summary = ! empty( $results ) ? implode( ' ', $results ) : $event->post_title;
$summary = preg_replace_callback(
'/(&#[0-9]+;)/',
static function( $matches ) {
return mb_convert_encoding( $matches[1], 'UTF-8', 'HTML-ENTITIES' );
},
$summary
);
return apply_filters( 'sportspress_ical_feed_summary', $summary, $event );
}
/**
* Get the ICS location payload for an event.
*
* @param WP_Post $event Event post.
* @return array
*/
function tse_sp_event_export_get_ical_location_data( $event ) {
$location = '';
$geo = false;
$venues = get_the_terms( $event->ID, 'sp_venue' );
if ( ! is_array( $venues ) || empty( $venues ) ) {
return array(
'location' => $location,
'geo' => $geo,
);
}
$venue = reset( $venues );
$location = $venue->name;
$meta = get_option( 'taxonomy_' . $venue->term_id );
$address = is_array( $meta ) && isset( $meta['sp_address'] ) ? $meta['sp_address'] : false;
if ( false !== $address && '' !== (string) $address ) {
$location = $venue->name . ', ' . $address;
}
$latitude = is_array( $meta ) && isset( $meta['sp_latitude'] ) ? $meta['sp_latitude'] : false;
$longitude = is_array( $meta ) && isset( $meta['sp_longitude'] ) ? $meta['sp_longitude'] : false;
if ( false !== $latitude && false !== $longitude && '' !== (string) $latitude && '' !== (string) $longitude ) {
$geo = $latitude . ';' . $longitude;
}
return array(
'location' => (string) $location,
'geo' => $geo,
);
}
/**
* Build iCalendar output for the requested filters.
*
* @param array $filters Export filters.
* @return string
*/
function tse_sp_event_export_build_ical_output( $filters ) {
$query_posts = tse_sp_event_export_query_posts( $filters );
$locale = substr( get_locale(), 0, 2 );
$timezone = sanitize_option( 'timezone_string', get_option( 'timezone_string' ) );
$url = tse_sp_event_export_get_feed_url( $filters, 'ics' );
$calendar = tse_sp_event_export_get_feed_title( $filters );
$output =
"BEGIN:VCALENDAR\r\n" .
"VERSION:2.0\r\n" .
'PRODID:-//ThemeBoy//SportsPress//' . strtoupper( $locale ) . "\r\n" .
"CALSCALE:GREGORIAN\r\n" .
"METHOD:PUBLISH\r\n" .
'URL:' . tse_sp_event_export_fold_ical_line( $url ) . "\r\n" .
'X-FROM-URL:' . tse_sp_event_export_fold_ical_line( $url ) . "\r\n" .
'NAME:' . tse_sp_event_export_escape_ical_text( $calendar ) . "\r\n" .
'X-WR-CALNAME:' . tse_sp_event_export_escape_ical_text( $calendar ) . "\r\n" .
'DESCRIPTION:' . tse_sp_event_export_escape_ical_text( $calendar ) . "\r\n" .
'X-WR-CALDESC:' . tse_sp_event_export_escape_ical_text( $calendar ) . "\r\n" .
"REFRESH-INTERVAL;VALUE=DURATION:PT2M\r\n" .
"X-PUBLISHED-TTL:PT2M\r\n" .
'TZID:' . $timezone . "\r\n" .
'X-WR-TIMEZONE:' . $timezone . "\r\n";
foreach ( $query_posts as $event ) {
if ( ! $event instanceof WP_Post ) {
continue;
}
$date_format = 'Ymd\THis';
$description = tse_sp_event_export_escape_ical_text( $event->post_content );
$summary = tse_sp_event_export_get_ical_summary( $event, $filters );
$minutes = get_post_meta( $event->ID, 'sp_minutes', true );
$minutes = '' === $minutes ? get_option( 'sportspress_event_minutes', 90 ) : $minutes;
$end = new DateTime( $event->post_date );
$end->add( new DateInterval( 'PT' . absint( $minutes ) . 'M' ) );
$location = tse_sp_event_export_get_ical_location_data( $event );
$event_url = get_permalink( $event );
$output .= "BEGIN:VEVENT\r\n";
$output .= tse_sp_event_export_fold_ical_line( 'SUMMARY:' . tse_sp_event_export_escape_ical_text( $summary ) ) . "\r\n";
$output .= 'UID:' . $event->ID . "\r\n";
$output .= "STATUS:CONFIRMED\r\n";
$output .= "DTSTAMP:19700101T000000\r\n";
$output .= 'DTSTART:' . mysql2date( $date_format, $event->post_date ) . "\r\n";
$output .= 'DTEND:' . $end->format( $date_format ) . "\r\n";
$output .= 'LAST-MODIFIED:' . mysql2date( $date_format, $event->post_modified_gmt ) . "\r\n";
if ( '' !== $description ) {
$output .= tse_sp_event_export_fold_ical_line( 'DESCRIPTION:' . $description ) . "\r\n";
}
if ( '' !== $location['location'] ) {
$output .= tse_sp_event_export_fold_ical_line( 'LOCATION:' . tse_sp_event_export_escape_ical_text( $location['location'] ) ) . "\r\n";
}
if ( ! empty( $location['geo'] ) ) {
$output .= 'GEO:' . $location['geo'] . "\r\n";
}
if ( is_string( $event_url ) && '' !== $event_url ) {
$output .= tse_sp_event_export_fold_ical_line( 'URL:' . esc_url_raw( $event_url ) ) . "\r\n";
}
$output .= "END:VEVENT\r\n";
}
$output .= 'END:VCALENDAR';
return $output;
}
/**
* Stream ICS output.
*
* @param array $filters Export filters.
* @param array $args Optional render args.
* @return void
*/
function tse_sp_event_export_stream_ical( $filters, $args = array() ) {
tse_sp_event_export_validate_filters( $filters );
$disposition = isset( $args['disposition'] ) ? sanitize_key( $args['disposition'] ) : 'inline';
$disposition = in_array( $disposition, array( 'inline', 'attachment' ), true ) ? $disposition : 'inline';
$output = tse_sp_event_export_build_ical_output( $filters );
$filename = tse_sp_event_export_build_filename( $filters ) . '.ics';
$etag = md5( $output );
header( 'Content-type: text/calendar; charset=utf-8' );
header( 'Etag:' . '"' . $etag . '"' );
header( 'Content-Disposition: ' . $disposition . '; filename=' . $filename );
echo $output; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
exit;
}
/**
* Build a CSV row value for a logical column.
*
* @param array $event Normalized event row.
* @param string $column Column key.
* @return string
*/
function tse_sp_event_export_get_row_value( $event, $column ) {
return isset( $event[ $column ] ) ? (string) $event[ $column ] : '';
}
/**
* Build a CSV filename for the export.
*
* @param array $filters Export filters.
* @return string
*/
function tse_sp_event_export_build_filename( $filters ) {
$parts = array( 'schedule' );
$parts = array_merge( $parts, tse_sp_event_export_get_post_slugs( isset( $filters['team_ids'] ) ? $filters['team_ids'] : array() ) );
$parts = array_merge( $parts, tse_sp_event_export_get_term_slugs_by_ids( isset( $filters['season_ids'] ) ? $filters['season_ids'] : array(), 'sp_season' ) );
$parts = array_merge( $parts, tse_sp_event_export_get_term_slugs_by_ids( isset( $filters['league_ids'] ) ? $filters['league_ids'] : array(), 'sp_league' ) );
$parts = array_merge( $parts, tse_sp_event_export_get_term_slugs_by_ids( isset( $filters['field_ids'] ) ? $filters['field_ids'] : array(), 'sp_venue' ) );
$parts[] = tse_sp_event_export_sanitize_format( isset( $filters['format'] ) ? $filters['format'] : 'matchup' );
$parts = array_values( array_filter( $parts ) );
return implode( '-', $parts );
}
/**
* Build a human-readable feed title from filters.
*
* @param array $filters Export filters.
* @return string
*/
function tse_sp_event_export_get_feed_title( $filters ) {
$site_title = trim( (string) get_bloginfo( 'name' ) );
$team_name = tse_sp_event_export_get_post_titles( isset( $filters['team_ids'] ) ? $filters['team_ids'] : array() );
$season_name = tse_sp_event_export_get_term_names_by_ids( isset( $filters['season_ids'] ) ? $filters['season_ids'] : array(), 'sp_season' );
$league_name = tse_sp_event_export_get_term_names_by_ids( isset( $filters['league_ids'] ) ? $filters['league_ids'] : array(), 'sp_league' );
$field_name = tse_sp_event_export_get_term_names_by_ids( isset( $filters['field_ids'] ) ? $filters['field_ids'] : array(), 'sp_venue' );
$title = '' !== $site_title ? $site_title . ' ' . __( 'Event Feed', 'tonys-sportspress-enhancements' ) : __( 'Event Feed', 'tonys-sportspress-enhancements' );
$details = array_filter( array( $team_name, $season_name, $league_name, $field_name ) );
if ( ! empty( $details ) ) {
$title .= ' (' . implode( ' • ', $details ) . ')';
}
return $title;
}
/**
* Get term names for a list of term IDs.
*
* @param array $ids Term IDs.
* @param string $taxonomy Taxonomy name.
* @return string
*/
function tse_sp_event_export_get_term_names_by_ids( $ids, $taxonomy ) {
$names = array();
foreach ( (array) $ids as $id ) {
$term = get_term( absint( $id ), $taxonomy );
if ( $term instanceof WP_Term ) {
$names[] = $term->name;
}
}
$names = array_values( array_unique( array_filter( $names ) ) );
return implode( '; ', $names );
}
/**
* Get post titles for a list of post IDs.
*
* @param array $ids Post IDs.
* @return string
*/
function tse_sp_event_export_get_post_titles( $ids ) {
$titles = array();
foreach ( (array) $ids as $id ) {
$post = get_post( absint( $id ) );
if ( $post instanceof WP_Post ) {
$titles[] = $post->post_title;
}
}
$titles = array_values( array_unique( array_filter( $titles ) ) );
return implode( '; ', $titles );
}
/**
* Get term slugs for a list of term IDs.
*
* @param array $ids Term IDs.
* @param string $taxonomy Taxonomy name.
* @return string[]
*/
function tse_sp_event_export_get_term_slugs_by_ids( $ids, $taxonomy ) {
$slugs = array();
foreach ( (array) $ids as $id ) {
$term = get_term( absint( $id ), $taxonomy );
if ( $term instanceof WP_Term && ! empty( $term->slug ) ) {
$slugs[] = sanitize_title( $term->slug );
}
}
return array_values( array_unique( array_filter( $slugs ) ) );
}
/**
* Get post slugs for a list of post IDs.
*
* @param array $ids Post IDs.
* @return string[]
*/
function tse_sp_event_export_get_post_slugs( $ids ) {
$slugs = array();
foreach ( (array) $ids as $id ) {
$post = get_post( absint( $id ) );
if ( $post instanceof WP_Post ) {
$slugs[] = sanitize_title( $post->post_name ? $post->post_name : $post->post_title );
}
}
return array_values( array_unique( array_filter( $slugs ) ) );
}
/**
* Stream CSV output for the requested export.
*
* @param array $filters Export filters.
* @param array $args Optional render args.
* @return void
*/
function tse_sp_event_export_stream_csv( $filters, $args = array() ) {
$filters['format'] = tse_sp_event_export_sanitize_format( isset( $filters['format'] ) ? $filters['format'] : 'matchup' );
$filters['columns'] = tse_sp_event_export_sanitize_columns( $filters['format'], isset( $filters['columns'] ) ? $filters['columns'] : array() );
tse_sp_event_export_validate_filters( $filters );
$definitions = tse_sp_event_export_get_column_definitions();
$headers = array();
foreach ( $filters['columns'] as $column ) {
$headers[] = $definitions[ $filters['format'] ][ $column ];
}
$events = tse_sp_event_export_get_events( $filters );
$disposition = isset( $args['disposition'] ) ? sanitize_key( $args['disposition'] ) : 'inline';
$disposition = in_array( $disposition, array( 'inline', 'attachment' ), true ) ? $disposition : 'inline';
$filename = tse_sp_event_export_build_filename( $filters ) . '.csv';
header( 'Content-Type: text/csv; charset=utf-8' );
header( 'Content-Disposition: ' . $disposition . '; filename=' . $filename );
$output = fopen( 'php://output', 'w' );
if ( false === $output ) {
wp_die( esc_html__( 'Unable to generate the CSV export.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 500 ) );
}
fwrite( $output, "\xEF\xBB\xBF" );
fputcsv( $output, $headers );
foreach ( $events as $event ) {
$row = array();
foreach ( $filters['columns'] as $column ) {
$row[] = tse_sp_event_export_get_row_value( $event, $column );
}
fputcsv( $output, $row );
}
fclose( $output );
exit;
}
/**
* Render the standalone CSV feed.
*
* @return void
*/
function tse_sp_event_export_render_feed() {
if ( ! class_exists( 'SP_Calendar' ) && ! post_type_exists( 'sp_event' ) ) {
wp_die( esc_html__( 'SportsPress is required for this feed.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 500 ) );
}
tse_sp_event_export_stream_csv( tse_sp_event_export_normalize_request_args() );
}
/**
* Render the standalone ICS feed.
*
* @return void
*/
function tse_sp_event_export_render_ical_feed() {
if ( ! class_exists( 'SP_Calendar' ) && ! post_type_exists( 'sp_event' ) ) {
wp_die( esc_html__( 'SportsPress is required for this feed.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 500 ) );
}
tse_sp_event_export_stream_ical( tse_sp_event_export_normalize_request_args() );
}
/**
* Build a shareable feed URL.
*
* @param array $args Export args.
* @param string $feed_type Feed type.
* @return string
*/
function tse_sp_event_export_get_feed_url( $args = array(), $feed_type = 'csv' ) {
$filters = tse_sp_event_export_normalize_request_args( $args );
$feed = 'ics' === sanitize_key( $feed_type ) ? 'sp-ics' : 'sp-csv';
$query = array(
'feed' => $feed,
'format' => $filters['format'],
'team_id' => ! empty( $filters['team_ids'] ) ? implode( ',', $filters['team_ids'] ) : '',
'season_id' => ! empty( $filters['season_ids'] ) ? implode( ',', $filters['season_ids'] ) : '',
'league_id' => ! empty( $filters['league_ids'] ) ? implode( ',', $filters['league_ids'] ) : '',
'field_id' => ! empty( $filters['field_ids'] ) ? implode( ',', $filters['field_ids'] ) : '',
'columns' => implode( ',', $filters['columns'] ),
);
return add_query_arg( array_filter( $query, 'strlen' ), home_url( '/' ) );
}

View File

@@ -244,6 +244,73 @@ if ( ! class_exists( 'Tony_Sportspress_Printable_Calendars' ) ) {
}
$current_tab = $this->current_settings_tab();
echo '<div class="wrap">';
echo '<h1>' . esc_html__( 'Tony\'s Settings', 'tonys-sportspress-enhancements' ) . '</h1>';
$this->render_settings_tabs( $current_tab );
if ( self::TAB_PRINTABLE === $current_tab ) {
$this->render_printable_settings_tab( $current_tab );
} else {
do_action( 'tse_tonys_settings_render_tab_' . $current_tab );
}
echo '</div>';
}
/**
* Render Tony's settings tabs.
*
* @param string $current_tab Current tab key.
*/
private function render_settings_tabs( $current_tab ) {
$tabs = apply_filters(
'tse_tonys_settings_tabs',
array(
self::TAB_PRINTABLE => __( 'Printable Calendars', 'tonys-sportspress-enhancements' ),
)
);
echo '<nav class="nav-tab-wrapper" style="margin-bottom:20px;">';
foreach ( $tabs as $tab => $label ) {
$url = add_query_arg(
array(
'page' => self::PAGE_SLUG,
'tab' => $tab,
),
admin_url( 'admin.php' )
);
$class = $tab === $current_tab ? ' nav-tab-active' : '';
echo '<a class="nav-tab' . esc_attr( $class ) . '" href="' . esc_url( $url ) . '">' . esc_html( $label ) . '</a>';
}
echo '</nav>';
}
/**
* Resolve the current settings tab.
*
* @return string
*/
private function current_settings_tab() {
$tab = isset( $_GET['tab'] ) ? sanitize_key( wp_unslash( $_GET['tab'] ) ) : self::TAB_PRINTABLE;
$tabs = apply_filters(
'tse_tonys_settings_tabs',
array(
self::TAB_PRINTABLE => __( 'Printable Calendars', 'tonys-sportspress-enhancements' ),
)
);
return isset( $tabs[ $tab ] ) ? $tab : self::TAB_PRINTABLE;
}
/**
* Render printable settings tab content.
*
* @param string $current_tab Current tab key.
* @return void
*/
private function render_printable_settings_tab( $current_tab ) {
$season_id = $this->selected_season_id();
$seasons = $this->get_seasons();
$venues = $this->get_venues_for_season( $season_id );
@@ -253,9 +320,6 @@ if ( ! class_exists( 'Tony_Sportspress_Printable_Calendars' ) ) {
$season_overrides = isset( $overrides[ $season_key ] ) && is_array( $overrides[ $season_key ] ) ? $overrides[ $season_key ] : array();
$season_primary_flags = isset( $primary_flags[ $season_key ] ) && is_array( $primary_flags[ $season_key ] ) ? $primary_flags[ $season_key ] : array();
echo '<div class="wrap">';
echo '<h1>' . esc_html__( 'Tony\'s Settings', 'tonys-sportspress-enhancements' ) . '</h1>';
$this->render_settings_tabs( $current_tab );
echo '<form method="post" action="options.php">';
settings_fields( self::OPTION_GROUP );
@@ -349,46 +413,203 @@ if ( ! class_exists( 'Tony_Sportspress_Printable_Calendars' ) ) {
submit_button( __( 'Save Settings', 'tonys-sportspress-enhancements' ) );
echo '</form>';
$this->render_printable_url_builder( $season_id );
}
/**
* Render printable calendar URL builder.
*
* @param int $season_id Current season context.
* @return void
*/
private function render_printable_url_builder( $season_id ) {
$leagues = function_exists( 'tse_sp_schedule_exporter_get_leagues' ) ? tse_sp_schedule_exporter_get_leagues() : array();
$teams = function_exists( 'tse_sp_schedule_exporter_get_teams' ) ? tse_sp_schedule_exporter_get_teams() : array();
$paper = '11x17';
echo '<div class="tse-printable-url-builder" style="max-width:1100px;margin-top:28px;padding:20px 24px;border:1px solid #dcdcde;background:#fff;">';
echo '<h2 style="margin-top:0;">' . esc_html__( 'Printable Calendar URL Builder', 'tonys-sportspress-enhancements' ) . '</h2>';
echo '<p>' . esc_html__( 'Build a shareable printable calendar URL with team, season, league, paper size, and optional auto-print.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '<table class="form-table" role="presentation"><tbody>';
echo '<tr><th scope="row"><label for="tse-printable-builder-team">' . esc_html__( 'Team', 'tonys-sportspress-enhancements' ) . '</label></th><td>';
echo '<select id="tse-printable-builder-team" style="min-width:280px;">';
echo '<option value="0">' . esc_html__( 'Choose a team', 'tonys-sportspress-enhancements' ) . '</option>';
foreach ( $teams as $team ) {
if ( ! $team instanceof WP_Post ) {
continue;
}
echo '<option value="' . esc_attr( (string) $team->ID ) . '">' . esc_html( $team->post_title ) . '</option>';
}
echo '</select>';
echo '</td></tr>';
echo '<tr><th scope="row"><label for="tse-printable-builder-season">' . esc_html__( 'Season', 'tonys-sportspress-enhancements' ) . '</label></th><td>';
echo '<select id="tse-printable-builder-season" style="min-width:280px;">';
echo '<option value="0">' . esc_html__( 'Current season', 'tonys-sportspress-enhancements' ) . '</option>';
foreach ( $this->get_seasons() as $season ) {
if ( ! $season instanceof WP_Term ) {
continue;
}
echo '<option value="' . esc_attr( (string) $season->term_id ) . '" ' . selected( $season_id, (int) $season->term_id, false ) . '>' . esc_html( $season->name ) . '</option>';
}
echo '</select>';
echo '</td></tr>';
echo '<tr><th scope="row"><label for="tse-printable-builder-league">' . esc_html__( 'League', 'tonys-sportspress-enhancements' ) . '</label></th><td>';
echo '<select id="tse-printable-builder-league" style="min-width:280px;">';
echo '<option value="0">' . esc_html__( 'Any league', 'tonys-sportspress-enhancements' ) . '</option>';
foreach ( $leagues as $league ) {
if ( ! $league instanceof WP_Term ) {
continue;
}
echo '<option value="' . esc_attr( (string) $league->term_id ) . '">' . esc_html( $league->name ) . '</option>';
}
echo '</select>';
echo '</td></tr>';
echo '<tr><th scope="row"><label for="tse-printable-builder-paper">' . esc_html__( 'Paper Size', 'tonys-sportspress-enhancements' ) . '</label></th><td>';
echo '<select id="tse-printable-builder-paper" style="min-width:280px;">';
foreach ( array( 'letter' => __( 'Letter', 'tonys-sportspress-enhancements' ), '11x17' => __( '11x17 / Ledger', 'tonys-sportspress-enhancements' ) ) as $paper_value => $paper_label ) {
echo '<option value="' . esc_attr( $paper_value ) . '" ' . selected( $paper, $paper_value, false ) . '>' . esc_html( $paper_label ) . '</option>';
}
echo '</select>';
echo '</td></tr>';
echo '<tr><th scope="row">' . esc_html__( 'Options', 'tonys-sportspress-enhancements' ) . '</th><td>';
echo '<label for="tse-printable-builder-autoprint" style="display:inline-flex;align-items:center;gap:6px;">';
echo '<input id="tse-printable-builder-autoprint" type="checkbox" value="1" />';
echo esc_html__( 'Auto-open print dialog', 'tonys-sportspress-enhancements' );
echo '</label>';
echo '</td></tr>';
echo '</tbody></table>';
echo '<h3 style="margin-top:24px;">' . esc_html__( 'Generated URL', 'tonys-sportspress-enhancements' ) . '</h3>';
echo '<div style="display:flex;align-items:center;gap:8px;max-width:100%;">';
echo '<input type="text" id="tse-printable-builder-output" class="large-text code" readonly="readonly" />';
echo '<button type="button" id="tse-printable-builder-copy" class="button" aria-label="' . esc_attr__( 'Copy URL', 'tonys-sportspress-enhancements' ) . '" title="' . esc_attr__( 'Copy URL', 'tonys-sportspress-enhancements' ) . '" style="display:inline-flex;align-items:center;justify-content:center;min-width:40px;padding:0 10px;">';
echo '<span aria-hidden="true" style="font-size:16px;line-height:1;">⧉</span>';
echo '</button>';
echo '</div>';
echo '<p><a id="tse-printable-builder-open" class="button button-primary" href="' . esc_url( home_url( '/' ) ) . '" target="_blank" rel="noopener noreferrer">' . esc_html__( 'Open Printable URL', 'tonys-sportspress-enhancements' ) . '</a></p>';
echo '<p class="description">' . esc_html__( 'The printable route requires a single team selection.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</div>';
$this->render_printable_url_builder_script();
}
/**
* Render Tony's settings tabs.
* Render printable URL builder script.
*
* @param string $current_tab Current tab key.
* @return void
*/
private function render_settings_tabs( $current_tab ) {
$tabs = array(
self::TAB_PRINTABLE => __( 'Printable Calendars', 'tonys-sportspress-enhancements' ),
);
echo '<nav class="nav-tab-wrapper" style="margin-bottom:20px;">';
foreach ( $tabs as $tab => $label ) {
$url = add_query_arg(
array(
'page' => self::PAGE_SLUG,
'tab' => $tab,
),
admin_url( 'admin.php' )
);
$class = self::TAB_PRINTABLE === $tab && self::TAB_PRINTABLE === $current_tab ? ' nav-tab-active' : '';
echo '<a class="nav-tab' . esc_attr( $class ) . '" href="' . esc_url( $url ) . '">' . esc_html( $label ) . '</a>';
}
echo '</nav>';
private function render_printable_url_builder_script() {
$base_url = home_url( '/' );
$query_flag = self::QUERY_FLAG;
?>
<script>
(function(){
var root = document.querySelector('.tse-printable-url-builder');
if (!root) {
return;
}
/**
* Resolve the current settings tab.
*
* @return string
*/
private function current_settings_tab() {
$tab = isset( $_GET['tab'] ) ? sanitize_key( wp_unslash( $_GET['tab'] ) ) : self::TAB_PRINTABLE;
var baseUrl = <?php echo wp_json_encode( $base_url ); ?>;
var queryFlag = <?php echo wp_json_encode( $query_flag ); ?>;
var team = root.querySelector('#tse-printable-builder-team');
var season = root.querySelector('#tse-printable-builder-season');
var league = root.querySelector('#tse-printable-builder-league');
var paper = root.querySelector('#tse-printable-builder-paper');
var autoprint = root.querySelector('#tse-printable-builder-autoprint');
var output = root.querySelector('#tse-printable-builder-output');
var copyButton = root.querySelector('#tse-printable-builder-copy');
var openLink = root.querySelector('#tse-printable-builder-open');
return self::TAB_PRINTABLE === $tab ? $tab : self::TAB_PRINTABLE;
function buildUrl() {
var url = new URL(baseUrl, window.location.origin);
url.searchParams.set(queryFlag, '1');
if (team.value && team.value !== '0') {
url.searchParams.set('sp_team', team.value);
} else {
url.searchParams.delete('sp_team');
}
if (season.value && season.value !== '0') {
url.searchParams.set('sp_season', season.value);
} else {
url.searchParams.delete('sp_season');
}
if (league.value && league.value !== '0') {
url.searchParams.set('sp_league', league.value);
} else {
url.searchParams.delete('sp_league');
}
if (paper.value) {
url.searchParams.set('paper', paper.value);
}
if (autoprint.checked) {
url.searchParams.set('autoprint', '1');
} else {
url.searchParams.delete('autoprint');
}
output.value = url.toString();
openLink.href = url.toString();
openLink.toggleAttribute('disabled', !(team.value && team.value !== '0'));
}
[team, season, league, paper, autoprint].forEach(function(input){
input.addEventListener('change', buildUrl);
});
if (copyButton) {
copyButton.addEventListener('click', function(){
var value = output.value || '';
if (!value) {
return;
}
var defaultTitle = copyButton.getAttribute('data-default-title') || copyButton.title || 'Copy URL';
copyButton.setAttribute('data-default-title', defaultTitle);
function markCopied() {
copyButton.title = 'Copied';
window.setTimeout(function(){
copyButton.title = defaultTitle;
}, 1200);
}
if (navigator.clipboard && navigator.clipboard.writeText) {
navigator.clipboard.writeText(value).then(markCopied).catch(function(){
output.focus();
output.select();
document.execCommand('copy');
markCopied();
});
return;
}
output.focus();
output.select();
document.execCommand('copy');
markCopied();
});
}
buildUrl();
})();
</script>
<?php
}
/**
* Render the printable page when the flag is present.
*/

View File

@@ -40,104 +40,18 @@ function tse_sp_schedule_exporter_handle_download() {
check_admin_referer( 'tse_schedule_export' );
$team_id = isset( $_GET['team_id'] ) ? absint( wp_unslash( $_GET['team_id'] ) ) : 0;
$season_id = isset( $_GET['season_id'] ) ? absint( wp_unslash( $_GET['season_id'] ) ) : 0;
$league_id = isset( $_GET['league_id'] ) ? absint( wp_unslash( $_GET['league_id'] ) ) : 0;
$format = isset( $_GET['format'] ) ? sanitize_key( wp_unslash( $_GET['format'] ) ) : '';
$filters = tse_sp_event_export_normalize_request_args();
if ( $team_id <= 0 || 'sp_team' !== get_post_type( $team_id ) ) {
if ( $filters['team_id'] <= 0 || 'sp_team' !== get_post_type( $filters['team_id'] ) ) {
wp_die( esc_html__( 'Choose a valid team before exporting.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 400 ) );
}
if ( ! in_array( $format, array( 'matchup', 'team' ), true ) ) {
wp_die( esc_html__( 'Choose a valid export format.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 400 ) );
}
$events = tse_sp_schedule_exporter_get_events( $team_id, $season_id, $league_id );
$team = get_post( $team_id );
if ( ! $team instanceof WP_Post ) {
wp_die( esc_html__( 'The selected team could not be loaded.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 404 ) );
}
$filename = sanitize_title( $team->post_name ? $team->post_name : $team->post_title );
if ( '' === $filename ) {
$filename = 'schedule';
}
if ( $season_id > 0 ) {
$season = get_term( $season_id, 'sp_season' );
if ( $season && ! is_wp_error( $season ) && ! empty( $season->slug ) ) {
$filename .= '-' . sanitize_title( $season->slug );
}
}
$filename .= '-' . $format . '.csv';
header( 'Content-Type: text/csv; charset=utf-8' );
header( 'Content-Disposition: attachment; filename=' . $filename );
$output = fopen( 'php://output', 'w' );
if ( false === $output ) {
wp_die( esc_html__( 'Unable to start the CSV export.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 500 ) );
}
fwrite( $output, "\xEF\xBB\xBF" );
if ( 'matchup' === $format ) {
fputcsv(
$output,
tse_sp_event_export_stream_csv(
$filters,
array(
'Date',
'Time',
'Away Team',
'Home Team',
'Field Name',
'disposition' => 'attachment',
)
);
foreach ( $events as $event ) {
fputcsv(
$output,
array(
$event['date'],
$event['time'],
$event['away_team'],
$event['home_team'],
$event['venue_name'],
)
);
}
} else {
fputcsv(
$output,
array(
'Extra Label',
'Date',
'Time',
'Opponent',
'Home/Away',
'Venue',
)
);
foreach ( $events as $event ) {
fputcsv(
$output,
array(
$event['label'],
$event['date'],
$event['time'],
$event['opponent_name'],
$event['location_flag'],
$event['venue_name'],
)
);
}
}
fclose( $output );
exit;
}
add_action( 'admin_post_tse_schedule_export', 'tse_sp_schedule_exporter_handle_download' );
add_action( 'admin_post_nopriv_tse_schedule_export', 'tse_sp_schedule_exporter_handle_download' );
@@ -165,7 +79,7 @@ function tse_sp_schedule_exporter_register_block() {
array(
'api_version' => 3,
'title' => __( 'Schedule Exporter', 'tonys-sportspress-enhancements' ),
'description' => __( 'Shows the public schedule exporter with CSV and printable PDF options.', 'tonys-sportspress-enhancements' ),
'description' => __( 'Shows the public schedule exporter with CSV, iCal, and printable page options.', 'tonys-sportspress-enhancements' ),
'category' => 'widgets',
'icon' => 'calendar-alt',
'editor_script' => 'tse-schedule-exporter-block',
@@ -193,19 +107,52 @@ function tse_sp_schedule_exporter_render_admin_page() {
$season_id = tse_sp_schedule_exporter_resolve_season_id( $seasons );
$teams = tse_sp_schedule_exporter_get_teams( $league_id, $season_id );
$team_id = tse_sp_schedule_exporter_resolve_team_id( $teams );
$paper = tse_sp_schedule_exporter_resolve_paper_size();
$fields = tse_sp_schedule_exporter_get_fields();
$field_id = tse_sp_schedule_exporter_resolve_field_id( $fields );
$export_type = tse_sp_schedule_exporter_resolve_export_type();
$subformat = tse_sp_schedule_exporter_resolve_subformat();
echo '<div class="wrap">';
echo '<h1>' . esc_html__( 'Schedule Exporter', 'tonys-sportspress-enhancements' ) . '</h1>';
echo '<p>' . esc_html__( 'Choose a team and season, then export the schedule as CSV or open the printable schedule in a PDF-ready print view.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '<p>' . esc_html__( 'Choose filters once, then generate the CSV feed, iCal link, or printable page URL from the same controls.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '<form method="get" action="' . esc_url( admin_url( 'admin.php' ) ) . '" class="tse-schedule-exporter-form" style="max-width:960px;margin:20px 0 28px;">';
echo '<form method="get" action="' . esc_url( admin_url( 'admin.php' ) ) . '" class="tse-schedule-exporter-form" style="max-width:720px;margin:20px 0 28px;">';
echo '<input type="hidden" name="page" value="tse-schedule-exporter" />';
echo '<table class="form-table" role="presentation"><tbody>';
echo '<div>';
echo '<tr>';
echo '<th scope="row"><label for="tse-schedule-exporter-league">' . esc_html__( 'League', 'tonys-sportspress-enhancements' ) . '</label></th>';
echo '<td><select id="tse-schedule-exporter-league" name="league_id" data-auto-submit="1">';
echo '<div style="margin-bottom:16px;">';
echo '<label for="tse-schedule-exporter-export-type"><strong>' . esc_html__( 'Format', 'tonys-sportspress-enhancements' ) . '</strong></label><br />';
echo '<select id="tse-schedule-exporter-export-type" name="export_type">';
foreach ( tse_sp_schedule_exporter_get_export_types() as $type_key => $type_label ) {
printf(
'<option value="%1$s" %2$s>%3$s</option>',
esc_attr( $type_key ),
selected( $export_type, $type_key, false ),
esc_html( $type_label )
);
}
echo '</select>';
echo '<p class="description">' . esc_html__( 'CSV builds a feed URL, iCal Link builds a subscription URL, and Printable opens the printable page.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</div>';
echo '<div data-subformat-wrap="1" style="margin-bottom:16px;">';
echo '<label for="tse-schedule-exporter-subformat"><strong>' . esc_html__( 'CSV Layout', 'tonys-sportspress-enhancements' ) . '</strong></label><br />';
echo '<select id="tse-schedule-exporter-subformat" name="subformat">';
foreach ( tse_sp_event_export_get_formats() as $format_key => $format_definition ) {
printf(
'<option value="%1$s" %2$s>%3$s</option>',
esc_attr( $format_key ),
selected( $subformat, $format_key, false ),
esc_html( $format_definition['label'] )
);
}
echo '</select>';
echo '<p class="description">' . esc_html__( 'Matchup is away vs home. Team is opponent-based and requires one specific team.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</div>';
echo '<div style="margin-bottom:16px;">';
echo '<label for="tse-schedule-exporter-league"><strong>' . esc_html__( 'League', 'tonys-sportspress-enhancements' ) . '</strong></label><br />';
echo '<select id="tse-schedule-exporter-league" name="league_id" data-auto-submit="1">';
foreach ( $leagues as $league ) {
printf(
'<option value="%1$s" %2$s>%3$s</option>',
@@ -214,12 +161,12 @@ function tse_sp_schedule_exporter_render_admin_page() {
esc_html( $league->name )
);
}
echo '</select></td>';
echo '</tr>';
echo '</select>';
echo '</div>';
echo '<tr>';
echo '<th scope="row"><label for="tse-schedule-exporter-season">' . esc_html__( 'Season', 'tonys-sportspress-enhancements' ) . '</label></th>';
echo '<td><select id="tse-schedule-exporter-season" name="season_id" data-auto-submit="1">';
echo '<div style="margin-bottom:16px;">';
echo '<label for="tse-schedule-exporter-season"><strong>' . esc_html__( 'Season', 'tonys-sportspress-enhancements' ) . '</strong></label><br />';
echo '<select id="tse-schedule-exporter-season" name="season_id" data-auto-submit="1">';
echo '<option value="0">' . esc_html__( 'Current / All matching events', 'tonys-sportspress-enhancements' ) . '</option>';
foreach ( $seasons as $season ) {
printf(
@@ -229,28 +176,13 @@ function tse_sp_schedule_exporter_render_admin_page() {
esc_html( $season->name )
);
}
echo '</select></td>';
echo '</tr>';
echo '<tr>';
echo '<th scope="row"><label for="tse-schedule-exporter-paper">' . esc_html__( 'Paper Size', 'tonys-sportspress-enhancements' ) . '</label></th>';
echo '<td><select id="tse-schedule-exporter-paper" name="paper">';
foreach ( tse_sp_schedule_exporter_get_paper_sizes() as $paper_value => $paper_label ) {
printf(
'<option value="%1$s" %2$s>%3$s</option>',
esc_attr( $paper_value ),
selected( $paper, $paper_value, false ),
esc_html( $paper_label )
);
}
echo '</select>';
echo '<p class="description">' . esc_html__( 'The PDF option opens the existing printable schedule and triggers the browser print dialog so you can save it as a PDF.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</td>';
echo '</tr>';
echo '</div>';
echo '<tr>';
echo '<th scope="row"><label for="tse-schedule-exporter-team">' . esc_html__( 'Team', 'tonys-sportspress-enhancements' ) . '</label></th>';
echo '<td><select id="tse-schedule-exporter-team" name="team_id">';
echo '<div style="margin-bottom:16px;">';
echo '<label for="tse-schedule-exporter-team"><strong>' . esc_html__( 'Team', 'tonys-sportspress-enhancements' ) . '</strong></label><br />';
echo '<select id="tse-schedule-exporter-team" name="team_id">';
echo '<option value="0">' . esc_html__( 'All teams', 'tonys-sportspress-enhancements' ) . '</option>';
foreach ( $teams as $team ) {
printf(
'<option value="%1$s" %2$s>%3$s</option>',
@@ -261,10 +193,25 @@ function tse_sp_schedule_exporter_render_admin_page() {
}
echo '</select>';
echo '<p class="description">' . esc_html__( 'Teams are filtered by the selected league and season.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</td>';
echo '</tr>';
echo '</div>';
echo '</tbody></table>';
echo '<div style="margin-bottom:16px;">';
echo '<label for="tse-schedule-exporter-field"><strong>' . esc_html__( 'Field', 'tonys-sportspress-enhancements' ) . '</strong></label><br />';
echo '<select id="tse-schedule-exporter-field" name="field_id">';
echo '<option value="0">' . esc_html__( 'All fields', 'tonys-sportspress-enhancements' ) . '</option>';
foreach ( $fields as $field ) {
printf(
'<option value="%1$s" %2$s>%3$s</option>',
esc_attr( (string) $field->term_id ),
selected( $field_id, (int) $field->term_id, false ),
esc_html( $field->name )
);
}
echo '</select>';
echo '<p class="description">' . esc_html__( 'Use the field filter to narrow the feed to a specific venue.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</div>';
echo '</div>';
echo '</form>';
if ( empty( $teams ) ) {
@@ -274,47 +221,40 @@ function tse_sp_schedule_exporter_render_admin_page() {
}
echo '<div style="max-width:960px;padding:20px 24px;border:1px solid #dcdcde;background:#fff;">';
echo '<h2 style="margin-top:0;">' . esc_html__( 'Exports', 'tonys-sportspress-enhancements' ) . '</h2>';
echo '<h2 style="margin-top:0;">' . esc_html__( 'Output URL', 'tonys-sportspress-enhancements' ) . '</h2>';
echo '<p>' . esc_html__( 'The generated URL below updates from the shared controls above.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '<table class="widefat striped" style="max-width:100%;margin-top:16px;"><tbody>';
foreach ( array(
array(
'format' => 'matchup',
'label' => __( 'Download Matchup CSV', 'tonys-sportspress-enhancements' ),
'description' => __( 'Date, time, away team, home team, and field name.', 'tonys-sportspress-enhancements' ),
),
array(
'format' => 'team',
'label' => __( 'Download Team CSV', 'tonys-sportspress-enhancements' ),
'description' => __( 'TeamSnap-compatible layout with game label, opponent, home/away flag, and venue.', 'tonys-sportspress-enhancements' ),
),
) as $export_option ) {
$url = wp_nonce_url(
add_query_arg(
array(
'action' => 'tse_schedule_export',
tse_sp_schedule_exporter_render_column_picker( 'matchup', 'admin', $subformat );
tse_sp_schedule_exporter_render_column_picker( 'team', 'admin', $subformat );
$base_args = array(
'league_id' => $league_id,
'team_id' => $team_id,
'season_id' => $season_id,
'format' => $export_option['format'],
),
admin_url( 'admin-post.php' )
),
'tse_schedule_export'
'field_id' => $field_id,
'format' => $subformat,
);
$csv_url = tse_sp_event_export_get_feed_url( $base_args, 'csv' );
$ics_url = tse_sp_event_export_get_feed_url(
array(
'league_id' => $league_id,
'team_id' => $team_id,
'season_id' => $season_id,
'field_id' => $field_id,
),
'ics'
);
$print_url = tse_sp_schedule_exporter_get_printable_url( $team_id, $season_id, 'letter', $league_id );
$current_url = tse_sp_schedule_exporter_get_output_url( $export_type, $csv_url, $ics_url, $print_url );
echo '<tr>';
echo '<td style="width:240px;"><a class="button button-primary tse-export-link" data-format="' . esc_attr( $export_option['format'] ) . '" href="' . esc_url( $url ) . '">' . esc_html( $export_option['label'] ) . '</a></td>';
echo '<td>' . esc_html( $export_option['description'] ) . '</td>';
echo '</tr>';
}
$pdf_url = tse_sp_schedule_exporter_get_pdf_url( $team_id, $season_id, $paper, $league_id );
echo '<tr>';
echo '<td style="width:240px;"><a class="button tse-pdf-link" href="' . esc_url( $pdf_url ) . '" target="_blank" rel="noopener noreferrer">' . esc_html__( 'Open Printable PDF View', 'tonys-sportspress-enhancements' ) . '</a></td>';
echo '<td>' . esc_html__( 'Opens the printable schedule and launches the browser print dialog so you can save a PDF.', 'tonys-sportspress-enhancements' ) . '</td>';
echo '</tr>';
echo '</tbody></table>';
echo '<div style="display:flex;align-items:center;gap:8px;max-width:100%;margin-top:16px;">';
echo '<input type="text" class="large-text code tse-output-url" readonly="readonly" value="' . esc_attr( $current_url ) . '" />';
echo '<button type="button" class="button tse-copy-link" title="' . esc_attr__( 'Copy URL', 'tonys-sportspress-enhancements' ) . '">' . esc_html__( 'Copy URL', 'tonys-sportspress-enhancements' ) . '</button>';
echo '<button type="button" class="button button-primary tse-open-link" data-csv-url="' . esc_url( $csv_url ) . '" data-ics-url="' . esc_url( $ics_url ) . '" data-print-url="' . esc_url( $print_url ) . '" title="' . esc_attr__( 'Open URL in new tab', 'tonys-sportspress-enhancements' ) . '">' . esc_html__( 'Open URL in New Tab', 'tonys-sportspress-enhancements' ) . '</button>';
echo '<button type="button" class="button button-primary tse-ics-ios-link" data-ics-url="' . esc_url( $ics_url ) . '" title="' . esc_attr__( 'Subscribe on iPhone or iPad', 'tonys-sportspress-enhancements' ) . '" style="display:none;">' . esc_html__( 'Subscribe on iPhone/iPad', 'tonys-sportspress-enhancements' ) . '</button>';
echo '<button type="button" class="button tse-ics-android-link" data-ics-url="' . esc_url( $ics_url ) . '" title="' . esc_attr__( 'Subscribe on Android', 'tonys-sportspress-enhancements' ) . '" style="display:none;">' . esc_html__( 'Subscribe on Android', 'tonys-sportspress-enhancements' ) . '</button>';
echo '</div>';
echo '<p class="description tse-output-note">' . esc_html__( 'Use the buttons to copy the generated URL or open the right destination for this export type.', 'tonys-sportspress-enhancements' ) . '</p>';
tse_sp_schedule_exporter_render_link_sync_script( true );
echo '</div>';
echo '</div>';
@@ -332,7 +272,10 @@ function tse_sp_schedule_exporter_render_shortcode() {
$season_id = tse_sp_schedule_exporter_resolve_season_id( $seasons );
$teams = tse_sp_schedule_exporter_get_teams( $league_id, $season_id );
$team_id = tse_sp_schedule_exporter_resolve_team_id( $teams );
$paper = tse_sp_schedule_exporter_resolve_paper_size();
$fields = tse_sp_schedule_exporter_get_fields();
$field_id = tse_sp_schedule_exporter_resolve_field_id( $fields );
$export_type = tse_sp_schedule_exporter_resolve_export_type();
$subformat = tse_sp_schedule_exporter_resolve_subformat();
if ( empty( $teams ) ) {
return '<p>' . esc_html__( 'No SportsPress teams match the selected league and season.', 'tonys-sportspress-enhancements' ) . '</p>';
@@ -342,10 +285,34 @@ function tse_sp_schedule_exporter_render_shortcode() {
?>
<div class="tse-schedule-exporter" style="max-width:960px;margin:0 auto;padding:24px;border:1px solid #d7d7db;background:#fff;">
<h2 style="margin-top:0;"><?php esc_html_e( 'Schedule Exporter', 'tonys-sportspress-enhancements' ); ?></h2>
<p><?php esc_html_e( 'Export schedules as CSV or open the printable version and save it as a PDF.', 'tonys-sportspress-enhancements' ); ?></p>
<p><?php esc_html_e( 'Choose filters once, then generate the CSV feed, iCal link, or printable page URL from the same controls.', 'tonys-sportspress-enhancements' ); ?></p>
<form method="get" action="<?php echo esc_url( get_permalink() ); ?>" class="tse-schedule-exporter-form" style="display:grid;gap:16px;margin:24px 0;">
<div>
<form method="get" action="<?php echo esc_url( get_permalink() ); ?>" class="tse-schedule-exporter-form" style="max-width:720px;margin:24px 0;">
<div style="margin-bottom:16px;">
<label for="tse-public-export-type"><strong><?php esc_html_e( 'Format', 'tonys-sportspress-enhancements' ); ?></strong></label><br />
<select id="tse-public-export-type" name="export_type">
<?php foreach ( tse_sp_schedule_exporter_get_export_types() as $type_key => $type_label ) : ?>
<option value="<?php echo esc_attr( $type_key ); ?>" <?php selected( $export_type, $type_key ); ?>>
<?php echo esc_html( $type_label ); ?>
</option>
<?php endforeach; ?>
</select>
<p class="description"><?php esc_html_e( 'CSV builds a feed URL, iCal Link builds a subscription URL, and Printable opens the printable page.', 'tonys-sportspress-enhancements' ); ?></p>
</div>
<div data-subformat-wrap="1" style="margin-bottom:16px;">
<label for="tse-public-subformat"><strong><?php esc_html_e( 'CSV Layout', 'tonys-sportspress-enhancements' ); ?></strong></label><br />
<select id="tse-public-subformat" name="subformat">
<?php foreach ( tse_sp_event_export_get_formats() as $format_key => $format_definition ) : ?>
<option value="<?php echo esc_attr( $format_key ); ?>" <?php selected( $subformat, $format_key ); ?>>
<?php echo esc_html( $format_definition['label'] ); ?>
</option>
<?php endforeach; ?>
</select>
<p class="description"><?php esc_html_e( 'Matchup is away vs home. Team is opponent-based and requires one specific team.', 'tonys-sportspress-enhancements' ); ?></p>
</div>
<div style="margin-bottom:16px;">
<label for="tse-public-league"><strong><?php esc_html_e( 'League', 'tonys-sportspress-enhancements' ); ?></strong></label><br />
<select id="tse-public-league" name="league_id" data-auto-submit="1">
<?php foreach ( $leagues as $league ) : ?>
@@ -356,7 +323,7 @@ function tse_sp_schedule_exporter_render_shortcode() {
</select>
</div>
<div>
<div style="margin-bottom:16px;">
<label for="tse-public-season"><strong><?php esc_html_e( 'Season', 'tonys-sportspress-enhancements' ); ?></strong></label><br />
<select id="tse-public-season" name="season_id" data-auto-submit="1">
<option value="0"><?php esc_html_e( 'Current / All matching events', 'tonys-sportspress-enhancements' ); ?></option>
@@ -368,20 +335,10 @@ function tse_sp_schedule_exporter_render_shortcode() {
</select>
</div>
<div>
<label for="tse-public-paper"><strong><?php esc_html_e( 'Paper Size', 'tonys-sportspress-enhancements' ); ?></strong></label><br />
<select id="tse-public-paper" name="paper">
<?php foreach ( tse_sp_schedule_exporter_get_paper_sizes() as $paper_value => $paper_label ) : ?>
<option value="<?php echo esc_attr( $paper_value ); ?>" <?php selected( $paper, $paper_value ); ?>>
<?php echo esc_html( $paper_label ); ?>
</option>
<?php endforeach; ?>
</select>
</div>
<div>
<div style="margin-bottom:16px;">
<label for="tse-public-team"><strong><?php esc_html_e( 'Team', 'tonys-sportspress-enhancements' ); ?></strong></label><br />
<select id="tse-public-team" name="team_id">
<option value="0"><?php esc_html_e( 'All teams', 'tonys-sportspress-enhancements' ); ?></option>
<?php foreach ( $teams as $team ) : ?>
<option value="<?php echo esc_attr( (string) $team->ID ); ?>" <?php selected( $team_id, (int) $team->ID ); ?>>
<?php echo esc_html( $team->post_title ); ?>
@@ -390,30 +347,37 @@ function tse_sp_schedule_exporter_render_shortcode() {
</select>
</div>
<div style="margin-bottom:16px;">
<label for="tse-public-field"><strong><?php esc_html_e( 'Field', 'tonys-sportspress-enhancements' ); ?></strong></label><br />
<select id="tse-public-field" name="field_id">
<option value="0"><?php esc_html_e( 'All fields', 'tonys-sportspress-enhancements' ); ?></option>
<?php foreach ( $fields as $field ) : ?>
<option value="<?php echo esc_attr( (string) $field->term_id ); ?>" <?php selected( $field_id, (int) $field->term_id ); ?>>
<?php echo esc_html( $field->name ); ?>
</option>
<?php endforeach; ?>
</select>
</div>
</form>
<table style="width:100%;border-collapse:collapse;margin-top:16px;">
<tbody>
<tr>
<td style="width:240px;padding:10px 12px;border:1px solid #d7d7db;">
<a class="button button-primary tse-export-link" data-format="matchup" href="<?php echo esc_url( tse_sp_schedule_exporter_get_export_url( $team_id, $season_id, 'matchup', $league_id ) ); ?>"><?php esc_html_e( 'Download Matchup CSV', 'tonys-sportspress-enhancements' ); ?></a>
</td>
<td style="padding:10px 12px;border:1px solid #d7d7db;"><?php esc_html_e( 'Date, time, away team, home team, and field name.', 'tonys-sportspress-enhancements' ); ?></td>
</tr>
<tr>
<td style="width:240px;padding:10px 12px;border:1px solid #d7d7db;">
<a class="button button-primary tse-export-link" data-format="team" href="<?php echo esc_url( tse_sp_schedule_exporter_get_export_url( $team_id, $season_id, 'team', $league_id ) ); ?>"><?php esc_html_e( 'Download Team CSV', 'tonys-sportspress-enhancements' ); ?></a>
</td>
<td style="padding:10px 12px;border:1px solid #d7d7db;"><?php esc_html_e( 'TeamSnap-compatible layout with game label, opponent, home/away flag, and venue.', 'tonys-sportspress-enhancements' ); ?></td>
</tr>
<tr>
<td style="width:240px;padding:10px 12px;border:1px solid #d7d7db;">
<a class="button tse-pdf-link" href="<?php echo esc_url( tse_sp_schedule_exporter_get_pdf_url( $team_id, $season_id, $paper, $league_id ) ); ?>" target="_blank" rel="noopener noreferrer"><?php esc_html_e( 'Open Printable PDF View', 'tonys-sportspress-enhancements' ); ?></a>
</td>
<td style="padding:10px 12px;border:1px solid #d7d7db;"><?php esc_html_e( 'Opens the printable schedule and starts the print dialog so visitors can save a PDF.', 'tonys-sportspress-enhancements' ); ?></td>
</tr>
</tbody>
</table>
<?php tse_sp_schedule_exporter_render_column_picker( 'matchup', 'public', $subformat ); ?>
<?php tse_sp_schedule_exporter_render_column_picker( 'team', 'public', $subformat ); ?>
<?php
$csv_url = tse_sp_event_export_get_feed_url( array( 'team_id' => $team_id, 'season_id' => $season_id, 'league_id' => $league_id, 'field_id' => $field_id, 'format' => $subformat ), 'csv' );
$ics_url = tse_sp_event_export_get_feed_url( array( 'team_id' => $team_id, 'season_id' => $season_id, 'league_id' => $league_id, 'field_id' => $field_id ), 'ics' );
$print_url = tse_sp_schedule_exporter_get_printable_url( $team_id, $season_id, 'letter', $league_id );
$current_url = tse_sp_schedule_exporter_get_output_url( $export_type, $csv_url, $ics_url, $print_url );
?>
<div style="display:flex;align-items:center;gap:8px;max-width:100%;margin-top:16px;">
<input type="text" class="large-text code tse-output-url" readonly="readonly" value="<?php echo esc_attr( $current_url ); ?>" />
<button type="button" class="button tse-copy-link" title="<?php esc_attr_e( 'Copy URL', 'tonys-sportspress-enhancements' ); ?>"><?php esc_html_e( 'Copy URL', 'tonys-sportspress-enhancements' ); ?></button>
<button type="button" class="button button-primary tse-open-link" data-csv-url="<?php echo esc_url( $csv_url ); ?>" data-ics-url="<?php echo esc_url( $ics_url ); ?>" data-print-url="<?php echo esc_url( $print_url ); ?>" title="<?php esc_attr_e( 'Open URL in new tab', 'tonys-sportspress-enhancements' ); ?>"><?php esc_html_e( 'Open URL in New Tab', 'tonys-sportspress-enhancements' ); ?></button>
<button type="button" class="button button-primary tse-ics-ios-link" data-ics-url="<?php echo esc_url( $ics_url ); ?>" title="<?php esc_attr_e( 'Subscribe on iPhone or iPad', 'tonys-sportspress-enhancements' ); ?>" style="display:none;"><?php esc_html_e( 'Subscribe on iPhone/iPad', 'tonys-sportspress-enhancements' ); ?></button>
<button type="button" class="button tse-ics-android-link" data-ics-url="<?php echo esc_url( $ics_url ); ?>" title="<?php esc_attr_e( 'Subscribe on Android', 'tonys-sportspress-enhancements' ); ?>" style="display:none;"><?php esc_html_e( 'Subscribe on Android', 'tonys-sportspress-enhancements' ); ?></button>
</div>
<p class="description tse-output-note"><?php esc_html_e( 'Use the buttons to copy the generated URL or open the right destination for this export type.', 'tonys-sportspress-enhancements' ); ?></p>
</div>
<?php
$output = (string) ob_get_clean();
@@ -478,6 +442,28 @@ function tse_sp_schedule_exporter_get_seasons() {
return $seasons;
}
/**
* Get fields for the exporter.
*
* @return WP_Term[]
*/
function tse_sp_schedule_exporter_get_fields() {
$fields = get_terms(
array(
'taxonomy' => 'sp_venue',
'hide_empty' => false,
'orderby' => 'name',
'order' => 'ASC',
)
);
if ( is_wp_error( $fields ) || ! is_array( $fields ) ) {
return array();
}
return $fields;
}
/**
* Get teams for the exporter.
*
@@ -552,6 +538,70 @@ function tse_sp_schedule_exporter_resolve_team_id( $teams ) {
return 0;
}
/**
* Resolve selected field ID.
*
* @param WP_Term[] $fields Field terms.
* @return int
*/
function tse_sp_schedule_exporter_resolve_field_id( $fields ) {
$requested = isset( $_GET['field_id'] ) ? absint( wp_unslash( $_GET['field_id'] ) ) : 0;
if ( 0 === $requested ) {
return 0;
}
foreach ( $fields as $field ) {
if ( $field instanceof WP_Term && (int) $field->term_id === $requested ) {
return $requested;
}
}
return 0;
}
/**
* Render selectable columns for a format.
*
* @param string $format Export format.
* @param string $context Render context suffix.
* @return void
*/
function tse_sp_schedule_exporter_render_column_picker( $format, $context, $active_format = '' ) {
$format = tse_sp_event_export_sanitize_format( $format );
$definitions = tse_sp_event_export_get_column_definitions();
$columns = isset( $definitions[ $format ] ) ? $definitions[ $format ] : array();
$selected = tse_sp_event_export_get_default_columns( $format );
$formats = tse_sp_event_export_get_formats();
$legend = isset( $formats[ $format ]['label'] ) ? $formats[ $format ]['label'] : ucfirst( $format );
if ( empty( $columns ) ) {
return;
}
$style = 'margin:18px 0;padding:16px;border:1px solid #d7d7db;';
if ( $active_format && $active_format !== $format ) {
$style .= 'display:none;';
}
echo '<fieldset data-column-group="' . esc_attr( $format ) . '" style="' . esc_attr( $style ) . '">';
echo '<legend><strong>' . esc_html( sprintf( __( '%s Columns', 'tonys-sportspress-enhancements' ), $legend ) ) . '</strong></legend>';
echo '<div style="display:flex;flex-wrap:wrap;gap:12px 18px;">';
foreach ( $columns as $column_key => $column_label ) {
$input_id = sprintf( 'tse-columns-%1$s-%2$s-%3$s', sanitize_html_class( $context ), sanitize_html_class( $format ), sanitize_html_class( $column_key ) );
echo '<label for="' . esc_attr( $input_id ) . '" style="display:inline-flex;align-items:center;gap:6px;">';
echo '<input id="' . esc_attr( $input_id ) . '" type="checkbox" data-columns-format="' . esc_attr( $format ) . '" value="' . esc_attr( $column_key ) . '" ' . checked( in_array( $column_key, $selected, true ), true, false ) . ' />';
echo esc_html( $column_label );
echo '</label>';
}
echo '</div>';
echo '<p class="description" style="margin:10px 0 0;">' . esc_html__( 'These checkboxes only change the CSV feed link. iCal and printable links use the same shared filters but ignore columns.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</fieldset>';
}
/**
* Resolve selected league ID.
*
@@ -631,6 +681,74 @@ function tse_sp_schedule_exporter_resolve_paper_size() {
return array_key_exists( $paper, tse_sp_schedule_exporter_get_paper_sizes() ) ? $paper : 'letter';
}
/**
* Resolve selected export format.
*
* @return string
*/
function tse_sp_schedule_exporter_resolve_format() {
$requested = isset( $_GET['format'] ) ? sanitize_key( wp_unslash( $_GET['format'] ) ) : 'matchup';
return tse_sp_event_export_sanitize_format( $requested );
}
/**
* Get supported exporter output types.
*
* @return array
*/
function tse_sp_schedule_exporter_get_export_types() {
return array(
'csv' => __( 'CSV', 'tonys-sportspress-enhancements' ),
'ics' => __( 'iCal Link', 'tonys-sportspress-enhancements' ),
'printable' => __( 'Printable', 'tonys-sportspress-enhancements' ),
);
}
/**
* Resolve selected exporter output type.
*
* @return string
*/
function tse_sp_schedule_exporter_resolve_export_type() {
$requested = isset( $_GET['export_type'] ) ? sanitize_key( wp_unslash( $_GET['export_type'] ) ) : 'csv';
$types = tse_sp_schedule_exporter_get_export_types();
return isset( $types[ $requested ] ) ? $requested : 'csv';
}
/**
* Resolve selected CSV subformat.
*
* @return string
*/
function tse_sp_schedule_exporter_resolve_subformat() {
$requested = isset( $_GET['subformat'] ) ? sanitize_key( wp_unslash( $_GET['subformat'] ) ) : 'matchup';
return tse_sp_event_export_sanitize_format( $requested );
}
/**
* Get current output URL for the selected export type.
*
* @param string $export_type Export type.
* @param string $csv_url CSV URL.
* @param string $ics_url ICS URL.
* @param string $print_url Printable URL.
* @return string
*/
function tse_sp_schedule_exporter_get_output_url( $export_type, $csv_url, $ics_url, $print_url ) {
if ( 'ics' === $export_type ) {
return $ics_url;
}
if ( 'printable' === $export_type ) {
return $print_url;
}
return $csv_url;
}
/**
* Collect team schedule events for export.
*
@@ -761,32 +879,7 @@ function tse_sp_schedule_exporter_get_primary_venue( $event_id ) {
}
/**
* Build an export download URL.
*
* @param int $team_id Team ID.
* @param int $season_id Season ID.
* @param string $format Export format.
* @param int $league_id League ID.
* @return string
*/
function tse_sp_schedule_exporter_get_export_url( $team_id, $season_id, $format, $league_id = 0 ) {
return wp_nonce_url(
add_query_arg(
array(
'action' => 'tse_schedule_export',
'league_id' => absint( $league_id ),
'team_id' => absint( $team_id ),
'season_id' => absint( $season_id ),
'format' => sanitize_key( $format ),
),
admin_url( 'admin-post.php' )
),
'tse_schedule_export'
);
}
/**
* Build the printable PDF URL.
* Build the printable page URL.
*
* @param int $team_id Team ID.
* @param int $season_id Season ID.
@@ -794,7 +887,7 @@ function tse_sp_schedule_exporter_get_export_url( $team_id, $season_id, $format,
* @param int $league_id League ID.
* @return string
*/
function tse_sp_schedule_exporter_get_pdf_url( $team_id, $season_id, $paper, $league_id = 0 ) {
function tse_sp_schedule_exporter_get_printable_url( $team_id, $season_id, $paper, $league_id = 0, $autoprint = false ) {
return add_query_arg(
array(
Tony_Sportspress_Printable_Calendars::QUERY_FLAG => '1',
@@ -802,7 +895,7 @@ function tse_sp_schedule_exporter_get_pdf_url( $team_id, $season_id, $paper, $le
'sp_season' => $season_id > 0 ? (string) absint( $season_id ) : '',
'sp_league' => $league_id > 0 ? (string) absint( $league_id ) : '',
'paper' => $paper,
'autoprint' => '1',
'autoprint' => $autoprint ? '1' : '',
),
home_url( '/' )
);
@@ -818,6 +911,30 @@ function tse_sp_schedule_exporter_render_link_sync_script( $echo = false ) {
$script = <<<HTML
<script>
(function(){
function copyText(text, done){
if (navigator.clipboard && navigator.clipboard.writeText) {
navigator.clipboard.writeText(text).then(done).catch(function(){
var input = document.createElement('input');
input.value = text;
document.body.appendChild(input);
input.focus();
input.select();
document.execCommand('copy');
document.body.removeChild(input);
done();
});
return;
}
var input = document.createElement('input');
input.value = text;
document.body.appendChild(input);
input.focus();
input.select();
document.execCommand('copy');
document.body.removeChild(input);
done();
}
function syncLinks(scope){
var form = scope.querySelector('.tse-schedule-exporter-form');
if (!form) {
@@ -827,27 +944,125 @@ function tse_sp_schedule_exporter_render_link_sync_script( $echo = false ) {
var league = form.querySelector('[name="league_id"]');
var season = form.querySelector('[name="season_id"]');
var team = form.querySelector('[name="team_id"]');
var paper = form.querySelector('[name="paper"]');
var exportType = form.querySelector('[name="export_type"]');
var subformat = form.querySelector('[name="subformat"]');
var field = form.querySelector('[name="field_id"]');
var outputUrl = scope.querySelector('.tse-output-url');
var openButton = scope.querySelector('.tse-open-link');
var iosButton = scope.querySelector('.tse-ics-ios-link');
var androidButton = scope.querySelector('.tse-ics-android-link');
var outputNote = scope.querySelector('.tse-output-note');
var copyButton = scope.querySelector('.tse-copy-link');
var teamValue = team ? (team.value || '0') : '0';
var activeSubformat = subformat ? (subformat.value || 'matchup') : 'matchup';
var selectedExportType = exportType ? (exportType.value || 'csv') : 'csv';
scope.querySelectorAll('.tse-export-link').forEach(function(link){
var url = new URL(link.href, window.location.origin);
if (league) url.searchParams.set('league_id', league.value || '0');
if (season) url.searchParams.set('season_id', season.value || '0');
if (team) url.searchParams.set('team_id', team.value || '0');
if (link.dataset.format) url.searchParams.set('format', link.dataset.format);
link.href = url.toString();
scope.querySelectorAll('[data-column-group]').forEach(function(group){
var visible = selectedExportType === 'csv' && group.getAttribute('data-column-group') === activeSubformat;
group.style.display = visible ? 'block' : 'none';
});
scope.querySelectorAll('.tse-pdf-link').forEach(function(link){
var url = new URL(link.href, window.location.origin);
if (league) url.searchParams.set('sp_league', league.value || '0');
if (season) url.searchParams.set('sp_season', season.value || '0');
if (team) url.searchParams.set('sp_team', team.value || '0');
if (paper) url.searchParams.set('paper', paper.value || 'letter');
link.href = url.toString();
if (scope.querySelector('[data-subformat-wrap]')) {
scope.querySelectorAll('[data-subformat-wrap]').forEach(function(wrap){
wrap.style.display = selectedExportType === 'csv' ? 'block' : 'none';
});
}
var csvUrl = openButton ? new URL(openButton.dataset.csvUrl, window.location.origin) : null;
var icsUrl = openButton ? new URL(openButton.dataset.icsUrl, window.location.origin) : null;
var printUrl = openButton ? new URL(openButton.dataset.printUrl, window.location.origin) : null;
if (csvUrl) {
if (league) csvUrl.searchParams.set('league_id', league.value || '0');
if (season) csvUrl.searchParams.set('season_id', season.value || '0');
if (team) csvUrl.searchParams.set('team_id', teamValue);
if (field) csvUrl.searchParams.set('field_id', field.value || '0');
csvUrl.searchParams.set('format', activeSubformat);
var columns = Array.prototype.slice.call(scope.querySelectorAll('[data-columns-format="' + activeSubformat + '"]:checked')).map(function(input){
return input.value;
}).filter(Boolean);
if (columns.length) {
csvUrl.searchParams.set('columns', columns.join(','));
} else {
csvUrl.searchParams.delete('columns');
}
}
if (icsUrl) {
if (league) icsUrl.searchParams.set('league_id', league.value || '0');
if (season) icsUrl.searchParams.set('season_id', season.value || '0');
if (team) icsUrl.searchParams.set('team_id', teamValue);
if (field) icsUrl.searchParams.set('field_id', field.value || '0');
icsUrl.searchParams.delete('format');
icsUrl.searchParams.delete('columns');
}
if (printUrl) {
if (league) printUrl.searchParams.set('sp_league', league.value || '0');
if (season) printUrl.searchParams.set('sp_season', season.value || '0');
if (team) printUrl.searchParams.set('sp_team', teamValue);
printUrl.searchParams.set('paper', 'letter');
}
var resolvedUrl = csvUrl ? csvUrl.toString() : '';
var label = 'Open URL in New Tab';
var disabled = false;
var note = 'Use the buttons to copy the generated URL or open the right destination for this export type.';
var iosUrl = icsUrl ? icsUrl.toString().replace(/^https?:\/\//, 'webcal://') : '';
var androidUrl = icsUrl ? 'https://calendar.google.com/calendar/render?cid=' + encodeURIComponent(icsUrl.toString()) : '';
if (selectedExportType === 'ics' && icsUrl) {
resolvedUrl = icsUrl.toString();
note = 'Use the iPhone/iPad or Android button to subscribe, or copy the feed URL.';
} else if (selectedExportType === 'printable' && printUrl) {
resolvedUrl = printUrl.toString();
if (teamValue === '0') {
disabled = true;
note = 'Printable requires a specific team. All teams is not supported.';
}
} else if (selectedExportType === 'csv' && activeSubformat === 'team' && teamValue === '0') {
disabled = true;
note = 'CSV team layout requires a specific team. All teams is not supported.';
}
if (outputUrl) {
outputUrl.value = resolvedUrl;
}
if (openButton) {
openButton.dataset.currentUrl = resolvedUrl;
openButton.textContent = label;
openButton.style.display = selectedExportType === 'ics' ? 'none' : 'inline-flex';
openButton.disabled = disabled;
openButton.setAttribute('aria-disabled', disabled ? 'true' : 'false');
openButton.style.opacity = disabled ? '0.55' : '1';
}
if (iosButton) {
iosButton.dataset.currentUrl = iosUrl;
iosButton.style.display = selectedExportType === 'ics' ? 'inline-flex' : 'none';
iosButton.disabled = !iosUrl;
iosButton.setAttribute('aria-disabled', !iosUrl ? 'true' : 'false');
iosButton.style.opacity = !iosUrl ? '0.55' : '1';
}
if (androidButton) {
androidButton.dataset.currentUrl = androidUrl;
androidButton.style.display = selectedExportType === 'ics' ? 'inline-flex' : 'none';
androidButton.disabled = !androidUrl;
androidButton.setAttribute('aria-disabled', !androidUrl ? 'true' : 'false');
androidButton.style.opacity = !androidUrl ? '0.55' : '1';
}
if (outputNote) {
outputNote.textContent = note;
}
if (copyButton) {
copyButton.disabled = disabled;
}
}
document.querySelectorAll('.tse-schedule-exporter, .wrap').forEach(function(scope){
if (!scope.querySelector('.tse-schedule-exporter-form')) {
return;
@@ -865,6 +1080,67 @@ function tse_sp_schedule_exporter_render_link_sync_script( $echo = false ) {
syncLinks(scope);
});
});
scope.querySelectorAll('[data-columns-format]').forEach(function(input){
input.addEventListener('change', function(){
syncLinks(scope);
});
});
var copyButton = scope.querySelector('.tse-copy-link');
var openButton = scope.querySelector('.tse-open-link');
var iosButton = scope.querySelector('.tse-ics-ios-link');
var androidButton = scope.querySelector('.tse-ics-android-link');
var outputUrl = scope.querySelector('.tse-output-url');
if (copyButton && outputUrl) {
copyButton.addEventListener('click', function(){
if (copyButton.disabled || !outputUrl.value) {
return;
}
var defaultTitle = copyButton.getAttribute('data-default-title') || copyButton.title || 'Copy URL';
copyButton.setAttribute('data-default-title', defaultTitle);
copyText(outputUrl.value, function(){
copyButton.title = 'Copied';
window.setTimeout(function(){
copyButton.title = defaultTitle;
}, 1200);
});
});
}
if (openButton && outputUrl) {
openButton.addEventListener('click', function(){
if (openButton.disabled || !outputUrl.value) {
return;
}
window.open(outputUrl.value, '_blank', 'noopener,noreferrer');
});
}
if (iosButton) {
iosButton.addEventListener('click', function(){
var targetUrl = iosButton.dataset.currentUrl || '';
if (iosButton.disabled || !targetUrl) {
return;
}
window.location.href = targetUrl;
});
}
if (androidButton) {
androidButton.addEventListener('click', function(){
var targetUrl = androidButton.dataset.currentUrl || '';
if (androidButton.disabled || !targetUrl) {
return;
}
window.open(targetUrl, '_blank', 'noopener,noreferrer');
});
}
});
})();
</script>

285
includes/sp-url-builder.php Normal file
View File

@@ -0,0 +1,285 @@
<?php
/**
* Tony's Settings CSV URL builder tab.
*
* @package Tonys_Sportspress_Enhancements
*/
if ( ! defined( 'ABSPATH' ) ) {
exit;
}
/**
* Register the URL builder tab label.
*
* @param array $tabs Existing tab map.
* @return array
*/
function tse_sp_url_builder_register_tab( $tabs ) {
$tabs['url-builder'] = __( 'Feed Builder', 'tonys-sportspress-enhancements' );
return $tabs;
}
add_filter( 'tse_tonys_settings_tabs', 'tse_sp_url_builder_register_tab' );
/**
* Render the URL builder tab.
*
* @return void
*/
function tse_sp_url_builder_render_tab() {
$leagues = function_exists( 'tse_sp_schedule_exporter_get_leagues' ) ? tse_sp_schedule_exporter_get_leagues() : array();
$seasons = function_exists( 'tse_sp_schedule_exporter_get_seasons' ) ? tse_sp_schedule_exporter_get_seasons() : array();
$teams = function_exists( 'tse_sp_schedule_exporter_get_teams' ) ? tse_sp_schedule_exporter_get_teams() : array();
$fields = function_exists( 'tse_sp_schedule_exporter_get_fields' ) ? tse_sp_schedule_exporter_get_fields() : array();
$formats = function_exists( 'tse_sp_event_export_get_formats' ) ? tse_sp_event_export_get_formats() : array();
$columns = function_exists( 'tse_sp_event_export_get_column_definitions' ) ? tse_sp_event_export_get_column_definitions() : array();
echo '<h2>' . esc_html__( 'Feed Builder', 'tonys-sportspress-enhancements' ) . '</h2>';
echo '<p>' . esc_html__( 'Build a shareable CSV feed URL with format, filters, and custom columns. This does not save settings.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '<div class="tse-url-builder" style="max-width:1100px;padding:20px 24px;border:1px solid #dcdcde;background:#fff;">';
echo '<table class="form-table" role="presentation"><tbody>';
echo '<tr><th scope="row"><label for="tse-url-builder-feed-type">' . esc_html__( 'Feed Type', 'tonys-sportspress-enhancements' ) . '</label></th><td>';
echo '<select id="tse-url-builder-feed-type">';
echo '<option value="csv">' . esc_html__( 'CSV', 'tonys-sportspress-enhancements' ) . '</option>';
echo '<option value="ics">' . esc_html__( 'iCal / ICS', 'tonys-sportspress-enhancements' ) . '</option>';
echo '</select>';
echo '</td></tr>';
echo '<tr><th scope="row"><label for="tse-url-builder-format">' . esc_html__( 'Format', 'tonys-sportspress-enhancements' ) . '</label></th><td>';
echo '<select id="tse-url-builder-format">';
foreach ( $formats as $format_key => $format ) {
echo '<option value="' . esc_attr( $format_key ) . '">' . esc_html( $format['label'] ) . '</option>';
}
echo '</select>';
echo '</td></tr>';
echo '<tr><th scope="row">' . esc_html__( 'League', 'tonys-sportspress-enhancements' ) . '</th><td>';
echo '<div id="tse-url-builder-league" style="display:flex;flex-wrap:wrap;gap:10px 14px;max-width:720px;">';
foreach ( $leagues as $league ) {
$input_id = 'tse-url-builder-league-' . absint( $league->term_id );
echo '<label for="' . esc_attr( $input_id ) . '" style="display:inline-flex;align-items:center;gap:6px;">';
echo '<input id="' . esc_attr( $input_id ) . '" type="checkbox" data-tse-builder-filter="league_id" value="' . esc_attr( (string) $league->term_id ) . '" />';
echo esc_html( $league->name );
echo '</label>';
}
echo '</div>';
echo '<p class="description">' . esc_html__( 'Select one or more leagues. Leave empty to include all leagues.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</td></tr>';
echo '<tr><th scope="row">' . esc_html__( 'Season', 'tonys-sportspress-enhancements' ) . '</th><td>';
echo '<div id="tse-url-builder-season" style="display:flex;flex-wrap:wrap;gap:10px 14px;max-width:720px;">';
foreach ( $seasons as $season ) {
$input_id = 'tse-url-builder-season-' . absint( $season->term_id );
echo '<label for="' . esc_attr( $input_id ) . '" style="display:inline-flex;align-items:center;gap:6px;">';
echo '<input id="' . esc_attr( $input_id ) . '" type="checkbox" data-tse-builder-filter="season_id" value="' . esc_attr( (string) $season->term_id ) . '" />';
echo esc_html( $season->name );
echo '</label>';
}
echo '</div>';
echo '<p class="description">' . esc_html__( 'Select one or more seasons. Leave empty to include all seasons.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</td></tr>';
echo '<tr><th scope="row">' . esc_html__( 'Team', 'tonys-sportspress-enhancements' ) . '</th><td>';
echo '<div id="tse-url-builder-team" style="display:flex;flex-wrap:wrap;gap:10px 14px;max-width:720px;">';
foreach ( $teams as $team ) {
$input_id = 'tse-url-builder-team-' . absint( $team->ID );
echo '<label for="' . esc_attr( $input_id ) . '" style="display:inline-flex;align-items:center;gap:6px;">';
echo '<input id="' . esc_attr( $input_id ) . '" type="checkbox" data-tse-builder-filter="team_id" value="' . esc_attr( (string) $team->ID ) . '" />';
echo esc_html( $team->post_title );
echo '</label>';
}
echo '</div>';
echo '<p class="description">' . esc_html__( 'Select one or more teams. Team format requires exactly one team.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</td></tr>';
echo '<tr><th scope="row">' . esc_html__( 'Field', 'tonys-sportspress-enhancements' ) . '</th><td>';
echo '<div id="tse-url-builder-field" style="display:flex;flex-wrap:wrap;gap:10px 14px;max-width:720px;">';
foreach ( $fields as $field ) {
$input_id = 'tse-url-builder-field-' . absint( $field->term_id );
echo '<label for="' . esc_attr( $input_id ) . '" style="display:inline-flex;align-items:center;gap:6px;">';
echo '<input id="' . esc_attr( $input_id ) . '" type="checkbox" data-tse-builder-filter="field_id" value="' . esc_attr( (string) $field->term_id ) . '" />';
echo esc_html( $field->name );
echo '</label>';
}
echo '</div>';
echo '<p class="description">' . esc_html__( 'Select one or more fields. Leave empty to include all fields.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</td></tr>';
echo '</tbody></table>';
echo '<div style="display:grid;gap:16px;margin-top:20px;">';
foreach ( $columns as $format_key => $format_columns ) {
$default_columns = function_exists( 'tse_sp_event_export_get_default_columns' ) ? tse_sp_event_export_get_default_columns( $format_key ) : array();
$label = isset( $formats[ $format_key ]['label'] ) ? $formats[ $format_key ]['label'] : ucfirst( $format_key );
echo '<fieldset data-tse-builder-columns="' . esc_attr( $format_key ) . '" style="margin:0;padding:16px;border:1px solid #d7d7db;">';
echo '<legend><strong>' . esc_html( sprintf( __( '%s Columns', 'tonys-sportspress-enhancements' ), $label ) ) . '</strong></legend>';
echo '<div style="display:flex;flex-wrap:wrap;gap:12px 18px;">';
foreach ( $format_columns as $column_key => $column_label ) {
$input_id = 'tse-url-builder-' . sanitize_html_class( $format_key . '-' . $column_key );
echo '<label for="' . esc_attr( $input_id ) . '" style="display:inline-flex;align-items:center;gap:6px;">';
echo '<input id="' . esc_attr( $input_id ) . '" type="checkbox" data-tse-builder-column="' . esc_attr( $format_key ) . '" value="' . esc_attr( $column_key ) . '" ' . checked( in_array( $column_key, $default_columns, true ), true, false ) . ' />';
echo esc_html( $column_label );
echo '</label>';
}
echo '</div>';
echo '</fieldset>';
}
echo '</div>';
echo '<h3 style="margin-top:24px;">' . esc_html__( 'Generated URL', 'tonys-sportspress-enhancements' ) . '</h3>';
echo '<div style="display:flex;align-items:center;gap:8px;max-width:100%;">';
echo '<input type="text" id="tse-url-builder-output" class="large-text code" readonly="readonly" />';
echo '<button type="button" id="tse-url-builder-copy" class="button" aria-label="' . esc_attr__( 'Copy URL', 'tonys-sportspress-enhancements' ) . '" title="' . esc_attr__( 'Copy URL', 'tonys-sportspress-enhancements' ) . '" style="display:inline-flex;align-items:center;justify-content:center;min-width:40px;padding:0 10px;">';
echo '<span aria-hidden="true" style="font-size:16px;line-height:1;">⧉</span>';
echo '</button>';
echo '</div>';
echo '<p><a id="tse-url-builder-open" class="button button-primary" href="' . esc_url( home_url( '/' ) ) . '" target="_blank" rel="noopener noreferrer">' . esc_html__( 'Open Feed URL', 'tonys-sportspress-enhancements' ) . '</a></p>';
echo '<p class="description">' . esc_html__( 'The builder always generates the standalone CSV feed endpoint using the selected filters and columns.', 'tonys-sportspress-enhancements' ) . '</p>';
echo '</div>';
tse_sp_url_builder_render_script();
}
add_action( 'tse_tonys_settings_render_tab_url-builder', 'tse_sp_url_builder_render_tab' );
/**
* Render builder script.
*
* @return void
*/
function tse_sp_url_builder_render_script() {
$base_url = home_url( '/' );
?>
<script>
(function(){
var root = document.querySelector('.tse-url-builder');
if (!root) {
return;
}
var baseUrl = <?php echo wp_json_encode( $base_url ); ?>;
var feedType = root.querySelector('#tse-url-builder-feed-type');
var format = root.querySelector('#tse-url-builder-format');
var output = root.querySelector('#tse-url-builder-output');
var copyButton = root.querySelector('#tse-url-builder-copy');
var openLink = root.querySelector('#tse-url-builder-open');
function getSelectedValues(filterName) {
return Array.prototype.slice.call(root.querySelectorAll('[data-tse-builder-filter="' + filterName + '"]:checked')).map(function(input){
return input.value;
});
}
function syncColumnGroups() {
var selectedFormat = format.value || 'matchup';
var isCsv = (feedType.value || 'csv') === 'csv';
root.querySelectorAll('[data-tse-builder-columns]').forEach(function(group){
group.style.display = (isCsv && group.getAttribute('data-tse-builder-columns') === selectedFormat) ? 'block' : 'none';
});
}
function buildUrl() {
var selectedFeedType = feedType.value || 'csv';
var selectedFormat = format.value || 'matchup';
var url = new URL(baseUrl, window.location.origin);
var leagues = getSelectedValues('league_id');
var seasons = getSelectedValues('season_id');
var teams = getSelectedValues('team_id');
var fields = getSelectedValues('field_id');
var selectedColumns = Array.prototype.slice.call(root.querySelectorAll('[data-tse-builder-column="' + selectedFormat + '"]:checked')).map(function(input){
return input.value;
}).filter(Boolean);
url.searchParams.set('feed', selectedFeedType === 'ics' ? 'sp-ics' : 'sp-csv');
url.searchParams.set('format', selectedFormat);
if (leagues.length) {
url.searchParams.set('league_id', leagues.join(','));
} else {
url.searchParams.delete('league_id');
}
if (seasons.length) {
url.searchParams.set('season_id', seasons.join(','));
} else {
url.searchParams.delete('season_id');
}
if (teams.length) {
url.searchParams.set('team_id', teams.join(','));
} else {
url.searchParams.delete('team_id');
}
if (fields.length) {
url.searchParams.set('field_id', fields.join(','));
} else {
url.searchParams.delete('field_id');
}
if (selectedFeedType === 'csv' && selectedColumns.length) {
url.searchParams.set('columns', selectedColumns.join(','));
} else {
url.searchParams.delete('columns');
}
output.value = url.toString();
openLink.href = url.toString();
openLink.textContent = selectedFeedType === 'ics' ? 'Open ICS Feed' : 'Open Feed URL';
}
syncColumnGroups();
buildUrl();
[feedType, format].forEach(function(input){
input.addEventListener('change', function(){
syncColumnGroups();
buildUrl();
});
});
root.querySelectorAll('[data-tse-builder-filter]').forEach(function(input){
input.addEventListener('change', buildUrl);
});
root.querySelectorAll('[data-tse-builder-column]').forEach(function(input){
input.addEventListener('change', buildUrl);
});
if (copyButton) {
copyButton.addEventListener('click', function(){
var value = output.value || '';
if (!value) {
return;
}
var defaultTitle = copyButton.getAttribute('data-default-title') || copyButton.title || 'Copy URL';
copyButton.setAttribute('data-default-title', defaultTitle);
function markCopied() {
copyButton.title = 'Copied';
window.setTimeout(function(){
copyButton.title = defaultTitle;
}, 1200);
}
if (navigator.clipboard && navigator.clipboard.writeText) {
navigator.clipboard.writeText(value).then(function(){
markCopied();
}).catch(function(){
output.focus();
output.select();
document.execCommand('copy');
markCopied();
});
return;
}
output.focus();
output.select();
document.execCommand('copy');
markCopied();
});
}
})();
</script>
<?php
}

View File

@@ -7,13 +7,13 @@
* Author URI: https://github.com/anthonyscorrea/
* Text Domain: tonys-sportspress-enhancements
* Domain Path: /languages
* Version: 0.1.6
* Version: 0.1.7
*
* @package Tonys_Sportspress_Enhancements
*/
if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_VERSION' ) ) {
define( 'TONY_SPORTSPRESS_ENHANCEMENTS_VERSION', '0.1.6' );
define( 'TONY_SPORTSPRESS_ENHANCEMENTS_VERSION', '0.1.7' );
}
if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_FILE' ) ) {
@@ -32,10 +32,12 @@ if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_URL' ) ) {
require_once plugin_dir_path(__FILE__) . 'includes/open-graph-tags.php';
require_once plugin_dir_path(__FILE__) . 'includes/featured-image-generator.php';
require_once plugin_dir_path(__FILE__) . 'includes/sp-event-permalink.php';
require_once plugin_dir_path(__FILE__) . 'includes/sp-event-export.php';
require_once plugin_dir_path(__FILE__) . 'includes/sp-event-csv.php';
require_once plugin_dir_path(__FILE__) . 'includes/sp-event-admin-week-filter.php';
require_once plugin_dir_path(__FILE__) . 'includes/sp-event-quick-edit-officials.php';
require_once plugin_dir_path(__FILE__) . 'includes/sp-event-team-ordering.php';
require_once plugin_dir_path(__FILE__) . 'includes/sp-printable-calendars.php';
require_once plugin_dir_path(__FILE__) . 'includes/sp-url-builder.php';
require_once plugin_dir_path(__FILE__) . 'includes/sp-schedule-exporter.php';
require_once plugin_dir_path(__FILE__) . 'includes/sp-venue-meta.php';