Refactor schedule exports into feed builders
This commit is contained in:
@@ -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.
|
||||
|
||||
922
includes/sp-event-export.php
Normal file
922
includes/sp-event-export.php
Normal 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( '/' ) );
|
||||
}
|
||||
@@ -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.
|
||||
*/
|
||||
|
||||
@@ -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,38 @@ 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 '</div>';
|
||||
echo '<p class="description tse-output-note">' . esc_html__( 'Use the buttons to copy or open the generated URL.', 'tonys-sportspress-enhancements' ) . '</p>';
|
||||
tse_sp_schedule_exporter_render_link_sync_script( true );
|
||||
echo '</div>';
|
||||
echo '</div>';
|
||||
@@ -332,7 +270,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 +283,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 +321,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 +333,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 +345,35 @@ 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>
|
||||
</div>
|
||||
<p class="description tse-output-note"><?php esc_html_e( 'Use the buttons to copy or open the generated URL.', 'tonys-sportspress-enhancements' ); ?></p>
|
||||
</div>
|
||||
<?php
|
||||
$output = (string) ob_get_clean();
|
||||
@@ -478,6 +438,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 +534,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 +677,92 @@ 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;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get current output button label for the selected export type.
|
||||
*
|
||||
* @param string $export_type Export type.
|
||||
* @return string
|
||||
*/
|
||||
function tse_sp_schedule_exporter_get_output_label( $export_type ) {
|
||||
if ( 'ics' === $export_type ) {
|
||||
return __( 'Subscribe to iCal / ICS', 'tonys-sportspress-enhancements' );
|
||||
}
|
||||
|
||||
if ( 'printable' === $export_type ) {
|
||||
return __( 'Open Printable Page', 'tonys-sportspress-enhancements' );
|
||||
}
|
||||
|
||||
return __( 'Open CSV Feed', 'tonys-sportspress-enhancements' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Collect team schedule events for export.
|
||||
*
|
||||
@@ -761,32 +893,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 +901,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 +909,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 +925,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 +958,105 @@ 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 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 CSV Feed';
|
||||
var disabled = false;
|
||||
var note = 'Use the buttons to copy or open the generated URL.';
|
||||
|
||||
if (selectedExportType === 'ics' && icsUrl) {
|
||||
resolvedUrl = icsUrl.toString();
|
||||
label = 'Subscribe to iCal / ICS';
|
||||
} else if (selectedExportType === 'printable' && printUrl) {
|
||||
resolvedUrl = printUrl.toString();
|
||||
label = 'Open Printable Page';
|
||||
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.disabled = disabled;
|
||||
openButton.setAttribute('aria-disabled', disabled ? 'true' : 'false');
|
||||
openButton.style.opacity = disabled ? '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 +1074,43 @@ 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 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');
|
||||
});
|
||||
}
|
||||
});
|
||||
})();
|
||||
</script>
|
||||
|
||||
285
includes/sp-url-builder.php
Normal file
285
includes/sp-url-builder.php
Normal 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
|
||||
}
|
||||
@@ -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';
|
||||
|
||||
Reference in New Issue
Block a user