Compare commits
29 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
083d0320e1
|
|||
|
518adda534
|
|||
|
43033ddfb2
|
|||
|
ac71e880a4
|
|||
|
5fbd902c6c
|
|||
|
635a76342e
|
|||
|
78a2229bb2
|
|||
|
e6a336751c
|
|||
|
b6ca3c3d38
|
|||
|
10ef2dd28c
|
|||
|
2afe98bc99
|
|||
|
1df307dfbe
|
|||
|
f38ceccdb3
|
|||
|
94b51c3959
|
|||
|
8f1a819a2d
|
|||
|
5a5bf5acdb
|
|||
|
dbe3048af7
|
|||
|
4ed968a045
|
|||
|
360b971880
|
|||
|
2dbdc5fae9
|
|||
|
4c07787a44
|
|||
|
25014f6368
|
|||
|
37d1037238
|
|||
|
65df3525a4
|
|||
|
bc0e913dfb
|
|||
|
2baebf9c30
|
|||
|
b2d4c240ad
|
|||
|
6837907aa9
|
|||
|
bfc74fcab6
|
93
assets/fonts/BebasNeue-OFL.txt
Normal file
93
assets/fonts/BebasNeue-OFL.txt
Normal file
@@ -0,0 +1,93 @@
|
||||
Copyright © 2010 by Dharma Type.
|
||||
|
||||
This Font Software is licensed under the SIL Open Font License, Version 1.1.
|
||||
This license is copied below, and is also available with a FAQ at:
|
||||
http://scripts.sil.org/OFL
|
||||
|
||||
|
||||
-----------------------------------------------------------
|
||||
SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007
|
||||
-----------------------------------------------------------
|
||||
|
||||
PREAMBLE
|
||||
The goals of the Open Font License (OFL) are to stimulate worldwide
|
||||
development of collaborative font projects, to support the font creation
|
||||
efforts of academic and linguistic communities, and to provide a free and
|
||||
open framework in which fonts may be shared and improved in partnership
|
||||
with others.
|
||||
|
||||
The OFL allows the licensed fonts to be used, studied, modified and
|
||||
redistributed freely as long as they are not sold by themselves. The
|
||||
fonts, including any derivative works, can be bundled, embedded,
|
||||
redistributed and/or sold with any software provided that any reserved
|
||||
names are not used by derivative works. The fonts and derivatives,
|
||||
however, cannot be released under any other type of license. The
|
||||
requirement for fonts to remain under this license does not apply
|
||||
to any document created using the fonts or their derivatives.
|
||||
|
||||
DEFINITIONS
|
||||
"Font Software" refers to the set of files released by the Copyright
|
||||
Holder(s) under this license and clearly marked as such. This may
|
||||
include source files, build scripts and documentation.
|
||||
|
||||
"Reserved Font Name" refers to any names specified as such after the
|
||||
copyright statement(s).
|
||||
|
||||
"Original Version" refers to the collection of Font Software components as
|
||||
distributed by the Copyright Holder(s).
|
||||
|
||||
"Modified Version" refers to any derivative made by adding to, deleting,
|
||||
or substituting -- in part or in whole -- any of the components of the
|
||||
Original Version, by changing formats or by porting the Font Software to a
|
||||
new environment.
|
||||
|
||||
"Author" refers to any designer, engineer, programmer, technical
|
||||
writer or other person who contributed to the Font Software.
|
||||
|
||||
PERMISSION & CONDITIONS
|
||||
Permission is hereby granted, free of charge, to any person obtaining
|
||||
a copy of the Font Software, to use, study, copy, merge, embed, modify,
|
||||
redistribute, and sell modified and unmodified copies of the Font
|
||||
Software, subject to the following conditions:
|
||||
|
||||
1) Neither the Font Software nor any of its individual components,
|
||||
in Original or Modified Versions, may be sold by itself.
|
||||
|
||||
2) Original or Modified Versions of the Font Software may be bundled,
|
||||
redistributed and/or sold with any software, provided that each copy
|
||||
contains the above copyright notice and this license. These can be
|
||||
included either as stand-alone text files, human-readable headers or
|
||||
in the appropriate machine-readable metadata fields within text or
|
||||
binary files as long as those fields can be easily viewed by the user.
|
||||
|
||||
3) No Modified Version of the Font Software may use the Reserved Font
|
||||
Name(s) unless explicit written permission is granted by the corresponding
|
||||
Copyright Holder. This restriction only applies to the primary font name as
|
||||
presented to the users.
|
||||
|
||||
4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font
|
||||
Software shall not be used to promote, endorse or advertise any
|
||||
Modified Version, except to acknowledge the contribution(s) of the
|
||||
Copyright Holder(s) and the Author(s) or with their explicit written
|
||||
permission.
|
||||
|
||||
5) The Font Software, modified or unmodified, in part or in whole,
|
||||
must be distributed entirely under this license, and must not be
|
||||
distributed under any other license. The requirement for fonts to
|
||||
remain under this license does not apply to any document created
|
||||
using the Font Software.
|
||||
|
||||
TERMINATION
|
||||
This license becomes null and void if any of the above conditions are
|
||||
not met.
|
||||
|
||||
DISCLAIMER
|
||||
THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF
|
||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
|
||||
OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE
|
||||
COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||
INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL
|
||||
DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM
|
||||
OTHER DEALINGS IN THE FONT SOFTWARE.
|
||||
BIN
assets/fonts/BebasNeue-Regular.ttf
Normal file
BIN
assets/fonts/BebasNeue-Regular.ttf
Normal file
Binary file not shown.
@@ -27,10 +27,7 @@ body {
|
||||
|
||||
.print-shell {
|
||||
margin: 0;
|
||||
width: calc(100% / var(--sheet-scale));
|
||||
background: #fff;
|
||||
transform-origin: top left;
|
||||
transform: scale(var(--sheet-scale));
|
||||
}
|
||||
|
||||
.print-page {
|
||||
@@ -48,7 +45,10 @@ body {
|
||||
}
|
||||
|
||||
body.print-preview .print-shell {
|
||||
flex: 0 0 auto;
|
||||
box-shadow: 0 10px 30px rgba(17, 24, 39, 0.2);
|
||||
transform-origin: top left;
|
||||
transform: scale(var(--sheet-scale));
|
||||
}
|
||||
|
||||
body.print-preview .print-shell.letter {
|
||||
@@ -60,6 +60,104 @@ body {
|
||||
width: 11in;
|
||||
min-height: 17in;
|
||||
}
|
||||
|
||||
body.print-preview.month-pages {
|
||||
display: block;
|
||||
overflow-x: auto;
|
||||
}
|
||||
|
||||
body.print-preview.month-pages .print-shell,
|
||||
body.print-preview.month-pages .print-shell.letter,
|
||||
body.print-preview.month-pages .print-shell.ledger {
|
||||
width: auto;
|
||||
min-height: auto;
|
||||
background: transparent;
|
||||
box-shadow: none;
|
||||
transform: none;
|
||||
}
|
||||
|
||||
body.print-preview.month-pages .print-page {
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
body.print-preview.month-pages .month-page {
|
||||
box-sizing: border-box;
|
||||
width: 8.5in;
|
||||
min-height: 11in;
|
||||
margin-right: auto;
|
||||
margin-left: auto;
|
||||
padding: var(--pc-page-padding);
|
||||
background: #fff;
|
||||
box-shadow: 0 10px 30px rgba(17, 24, 39, 0.2);
|
||||
}
|
||||
|
||||
body.print-preview.month-pages.ledger .month-page {
|
||||
width: 11in;
|
||||
min-height: 17in;
|
||||
}
|
||||
|
||||
body.print-preview.month-pages .month-page + .month-page {
|
||||
margin-top: 24px;
|
||||
}
|
||||
|
||||
body.print-preview.month-pages .month-page .header {
|
||||
margin-bottom: 24px;
|
||||
}
|
||||
|
||||
body.print-preview.month-pages .month-page .footer-meta {
|
||||
margin-top: 24px;
|
||||
}
|
||||
|
||||
body.print-preview.month-pages .sheet-grid {
|
||||
display: block;
|
||||
}
|
||||
|
||||
body.print-preview.month-pages .month {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
body.month-pages .month-title {
|
||||
font-size: calc(26px * var(--month-font-scale));
|
||||
padding: 4px;
|
||||
}
|
||||
|
||||
body.month-pages .dow span {
|
||||
font-size: calc(12px * var(--month-font-scale));
|
||||
padding: 4px 2px;
|
||||
}
|
||||
|
||||
body.month-pages .day-num {
|
||||
font-size: calc(14px * var(--month-font-scale));
|
||||
}
|
||||
|
||||
body.month-pages .event-name {
|
||||
font-size: calc(13px * var(--month-font-scale));
|
||||
line-height: 0.98;
|
||||
}
|
||||
|
||||
body.month-pages .matchup-name {
|
||||
font-size: calc(13px * var(--month-font-scale));
|
||||
line-height: 0.92;
|
||||
}
|
||||
|
||||
body.month-pages .event-time {
|
||||
font-size: calc(14px * var(--month-font-scale));
|
||||
}
|
||||
|
||||
body.month-pages .event.matchup .event-time {
|
||||
font-size: calc(11px * var(--month-font-scale));
|
||||
}
|
||||
|
||||
body.month-pages .event.matchup .event-venue {
|
||||
font-size: calc(10px * var(--month-font-scale));
|
||||
}
|
||||
}
|
||||
|
||||
@media screen and (max-width: 900px) {
|
||||
body.print-preview {
|
||||
justify-content: flex-start;
|
||||
overflow-x: auto;
|
||||
}
|
||||
}
|
||||
|
||||
.header {
|
||||
@@ -235,7 +333,7 @@ body {
|
||||
position: absolute;
|
||||
top: var(--corner-badge-offset);
|
||||
left: var(--corner-badge-offset);
|
||||
z-index: 4;
|
||||
z-index: 20;
|
||||
width: var(--corner-badge-size);
|
||||
height: var(--corner-badge-size);
|
||||
display: flex;
|
||||
@@ -252,12 +350,46 @@ body {
|
||||
color: var(--day-num-color, #fff);
|
||||
}
|
||||
|
||||
.day.has-matchups {
|
||||
background: #fff;
|
||||
border: 1px solid var(--pc-border);
|
||||
}
|
||||
|
||||
.day.has-matchups .day-num {
|
||||
position: absolute;
|
||||
z-index: 20;
|
||||
background: #fff;
|
||||
color: var(--team-ink, #111) !important;
|
||||
text-shadow: none !important;
|
||||
box-shadow: 0 0 0 1px var(--pc-border);
|
||||
}
|
||||
|
||||
body.month-pages .day .day-num {
|
||||
top: 1px;
|
||||
left: 1px;
|
||||
width: var(--corner-badge-size);
|
||||
height: var(--corner-badge-size);
|
||||
border-radius: 0;
|
||||
background: #fff;
|
||||
color: var(--team-ink, #111);
|
||||
text-shadow: none;
|
||||
}
|
||||
|
||||
.events-stack {
|
||||
position: relative;
|
||||
z-index: 1;
|
||||
height: 100%;
|
||||
display: grid;
|
||||
grid-template-rows: repeat(var(--event-count), minmax(0, 1fr));
|
||||
}
|
||||
|
||||
.day.has-matchups .events-stack {
|
||||
box-sizing: border-box;
|
||||
grid-template-rows: repeat(var(--event-count), minmax(0, 1fr));
|
||||
gap: 2px;
|
||||
padding: 2px;
|
||||
}
|
||||
|
||||
.event {
|
||||
--event-top-band: calc(var(--corner-badge-size, 11px) + var(--corner-badge-offset, 2px));
|
||||
--event-bottom-band: 26px;
|
||||
@@ -275,6 +407,17 @@ body {
|
||||
--event-bg: var(--team-link-color, var(--team-secondary, #8b3f1f));
|
||||
}
|
||||
|
||||
.event.matchup {
|
||||
display: grid;
|
||||
grid-template-columns: minmax(0, 1fr);
|
||||
grid-template-rows: minmax(0, 1fr) auto;
|
||||
gap: 1px;
|
||||
min-height: 0;
|
||||
padding: 2px 3px;
|
||||
border-radius: 2px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.event-center {
|
||||
position: absolute;
|
||||
top: var(--event-top-band);
|
||||
@@ -293,6 +436,17 @@ body {
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.event.matchup .event-center {
|
||||
position: relative;
|
||||
top: auto;
|
||||
left: auto;
|
||||
right: auto;
|
||||
bottom: auto;
|
||||
min-height: 0;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.event-center img {
|
||||
width: auto;
|
||||
height: var(--event-logo-height);
|
||||
@@ -315,6 +469,8 @@ body {
|
||||
hyphens: none;
|
||||
line-height: 1.05;
|
||||
font-weight: 700;
|
||||
font-stretch: condensed;
|
||||
font-variation-settings: "wdth" 42, "wght" 700;
|
||||
opacity: 0.85;
|
||||
}
|
||||
|
||||
@@ -322,10 +478,38 @@ body {
|
||||
font-family: var(--pc-font-display);
|
||||
font-weight: 700;
|
||||
font-variation-settings: "wdth" 30, "wght" 700;
|
||||
letter-spacing: -0.01em;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
|
||||
.matchup-name {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
gap: 0;
|
||||
padding: 0;
|
||||
font-size: calc(8.5px * var(--month-font-scale));
|
||||
line-height: 0.98;
|
||||
font-variation-settings: "wdth" 36, "wght" 800;
|
||||
opacity: 1;
|
||||
}
|
||||
|
||||
.matchup-vs {
|
||||
font-size: 0.66em;
|
||||
font-weight: 900;
|
||||
opacity: 0.78;
|
||||
}
|
||||
|
||||
.matchup-team {
|
||||
display: block;
|
||||
width: 100%;
|
||||
overflow: hidden;
|
||||
text-align: center;
|
||||
text-overflow: clip;
|
||||
white-space: nowrap;
|
||||
font-stretch: condensed;
|
||||
}
|
||||
|
||||
.ha-flag {
|
||||
position: absolute;
|
||||
top: var(--corner-badge-offset, 2px);
|
||||
@@ -341,7 +525,7 @@ body {
|
||||
font-weight: 900;
|
||||
font-variation-settings: "wdth" 84, "wght" 900;
|
||||
line-height: 1;
|
||||
letter-spacing: -0.01em;
|
||||
letter-spacing: 0;
|
||||
background: #111;
|
||||
color: #fff;
|
||||
}
|
||||
@@ -366,11 +550,26 @@ body {
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.event.matchup .event-meta {
|
||||
position: relative;
|
||||
left: auto;
|
||||
right: auto;
|
||||
bottom: auto;
|
||||
display: grid;
|
||||
grid-template-columns: minmax(0, auto) minmax(0, 1fr);
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
column-gap: 3px;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
.event-time {
|
||||
order: 1;
|
||||
max-width: 100%;
|
||||
font-size: calc(12px * var(--month-font-scale));
|
||||
font-weight: 800;
|
||||
font-stretch: condensed;
|
||||
font-variation-settings: "wdth" 56, "wght" 800;
|
||||
line-height: 1;
|
||||
text-transform: uppercase;
|
||||
text-align: center;
|
||||
@@ -381,11 +580,19 @@ body {
|
||||
opacity: 0.95;
|
||||
}
|
||||
|
||||
.event.matchup .event-time {
|
||||
min-width: 0;
|
||||
font-size: calc(7.5px * var(--month-font-scale));
|
||||
font-weight: 900;
|
||||
}
|
||||
|
||||
.event-venue {
|
||||
order: 2;
|
||||
max-width: 100%;
|
||||
font-size: calc(8px * var(--month-font-scale));
|
||||
font-weight: 700;
|
||||
font-stretch: condensed;
|
||||
font-variation-settings: "wdth" 50, "wght" 700;
|
||||
line-height: 1;
|
||||
text-align: center;
|
||||
text-transform: uppercase;
|
||||
@@ -395,6 +602,12 @@ body {
|
||||
opacity: 0.88;
|
||||
}
|
||||
|
||||
.event.matchup .event-venue {
|
||||
min-width: 0;
|
||||
font-size: calc(7px * var(--month-font-scale));
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
.empty {
|
||||
padding: 16px;
|
||||
border: 2px dashed #c8d2de;
|
||||
@@ -472,6 +685,100 @@ body {
|
||||
background: #fff;
|
||||
}
|
||||
|
||||
body.black-white .header,
|
||||
body.black-white .month-title,
|
||||
body.black-white .dow span,
|
||||
body.black-white .day,
|
||||
body.black-white .event,
|
||||
body.black-white .empty,
|
||||
body.black-white .footer-meta,
|
||||
body.black-white .footer-qr-image {
|
||||
border-color: #2f3337 !important;
|
||||
}
|
||||
|
||||
body.black-white .month-title,
|
||||
body.black-white .ha-flag {
|
||||
background: #1f2328 !important;
|
||||
color: #fff !important;
|
||||
text-shadow: none !important;
|
||||
}
|
||||
|
||||
body.black-white .dow span,
|
||||
body.black-white .day,
|
||||
body.black-white .day.muted,
|
||||
body.black-white .day.no-events,
|
||||
body.black-white .day.has-matchups,
|
||||
body.black-white .empty,
|
||||
body.black-white .legend-item,
|
||||
body.black-white .event.a .ha-flag {
|
||||
background: #f7f7f7 !important;
|
||||
color: #111 !important;
|
||||
text-shadow: none !important;
|
||||
}
|
||||
|
||||
body.black-white .dow span {
|
||||
background: #e4e7eb !important;
|
||||
}
|
||||
|
||||
body.black-white .day,
|
||||
body.black-white .day.has-matchups,
|
||||
body.black-white .legend-item {
|
||||
background: #fff !important;
|
||||
}
|
||||
|
||||
body.black-white .event,
|
||||
body.black-white .event.h,
|
||||
body.black-white .event.a,
|
||||
body.black-white .event.matchup {
|
||||
background: #e4e7eb !important;
|
||||
color: #111 !important;
|
||||
text-shadow: none !important;
|
||||
}
|
||||
|
||||
body.black-white .day.muted,
|
||||
body.black-white .day.no-events {
|
||||
background: #f2f3f5 !important;
|
||||
color: #4b5563 !important;
|
||||
}
|
||||
|
||||
body.black-white .month-title,
|
||||
body.black-white .dow span,
|
||||
body.black-white .day,
|
||||
body.black-white .event {
|
||||
border: 1px solid #2f3337;
|
||||
}
|
||||
|
||||
body.black-white .day-num,
|
||||
body.black-white .day.has-events .day-num,
|
||||
body.black-white .day.has-matchups .day-num,
|
||||
body.black-white.month-pages .day .day-num {
|
||||
background: #fff !important;
|
||||
color: #111 !important;
|
||||
box-shadow: 0 0 0 1px #2f3337 !important;
|
||||
text-shadow: none !important;
|
||||
}
|
||||
|
||||
body.black-white .ha-flag,
|
||||
body.black-white .event.a .ha-flag {
|
||||
box-shadow: 0 0 0 1px #2f3337;
|
||||
}
|
||||
|
||||
body.black-white .event-name,
|
||||
body.black-white .event-time,
|
||||
body.black-white .event-venue,
|
||||
body.black-white .matchup-vs,
|
||||
body.black-white .footer-qr-label,
|
||||
body.black-white .footer-qr-link,
|
||||
body.black-white .meta {
|
||||
color: #111 !important;
|
||||
opacity: 1 !important;
|
||||
}
|
||||
|
||||
body.black-white .meta,
|
||||
body.black-white .footer-qr-label {
|
||||
color: #374151 !important;
|
||||
}
|
||||
|
||||
@media print {
|
||||
body,
|
||||
body.print-preview {
|
||||
@@ -487,6 +794,7 @@ body {
|
||||
width: auto;
|
||||
min-height: auto;
|
||||
box-shadow: none;
|
||||
transform: none;
|
||||
}
|
||||
|
||||
.print-page {
|
||||
@@ -500,4 +808,27 @@ body {
|
||||
.title {
|
||||
font-size: calc(26px * var(--month-font-scale));
|
||||
}
|
||||
|
||||
body.month-pages .sheet-grid {
|
||||
display: block;
|
||||
}
|
||||
|
||||
body.month-pages .month-page {
|
||||
break-after: page;
|
||||
page-break-after: always;
|
||||
}
|
||||
|
||||
body.month-pages .month-page:last-child {
|
||||
break-after: auto;
|
||||
page-break-after: auto;
|
||||
}
|
||||
|
||||
body.month-pages .month {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
body.month-pages .grid {
|
||||
break-inside: avoid;
|
||||
page-break-inside: avoid;
|
||||
}
|
||||
}
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -1,39 +1,151 @@
|
||||
<?php
|
||||
/*
|
||||
Plugin Name: Custom Open Graph Tags with SportsPress Integration
|
||||
Description: Adds custom Open Graph tags to posts based on their type, specifically handling sp_event post types with methods from the SportsPress SP_Event class.
|
||||
Version: 1.0
|
||||
Author: Your Name
|
||||
*/
|
||||
/**
|
||||
* Open Graph tags for SportsPress events.
|
||||
*
|
||||
* @package Tonys_Sportspress_Enhancements
|
||||
*/
|
||||
|
||||
add_action('wp_head', 'custom_open_graph_tags_with_sportspress_integration');
|
||||
add_action( 'wp_head', 'custom_open_graph_tags_with_sportspress_integration' );
|
||||
|
||||
function asc_generate_sp_event_title( $post ) {
|
||||
// See https://github.com/ThemeBoy/SportsPress/blob/770fa8c6654d7d6648791e877709c2428677635b/includes/admin/post-types/class-sp-admin-cpt-event.php#L99C40-L99C55
|
||||
if ( is_numeric( $post ) ) {
|
||||
$post = get_post( $post );
|
||||
}
|
||||
if ( ! $post || $post->post_type !== 'sp_event' ) {
|
||||
return get_the_title();
|
||||
/**
|
||||
* Build the dynamic matchup image URL for an event.
|
||||
*
|
||||
* @param int|WP_Post $post Event post or ID.
|
||||
* @param string $variant Image variant.
|
||||
* @return string
|
||||
*/
|
||||
function asc_sp_event_matchup_image_url( $post, $variant = 'wide' ) {
|
||||
$post = asc_sp_event_get_post( $post );
|
||||
|
||||
if ( ! $post ) {
|
||||
return '';
|
||||
}
|
||||
|
||||
$teams = get_post_meta( $post->ID, 'sp_team', false );
|
||||
$teams = array_filter( $teams );
|
||||
$args = array(
|
||||
'post' => $post->ID,
|
||||
);
|
||||
|
||||
if ( 'wide' !== $variant ) {
|
||||
$args['variant'] = $variant;
|
||||
}
|
||||
|
||||
if ( function_exists( 'asc_sp_event_image_url_version' ) ) {
|
||||
$args['v'] = asc_sp_event_image_url_version();
|
||||
}
|
||||
|
||||
return add_query_arg( $args, home_url( '/head-to-head' ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Build Open Graph image descriptors for an event.
|
||||
*
|
||||
* @param WP_Post $post Event post.
|
||||
* @return array<int,array<string,string>>
|
||||
*/
|
||||
function asc_sp_event_open_graph_images( WP_Post $post ) {
|
||||
return array(
|
||||
array(
|
||||
'url' => asc_sp_event_matchup_image_url( $post, 'wide' ),
|
||||
'width' => '1200',
|
||||
'height' => '628',
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Normalize an event post argument.
|
||||
*
|
||||
* @param int|WP_Post|null $post Post object or ID.
|
||||
* @return WP_Post|null
|
||||
*/
|
||||
function asc_sp_event_get_post( $post = null ) {
|
||||
if ( null === $post ) {
|
||||
$post = get_post();
|
||||
} elseif ( is_numeric( $post ) ) {
|
||||
$post = get_post( absint( $post ) );
|
||||
}
|
||||
|
||||
if ( ! $post instanceof WP_Post || 'sp_event' !== $post->post_type ) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return $post;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get event team IDs in SportsPress display order.
|
||||
*
|
||||
* @param int|WP_Post $post Event post or ID.
|
||||
* @return int[]
|
||||
*/
|
||||
function asc_sp_event_team_ids( $post ) {
|
||||
$post = asc_sp_event_get_post( $post );
|
||||
|
||||
if ( ! $post ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
$teams = get_post_meta( $post->ID, 'sp_team', false );
|
||||
$team_ids = array();
|
||||
|
||||
$team_names = array();
|
||||
foreach ( $teams as $team ) {
|
||||
while ( is_array( $team ) ) {
|
||||
$team = array_shift( array_filter( $team ) );
|
||||
}
|
||||
|
||||
$team = absint( $team );
|
||||
if ( $team > 0 ) {
|
||||
$team_names[] = sp_team_short_name( $team );
|
||||
$team_ids[] = $team;
|
||||
}
|
||||
}
|
||||
|
||||
$team_names = array_unique( $team_names );
|
||||
$team_ids = array_values( array_unique( $team_ids ) );
|
||||
|
||||
if ( get_option( 'sportspress_event_reverse_teams', 'no' ) === 'yes' ) {
|
||||
$team_names = array_reverse( $team_names );
|
||||
if ( 'yes' === get_option( 'sportspress_event_reverse_teams', 'no' ) ) {
|
||||
$team_ids = array_reverse( $team_ids );
|
||||
}
|
||||
|
||||
return $team_ids;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get a safe team short name with fallbacks for test and partial SportsPress environments.
|
||||
*
|
||||
* @param int $team_id Team post ID.
|
||||
* @return string
|
||||
*/
|
||||
function asc_sp_team_short_name( $team_id ) {
|
||||
$name = '';
|
||||
|
||||
if ( function_exists( 'sp_team_short_name' ) ) {
|
||||
$name = (string) sp_team_short_name( $team_id );
|
||||
}
|
||||
|
||||
if ( '' === trim( $name ) ) {
|
||||
$name = get_the_title( $team_id );
|
||||
}
|
||||
|
||||
return '' !== trim( $name ) ? $name : __( 'Team TBD', 'tonys-sportspress-enhancements' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Generate a matchup title from event teams.
|
||||
*
|
||||
* @param int|WP_Post $post Event post or ID.
|
||||
* @return string
|
||||
*/
|
||||
function asc_generate_sp_event_title( $post ) {
|
||||
$post = asc_sp_event_get_post( $post );
|
||||
|
||||
if ( ! $post ) {
|
||||
return get_the_title();
|
||||
}
|
||||
|
||||
$team_names = array_map( 'asc_sp_team_short_name', asc_sp_event_team_ids( $post ) );
|
||||
$team_names = array_values( array_filter( array_unique( $team_names ) ) );
|
||||
|
||||
if ( empty( $team_names ) ) {
|
||||
return get_the_title( $post );
|
||||
}
|
||||
|
||||
$delimiter = ' ' . get_option( 'sportspress_event_teams_delimiter', 'vs' ) . ' ';
|
||||
@@ -41,147 +153,343 @@ function asc_generate_sp_event_title( $post ) {
|
||||
return implode( $delimiter, $team_names );
|
||||
}
|
||||
|
||||
/**
|
||||
* Generate compact event date text.
|
||||
*
|
||||
* @param int|WP_Post $post Event post or ID.
|
||||
* @param bool $withTime Include time.
|
||||
* @return string
|
||||
*/
|
||||
function asc_generate_short_date( $post, $withTime = true ) {
|
||||
$formatted_date = get_the_date('D n/j/y', $post);
|
||||
$post = asc_sp_event_get_post( $post );
|
||||
|
||||
if (!$withTime){
|
||||
return $formatted_date;
|
||||
}
|
||||
if ( ! $post ) {
|
||||
return '';
|
||||
}
|
||||
|
||||
if ( get_the_date('i', $post) == "00") {
|
||||
$formatted_time = get_the_date('gA', $post);
|
||||
} else {
|
||||
$formatted_time = get_the_date('g:iA', $post);
|
||||
}
|
||||
return $formatted_date . " " . $formatted_time ;
|
||||
$formatted_date = get_the_date( 'D n/j/y', $post );
|
||||
|
||||
if ( ! $withTime ) {
|
||||
return $formatted_date;
|
||||
}
|
||||
|
||||
$formatted_time = '00' === get_the_date( 'i', $post ) ? get_the_date( 'gA', $post ) : get_the_date( 'g:iA', $post );
|
||||
|
||||
return trim( $formatted_date . ' ' . $formatted_time );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get venue name for an event.
|
||||
*
|
||||
* @param WP_Post $post Event post.
|
||||
* @return string
|
||||
*/
|
||||
function asc_sp_event_venue_name( WP_Post $post ) {
|
||||
$venue_terms = get_the_terms( $post->ID, 'sp_venue' );
|
||||
|
||||
if ( is_wp_error( $venue_terms ) || empty( $venue_terms ) ) {
|
||||
return __( 'Venue TBD', 'tonys-sportspress-enhancements' );
|
||||
}
|
||||
|
||||
return $venue_terms[0]->name;
|
||||
}
|
||||
|
||||
/**
|
||||
* Normalize event body content for meta descriptions.
|
||||
*
|
||||
* @param WP_Post $post Event post.
|
||||
* @return string
|
||||
*/
|
||||
function asc_sp_event_body_excerpt( WP_Post $post ) {
|
||||
$content = strip_shortcodes( $post->post_content );
|
||||
$content = wp_strip_all_tags( $content, true );
|
||||
$content = html_entity_decode( $content, ENT_QUOTES, get_bloginfo( 'charset' ) );
|
||||
$content = preg_replace( '/\s+/', ' ', $content );
|
||||
$content = trim( (string) $content );
|
||||
|
||||
if ( '' === $content ) {
|
||||
return '';
|
||||
}
|
||||
|
||||
return wp_trim_words( $content, 35, '' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Safely instantiate a SportsPress event object.
|
||||
*
|
||||
* @param WP_Post $post Event post.
|
||||
* @return object|null
|
||||
*/
|
||||
function asc_sp_event_object( WP_Post $post ) {
|
||||
if ( ! class_exists( 'SP_Event' ) ) {
|
||||
return null;
|
||||
}
|
||||
|
||||
try {
|
||||
return new SP_Event( $post->ID );
|
||||
} catch ( Throwable $e ) {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the SportsPress event status with fallbacks.
|
||||
*
|
||||
* @param WP_Post $post Event post.
|
||||
* @param object|null $event SportsPress event object.
|
||||
* @return string
|
||||
*/
|
||||
function asc_sp_event_status( WP_Post $post, $event = null ) {
|
||||
if ( $event && is_callable( array( $event, 'status' ) ) ) {
|
||||
try {
|
||||
$status = (string) $event->status();
|
||||
if ( '' !== $status ) {
|
||||
return $status;
|
||||
}
|
||||
} catch ( Throwable $e ) {
|
||||
return '';
|
||||
}
|
||||
}
|
||||
|
||||
return 'future' === $post->post_status ? 'future' : '';
|
||||
}
|
||||
|
||||
/**
|
||||
* Get SportsPress result rows safely.
|
||||
*
|
||||
* @param object|null $event SportsPress event object.
|
||||
* @return array
|
||||
*/
|
||||
function asc_sp_event_results( $event = null ) {
|
||||
if ( ! $event || ! is_callable( array( $event, 'results' ) ) ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
try {
|
||||
$results = $event->results();
|
||||
return is_array( $results ) ? $results : array();
|
||||
} catch ( Throwable $e ) {
|
||||
return array();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert a result row into outcome labels.
|
||||
*
|
||||
* @param array $result Result row.
|
||||
* @return array
|
||||
*/
|
||||
function asc_sp_event_result_outcomes( array $result ) {
|
||||
$result_outcome = isset( $result['outcome'] ) ? $result['outcome'] : null;
|
||||
|
||||
if ( ! is_array( $result_outcome ) ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
$outcomes = array();
|
||||
|
||||
foreach ( $result_outcome as $outcome ) {
|
||||
$the_outcome = get_page_by_path( $outcome, OBJECT, 'sp_outcome' );
|
||||
|
||||
if ( $the_outcome instanceof WP_Post ) {
|
||||
$outcome_abbreviation = get_post_meta( $the_outcome->ID, 'sp_abbreviation', true );
|
||||
if ( ! $outcome_abbreviation ) {
|
||||
$outcome_abbreviation = function_exists( 'sp_substr' ) ? sp_substr( $the_outcome->post_title, 0, 1 ) : substr( $the_outcome->post_title, 0, 1 );
|
||||
}
|
||||
|
||||
$outcomes[] = array(
|
||||
'title' => $the_outcome->post_title,
|
||||
'abbreviation' => $outcome_abbreviation,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
return $outcomes;
|
||||
}
|
||||
|
||||
/**
|
||||
* Build a result title/description from SportsPress result data.
|
||||
*
|
||||
* @param WP_Post $post Event post.
|
||||
* @param array $results Event results data.
|
||||
* @param string $description Existing description.
|
||||
* @return array{title:string,description:string}|null
|
||||
*/
|
||||
function asc_sp_event_result_meta( WP_Post $post, array $results, $description ) {
|
||||
unset( $results[0] );
|
||||
|
||||
$results = array_filter( $results );
|
||||
|
||||
if ( count( $results ) < 2 ) {
|
||||
return null;
|
||||
}
|
||||
|
||||
if ( 'yes' === get_option( 'sportspress_event_reverse_teams', 'no' ) ) {
|
||||
$results = array_reverse( $results, true );
|
||||
}
|
||||
|
||||
$teams_result_array = array();
|
||||
|
||||
foreach ( $results as $team_id => $result ) {
|
||||
if ( ! is_array( $result ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$outcomes = asc_sp_event_result_outcomes( $result );
|
||||
$first_outcome = ! empty( $outcomes ) ? $outcomes[0] : array( 'title' => __( 'Result', 'tonys-sportspress-enhancements' ), 'abbreviation' => '' );
|
||||
$team_name = asc_sp_team_short_name( $team_id );
|
||||
$team_score = isset( $result['r'] ) && '' !== $result['r'] ? $result['r'] : null;
|
||||
$team_score = null !== $team_score ? (string) $team_score : '';
|
||||
|
||||
if ( '' === $team_score ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$teams_result_array[] = array(
|
||||
'score' => $team_score,
|
||||
'outcome' => $first_outcome['title'],
|
||||
'outcome_abbreviation' => $first_outcome['abbreviation'],
|
||||
'team_name' => $team_name,
|
||||
);
|
||||
}
|
||||
|
||||
if ( count( $teams_result_array ) < 2 ) {
|
||||
return null;
|
||||
}
|
||||
|
||||
$special_result_suffix_abbreviation = '';
|
||||
$special_result_suffix = '';
|
||||
|
||||
foreach ( $teams_result_array as $team ) {
|
||||
$outcome_abbreviation = strtoupper( (string) $team['outcome_abbreviation'] );
|
||||
|
||||
if ( 'TF-W' === $outcome_abbreviation ) {
|
||||
$special_result_suffix_abbreviation = 'TF-W';
|
||||
$special_result_suffix = 'Technical Forfeit Win';
|
||||
break;
|
||||
}
|
||||
|
||||
if ( 'TF-L' === $outcome_abbreviation ) {
|
||||
$special_result_suffix_abbreviation = 'TF';
|
||||
$special_result_suffix = 'Technical Forfeit';
|
||||
break;
|
||||
}
|
||||
|
||||
if ( 'F-W' === $outcome_abbreviation || 'F-L' === $outcome_abbreviation ) {
|
||||
$special_result_suffix_abbreviation = 'Forfeit';
|
||||
$special_result_suffix = 'Forfeit';
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
$publish_date = asc_generate_short_date( $post, false );
|
||||
$title = sprintf(
|
||||
'%1$s %2$s-%3$s %4$s%s',
|
||||
$teams_result_array[0]['team_name'],
|
||||
$teams_result_array[0]['score'],
|
||||
$teams_result_array[1]['score'],
|
||||
$teams_result_array[1]['team_name'],
|
||||
$publish_date ? ' - ' . $publish_date : ''
|
||||
);
|
||||
|
||||
if ( $special_result_suffix ) {
|
||||
$title .= " ({$special_result_suffix_abbreviation})";
|
||||
}
|
||||
|
||||
$description .= sprintf(
|
||||
' %1$s (%2$s), %3$s (%4$s).',
|
||||
$teams_result_array[0]['team_name'],
|
||||
$teams_result_array[0]['outcome'],
|
||||
$teams_result_array[1]['team_name'],
|
||||
$teams_result_array[1]['outcome']
|
||||
);
|
||||
|
||||
return array(
|
||||
'title' => $title,
|
||||
'description' => $description,
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Build all Open Graph values for an event.
|
||||
*
|
||||
* @param int|WP_Post $post Event post or ID.
|
||||
* @return array<string,mixed>
|
||||
*/
|
||||
function asc_sp_event_open_graph_data( $post ) {
|
||||
$post = asc_sp_event_get_post( $post );
|
||||
|
||||
if ( ! $post ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
$event = asc_sp_event_object( $post );
|
||||
$venue_name = asc_sp_event_venue_name( $post );
|
||||
$publish_date_and_time = get_the_date( 'F j, Y g:i A', $post );
|
||||
$description = trim( "{$publish_date_and_time} at {$venue_name}." );
|
||||
$title = asc_generate_sp_event_title( $post );
|
||||
$sp_status = get_post_meta( $post->ID, 'sp_status', true );
|
||||
$status = asc_sp_event_status( $post, $event );
|
||||
|
||||
if ( in_array( $sp_status, array( 'postponed', 'cancelled', 'tbd' ), true ) ) {
|
||||
$status_label = strtoupper( $sp_status );
|
||||
$description = "{$status_label} - {$description}";
|
||||
$title = trim( "{$status_label} - {$title} - " . asc_generate_short_date( $post ) . " - {$venue_name}", ' -' );
|
||||
} elseif ( 'future' === $status ) {
|
||||
$title = trim( $title . ' - ' . asc_generate_short_date( $post ) . " - {$venue_name}", ' -' );
|
||||
} elseif ( 'results' === $status ) {
|
||||
$result_meta = asc_sp_event_result_meta( $post, asc_sp_event_results( $event ), $description );
|
||||
|
||||
if ( $result_meta ) {
|
||||
$title = $result_meta['title'];
|
||||
$description = $result_meta['description'];
|
||||
}
|
||||
}
|
||||
|
||||
$body_excerpt = asc_sp_event_body_excerpt( $post );
|
||||
if ( '' !== $body_excerpt ) {
|
||||
$description = trim( $description . ' ' . $body_excerpt );
|
||||
}
|
||||
|
||||
return array(
|
||||
'type' => 'article',
|
||||
'images' => asc_sp_event_open_graph_images( $post ),
|
||||
'image' => asc_sp_event_matchup_image_url( $post, 'wide' ),
|
||||
'image_width' => '1200',
|
||||
'image_height' => '628',
|
||||
'title' => $title,
|
||||
'description' => $description,
|
||||
'url' => get_permalink( $post ),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Echo Open Graph meta tags for single SportsPress events.
|
||||
*/
|
||||
function custom_open_graph_tags_with_sportspress_integration() {
|
||||
if (is_single()) {
|
||||
global $post;
|
||||
if ($post->post_type === 'sp_event') {
|
||||
// Instantiate SP_Event object
|
||||
$event = new SP_Event($post->ID);
|
||||
if ( ! is_single() ) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Fetch details using SP_Event methods
|
||||
$publish_date = get_the_date('F j, Y', $post);
|
||||
$venue_terms = get_the_terms($post->ID, 'sp_venue');
|
||||
$venue_name = $venue_terms ? $venue_terms[0]->name : 'Venue TBD';
|
||||
$results = $event->results(); // Using SP_Event method
|
||||
$title = asc_generate_sp_event_title($post);
|
||||
$sp_status = get_post_meta( $post->ID, 'sp_status', true );
|
||||
$status = $event->status(); // Using SP_Event method
|
||||
$publish_date_and_time = get_the_date('F j, Y g:i A', $post);
|
||||
$description = "{$publish_date_and_time} at {$venue_name}.";
|
||||
|
||||
if ( 'postponed' == $sp_status || 'cancelled' == $sp_status || 'tbd' == $sp_status) {
|
||||
$description = strtoupper($sp_status) . " — " . $description;
|
||||
$title = strtoupper($sp_status) . " — " . $title . " — " . asc_generate_short_date($post) . " — " . $venue_name;
|
||||
}
|
||||
$post = asc_sp_event_get_post();
|
||||
|
||||
if ( 'future' == $status ) {
|
||||
$description = $description;
|
||||
$title = $title . " — " . asc_generate_short_date($post) . " — " . $venue_name;
|
||||
}
|
||||
if ( ! $post ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( 'results' == $status ) { // checks if there is a final score
|
||||
// Get event result data
|
||||
$data = $event->results();
|
||||
$meta = asc_sp_event_open_graph_data( $post );
|
||||
|
||||
// The first row should be column labels
|
||||
$labels = $data[0];
|
||||
if ( empty( $meta ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Remove the first row to leave us with the actual data
|
||||
unset( $data[0] );
|
||||
|
||||
$data = array_filter( $data );
|
||||
|
||||
if ( empty( $data ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// Initialize
|
||||
$i = 0;
|
||||
$result_string = '';
|
||||
$title_string = '';
|
||||
|
||||
// Reverse teams order if the option "Events > Teams > Order > Reverse order" is enabled.
|
||||
$reverse_teams = get_option( 'sportspress_event_reverse_teams', 'no' ) === 'yes' ? true : false;
|
||||
if ( $reverse_teams ) {
|
||||
$data = array_reverse( $data, true );
|
||||
}
|
||||
|
||||
$teams_result_array = [];
|
||||
|
||||
foreach ( $data as $team_id => $result ) :
|
||||
$outcomes = array();
|
||||
$result_outcome = sp_array_value( $result, 'outcome' );
|
||||
if ( ! is_array( $result_outcome ) ) :
|
||||
$outcomes = array( '—' );
|
||||
else :
|
||||
foreach ( $result_outcome as $outcome ) :
|
||||
$the_outcome = get_page_by_path( $outcome, OBJECT, 'sp_outcome' );
|
||||
if ( is_object( $the_outcome ) ) :
|
||||
$outcomes[] = $the_outcome->post_title;
|
||||
endif;
|
||||
endforeach;
|
||||
endif;
|
||||
|
||||
unset( $result['outcome'] );
|
||||
|
||||
$team_name = sp_team_short_name( $team_id );
|
||||
$team_abbreviation = sp_team_abbreviation( $team_id );
|
||||
|
||||
$outcome_abbreviation = get_post_meta( $the_outcome->ID, 'sp_abbreviation', true );
|
||||
if ( ! $outcome_abbreviation ) {
|
||||
$outcome_abbreviation = sp_substr( $the_outcome->post_title, 0, 1 );
|
||||
}
|
||||
|
||||
array_push($teams_result_array, [
|
||||
"result" => $result,
|
||||
"outcome" => $the_outcome->post_title,
|
||||
"outcome_abbreviation" => $outcome_abbreviation,
|
||||
"team_name" => $team_name,
|
||||
"team_abbreviation" => $team_abbreviation
|
||||
]
|
||||
);
|
||||
$i++;
|
||||
endforeach;
|
||||
$publish_date = asc_generate_short_date($post, false);
|
||||
|
||||
$special_result_suffix_abbreviation = '';
|
||||
$special_result_suffix= '';
|
||||
|
||||
foreach ( $teams_result_array as $team ) {
|
||||
$outcome_abbreviation = strtoupper( $team['outcome_abbreviation'] ); // Normalize case
|
||||
|
||||
if ( $outcome_abbreviation === 'TF-W' ) {
|
||||
$special_result_suffix_abbreviation = 'TF-W';
|
||||
$special_result_suffix = 'Technical Forfeit Win';
|
||||
break;
|
||||
} elseif ( $outcome_abbreviation === 'TF-L' ) {
|
||||
$special_result_suffix_abbreviation = 'TF';
|
||||
$special_result_suffix = 'Technical Forfeit';
|
||||
break;
|
||||
} elseif ( $outcome_abbreviation === 'F-W' || $outcome_abbreviation === 'F-L' ) {
|
||||
$special_result_suffix_abbreviation = 'Forfeit';
|
||||
$special_result_suffix = 'Forfeit';
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
$title = "{$teams_result_array[0]['team_name']} {$teams_result_array[0]['result']['r']}-{$teams_result_array[1]['result']['r']} {$teams_result_array[1]['team_name']} — {$publish_date}" . ($special_result_suffix ? "({$special_result_suffix_abbreviation})" : "");
|
||||
$description .= " " . "{$teams_result_array[0]['team_name']} ({$teams_result_array[0]['outcome']}), {$teams_result_array[1]['team_name']} ({$teams_result_array[1]['outcome']})." ;
|
||||
}
|
||||
$description .= " " . $post->post_content;
|
||||
$image = get_site_url() . "/head-to-head?post={$post->ID}";
|
||||
echo '<meta property="og:type" content="article" />' . "\n";
|
||||
echo '<meta property="og:image" content="'. $image . '" />' . "\n";
|
||||
echo '<meta property="og:title" content="' . $title . '" />' . "\n";
|
||||
echo '<meta property="og:description" content="' . $description . '" />' . "\n";
|
||||
echo '<meta property="og:url" content="' . get_permalink() . '" />' . "\n";
|
||||
}
|
||||
}
|
||||
echo '<meta property="og:type" content="' . esc_attr( $meta['type'] ) . '" />' . "\n";
|
||||
foreach ( $meta['images'] as $image ) {
|
||||
echo '<meta property="og:image" content="' . esc_url( $image['url'] ) . '" />' . "\n";
|
||||
echo '<meta property="og:image:width" content="' . esc_attr( $image['width'] ) . '" />' . "\n";
|
||||
echo '<meta property="og:image:height" content="' . esc_attr( $image['height'] ) . '" />' . "\n";
|
||||
}
|
||||
echo '<meta property="og:title" content="' . esc_attr( $meta['title'] ) . '" />' . "\n";
|
||||
echo '<meta property="og:description" content="' . esc_attr( $meta['description'] ) . '" />' . "\n";
|
||||
echo '<meta property="og:url" content="' . esc_url( $meta['url'] ) . '" />' . "\n";
|
||||
}
|
||||
?>
|
||||
|
||||
@@ -18,7 +18,7 @@ if ( ! defined( 'ABSPATH' ) ) {
|
||||
function tony_sportspress_event_filter_defaults() {
|
||||
return array(
|
||||
'month' => true,
|
||||
'week' => true,
|
||||
'week' => false,
|
||||
'team' => true,
|
||||
'venue' => true,
|
||||
'league' => true,
|
||||
@@ -37,6 +37,52 @@ function tony_sportspress_event_filter_meta_key( $key ) {
|
||||
return 'tony_sp_event_filter_' . $key;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the current singular label for event venues.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
function tony_sportspress_get_event_venue_label() {
|
||||
$taxonomy = get_taxonomy( 'sp_venue' );
|
||||
|
||||
if ( $taxonomy && ! empty( $taxonomy->labels->singular_name ) ) {
|
||||
return (string) $taxonomy->labels->singular_name;
|
||||
}
|
||||
|
||||
return __( 'Venue', 'sportspress' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the current plural label for event venues.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
function tony_sportspress_get_event_venue_label_plural() {
|
||||
$taxonomy = get_taxonomy( 'sp_venue' );
|
||||
|
||||
if ( $taxonomy && ! empty( $taxonomy->labels->name ) ) {
|
||||
return (string) $taxonomy->labels->name;
|
||||
}
|
||||
|
||||
return __( 'Venues', 'sportspress' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Normalize event filter visibility rules.
|
||||
*
|
||||
* Month/Year and Week are mutually exclusive.
|
||||
*
|
||||
* @param array<string, bool> $filters Filter states keyed by filter name.
|
||||
* @return array<string, bool>
|
||||
*/
|
||||
function tony_sportspress_normalize_event_filter_states( $filters ) {
|
||||
if ( ! empty( $filters['month'] ) && ! empty( $filters['week'] ) ) {
|
||||
$filters['week'] = false;
|
||||
}
|
||||
|
||||
return $filters;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check whether a filter is enabled for the current user.
|
||||
*
|
||||
@@ -44,7 +90,7 @@ function tony_sportspress_event_filter_meta_key( $key ) {
|
||||
* @return bool
|
||||
*/
|
||||
function tony_sportspress_event_filter_enabled( $key ) {
|
||||
$defaults = tony_sportspress_event_filter_defaults();
|
||||
$defaults = tony_sportspress_normalize_event_filter_states( tony_sportspress_event_filter_defaults() );
|
||||
if ( ! array_key_exists( $key, $defaults ) ) {
|
||||
return true;
|
||||
}
|
||||
@@ -59,7 +105,15 @@ function tony_sportspress_event_filter_enabled( $key ) {
|
||||
return (bool) $defaults[ $key ];
|
||||
}
|
||||
|
||||
return '1' === (string) $stored;
|
||||
$states = array();
|
||||
foreach ( $defaults as $filter_key => $enabled ) {
|
||||
$current = get_user_meta( $user_id, tony_sportspress_event_filter_meta_key( $filter_key ), true );
|
||||
$states[ $filter_key ] = '' === $current ? (bool) $enabled : '1' === (string) $current;
|
||||
}
|
||||
|
||||
$states = tony_sportspress_normalize_event_filter_states( $states );
|
||||
|
||||
return ! empty( $states[ $key ] );
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -72,13 +126,21 @@ function tony_sportspress_save_event_filter_screen_options_ajax() {
|
||||
|
||||
check_ajax_referer( 'tony_sp_event_filters', 'nonce' );
|
||||
|
||||
$defaults = tony_sportspress_event_filter_defaults();
|
||||
$defaults = tony_sportspress_normalize_event_filter_states( tony_sportspress_event_filter_defaults() );
|
||||
$filters = isset( $_POST['filters'] ) && is_array( $_POST['filters'] ) ? $_POST['filters'] : array();
|
||||
$user_id = get_current_user_id();
|
||||
|
||||
$states = array();
|
||||
|
||||
foreach ( $defaults as $key => $_enabled ) {
|
||||
$value = isset( $filters[ $key ] ) ? sanitize_text_field( wp_unslash( $filters[ $key ] ) ) : '0';
|
||||
update_user_meta( $user_id, tony_sportspress_event_filter_meta_key( $key ), '1' === $value ? '1' : '0' );
|
||||
$states[ $key ] = '1' === $value;
|
||||
}
|
||||
|
||||
$states = tony_sportspress_normalize_event_filter_states( $states );
|
||||
|
||||
foreach ( $states as $key => $enabled ) {
|
||||
update_user_meta( $user_id, tony_sportspress_event_filter_meta_key( $key ), $enabled ? '1' : '0' );
|
||||
}
|
||||
|
||||
wp_send_json_success();
|
||||
@@ -99,9 +161,9 @@ function tony_sportspress_event_filter_screen_options_markup( $settings, $screen
|
||||
|
||||
$labels = array(
|
||||
'month' => __( 'Month/Year', 'tonys-sportspress-enhancements' ),
|
||||
'week' => __( 'Week', 'tonys-sportspress-enhancements' ),
|
||||
'week' => __( 'Year/Week', 'tonys-sportspress-enhancements' ),
|
||||
'team' => __( 'Team', 'tonys-sportspress-enhancements' ),
|
||||
'venue' => __( 'Venue', 'tonys-sportspress-enhancements' ),
|
||||
'venue' => tony_sportspress_get_event_venue_label(),
|
||||
'league' => __( 'League', 'tonys-sportspress-enhancements' ),
|
||||
'season' => __( 'Season', 'tonys-sportspress-enhancements' ),
|
||||
'match_day' => __( 'Match Day', 'tonys-sportspress-enhancements' ),
|
||||
@@ -135,7 +197,21 @@ function tony_sportspress_parse_admin_week_filter() {
|
||||
return null;
|
||||
}
|
||||
|
||||
$raw = sanitize_text_field( wp_unslash( $_GET['sp_week_filter'] ) );
|
||||
$raw = sanitize_text_field( wp_unslash( $_GET['sp_week_filter'] ) );
|
||||
$timezone = wp_timezone();
|
||||
|
||||
if ( 'this-week' === $raw || 'next-week' === $raw ) {
|
||||
$base = new DateTimeImmutable( 'now', $timezone );
|
||||
if ( 'next-week' === $raw ) {
|
||||
$base = $base->modify( '+1 week' );
|
||||
}
|
||||
|
||||
return array(
|
||||
'year' => (int) $base->format( 'o' ),
|
||||
'week' => (int) $base->format( 'W' ),
|
||||
);
|
||||
}
|
||||
|
||||
if ( ! preg_match( '/^(\d{4})-W(0[1-9]|[1-4][0-9]|5[0-3])$/', $raw, $matches ) ) {
|
||||
return null;
|
||||
}
|
||||
@@ -149,6 +225,66 @@ function tony_sportspress_parse_admin_week_filter() {
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get available ISO week options from event post dates.
|
||||
*
|
||||
* @return array<int, array{value:string,label:string}>
|
||||
*/
|
||||
function tony_sportspress_get_admin_week_filter_options() {
|
||||
global $wpdb;
|
||||
|
||||
$results = $wpdb->get_results(
|
||||
$wpdb->prepare(
|
||||
"SELECT DISTINCT DATE_FORMAT(post_date, '%%x-W%%v') AS iso_week
|
||||
FROM {$wpdb->posts}
|
||||
WHERE post_type = %s
|
||||
AND post_status NOT IN ('auto-draft', 'trash')
|
||||
AND post_date IS NOT NULL
|
||||
AND post_date <> '0000-00-00 00:00:00'
|
||||
ORDER BY iso_week DESC",
|
||||
'sp_event'
|
||||
),
|
||||
ARRAY_A
|
||||
);
|
||||
|
||||
if ( ! is_array( $results ) || empty( $results ) ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
$options = array();
|
||||
$timezone = wp_timezone();
|
||||
|
||||
foreach ( $results as $result ) {
|
||||
if ( empty( $result['iso_week'] ) || ! preg_match( '/^(\d{4})-W(0[1-9]|[1-4][0-9]|5[0-3])$/', $result['iso_week'], $matches ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$year = (int) $matches[1];
|
||||
$week = (int) $matches[2];
|
||||
$monday = ( new DateTimeImmutable( 'now', $timezone ) )->setISODate( $year, $week, 1 )->setTime( 0, 0, 0 );
|
||||
$sunday = $monday->modify( '+6 days' );
|
||||
$start_label = wp_date( 'M j', $monday->getTimestamp(), $timezone );
|
||||
$end_label = wp_date(
|
||||
$monday->format( 'n' ) === $sunday->format( 'n' ) ? 'j' : 'M j',
|
||||
$sunday->getTimestamp(),
|
||||
$timezone
|
||||
);
|
||||
|
||||
$options[] = array(
|
||||
'value' => $result['iso_week'],
|
||||
/* translators: 1: ISO week code, 2: Monday date, 3: Sunday date. */
|
||||
'label' => sprintf(
|
||||
__( '%1$s (%2$s to %3$s)', 'tonys-sportspress-enhancements' ),
|
||||
$result['iso_week'],
|
||||
$start_label,
|
||||
$end_label
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
return $options;
|
||||
}
|
||||
|
||||
/**
|
||||
* Render week filter control in event admin list.
|
||||
*
|
||||
@@ -166,31 +302,24 @@ function tony_sportspress_render_admin_week_filter( $post_type ) {
|
||||
if ( ! empty( $_GET['sp_week_filter'] ) ) {
|
||||
$value = sanitize_text_field( wp_unslash( $_GET['sp_week_filter'] ) );
|
||||
}
|
||||
|
||||
$summary_text = __( 'Select a week', 'tonys-sportspress-enhancements' );
|
||||
$parsed = tony_sportspress_parse_admin_week_filter();
|
||||
if ( is_array( $parsed ) ) {
|
||||
$timezone = wp_timezone();
|
||||
$monday = ( new DateTimeImmutable( 'now', $timezone ) )->setISODate( $parsed['year'], $parsed['week'], 1 )->setTime( 0, 0, 0 );
|
||||
$sunday = $monday->modify( '+6 days' )->setTime( 23, 59, 59 );
|
||||
/* translators: 1: Monday label/date, 2: Sunday label/date. */
|
||||
$summary_text = sprintf(
|
||||
__( '%1$s to %2$s', 'tonys-sportspress-enhancements' ),
|
||||
wp_date( 'D M j, Y', $monday->getTimestamp(), $timezone ),
|
||||
wp_date( 'D M j, Y', $sunday->getTimestamp(), $timezone )
|
||||
);
|
||||
}
|
||||
$options = tony_sportspress_get_admin_week_filter_options();
|
||||
?>
|
||||
<label for="sp_week_filter" class="screen-reader-text"><?php esc_html_e( 'Filter by week', 'tonys-sportspress-enhancements' ); ?></label>
|
||||
<input
|
||||
type="week"
|
||||
<select
|
||||
id="sp_week_filter"
|
||||
name="sp_week_filter"
|
||||
class="sp-week-filter-field"
|
||||
value="<?php echo esc_attr( $value ); ?>"
|
||||
title="<?php esc_attr_e( 'Week (Monday start)', 'tonys-sportspress-enhancements' ); ?>"
|
||||
/>
|
||||
<span id="sp-week-filter-summary" class="sp-week-filter-summary"><?php echo esc_html( $summary_text ); ?></span>
|
||||
>
|
||||
<option value=""><?php esc_html_e( 'Year/Week', 'tonys-sportspress-enhancements' ); ?></option>
|
||||
<option value="this-week" <?php selected( $value, 'this-week' ); ?>><?php esc_html_e( 'This week', 'tonys-sportspress-enhancements' ); ?></option>
|
||||
<option value="next-week" <?php selected( $value, 'next-week' ); ?>><?php esc_html_e( 'Next week', 'tonys-sportspress-enhancements' ); ?></option>
|
||||
<?php foreach ( $options as $option ) : ?>
|
||||
<option value="<?php echo esc_attr( $option['value'] ); ?>" <?php selected( $value, $option['value'] ); ?>>
|
||||
<?php echo esc_html( $option['label'] ); ?>
|
||||
</option>
|
||||
<?php endforeach; ?>
|
||||
</select>
|
||||
<?php
|
||||
}
|
||||
add_action( 'restrict_manage_posts', 'tony_sportspress_render_admin_week_filter' );
|
||||
@@ -241,8 +370,7 @@ function tony_sportspress_admin_week_filter_styles() {
|
||||
.post-type-sp_event .tablenav.top .alignleft.actions:not(.bulkactions) input[name="match_day"] { display: none !important; }
|
||||
<?php endif; ?>
|
||||
<?php if ( $hide['week'] ) : ?>
|
||||
.post-type-sp_event .tablenav.top .alignleft.actions:not(.bulkactions) #sp_week_filter,
|
||||
.post-type-sp_event .tablenav.top .alignleft.actions:not(.bulkactions) #sp-week-filter-summary { display: none !important; }
|
||||
.post-type-sp_event .tablenav.top .alignleft.actions:not(.bulkactions) #sp_week_filter { display: none !important; }
|
||||
<?php endif; ?>
|
||||
@media (max-width: 1200px) {
|
||||
.post-type-sp_event .tablenav.top .alignleft.actions:not(.bulkactions) {
|
||||
@@ -258,14 +386,6 @@ function tony_sportspress_admin_week_filter_styles() {
|
||||
.post-type-sp_event .tablenav.top .alignleft.actions:not(.bulkactions) .sp-week-filter-field {
|
||||
min-width: 145px;
|
||||
}
|
||||
.post-type-sp_event .tablenav.top .alignleft.actions:not(.bulkactions) .sp-week-filter-summary {
|
||||
display: block;
|
||||
width: 100%;
|
||||
margin-top: 2px;
|
||||
color: #50575e;
|
||||
font-size: 12px;
|
||||
line-height: 1.4;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
<?php
|
||||
@@ -273,19 +393,41 @@ function tony_sportspress_admin_week_filter_styles() {
|
||||
add_action( 'admin_head-edit.php', 'tony_sportspress_admin_week_filter_styles' );
|
||||
|
||||
/**
|
||||
* Update week summary text when week input changes.
|
||||
* Update admin filter labels and persist screen options.
|
||||
*/
|
||||
function tony_sportspress_admin_week_filter_script() {
|
||||
$screen = get_current_screen();
|
||||
if ( ! $screen || 'edit-sp_event' !== $screen->id ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$venue_filter_text = sprintf(
|
||||
/* translators: %s: plural venue label. */
|
||||
__( 'Show all %s', 'sportspress' ),
|
||||
strtolower( tony_sportspress_get_event_venue_label_plural() )
|
||||
);
|
||||
?>
|
||||
<script>
|
||||
(function() {
|
||||
const filterCheckboxes = Array.from(
|
||||
document.querySelectorAll('#screen-options-wrap input[type="checkbox"][id^="tony_sp_event_filter_"]')
|
||||
);
|
||||
const monthCheckbox = document.getElementById('tony_sp_event_filter_month');
|
||||
const weekCheckbox = document.getElementById('tony_sp_event_filter_week');
|
||||
|
||||
function syncExclusiveFilters(changedCheckbox) {
|
||||
if (!changedCheckbox || !changedCheckbox.checked) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (changedCheckbox === monthCheckbox && weekCheckbox) {
|
||||
weekCheckbox.checked = false;
|
||||
}
|
||||
|
||||
if (changedCheckbox === weekCheckbox && monthCheckbox) {
|
||||
monthCheckbox.checked = false;
|
||||
}
|
||||
}
|
||||
|
||||
function saveFilterPrefs() {
|
||||
if (!filterCheckboxes.length || typeof ajaxurl === 'undefined') {
|
||||
@@ -313,7 +455,10 @@ function tony_sportspress_admin_week_filter_script() {
|
||||
}
|
||||
|
||||
filterCheckboxes.forEach(function(checkbox) {
|
||||
checkbox.addEventListener('change', saveFilterPrefs);
|
||||
checkbox.addEventListener('change', function() {
|
||||
syncExclusiveFilters(checkbox);
|
||||
saveFilterPrefs();
|
||||
});
|
||||
});
|
||||
|
||||
const monthSelect = document.querySelector('select[name="m"]');
|
||||
@@ -324,46 +469,13 @@ function tony_sportspress_admin_week_filter_script() {
|
||||
}
|
||||
}
|
||||
|
||||
const input = document.getElementById('sp_week_filter');
|
||||
const summary = document.getElementById('sp-week-filter-summary');
|
||||
if (!input || !summary) {
|
||||
return;
|
||||
}
|
||||
|
||||
function updateSummary() {
|
||||
const raw = (input.value || '').trim();
|
||||
const match = raw.match(/^(\d{4})-W(\d{2})$/);
|
||||
if (!match) {
|
||||
summary.textContent = 'Select a week';
|
||||
return;
|
||||
const venueSelect = document.querySelector('select[name="sp_venue"]');
|
||||
if (venueSelect) {
|
||||
const allVenues = venueSelect.querySelector('option[value="0"]');
|
||||
if (allVenues) {
|
||||
allVenues.textContent = <?php echo wp_json_encode( $venue_filter_text ); ?>;
|
||||
}
|
||||
|
||||
const year = parseInt(match[1], 10);
|
||||
const week = parseInt(match[2], 10);
|
||||
|
||||
const jan4 = new Date(Date.UTC(year, 0, 4));
|
||||
const jan4Day = jan4.getUTCDay() || 7;
|
||||
const mondayWeek1 = new Date(jan4);
|
||||
mondayWeek1.setUTCDate(jan4.getUTCDate() - jan4Day + 1);
|
||||
|
||||
const monday = new Date(mondayWeek1);
|
||||
monday.setUTCDate(mondayWeek1.getUTCDate() + (week - 1) * 7);
|
||||
const sunday = new Date(monday);
|
||||
sunday.setUTCDate(monday.getUTCDate() + 6);
|
||||
|
||||
const fmt = new Intl.DateTimeFormat(undefined, {
|
||||
weekday: 'short',
|
||||
month: 'short',
|
||||
day: 'numeric',
|
||||
year: 'numeric',
|
||||
timeZone: 'UTC'
|
||||
});
|
||||
|
||||
summary.textContent = fmt.format(monday) + ' to ' + fmt.format(sunday);
|
||||
}
|
||||
|
||||
input.addEventListener('change', updateSummary);
|
||||
updateSummary();
|
||||
})();
|
||||
</script>
|
||||
<?php
|
||||
|
||||
@@ -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.
|
||||
|
||||
1042
includes/sp-event-export.php
Normal file
1042
includes/sp-event-export.php
Normal file
File diff suppressed because it is too large
Load Diff
@@ -7,6 +7,89 @@ if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Add an Officials column to the event admin list.
|
||||
*
|
||||
* @param array $columns Existing columns.
|
||||
* @return array
|
||||
*/
|
||||
function tony_sportspress_event_add_officials_column( $columns ) {
|
||||
$updated = array();
|
||||
|
||||
foreach ( $columns as $key => $label ) {
|
||||
$updated[ $key ] = $label;
|
||||
|
||||
if ( 'sp_team' === $key ) {
|
||||
$updated['tony_sp_officials'] = esc_html__( 'Officials', 'tonys-sportspress-enhancements' );
|
||||
}
|
||||
}
|
||||
|
||||
if ( ! isset( $updated['tony_sp_officials'] ) ) {
|
||||
$updated['tony_sp_officials'] = esc_html__( 'Officials', 'tonys-sportspress-enhancements' );
|
||||
}
|
||||
|
||||
return $updated;
|
||||
}
|
||||
add_filter( 'manage_edit-sp_event_columns', 'tony_sportspress_event_add_officials_column', 20 );
|
||||
|
||||
/**
|
||||
* Build a display-ready officials map for an event.
|
||||
*
|
||||
* @param int $post_id Post ID.
|
||||
* @return array<int, array{name: string, officials: string[]}>
|
||||
*/
|
||||
function tony_sportspress_event_get_officials_display( $post_id ) {
|
||||
$officials_by_duty = get_post_meta( $post_id, 'sp_officials', true );
|
||||
if ( ! is_array( $officials_by_duty ) || empty( $officials_by_duty ) ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
$duties = get_terms(
|
||||
array(
|
||||
'taxonomy' => 'sp_duty',
|
||||
'hide_empty' => false,
|
||||
)
|
||||
);
|
||||
|
||||
$duty_names = array();
|
||||
if ( is_array( $duties ) ) {
|
||||
foreach ( $duties as $duty ) {
|
||||
if ( isset( $duty->term_id, $duty->name ) ) {
|
||||
$duty_names[ (int) $duty->term_id ] = $duty->name;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$rows = array();
|
||||
foreach ( $officials_by_duty as $duty_id => $official_ids ) {
|
||||
$duty_id = absint( $duty_id );
|
||||
$official_ids = array_filter( array_map( 'absint', (array) $official_ids ) );
|
||||
|
||||
if ( $duty_id <= 0 || empty( $official_ids ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$names = array();
|
||||
foreach ( $official_ids as $official_id ) {
|
||||
$title = get_the_title( $official_id );
|
||||
if ( is_string( $title ) && '' !== $title ) {
|
||||
$names[] = $title;
|
||||
}
|
||||
}
|
||||
|
||||
if ( empty( $names ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$rows[] = array(
|
||||
'name' => isset( $duty_names[ $duty_id ] ) ? $duty_names[ $duty_id ] : (string) $duty_id,
|
||||
'officials' => $names,
|
||||
);
|
||||
}
|
||||
|
||||
return $rows;
|
||||
}
|
||||
|
||||
/**
|
||||
* Print hidden officials data on each event row for quick edit prefill.
|
||||
*
|
||||
@@ -14,7 +97,7 @@ if ( ! defined( 'ABSPATH' ) ) {
|
||||
* @param int $post_id Post ID.
|
||||
*/
|
||||
function tony_sportspress_event_quick_edit_officials_row_data( $column, $post_id ) {
|
||||
if ( 'sp_team' !== $column ) {
|
||||
if ( 'tony_sp_officials' !== $column ) {
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -28,6 +111,18 @@ function tony_sportspress_event_quick_edit_officials_row_data( $column, $post_id
|
||||
$serialized = '{}';
|
||||
}
|
||||
|
||||
$rows = tony_sportspress_event_get_officials_display( $post_id );
|
||||
if ( empty( $rows ) ) {
|
||||
echo '—';
|
||||
} else {
|
||||
foreach ( $rows as $row ) {
|
||||
echo '<div class="tony-sp-event-official-row">';
|
||||
echo '<strong>' . esc_html( $row['name'] ) . ':</strong> ';
|
||||
echo esc_html( implode( ', ', $row['officials'] ) );
|
||||
echo '</div>';
|
||||
}
|
||||
}
|
||||
|
||||
echo '<span class="hidden tony-event-officials-data" data-officials="' . esc_attr( $serialized ) . '"></span>';
|
||||
}
|
||||
add_action( 'manage_sp_event_posts_custom_column', 'tony_sportspress_event_quick_edit_officials_row_data', 20, 2 );
|
||||
@@ -39,7 +134,7 @@ add_action( 'manage_sp_event_posts_custom_column', 'tony_sportspress_event_quick
|
||||
* @param string $post_type Post type key.
|
||||
*/
|
||||
function tony_sportspress_event_quick_edit_officials_field( $column_name, $post_type ) {
|
||||
if ( 'sp_event' !== $post_type || 'sp_team' !== $column_name ) {
|
||||
if ( 'sp_event' !== $post_type || 'tony_sp_officials' !== $column_name ) {
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
312
includes/sp-github-updater.php
Normal file
312
includes/sp-github-updater.php
Normal file
@@ -0,0 +1,312 @@
|
||||
<?php
|
||||
/**
|
||||
* GitHub release updater for the plugin.
|
||||
*
|
||||
* @package Tonys_Sportspress_Enhancements
|
||||
*/
|
||||
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_GITHUB_REPO' ) ) {
|
||||
define( 'TONY_SPORTSPRESS_ENHANCEMENTS_GITHUB_REPO', 'anthonyscorrea/tonys-sportspress-enhancements' );
|
||||
}
|
||||
|
||||
if ( ! class_exists( 'Tony_Sportspress_GitHub_Updater' ) ) {
|
||||
/**
|
||||
* Integrates WordPress plugin updates with GitHub Releases.
|
||||
*/
|
||||
class Tony_Sportspress_GitHub_Updater {
|
||||
/**
|
||||
* GitHub API URL for the latest release.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
private $release_api_url;
|
||||
|
||||
/**
|
||||
* Plugin basename.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
private $plugin_basename;
|
||||
|
||||
/**
|
||||
* Plugin slug.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
private $plugin_slug;
|
||||
|
||||
/**
|
||||
* Cache key for release metadata.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
private $cache_key = 'tony_sportspress_github_release';
|
||||
|
||||
/**
|
||||
* Constructor.
|
||||
*/
|
||||
public function __construct() {
|
||||
$this->release_api_url = sprintf(
|
||||
'https://api.github.com/repos/%s/releases/latest',
|
||||
TONY_SPORTSPRESS_ENHANCEMENTS_GITHUB_REPO
|
||||
);
|
||||
$this->plugin_basename = TONY_SPORTSPRESS_ENHANCEMENTS_PLUGIN_BASENAME;
|
||||
$this->plugin_slug = dirname( $this->plugin_basename );
|
||||
|
||||
add_filter( 'pre_set_site_transient_update_plugins', array( $this, 'inject_update' ) );
|
||||
add_filter( 'plugins_api', array( $this, 'plugin_information' ), 20, 3 );
|
||||
add_filter( 'upgrader_source_selection', array( $this, 'normalize_source_directory' ), 10, 4 );
|
||||
add_action( 'upgrader_process_complete', array( $this, 'purge_release_cache' ), 10, 2 );
|
||||
}
|
||||
|
||||
/**
|
||||
* Adds plugin update data to WordPress' update transient.
|
||||
*
|
||||
* @param stdClass $transient Existing update transient.
|
||||
* @return stdClass
|
||||
*/
|
||||
public function inject_update( $transient ) {
|
||||
if ( ! is_object( $transient ) || empty( $transient->checked ) ) {
|
||||
return $transient;
|
||||
}
|
||||
|
||||
$release = $this->get_latest_release();
|
||||
|
||||
if ( ! $release ) {
|
||||
return $transient;
|
||||
}
|
||||
|
||||
$remote_version = $this->normalize_version( $release['version'] );
|
||||
$current_version = $this->normalize_version( TONY_SPORTSPRESS_ENHANCEMENTS_VERSION );
|
||||
|
||||
$plugin_data = (object) array(
|
||||
'id' => $release['url'],
|
||||
'slug' => $this->plugin_slug,
|
||||
'plugin' => $this->plugin_basename,
|
||||
'new_version' => $remote_version,
|
||||
'url' => $release['url'],
|
||||
'package' => $release['package'],
|
||||
'tested' => '',
|
||||
'requires_php' => '',
|
||||
'icons' => array(),
|
||||
'banners' => array(),
|
||||
'banners_rtl' => array(),
|
||||
'translations' => array(),
|
||||
);
|
||||
|
||||
if ( version_compare( $remote_version, $current_version, '<=' ) ) {
|
||||
unset( $transient->response[ $this->plugin_basename ] );
|
||||
$transient->no_update[ $this->plugin_basename ] = $plugin_data;
|
||||
|
||||
return $transient;
|
||||
}
|
||||
|
||||
unset( $transient->no_update[ $this->plugin_basename ] );
|
||||
$transient->response[ $this->plugin_basename ] = $plugin_data;
|
||||
|
||||
return $transient;
|
||||
}
|
||||
|
||||
/**
|
||||
* Provides plugin information for the update details modal.
|
||||
*
|
||||
* @param false|object|array $result Existing result.
|
||||
* @param string $action API action.
|
||||
* @param object $args API args.
|
||||
* @return false|object|array
|
||||
*/
|
||||
public function plugin_information( $result, $action, $args ) {
|
||||
if ( 'plugin_information' !== $action || empty( $args->slug ) || $this->plugin_slug !== $args->slug ) {
|
||||
return $result;
|
||||
}
|
||||
|
||||
$release = $this->get_latest_release();
|
||||
|
||||
if ( ! $release ) {
|
||||
return $result;
|
||||
}
|
||||
|
||||
return (object) array(
|
||||
'name' => 'Tonys SportsPress Enhancements',
|
||||
'slug' => $this->plugin_slug,
|
||||
'version' => $this->normalize_version( $release['version'] ),
|
||||
'author' => '<a href="https://github.com/anthonyscorrea/">Tony Correa</a>',
|
||||
'author_profile'=> 'https://github.com/anthonyscorrea/',
|
||||
'homepage' => $release['url'],
|
||||
'download_link' => $release['package'],
|
||||
'sections' => array(
|
||||
'description' => wp_kses_post( wpautop( 'Suite of SportsPress Enhancements.' ) ),
|
||||
'changelog' => wp_kses_post( wpautop( $release['body'] ) ),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Ensures GitHub's extracted directory name matches the installed plugin slug.
|
||||
*
|
||||
* @param string $source Source file location.
|
||||
* @param string $remote_source Remote file source location.
|
||||
* @param WP_Upgrader $upgrader Upgrader instance.
|
||||
* @param array $hook_extra Extra hook arguments.
|
||||
* @return string|WP_Error
|
||||
*/
|
||||
public function normalize_source_directory( $source, $remote_source, $upgrader, $hook_extra ) {
|
||||
global $wp_filesystem;
|
||||
|
||||
if ( empty( $hook_extra['plugin'] ) || $this->plugin_basename !== $hook_extra['plugin'] ) {
|
||||
return $source;
|
||||
}
|
||||
|
||||
$expected_dir = trailingslashit( $remote_source ) . $this->plugin_slug;
|
||||
|
||||
if ( untrailingslashit( $source ) === untrailingslashit( $expected_dir ) ) {
|
||||
return $source;
|
||||
}
|
||||
|
||||
if ( ! $wp_filesystem ) {
|
||||
return $source;
|
||||
}
|
||||
|
||||
if ( $wp_filesystem->exists( $expected_dir ) ) {
|
||||
$wp_filesystem->delete( $expected_dir, true );
|
||||
}
|
||||
|
||||
if ( ! $wp_filesystem->move( $source, $expected_dir ) ) {
|
||||
return new WP_Error(
|
||||
'tony_sportspress_updater_rename_failed',
|
||||
__( 'The plugin update package could not be prepared for installation.', 'tonys-sportspress-enhancements' )
|
||||
);
|
||||
}
|
||||
|
||||
return $expected_dir;
|
||||
}
|
||||
|
||||
/**
|
||||
* Clears cached release metadata after plugin updates complete.
|
||||
*
|
||||
* @param WP_Upgrader $upgrader Upgrader instance.
|
||||
* @param array $hook_extra Extra hook arguments.
|
||||
* @return void
|
||||
*/
|
||||
public function purge_release_cache( $upgrader, $hook_extra ) {
|
||||
if ( empty( $hook_extra['type'] ) || 'plugin' !== $hook_extra['type'] ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( empty( $hook_extra['plugins'] ) || ! in_array( $this->plugin_basename, (array) $hook_extra['plugins'], true ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
delete_site_transient( $this->cache_key );
|
||||
}
|
||||
|
||||
/**
|
||||
* Reads and caches the latest GitHub release metadata.
|
||||
*
|
||||
* @return array|null
|
||||
*/
|
||||
private function get_latest_release() {
|
||||
$cached = get_site_transient( $this->cache_key );
|
||||
|
||||
if ( is_array( $cached ) ) {
|
||||
return $cached;
|
||||
}
|
||||
|
||||
$response = wp_remote_get(
|
||||
$this->release_api_url,
|
||||
array(
|
||||
'timeout' => 15,
|
||||
'headers' => array(
|
||||
'Accept' => 'application/vnd.github+json',
|
||||
'User-Agent' => 'WordPress/' . get_bloginfo( 'version' ) . '; ' . home_url( '/' ),
|
||||
),
|
||||
)
|
||||
);
|
||||
|
||||
if ( is_wp_error( $response ) ) {
|
||||
return null;
|
||||
}
|
||||
|
||||
if ( 200 !== (int) wp_remote_retrieve_response_code( $response ) ) {
|
||||
return null;
|
||||
}
|
||||
|
||||
$data = json_decode( wp_remote_retrieve_body( $response ), true );
|
||||
|
||||
if ( ! is_array( $data ) || empty( $data['tag_name'] ) || empty( $data['html_url'] ) ) {
|
||||
return null;
|
||||
}
|
||||
|
||||
$release = array(
|
||||
'version' => $data['tag_name'],
|
||||
'url' => $data['html_url'],
|
||||
'body' => isset( $data['body'] ) ? (string) $data['body'] : '',
|
||||
'package' => $this->determine_package_url( $data ),
|
||||
);
|
||||
|
||||
if ( empty( $release['package'] ) ) {
|
||||
return null;
|
||||
}
|
||||
|
||||
set_site_transient( $this->cache_key, $release, 6 * HOUR_IN_SECONDS );
|
||||
|
||||
return $release;
|
||||
}
|
||||
|
||||
/**
|
||||
* Selects the best package URL from a release payload.
|
||||
*
|
||||
* @param array $release GitHub release payload.
|
||||
* @return string
|
||||
*/
|
||||
private function determine_package_url( $release ) {
|
||||
if ( ! empty( $release['assets'] ) && is_array( $release['assets'] ) ) {
|
||||
$fallback_asset = '';
|
||||
|
||||
foreach ( $release['assets'] as $asset ) {
|
||||
if ( empty( $asset['browser_download_url'] ) || empty( $asset['name'] ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if ( '.zip' !== strtolower( substr( $asset['name'], -4 ) ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if ( false !== strpos( $asset['name'], $this->plugin_slug ) ) {
|
||||
return $asset['browser_download_url'];
|
||||
}
|
||||
|
||||
if ( empty( $fallback_asset ) ) {
|
||||
$fallback_asset = $asset['browser_download_url'];
|
||||
}
|
||||
}
|
||||
|
||||
if ( ! empty( $fallback_asset ) ) {
|
||||
return $fallback_asset;
|
||||
}
|
||||
}
|
||||
|
||||
if ( ! empty( $release['zipball_url'] ) ) {
|
||||
return $release['zipball_url'];
|
||||
}
|
||||
|
||||
return '';
|
||||
}
|
||||
|
||||
/**
|
||||
* Normalizes release versions so Git tags like v1.2.3 compare correctly.
|
||||
*
|
||||
* @param string $version Version string.
|
||||
* @return string
|
||||
*/
|
||||
private function normalize_version( $version ) {
|
||||
return ltrim( (string) $version, "vV \t\n\r\0\x0B" );
|
||||
}
|
||||
}
|
||||
|
||||
new Tony_Sportspress_GitHub_Updater();
|
||||
}
|
||||
420
includes/sp-officials-manager-role.php
Normal file
420
includes/sp-officials-manager-role.php
Normal file
@@ -0,0 +1,420 @@
|
||||
<?php
|
||||
/**
|
||||
* Officials Manager role and capability restrictions.
|
||||
*/
|
||||
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
if ( ! defined( 'TONY_SPORTSPRESS_OFFICIALS_MANAGER_ROLE' ) ) {
|
||||
define( 'TONY_SPORTSPRESS_OFFICIALS_MANAGER_ROLE', 'sp_officials_manager' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Build the primitive capabilities for a custom post type.
|
||||
*
|
||||
* @param string $singular Singular capability base.
|
||||
* @param string $plural Plural capability base.
|
||||
* @return string[]
|
||||
*/
|
||||
function tony_sportspress_build_post_type_caps( $singular, $plural ) {
|
||||
return array(
|
||||
"edit_{$singular}",
|
||||
"read_{$singular}",
|
||||
"delete_{$singular}",
|
||||
"edit_{$plural}",
|
||||
"edit_others_{$plural}",
|
||||
"publish_{$plural}",
|
||||
"read_private_{$plural}",
|
||||
"delete_{$plural}",
|
||||
"delete_private_{$plural}",
|
||||
"delete_published_{$plural}",
|
||||
"delete_others_{$plural}",
|
||||
"edit_private_{$plural}",
|
||||
"edit_published_{$plural}",
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get officials manager role capabilities.
|
||||
*
|
||||
* @return array<string, bool>
|
||||
*/
|
||||
function tony_sportspress_get_officials_manager_caps() {
|
||||
$caps = array(
|
||||
'read' => true,
|
||||
'edit_posts' => true,
|
||||
'delete_posts' => true,
|
||||
'edit_published_posts' => true,
|
||||
);
|
||||
|
||||
foreach ( tony_sportspress_build_post_type_caps( 'sp_official', 'sp_officials' ) as $cap ) {
|
||||
$caps[ $cap ] = true;
|
||||
}
|
||||
|
||||
// Allow access to the event list and quick edit for assignments, without full event management.
|
||||
$caps['read_sp_event'] = true;
|
||||
$caps['edit_sp_event'] = true;
|
||||
$caps['edit_sp_events'] = true;
|
||||
$caps['edit_others_sp_events'] = true;
|
||||
$caps['edit_published_sp_events'] = true;
|
||||
$caps['read_private_sp_events'] = true;
|
||||
|
||||
return $caps;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the capabilities managed for the officials manager role.
|
||||
*
|
||||
* @return string[]
|
||||
*/
|
||||
function tony_sportspress_get_officials_manager_managed_caps() {
|
||||
return array_keys( tony_sportspress_get_officials_manager_caps() );
|
||||
}
|
||||
|
||||
/**
|
||||
* Grant custom official caps to existing roles that already have matching event caps.
|
||||
*
|
||||
* This preserves existing access after moving officials off the `sp_event` capability type.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
function tony_sportspress_sync_official_caps_to_existing_roles() {
|
||||
global $wp_roles;
|
||||
|
||||
if ( ! class_exists( 'WP_Roles' ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( ! isset( $wp_roles ) ) {
|
||||
$wp_roles = wp_roles();
|
||||
}
|
||||
|
||||
if ( ! $wp_roles instanceof WP_Roles ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$cap_map = array(
|
||||
'edit_sp_event' => 'edit_sp_official',
|
||||
'read_sp_event' => 'read_sp_official',
|
||||
'delete_sp_event' => 'delete_sp_official',
|
||||
'edit_sp_events' => 'edit_sp_officials',
|
||||
'edit_others_sp_events' => 'edit_others_sp_officials',
|
||||
'publish_sp_events' => 'publish_sp_officials',
|
||||
'read_private_sp_events' => 'read_private_sp_officials',
|
||||
'delete_sp_events' => 'delete_sp_officials',
|
||||
'delete_private_sp_events' => 'delete_private_sp_officials',
|
||||
'delete_published_sp_events' => 'delete_published_sp_officials',
|
||||
'delete_others_sp_events' => 'delete_others_sp_officials',
|
||||
'edit_private_sp_events' => 'edit_private_sp_officials',
|
||||
'edit_published_sp_events' => 'edit_published_sp_officials',
|
||||
);
|
||||
|
||||
foreach ( $wp_roles->role_objects as $role ) {
|
||||
if ( ! $role instanceof WP_Role ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
foreach ( $cap_map as $event_cap => $official_cap ) {
|
||||
if ( $role->has_cap( $event_cap ) ) {
|
||||
$role->add_cap( $official_cap );
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Create or update the officials manager role.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
function tony_sportspress_sync_officials_manager_roles() {
|
||||
$role = get_role( TONY_SPORTSPRESS_OFFICIALS_MANAGER_ROLE );
|
||||
|
||||
if ( ! $role ) {
|
||||
$role = add_role(
|
||||
TONY_SPORTSPRESS_OFFICIALS_MANAGER_ROLE,
|
||||
__( 'Officials Manager', 'tonys-sportspress-enhancements' ),
|
||||
array()
|
||||
);
|
||||
}
|
||||
|
||||
if ( ! $role instanceof WP_Role ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$desired_caps = tony_sportspress_get_officials_manager_caps();
|
||||
|
||||
foreach ( tony_sportspress_get_officials_manager_managed_caps() as $cap ) {
|
||||
if ( ! empty( $desired_caps[ $cap ] ) ) {
|
||||
$role->add_cap( $cap );
|
||||
} else {
|
||||
$role->remove_cap( $cap );
|
||||
}
|
||||
}
|
||||
|
||||
tony_sportspress_sync_official_caps_to_existing_roles();
|
||||
}
|
||||
add_action( 'init', 'tony_sportspress_sync_officials_manager_roles' );
|
||||
|
||||
/**
|
||||
* Assign custom capabilities to the officials post type.
|
||||
*
|
||||
* @param array $args Post type registration args.
|
||||
* @return array
|
||||
*/
|
||||
function tony_sportspress_officials_post_type_caps( $args ) {
|
||||
$args['capability_type'] = array( 'sp_official', 'sp_officials' );
|
||||
$args['map_meta_cap'] = true;
|
||||
$args['capabilities'] = array(
|
||||
'create_posts' => 'edit_sp_officials',
|
||||
);
|
||||
|
||||
return $args;
|
||||
}
|
||||
add_filter( 'sportspress_register_post_type_official', 'tony_sportspress_officials_post_type_caps' );
|
||||
|
||||
/**
|
||||
* Determine whether the current user should be restricted to assignment-only event access.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
function tony_sportspress_is_officials_manager_user() {
|
||||
$user = wp_get_current_user();
|
||||
|
||||
if ( ! $user instanceof WP_User || empty( $user->roles ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( ! in_array( TONY_SPORTSPRESS_OFFICIALS_MANAGER_ROLE, $user->roles, true ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( current_user_can( 'manage_options' ) || current_user_can( 'manage_sportspress' ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Prevent assignment-only users from opening full event edit screens.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
function tony_sportspress_lock_event_editor_for_officials_manager() {
|
||||
global $pagenow;
|
||||
|
||||
if ( ! is_admin() || ! tony_sportspress_is_officials_manager_user() ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( 'post-new.php' === $pagenow ) {
|
||||
$post_type = isset( $_GET['post_type'] ) ? sanitize_key( wp_unslash( $_GET['post_type'] ) ) : 'post';
|
||||
|
||||
if ( 'sp_event' === $post_type ) {
|
||||
wp_safe_redirect( admin_url( 'edit.php?post_type=sp_event&tse_event_editor_locked=1' ) );
|
||||
exit;
|
||||
}
|
||||
}
|
||||
|
||||
if ( 'post.php' !== $pagenow ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$post_id = isset( $_GET['post'] ) ? absint( wp_unslash( $_GET['post'] ) ) : 0;
|
||||
if ( $post_id > 0 && 'sp_event' === get_post_type( $post_id ) ) {
|
||||
wp_safe_redirect( admin_url( 'edit.php?post_type=sp_event&tse_event_editor_locked=1' ) );
|
||||
exit;
|
||||
}
|
||||
}
|
||||
add_action( 'admin_init', 'tony_sportspress_lock_event_editor_for_officials_manager' );
|
||||
|
||||
/**
|
||||
* Show an admin notice when event editor access is blocked.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
function tony_sportspress_officials_manager_admin_notice() {
|
||||
if ( ! tony_sportspress_is_officials_manager_user() ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( empty( $_GET['tse_event_editor_locked'] ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
echo '<div class="notice notice-info is-dismissible"><p>' . esc_html__( 'Officials Managers can assign officials from the events list via Quick Edit, but cannot open the full event editor.', 'tonys-sportspress-enhancements' ) . '</p></div>';
|
||||
}
|
||||
add_action( 'admin_notices', 'tony_sportspress_officials_manager_admin_notice' );
|
||||
|
||||
/**
|
||||
* Remove event row actions that would expose broader editing.
|
||||
*
|
||||
* @param array $actions Row actions.
|
||||
* @param WP_Post $post Post object.
|
||||
* @return array
|
||||
*/
|
||||
function tony_sportspress_limit_event_row_actions_for_officials_manager( $actions, $post ) {
|
||||
if ( ! tony_sportspress_is_officials_manager_user() || ! $post instanceof WP_Post || 'sp_event' !== $post->post_type ) {
|
||||
return $actions;
|
||||
}
|
||||
|
||||
$allowed = array();
|
||||
|
||||
if ( isset( $actions['inline hide-if-no-js'] ) ) {
|
||||
$allowed['inline hide-if-no-js'] = $actions['inline hide-if-no-js'];
|
||||
}
|
||||
|
||||
if ( isset( $actions['view'] ) ) {
|
||||
$allowed['view'] = $actions['view'];
|
||||
}
|
||||
|
||||
return $allowed;
|
||||
}
|
||||
add_filter( 'post_row_actions', 'tony_sportspress_limit_event_row_actions_for_officials_manager', 10, 2 );
|
||||
|
||||
/**
|
||||
* Remove bulk actions from the events list for assignment-only users.
|
||||
*
|
||||
* @param array $actions Bulk actions.
|
||||
* @return array
|
||||
*/
|
||||
function tony_sportspress_limit_event_bulk_actions_for_officials_manager( $actions ) {
|
||||
if ( ! tony_sportspress_is_officials_manager_user() ) {
|
||||
return $actions;
|
||||
}
|
||||
|
||||
return array();
|
||||
}
|
||||
add_filter( 'bulk_actions-edit-sp_event', 'tony_sportspress_limit_event_bulk_actions_for_officials_manager' );
|
||||
|
||||
/**
|
||||
* Remove the Add New events submenu for assignment-only users.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
function tony_sportspress_limit_event_admin_menu_for_officials_manager() {
|
||||
if ( ! tony_sportspress_is_officials_manager_user() ) {
|
||||
return;
|
||||
}
|
||||
|
||||
remove_submenu_page( 'edit.php?post_type=sp_event', 'post-new.php?post_type=sp_event' );
|
||||
}
|
||||
add_action( 'admin_menu', 'tony_sportspress_limit_event_admin_menu_for_officials_manager', 99 );
|
||||
|
||||
/**
|
||||
* Hide event Add New buttons on the list screen for assignment-only users.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
function tony_sportspress_limit_event_admin_ui_for_officials_manager() {
|
||||
$screen = function_exists( 'get_current_screen' ) ? get_current_screen() : null;
|
||||
if ( ! $screen || 'edit-sp_event' !== $screen->id || ! tony_sportspress_is_officials_manager_user() ) {
|
||||
return;
|
||||
}
|
||||
?>
|
||||
<style>
|
||||
.post-type-sp_event .page-title-action,
|
||||
.post-type-sp_event .wrap .bulkactions {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.post-type-sp_event .wp-list-table .column-title .row-title {
|
||||
color: inherit;
|
||||
cursor: default;
|
||||
text-decoration: none;
|
||||
}
|
||||
</style>
|
||||
<script>
|
||||
(function() {
|
||||
const replaceTitleLinks = function() {
|
||||
document.querySelectorAll('.post-type-sp_event .wp-list-table .column-title a.row-title').forEach(function(link) {
|
||||
const text = document.createTextNode(link.textContent || '');
|
||||
const span = document.createElement('span');
|
||||
|
||||
span.className = link.className;
|
||||
span.appendChild(text);
|
||||
link.replaceWith(span);
|
||||
});
|
||||
};
|
||||
|
||||
if (document.readyState === 'loading') {
|
||||
document.addEventListener('DOMContentLoaded', replaceTitleLinks);
|
||||
} else {
|
||||
replaceTitleLinks();
|
||||
}
|
||||
})();
|
||||
</script>
|
||||
<?php
|
||||
}
|
||||
add_action( 'admin_head', 'tony_sportspress_limit_event_admin_ui_for_officials_manager' );
|
||||
|
||||
/**
|
||||
* Prevent clickable edit links to events for assignment-only users.
|
||||
*
|
||||
* This keeps the list-table title as plain text while preserving Quick Edit.
|
||||
*
|
||||
* @param string|false $link Edit link.
|
||||
* @param int $post_id Post ID.
|
||||
* @param string $context Link context.
|
||||
* @return string|false
|
||||
*/
|
||||
function tony_sportspress_disable_event_edit_links_for_officials_manager( $link, $post_id, $context ) {
|
||||
if ( ! is_admin() || ! tony_sportspress_is_officials_manager_user() ) {
|
||||
return $link;
|
||||
}
|
||||
|
||||
if ( 'sp_event' !== get_post_type( $post_id ) ) {
|
||||
return $link;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
add_filter( 'get_edit_post_link', 'tony_sportspress_disable_event_edit_links_for_officials_manager', 10, 3 );
|
||||
|
||||
/**
|
||||
* Preserve core event fields so assignment-only users cannot alter them via Quick Edit.
|
||||
*
|
||||
* @param array $data Sanitized post data.
|
||||
* @param array $postarr Raw post array.
|
||||
* @return array
|
||||
*/
|
||||
function tony_sportspress_protect_event_fields_for_officials_manager( $data, $postarr ) {
|
||||
if ( ! is_admin() || ! tony_sportspress_is_officials_manager_user() ) {
|
||||
return $data;
|
||||
}
|
||||
|
||||
if ( empty( $postarr['ID'] ) || 'sp_event' !== $data['post_type'] ) {
|
||||
return $data;
|
||||
}
|
||||
|
||||
$existing_post = get_post( (int) $postarr['ID'], ARRAY_A );
|
||||
if ( ! is_array( $existing_post ) ) {
|
||||
return $data;
|
||||
}
|
||||
|
||||
$protected_fields = array(
|
||||
'post_author',
|
||||
'post_content',
|
||||
'post_content_filtered',
|
||||
'post_date',
|
||||
'post_date_gmt',
|
||||
'post_excerpt',
|
||||
'post_name',
|
||||
'post_parent',
|
||||
'post_password',
|
||||
'post_status',
|
||||
'post_title',
|
||||
);
|
||||
|
||||
foreach ( $protected_fields as $field ) {
|
||||
if ( isset( $existing_post[ $field ] ) ) {
|
||||
$data[ $field ] = $existing_post[ $field ];
|
||||
}
|
||||
}
|
||||
|
||||
return $data;
|
||||
}
|
||||
add_filter( 'wp_insert_post_data', 'tony_sportspress_protect_event_fields_for_officials_manager', 20, 2 );
|
||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
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
|
||||
}
|
||||
@@ -2,7 +2,7 @@
|
||||
/**
|
||||
* Venue term metadata support.
|
||||
*
|
||||
* Adds short name and abbreviation fields to SportsPress venues.
|
||||
* Adds short name, abbreviation, and ground rules fields to SportsPress venues.
|
||||
*/
|
||||
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
@@ -40,9 +40,319 @@ function tony_sportspress_register_venue_term_meta() {
|
||||
},
|
||||
)
|
||||
);
|
||||
|
||||
register_term_meta(
|
||||
'sp_venue',
|
||||
'tse_ground_rules',
|
||||
array(
|
||||
'type' => 'string',
|
||||
'single' => true,
|
||||
'sanitize_callback' => 'wp_kses_post',
|
||||
'show_in_rest' => true,
|
||||
'auth_callback' => static function() {
|
||||
return current_user_can( 'manage_categories' );
|
||||
},
|
||||
)
|
||||
);
|
||||
}
|
||||
add_action( 'init', 'tony_sportspress_register_venue_term_meta' );
|
||||
|
||||
/**
|
||||
* Determine whether the field page should show its event list.
|
||||
*
|
||||
* The setting is stored in the core SportsPress Events settings page under
|
||||
* the venue/field section.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
function tony_sportspress_field_event_list_enabled() {
|
||||
$enabled = get_option( 'sportspress_event_show_venue_list', 'yes' ) === 'yes';
|
||||
|
||||
return (bool) apply_filters( 'tony_sportspress_field_event_list_enabled', $enabled );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the venue map caption text.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
function tony_sportspress_get_venue_map_caption() {
|
||||
return (string) apply_filters( 'tony_sportspress_venue_map_caption', __( 'Field Map', 'tonys-sportspress-enhancements' ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Add the field page event list setting to SportsPress > Settings > Games > Fields.
|
||||
*
|
||||
* @param array $options Existing venue settings.
|
||||
* @return array
|
||||
*/
|
||||
function tony_sportspress_add_venue_settings( $options ) {
|
||||
$options[] = array(
|
||||
'title' => esc_attr__( 'Event List', 'tonys-sportspress-enhancements' ),
|
||||
'desc' => esc_attr__( 'Display event list on field pages', 'tonys-sportspress-enhancements' ),
|
||||
'id' => 'sportspress_event_show_venue_list',
|
||||
'default' => 'yes',
|
||||
'type' => 'checkbox',
|
||||
);
|
||||
|
||||
return $options;
|
||||
}
|
||||
add_filter( 'sportspress_venue_options', 'tony_sportspress_add_venue_settings' );
|
||||
|
||||
/**
|
||||
* Enqueue the visual editor on the venue taxonomy screen.
|
||||
*
|
||||
* This turns the built-in description textarea into the standard WordPress
|
||||
* TinyMCE editor so venue descriptions can use markup and links.
|
||||
*
|
||||
* @param string $hook_suffix Current admin page hook.
|
||||
*/
|
||||
function tony_sportspress_enqueue_venue_description_editor( $hook_suffix ) {
|
||||
if ( ! in_array( $hook_suffix, array( 'edit-tags.php', 'term.php' ), true ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$taxonomy = isset( $_GET['taxonomy'] ) ? sanitize_key( wp_unslash( $_GET['taxonomy'] ) ) : '';
|
||||
if ( 'sp_venue' !== $taxonomy ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( ! function_exists( 'wp_enqueue_editor' ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
wp_enqueue_editor();
|
||||
|
||||
$script = <<<'JS'
|
||||
(function() {
|
||||
function initEditor(id) {
|
||||
var textarea = document.getElementById(id);
|
||||
if (!textarea || textarea.dataset.tseEditorInitialized) {
|
||||
return;
|
||||
}
|
||||
|
||||
textarea.dataset.tseEditorInitialized = '1';
|
||||
window.wp.editor.initialize(id, {
|
||||
tinymce: {
|
||||
wpautop: true,
|
||||
menubar: false,
|
||||
statusbar: true,
|
||||
toolbar1: 'formatselect,bold,italic,bullist,numlist,blockquote,alignleft,aligncenter,alignright,link,unlink,wp_more,spellchecker,fullscreen',
|
||||
toolbar2: 'strikethrough,hr,forecolor,pastetext,removeformat,charmap,outdent,indent,undo,redo,wp_help',
|
||||
block_formats: 'Paragraph=p; Heading 2=h2; Heading 3=h3; Heading 4=h4; Preformatted=pre',
|
||||
},
|
||||
quicktags: true,
|
||||
mediaButtons: true
|
||||
});
|
||||
}
|
||||
|
||||
document.addEventListener('DOMContentLoaded', function() {
|
||||
if (!window.wp || !wp.editor) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Different taxonomy screens use different IDs for the same description field.
|
||||
initEditor('description');
|
||||
initEditor('tag-description');
|
||||
});
|
||||
})();
|
||||
JS;
|
||||
|
||||
wp_add_inline_script( 'editor', $script );
|
||||
}
|
||||
add_action( 'admin_enqueue_scripts', 'tony_sportspress_enqueue_venue_description_editor' );
|
||||
|
||||
/**
|
||||
* Hide the built-in venue archive description output.
|
||||
*
|
||||
* Venue content is rendered as SportsPress-style sections instead of inside
|
||||
* the archive header.
|
||||
*
|
||||
* @param string $description Archive description HTML.
|
||||
* @return string
|
||||
*/
|
||||
function tony_sportspress_hide_venue_archive_description( $description ) {
|
||||
if ( is_tax( 'sp_venue' ) ) {
|
||||
return '';
|
||||
}
|
||||
|
||||
return $description;
|
||||
}
|
||||
add_filter( 'get_the_archive_description', 'tony_sportspress_hide_venue_archive_description', 99 );
|
||||
|
||||
/**
|
||||
* Determine whether the current venue has map coordinates.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
function tony_sportspress_current_venue_has_map() {
|
||||
$term = get_queried_object();
|
||||
if ( ! $term instanceof WP_Term ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$latitude = trim( (string) get_term_meta( $term->term_id, 'sp_latitude', true ) );
|
||||
$longitude = trim( (string) get_term_meta( $term->term_id, 'sp_longitude', true ) );
|
||||
|
||||
return '' !== $latitude && '' !== $longitude;
|
||||
}
|
||||
|
||||
/**
|
||||
* Render the venue ground rules section.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
function tony_sportspress_render_venue_ground_rules_section() {
|
||||
if ( ! is_tax( 'sp_venue' ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$term = get_queried_object();
|
||||
if ( ! $term instanceof WP_Term ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$ground_rules = get_term_meta( $term->term_id, 'tse_ground_rules', true );
|
||||
if ( ! is_string( $ground_rules ) || '' === trim( $ground_rules ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$ground_rules = apply_filters( 'the_content', $ground_rules );
|
||||
|
||||
echo '<div class="sp-section-content sp-section-content-details"><div class="sp-template sp-template-venue-details tse-ground-rules"><h4 class="sp-table-caption">' . esc_html__( 'Ground Rules', 'tonys-sportspress-enhancements' ) . '</h4><div class="sp-table-wrapper tse-ground-rules-content">' . $ground_rules . '</div></div></div>';
|
||||
}
|
||||
add_action( 'sportspress_before_venue_map', 'tony_sportspress_render_venue_ground_rules_section', 5 );
|
||||
|
||||
/**
|
||||
* Open the venue map section wrapper.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
function tony_sportspress_open_venue_map_section() {
|
||||
if ( ! is_tax( 'sp_venue' ) || ! tony_sportspress_current_venue_has_map() ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$GLOBALS['tse_venue_map_section_open'] = true;
|
||||
echo '<div class="sp-section-content sp-section-content-venue"><div class="sp-template sp-template-venue-map"><h4 class="sp-table-caption tse-venue-map-caption">' . esc_html( tony_sportspress_get_venue_map_caption() ) . '</h4><div class="sp-table-wrapper tse-venue-map-content">';
|
||||
}
|
||||
add_action( 'sportspress_before_venue_map', 'tony_sportspress_open_venue_map_section', 15 );
|
||||
|
||||
/**
|
||||
* Close the venue map section wrapper.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
function tony_sportspress_close_venue_map_section() {
|
||||
if ( empty( $GLOBALS['tse_venue_map_section_open'] ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
$GLOBALS['tse_venue_map_section_open'] = false;
|
||||
echo '</div></div></div>';
|
||||
}
|
||||
add_action( 'sportspress_after_venue_map', 'tony_sportspress_close_venue_map_section', 5 );
|
||||
|
||||
/**
|
||||
* Enqueue venue section styles.
|
||||
*
|
||||
* The venue map and ground rules are rendered as SportsPress-style sections
|
||||
* rather than archive header content, so they need section-scoped typography
|
||||
* and list styling.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
function tony_sportspress_enqueue_venue_section_styles() {
|
||||
if ( ! is_tax( 'sp_venue' ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
wp_register_style( 'tony-sportspress-venue-sections', false, array(), TONY_SPORTSPRESS_ENHANCEMENTS_VERSION );
|
||||
wp_enqueue_style( 'tony-sportspress-venue-sections' );
|
||||
|
||||
$event_list_display = tony_sportspress_field_event_list_enabled() ? 'block' : 'none';
|
||||
|
||||
$css = <<<CSS
|
||||
body.tax-sp_venue .sp-template-venue-map .sp-table-wrapper,
|
||||
body.tax-sp_venue .sp-template-venue-details .sp-table-wrapper {
|
||||
background: #fff;
|
||||
border: 1px solid #e0e0e0;
|
||||
border-top: 0;
|
||||
}
|
||||
|
||||
body.tax-sp_venue .site-main article.sp_event,
|
||||
body.tax-sp_venue .sp-template-event-list,
|
||||
body.tax-sp_venue .sp-template-event-blocks,
|
||||
body.tax-sp_venue .sp-template-event-logos,
|
||||
body.tax-sp_venue .sp-template-event-results {
|
||||
display: {$event_list_display};
|
||||
}
|
||||
|
||||
body.tax-sp_venue .sp-template-venue-map .sp-google-map {
|
||||
display: block;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
body.tax-sp_venue .tse-ground-rules-content {
|
||||
line-height: 1.7;
|
||||
padding: 1em 15px 0;
|
||||
}
|
||||
|
||||
body.tax-sp_venue .tse-ground-rules-content h2,
|
||||
body.tax-sp_venue .tse-ground-rules-content h3,
|
||||
body.tax-sp_venue .tse-ground-rules-content h4,
|
||||
body.tax-sp_venue .tse-ground-rules-content h5,
|
||||
body.tax-sp_venue .tse-ground-rules-content h6 {
|
||||
margin: 1.25em 0 0.55em;
|
||||
font-weight: 700;
|
||||
line-height: 1.25;
|
||||
text-transform: none;
|
||||
}
|
||||
|
||||
body.tax-sp_venue .tse-ground-rules-content h2 {
|
||||
font-size: 2rem;
|
||||
}
|
||||
|
||||
body.tax-sp_venue .tse-ground-rules-content h3 {
|
||||
font-size: 1.75rem;
|
||||
}
|
||||
|
||||
body.tax-sp_venue .tse-ground-rules-content h4 {
|
||||
font-size: 1.5rem;
|
||||
}
|
||||
|
||||
body.tax-sp_venue .tse-ground-rules-content p,
|
||||
body.tax-sp_venue .tse-ground-rules-content ul,
|
||||
body.tax-sp_venue .tse-ground-rules-content ol,
|
||||
body.tax-sp_venue .tse-ground-rules-content blockquote,
|
||||
body.tax-sp_venue .tse-ground-rules-content pre {
|
||||
margin: 0 0 1em;
|
||||
}
|
||||
|
||||
body.tax-sp_venue .tse-ground-rules-content ul,
|
||||
body.tax-sp_venue .tse-ground-rules-content ol {
|
||||
margin-left: 1.5em;
|
||||
padding-left: 1.5em;
|
||||
list-style-position: outside;
|
||||
}
|
||||
|
||||
body.tax-sp_venue .tse-ground-rules-content ul {
|
||||
list-style: disc;
|
||||
}
|
||||
|
||||
body.tax-sp_venue .tse-ground-rules-content ol {
|
||||
list-style: decimal;
|
||||
}
|
||||
|
||||
body.tax-sp_venue .tse-ground-rules-content li {
|
||||
margin: 0.35em 0;
|
||||
}
|
||||
CSS;
|
||||
|
||||
wp_add_inline_style( 'tony-sportspress-venue-sections', $css );
|
||||
}
|
||||
add_action( 'wp_enqueue_scripts', 'tony_sportspress_enqueue_venue_section_styles' );
|
||||
|
||||
/**
|
||||
* Render add-form fields for venue metadata.
|
||||
*/
|
||||
@@ -58,6 +368,24 @@ function tony_sportspress_add_venue_meta_fields() {
|
||||
<input name="tse_abbreviation" id="tse_abbreviation" type="text" value="" maxlength="20" />
|
||||
<p><?php esc_html_e( 'Optional abbreviation such as CC East or Field 1.', 'tonys-sportspress-enhancements' ); ?></p>
|
||||
</div>
|
||||
<div class="form-field term-ground-rules-wrap">
|
||||
<label for="tse_ground_rules"><?php esc_html_e( 'Ground Rules', 'tonys-sportspress-enhancements' ); ?></label>
|
||||
<?php
|
||||
wp_editor(
|
||||
'',
|
||||
'tse_ground_rules',
|
||||
array(
|
||||
'textarea_name' => 'tse_ground_rules',
|
||||
'textarea_rows' => 10,
|
||||
'media_buttons' => true,
|
||||
'teeny' => false,
|
||||
'quicktags' => true,
|
||||
'drag_drop_upload' => true,
|
||||
)
|
||||
);
|
||||
?>
|
||||
<p><?php esc_html_e( 'Supports headings, lists, links, and images.', 'tonys-sportspress-enhancements' ); ?></p>
|
||||
</div>
|
||||
<?php
|
||||
}
|
||||
add_action( 'sp_venue_add_form_fields', 'tony_sportspress_add_venue_meta_fields' );
|
||||
@@ -70,6 +398,7 @@ add_action( 'sp_venue_add_form_fields', 'tony_sportspress_add_venue_meta_fields'
|
||||
function tony_sportspress_edit_venue_meta_fields( $term ) {
|
||||
$short_name = get_term_meta( $term->term_id, 'tse_short_name', true );
|
||||
$abbreviation = get_term_meta( $term->term_id, 'tse_abbreviation', true );
|
||||
$ground_rules = get_term_meta( $term->term_id, 'tse_ground_rules', true );
|
||||
?>
|
||||
<tr class="form-field term-short-name-wrap">
|
||||
<th scope="row">
|
||||
@@ -89,6 +418,28 @@ function tony_sportspress_edit_venue_meta_fields( $term ) {
|
||||
<p class="description"><?php esc_html_e( 'Optional abbreviation such as CC East or Field 1.', 'tonys-sportspress-enhancements' ); ?></p>
|
||||
</td>
|
||||
</tr>
|
||||
<tr class="form-field term-ground-rules-wrap">
|
||||
<th scope="row">
|
||||
<label for="tse_ground_rules"><?php esc_html_e( 'Ground Rules', 'tonys-sportspress-enhancements' ); ?></label>
|
||||
</th>
|
||||
<td>
|
||||
<?php
|
||||
wp_editor(
|
||||
(string) $ground_rules,
|
||||
'tse_ground_rules',
|
||||
array(
|
||||
'textarea_name' => 'tse_ground_rules',
|
||||
'textarea_rows' => 10,
|
||||
'media_buttons' => true,
|
||||
'teeny' => false,
|
||||
'quicktags' => true,
|
||||
'drag_drop_upload' => true,
|
||||
)
|
||||
);
|
||||
?>
|
||||
<p class="description"><?php esc_html_e( 'Supports headings, lists, links, and images.', 'tonys-sportspress-enhancements' ); ?></p>
|
||||
</td>
|
||||
</tr>
|
||||
<?php
|
||||
}
|
||||
add_action( 'sp_venue_edit_form_fields', 'tony_sportspress_edit_venue_meta_fields' );
|
||||
@@ -109,8 +460,12 @@ function tony_sportspress_save_venue_meta_fields( $term_id ) {
|
||||
$abbreviation = isset( $_POST['tse_abbreviation'] ) ? sanitize_text_field( wp_unslash( $_POST['tse_abbreviation'] ) ) : '';
|
||||
$abbreviation = is_string( $abbreviation ) ? trim( $abbreviation ) : '';
|
||||
|
||||
$ground_rules = isset( $_POST['tse_ground_rules'] ) ? wp_kses_post( wp_unslash( $_POST['tse_ground_rules'] ) ) : '';
|
||||
$ground_rules = is_string( $ground_rules ) ? trim( $ground_rules ) : '';
|
||||
|
||||
update_term_meta( $term_id, 'tse_short_name', $short_name );
|
||||
update_term_meta( $term_id, 'tse_abbreviation', $abbreviation );
|
||||
update_term_meta( $term_id, 'tse_ground_rules', $ground_rules );
|
||||
}
|
||||
add_action( 'created_sp_venue', 'tony_sportspress_save_venue_meta_fields' );
|
||||
add_action( 'edited_sp_venue', 'tony_sportspress_save_venue_meta_fields' );
|
||||
|
||||
2679
includes/sp-webhooks.php
Normal file
2679
includes/sp-webhooks.php
Normal file
File diff suppressed because it is too large
Load Diff
@@ -1,7 +1,7 @@
|
||||
|
||||
{
|
||||
"name": "tonys-sportspress-enhancements",
|
||||
"version": "0.1.5",
|
||||
"version": "0.1.15",
|
||||
"main": "Gruntfile.js",
|
||||
"author": "YOUR NAME HERE",
|
||||
"scripts" : {
|
||||
|
||||
222
tests/test-featured-image-generator.php
Normal file
222
tests/test-featured-image-generator.php
Normal file
@@ -0,0 +1,222 @@
|
||||
<?php
|
||||
/**
|
||||
* Tests for the SportsPress event image generator.
|
||||
*
|
||||
* @package Tonys_Sportspress_Enhancements
|
||||
*/
|
||||
|
||||
/**
|
||||
* Featured image generator tests.
|
||||
*/
|
||||
class Test_Featured_Image_Generator extends WP_UnitTestCase {
|
||||
|
||||
/**
|
||||
* Temp files created during a test.
|
||||
*
|
||||
* @var string[]
|
||||
*/
|
||||
private $temp_files = array();
|
||||
|
||||
/**
|
||||
* Clean up temp files.
|
||||
*/
|
||||
public function tear_down(): void {
|
||||
foreach ( $this->temp_files as $file ) {
|
||||
if ( file_exists( $file ) ) {
|
||||
unlink( $file );
|
||||
}
|
||||
}
|
||||
|
||||
$this->temp_files = array();
|
||||
parent::tear_down();
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a post.
|
||||
*
|
||||
* @param string $type Post type.
|
||||
* @param string $title Title.
|
||||
* @return int
|
||||
*/
|
||||
private function create_post_of_type( $type, $title ) {
|
||||
return self::factory()->post->create(
|
||||
array(
|
||||
'post_type' => $type,
|
||||
'post_title' => $title,
|
||||
'post_status' => 'publish',
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a small raster fixture.
|
||||
*
|
||||
* @param string $extension File extension.
|
||||
* @return string
|
||||
*/
|
||||
private function create_raster_fixture( $extension ) {
|
||||
$image = imagecreatetruecolor( 24, 24 );
|
||||
$red = imagecolorallocate( $image, 200, 0, 0 );
|
||||
imagefilledrectangle( $image, 0, 0, 23, 23, $red );
|
||||
|
||||
$file = tempnam( sys_get_temp_dir(), 'sp-img-' );
|
||||
$path = $file . '.' . $extension;
|
||||
rename( $file, $path );
|
||||
|
||||
switch ( $extension ) {
|
||||
case 'jpg':
|
||||
imagejpeg( $image, $path );
|
||||
break;
|
||||
case 'gif':
|
||||
imagegif( $image, $path );
|
||||
break;
|
||||
case 'webp':
|
||||
imagewebp( $image, $path );
|
||||
break;
|
||||
case 'png':
|
||||
default:
|
||||
imagepng( $image, $path );
|
||||
break;
|
||||
}
|
||||
|
||||
asc_sp_event_image_destroy( $image );
|
||||
|
||||
$this->temp_files[] = $path;
|
||||
|
||||
return $path;
|
||||
}
|
||||
|
||||
/**
|
||||
* Invalid IDs and non-event posts produce request errors.
|
||||
*/
|
||||
public function test_invalid_and_non_event_requests_prepare_404_errors() {
|
||||
$this->assertWPError( asc_sp_event_prepare_image_request( 999999 ) );
|
||||
|
||||
$post_id = $this->create_post_of_type( 'post', 'Regular Post' );
|
||||
$error = asc_sp_event_prepare_image_request( $post_id );
|
||||
|
||||
$this->assertWPError( $error );
|
||||
$this->assertSame( 'invalid_event', $error->get_error_code() );
|
||||
}
|
||||
|
||||
/**
|
||||
* Missing team logo paths fall back to generated text and valid dimensions.
|
||||
*/
|
||||
public function test_missing_logo_path_generates_png_with_expected_dimensions() {
|
||||
$image_data = generate_bisected_image( '#123456', '#abcdef', '/missing-left.png', '/missing-right.png', 'Hawks', 'Electrons' );
|
||||
$image = imagecreatefromstring( $image_data );
|
||||
|
||||
$this->assertNotFalse( $image );
|
||||
$this->assertSame( 1200, imagesx( $image ) );
|
||||
$this->assertSame( 628, imagesy( $image ) );
|
||||
|
||||
asc_sp_event_image_destroy( $image );
|
||||
}
|
||||
|
||||
/**
|
||||
* Square image variant generates square PNG dimensions.
|
||||
*/
|
||||
public function test_square_variant_generates_expected_dimensions() {
|
||||
$dimensions = asc_sp_event_image_variant_dimensions( 'square' );
|
||||
$image_data = generate_bisected_image( '#123456', '#abcdef', '/missing-left.png', '/missing-right.png', 'Hawks', 'Electrons', $dimensions['width'], $dimensions['height'] );
|
||||
$image = imagecreatefromstring( $image_data );
|
||||
|
||||
$this->assertNotFalse( $image );
|
||||
$this->assertSame( 1200, imagesx( $image ) );
|
||||
$this->assertSame( 1200, imagesy( $image ) );
|
||||
|
||||
asc_sp_event_image_destroy( $image );
|
||||
}
|
||||
|
||||
/**
|
||||
* Raster loader supports common GD-backed formats.
|
||||
*/
|
||||
public function test_raster_loader_supports_common_formats_when_available() {
|
||||
$formats = array(
|
||||
'png' => 'imagecreatefrompng',
|
||||
'jpg' => 'imagecreatefromjpeg',
|
||||
'gif' => 'imagecreatefromgif',
|
||||
);
|
||||
|
||||
if ( function_exists( 'imagewebp' ) && function_exists( 'imagecreatefromwebp' ) ) {
|
||||
$formats['webp'] = 'imagecreatefromwebp';
|
||||
}
|
||||
|
||||
foreach ( $formats as $extension => $function ) {
|
||||
if ( ! function_exists( $function ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$path = $this->create_raster_fixture( $extension );
|
||||
$image = asc_sp_event_image_create_from_file( $path );
|
||||
|
||||
$this->assertNotFalse( $image, "Failed loading {$extension}" );
|
||||
$this->assertSame( 24, imagesx( $image ) );
|
||||
$this->assertSame( 24, imagesy( $image ) );
|
||||
asc_sp_event_image_destroy( $image );
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Bundled sporty font is available for fallback text.
|
||||
*/
|
||||
public function test_bundled_bebas_neue_font_is_available() {
|
||||
$this->assertFileExists( asc_sp_event_image_font_path() );
|
||||
$this->assertIsReadable( asc_sp_event_image_font_path() );
|
||||
}
|
||||
|
||||
/**
|
||||
* Prepared event request includes fallback text for missing logos.
|
||||
*/
|
||||
public function test_prepare_image_request_uses_team_short_name_fallbacks() {
|
||||
$team1 = $this->create_post_of_type( 'sp_team', 'Hawks' );
|
||||
$team2 = $this->create_post_of_type( 'sp_team', 'Electrons' );
|
||||
$event = $this->create_post_of_type( 'sp_event', 'Hawks vs Electrons' );
|
||||
|
||||
add_post_meta( $event, 'sp_team', $team1 );
|
||||
add_post_meta( $event, 'sp_team', $team2 );
|
||||
|
||||
$request = asc_sp_event_prepare_image_request( $event );
|
||||
|
||||
$this->assertIsArray( $request );
|
||||
$this->assertSame( 'Hawks', $request['team1_fallback'] );
|
||||
$this->assertSame( 'Electrons', $request['team2_fallback'] );
|
||||
$this->assertSame( '', $request['team1_logo'] );
|
||||
$this->assertSame( '', $request['team2_logo'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Invalid colors are safely normalized.
|
||||
*/
|
||||
public function test_invalid_colors_fall_back_to_configured_defaults() {
|
||||
$this->assertSame( '#4B5563', asc_sp_event_image_color( 'not-a-color' ) );
|
||||
$this->assertSame( '#6B7280', asc_sp_event_image_color( 'not-a-color', '#6B7280' ) );
|
||||
$this->assertSame( '#112233', asc_sp_event_image_color( '#112233' ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Image cache keys include the generator version and style hash.
|
||||
*/
|
||||
public function test_prepare_image_request_uses_versioned_style_cache_key() {
|
||||
$team1 = $this->create_post_of_type( 'sp_team', 'Hawks' );
|
||||
$team2 = $this->create_post_of_type( 'sp_team', 'Electrons' );
|
||||
$event = $this->create_post_of_type( 'sp_event', 'Hawks vs Electrons' );
|
||||
|
||||
add_post_meta( $event, 'sp_team', $team1 );
|
||||
add_post_meta( $event, 'sp_team', $team2 );
|
||||
|
||||
$request = asc_sp_event_prepare_image_request( $event );
|
||||
|
||||
$this->assertStringStartsWith( 'team_image_v' . ASC_SP_EVENT_IMAGE_CACHE_VERSION . '_' . asc_sp_event_image_cache_style_hash(), $request['cache_key'] );
|
||||
$this->assertSame( 'wide', $request['variant'] );
|
||||
$this->assertSame( 1200, $request['width'] );
|
||||
$this->assertSame( 628, $request['height'] );
|
||||
|
||||
$square_request = asc_sp_event_prepare_image_request( $event, 'square' );
|
||||
|
||||
$this->assertStringContainsString( '_square_', $square_request['cache_key'] );
|
||||
$this->assertSame( 'square', $square_request['variant'] );
|
||||
$this->assertSame( 1200, $square_request['width'] );
|
||||
$this->assertSame( 1200, $square_request['height'] );
|
||||
}
|
||||
}
|
||||
267
tests/test-open-graph-tags.php
Normal file
267
tests/test-open-graph-tags.php
Normal file
@@ -0,0 +1,267 @@
|
||||
<?php
|
||||
/**
|
||||
* Tests for SportsPress event Open Graph output.
|
||||
*
|
||||
* @package Tonys_Sportspress_Enhancements
|
||||
*/
|
||||
|
||||
if ( ! class_exists( 'SP_Event' ) ) {
|
||||
/**
|
||||
* Minimal SportsPress event test double.
|
||||
*/
|
||||
class SP_Event {
|
||||
/**
|
||||
* Event post ID.
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
private $id;
|
||||
|
||||
/**
|
||||
* Status values by event ID.
|
||||
*
|
||||
* @var array<int,string>
|
||||
*/
|
||||
public static $statuses = array();
|
||||
|
||||
/**
|
||||
* Result values by event ID.
|
||||
*
|
||||
* @var array<int,array>
|
||||
*/
|
||||
public static $results = array();
|
||||
|
||||
/**
|
||||
* Constructor.
|
||||
*
|
||||
* @param int $id Event post ID.
|
||||
*/
|
||||
public function __construct( $id ) {
|
||||
$this->id = absint( $id );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get event status.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function status() {
|
||||
return self::$statuses[ $this->id ] ?? '';
|
||||
}
|
||||
|
||||
/**
|
||||
* Get event results.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function results() {
|
||||
return self::$results[ $this->id ] ?? array();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Open Graph tests.
|
||||
*/
|
||||
class Test_Open_Graph_Tags extends WP_UnitTestCase {
|
||||
|
||||
/**
|
||||
* Reset mock SportsPress state.
|
||||
*/
|
||||
public function set_up(): void {
|
||||
parent::set_up();
|
||||
|
||||
if ( property_exists( 'SP_Event', 'statuses' ) ) {
|
||||
SP_Event::$statuses = array();
|
||||
SP_Event::$results = array();
|
||||
}
|
||||
|
||||
update_option( 'sportspress_event_reverse_teams', 'no' );
|
||||
update_option( 'sportspress_event_teams_delimiter', 'vs' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a team.
|
||||
*
|
||||
* @param string $name Team name.
|
||||
* @return int
|
||||
*/
|
||||
private function create_team( $name ) {
|
||||
return self::factory()->post->create(
|
||||
array(
|
||||
'post_type' => 'sp_team',
|
||||
'post_title' => $name,
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Create an event.
|
||||
*
|
||||
* @param array $args Post args.
|
||||
* @return int
|
||||
*/
|
||||
private function create_event( array $args = array() ) {
|
||||
return self::factory()->post->create(
|
||||
wp_parse_args(
|
||||
$args,
|
||||
array(
|
||||
'post_type' => 'sp_event',
|
||||
'post_title' => 'Test Event',
|
||||
'post_status' => 'future',
|
||||
'post_date' => '2026-05-02 13:00:00',
|
||||
'post_content' => 'First pitch at one.',
|
||||
)
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Future event emits complete Open Graph data.
|
||||
*/
|
||||
public function test_future_event_emits_core_open_graph_values() {
|
||||
$home = $this->create_team( 'Hawks' );
|
||||
$away = $this->create_team( 'Electrons' );
|
||||
$event = $this->create_event();
|
||||
|
||||
add_post_meta( $event, 'sp_team', $home );
|
||||
add_post_meta( $event, 'sp_team', $away );
|
||||
|
||||
if ( property_exists( 'SP_Event', 'statuses' ) ) {
|
||||
SP_Event::$statuses[ $event ] = 'future';
|
||||
}
|
||||
|
||||
$meta = asc_sp_event_open_graph_data( $event );
|
||||
|
||||
$this->assertSame( 'article', $meta['type'] );
|
||||
$this->assertStringContainsString( 'Hawks vs Electrons', $meta['title'] );
|
||||
$this->assertStringContainsString( 'First pitch at one.', $meta['description'] );
|
||||
$this->assertCount( 1, $meta['images'] );
|
||||
$this->assertSame( '1200', $meta['images'][0]['width'] );
|
||||
$this->assertSame( '628', $meta['images'][0]['height'] );
|
||||
$this->assertSame( '1200', $meta['image_width'] );
|
||||
$this->assertSame( '628', $meta['image_height'] );
|
||||
$this->assertStringContainsString( '/head-to-head?post=' . $event, $meta['image'] );
|
||||
$this->assertNotEmpty( $meta['url'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Postponed, cancelled, and TBD labels appear in title and description.
|
||||
*
|
||||
* @dataProvider status_provider
|
||||
*
|
||||
* @param string $status Status slug.
|
||||
*/
|
||||
public function test_schedule_status_appears_in_title_and_description( $status ) {
|
||||
$home = $this->create_team( 'Hawks' );
|
||||
$away = $this->create_team( 'Electrons' );
|
||||
$event = $this->create_event();
|
||||
|
||||
add_post_meta( $event, 'sp_team', $home );
|
||||
add_post_meta( $event, 'sp_team', $away );
|
||||
update_post_meta( $event, 'sp_status', $status );
|
||||
|
||||
$meta = asc_sp_event_open_graph_data( $event );
|
||||
$label = strtoupper( $status );
|
||||
|
||||
$this->assertStringStartsWith( $label, $meta['title'] );
|
||||
$this->assertStringStartsWith( $label, $meta['description'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Status provider.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function status_provider() {
|
||||
return array(
|
||||
array( 'postponed' ),
|
||||
array( 'cancelled' ),
|
||||
array( 'tbd' ),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Result events with scores emit score titles.
|
||||
*/
|
||||
public function test_result_event_with_scores_emits_score_title() {
|
||||
$home = $this->create_team( 'Hawks' );
|
||||
$away = $this->create_team( 'Electrons' );
|
||||
$event = $this->create_event( array( 'post_status' => 'publish' ) );
|
||||
|
||||
add_post_meta( $event, 'sp_team', $home );
|
||||
add_post_meta( $event, 'sp_team', $away );
|
||||
|
||||
if ( property_exists( 'SP_Event', 'statuses' ) ) {
|
||||
SP_Event::$statuses[ $event ] = 'results';
|
||||
SP_Event::$results[ $event ] = array(
|
||||
0 => array( 'r' => 'R' ),
|
||||
$home => array( 'r' => '7' ),
|
||||
$away => array( 'r' => '4' ),
|
||||
);
|
||||
}
|
||||
|
||||
$meta = asc_sp_event_open_graph_data( $event );
|
||||
|
||||
$this->assertStringContainsString( 'Hawks 7-4 Electrons', $meta['title'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Missing teams/results/outcomes still produce valid data.
|
||||
*/
|
||||
public function test_missing_sportspress_data_does_not_break_meta_generation() {
|
||||
$event = $this->create_event(
|
||||
array(
|
||||
'post_title' => 'Sparse Event',
|
||||
'post_content' => '',
|
||||
)
|
||||
);
|
||||
|
||||
if ( property_exists( 'SP_Event', 'statuses' ) ) {
|
||||
SP_Event::$statuses[ $event ] = 'results';
|
||||
SP_Event::$results[ $event ] = array();
|
||||
}
|
||||
|
||||
$meta = asc_sp_event_open_graph_data( $event );
|
||||
|
||||
$this->assertSame( 'Sparse Event', $meta['title'] );
|
||||
$this->assertNotEmpty( $meta['description'] );
|
||||
$this->assertSame( '1200', $meta['image_width'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* HTML-heavy post content is stripped and escaped in rendered tags.
|
||||
*/
|
||||
public function test_description_strips_html_and_rendered_tags_are_escaped() {
|
||||
$home = $this->create_team( 'Hawks "A"' );
|
||||
$away = $this->create_team( 'Electrons <B>' );
|
||||
$event = $this->create_event(
|
||||
array(
|
||||
'post_content' => '<script>alert("x")</script><p>Bring <strong>bats</strong> & gloves.</p>',
|
||||
)
|
||||
);
|
||||
|
||||
add_post_meta( $event, 'sp_team', $home );
|
||||
add_post_meta( $event, 'sp_team', $away );
|
||||
|
||||
$meta = asc_sp_event_open_graph_data( $event );
|
||||
|
||||
$this->assertStringNotContainsString( '<script', $meta['description'] );
|
||||
$this->assertStringContainsString( 'Bring bats & gloves.', $meta['description'] );
|
||||
|
||||
$GLOBALS['post'] = get_post( $event );
|
||||
$GLOBALS['wp_query']->is_single = true;
|
||||
|
||||
ob_start();
|
||||
custom_open_graph_tags_with_sportspress_integration();
|
||||
$output = ob_get_clean();
|
||||
|
||||
$this->assertStringContainsString( 'og:image:width', $output );
|
||||
$this->assertSame( 1, substr_count( $output, 'property="og:image" content=' ) );
|
||||
$this->assertStringContainsString( 'content="628"', $output );
|
||||
$this->assertStringContainsString( 'content="1200"', $output );
|
||||
$this->assertStringContainsString( 'Hawks "A"', $output );
|
||||
$this->assertStringNotContainsString( '<B>', $output );
|
||||
$this->assertStringNotContainsString( '<script', $output );
|
||||
}
|
||||
}
|
||||
341
tests/test-sp-schedule-exporter.php
Normal file
341
tests/test-sp-schedule-exporter.php
Normal file
@@ -0,0 +1,341 @@
|
||||
<?php
|
||||
/**
|
||||
* Tests for schedule exporter and printable schedule helpers.
|
||||
*
|
||||
* @package Tonys_Sportspress_Enhancements
|
||||
*/
|
||||
|
||||
/**
|
||||
* Schedule exporter tests.
|
||||
*/
|
||||
class Test_SP_Schedule_Exporter extends WP_UnitTestCase {
|
||||
|
||||
/**
|
||||
* Original request globals.
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
private $original_get = array();
|
||||
|
||||
/**
|
||||
* Set up shared test fixtures.
|
||||
*/
|
||||
public function set_up() {
|
||||
parent::set_up();
|
||||
|
||||
$this->original_get = $_GET;
|
||||
|
||||
foreach ( array( 'sp_venue', 'sp_league', 'sp_season' ) as $taxonomy ) {
|
||||
if ( ! taxonomy_exists( $taxonomy ) ) {
|
||||
register_taxonomy( $taxonomy, 'sp_event' );
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Restore request globals.
|
||||
*/
|
||||
public function tear_down() {
|
||||
$_GET = $this->original_get;
|
||||
|
||||
parent::tear_down();
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a SportsPress team post.
|
||||
*
|
||||
* @param string $name Team name.
|
||||
* @return int
|
||||
*/
|
||||
private function create_team( $name ) {
|
||||
return self::factory()->post->create(
|
||||
array(
|
||||
'post_type' => 'sp_team',
|
||||
'post_status' => 'publish',
|
||||
'post_title' => $name,
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a SportsPress event with ordered teams.
|
||||
*
|
||||
* @param int $home_id Home team ID.
|
||||
* @param int $away_id Away team ID.
|
||||
* @param int[] $venue_ids Venue IDs.
|
||||
* @return int
|
||||
*/
|
||||
private function create_event( $home_id, $away_id, $venue_ids = array() ) {
|
||||
$event_id = self::factory()->post->create(
|
||||
array(
|
||||
'post_type' => 'sp_event',
|
||||
'post_status' => 'publish',
|
||||
'post_title' => 'Game',
|
||||
'post_date' => '2026-05-20 18:00:00',
|
||||
'post_date_gmt' => '2026-05-20 23:00:00',
|
||||
)
|
||||
);
|
||||
|
||||
add_post_meta( $event_id, 'sp_team', (string) $home_id );
|
||||
add_post_meta( $event_id, 'sp_team', (string) $away_id );
|
||||
|
||||
if ( ! empty( $venue_ids ) ) {
|
||||
wp_set_object_terms( $event_id, $venue_ids, 'sp_venue' );
|
||||
}
|
||||
|
||||
return $event_id;
|
||||
}
|
||||
|
||||
/**
|
||||
* Multiple selected team IDs should be retained in request order.
|
||||
*/
|
||||
public function test_resolve_team_ids_accepts_multiple_request_values() {
|
||||
$team_one = $this->create_team( 'Blue' );
|
||||
$team_two = $this->create_team( 'Red' );
|
||||
$teams = array( get_post( $team_one ), get_post( $team_two ) );
|
||||
|
||||
$_GET['team_id'] = array( (string) $team_one, (string) $team_two );
|
||||
|
||||
$this->assertSame( array( $team_one, $team_two ), tse_sp_schedule_exporter_resolve_team_ids( $teams ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Printable URLs should carry multiple team IDs and the selected field.
|
||||
*/
|
||||
public function test_printable_url_accepts_multiple_teams_and_field() {
|
||||
$url = tse_sp_schedule_exporter_get_printable_url( array( 12, 34 ), 56, 'letter', 78, false, 90, true );
|
||||
$query = array();
|
||||
|
||||
wp_parse_str( (string) wp_parse_url( $url, PHP_URL_QUERY ), $query );
|
||||
|
||||
$this->assertSame( '12,34', $query['sp_team'] );
|
||||
$this->assertSame( '90', $query['sp_field'] );
|
||||
$this->assertSame( 'name', $query['team_label'] );
|
||||
$this->assertSame( 'name', $query['field_label'] );
|
||||
$this->assertSame( 'selected_first', $query['title_format'] );
|
||||
$this->assertSame( '1', $query['month_pages'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Printable URLs should carry selected team and field label modes.
|
||||
*/
|
||||
public function test_printable_url_accepts_label_modes() {
|
||||
$url = tse_sp_schedule_exporter_get_printable_url( 12, 56, 'letter', 78, false, 90, false, 'shortname', 'abbreviation' );
|
||||
$query = array();
|
||||
|
||||
wp_parse_str( (string) wp_parse_url( $url, PHP_URL_QUERY ), $query );
|
||||
|
||||
$this->assertSame( 'shortname', $query['team_label'] );
|
||||
$this->assertSame( 'abbreviation', $query['field_label'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Printable URLs should carry black-and-white mode when requested.
|
||||
*/
|
||||
public function test_printable_url_accepts_black_white_mode() {
|
||||
$url = tse_sp_schedule_exporter_get_printable_url( 12, 56, 'letter', 78, false, 90, false, 'name', 'name', 'selected_first', true );
|
||||
$query = array();
|
||||
|
||||
wp_parse_str( (string) wp_parse_url( $url, PHP_URL_QUERY ), $query );
|
||||
|
||||
$this->assertSame( '1', $query['black_white'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Single-team printable entries should keep the existing opponent perspective.
|
||||
*/
|
||||
public function test_printable_single_team_entries_keep_opponent_perspective() {
|
||||
$home_id = $this->create_team( 'Home Team' );
|
||||
$away_id = $this->create_team( 'Away Team' );
|
||||
$venue_id = self::factory()->term->create( array( 'taxonomy' => 'sp_venue', 'name' => 'North Field' ) );
|
||||
$this->create_event( $home_id, $away_id, array( $venue_id ) );
|
||||
|
||||
$printable = Tony_Sportspress_Printable_Calendars::instance();
|
||||
$method = new ReflectionMethod( $printable, 'get_schedule_entries' );
|
||||
$method->setAccessible( true );
|
||||
|
||||
$entries = $method->invoke( $printable, $home_id, 0, 0, array() );
|
||||
|
||||
$this->assertCount( 1, $entries );
|
||||
$this->assertFalse( $entries[0]['is_matchup'] );
|
||||
$this->assertTrue( $entries[0]['is_home'] );
|
||||
$this->assertSame( 'Away Team', $entries[0]['opponent_name'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Printable entries should honor requested team and field labels.
|
||||
*/
|
||||
public function test_printable_entries_honor_label_modes() {
|
||||
$home_id = $this->create_team( 'Home Team' );
|
||||
$away_id = $this->create_team( 'Away Team' );
|
||||
$venue_id = self::factory()->term->create( array( 'taxonomy' => 'sp_venue', 'name' => 'North Field' ) );
|
||||
update_post_meta( $away_id, 'sp_abbreviation', 'AWY' );
|
||||
update_term_meta( $venue_id, 'tse_abbreviation', 'NF' );
|
||||
$this->create_event( $home_id, $away_id, array( $venue_id ) );
|
||||
|
||||
$printable = Tony_Sportspress_Printable_Calendars::instance();
|
||||
$method = new ReflectionMethod( $printable, 'get_schedule_entries' );
|
||||
$method->setAccessible( true );
|
||||
|
||||
$entries = $method->invoke( $printable, $home_id, 0, 0, array(), 'abbreviation', 'abbreviation' );
|
||||
|
||||
$this->assertCount( 1, $entries );
|
||||
$this->assertSame( 'AWY', $entries[0]['opponent_name'] );
|
||||
$this->assertSame( 'NF', $entries[0]['venue_label'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Team CSV layout should support multiple selected teams with away-first context.
|
||||
*/
|
||||
public function test_team_export_multiple_selected_teams_uses_away_team_first() {
|
||||
$home_id = $this->create_team( 'Home Team' );
|
||||
$away_id = $this->create_team( 'Away Team' );
|
||||
$this->create_event( $home_id, $away_id );
|
||||
|
||||
$events = tse_sp_event_export_get_events(
|
||||
array(
|
||||
'team_ids' => array( $home_id, $away_id ),
|
||||
)
|
||||
);
|
||||
|
||||
$this->assertCount( 1, $events );
|
||||
$this->assertSame( 'Away Team', $events[0]['team_name'] );
|
||||
$this->assertSame( 'Away Team at Home Team', $events[0]['title'] );
|
||||
$this->assertSame( 'at', $events[0]['separator'] );
|
||||
$this->assertSame( 'Home Team', $events[0]['opponent_name'] );
|
||||
$this->assertSame( 'Away', $events[0]['location_flag'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Team CSV layout should use vs when the selected team is home.
|
||||
*/
|
||||
public function test_team_export_home_selected_team_uses_vs_separator() {
|
||||
$home_id = $this->create_team( 'Home Team' );
|
||||
$away_id = $this->create_team( 'Away Team' );
|
||||
$this->create_event( $home_id, $away_id );
|
||||
|
||||
$events = tse_sp_event_export_get_events(
|
||||
array(
|
||||
'team_ids' => array( $home_id ),
|
||||
)
|
||||
);
|
||||
|
||||
$this->assertCount( 1, $events );
|
||||
$this->assertSame( 'Home Team', $events[0]['team_name'] );
|
||||
$this->assertSame( 'Away Team', $events[0]['title'] );
|
||||
$this->assertSame( 'vs', $events[0]['separator'] );
|
||||
$this->assertSame( 'Away Team', $events[0]['opponent_name'] );
|
||||
$this->assertSame( 'Home', $events[0]['location_flag'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Team CSV layout should use at when the selected team is away.
|
||||
*/
|
||||
public function test_team_export_away_selected_team_uses_at_separator() {
|
||||
$home_id = $this->create_team( 'Home Team' );
|
||||
$away_id = $this->create_team( 'Away Team' );
|
||||
$this->create_event( $home_id, $away_id );
|
||||
|
||||
$events = tse_sp_event_export_get_events(
|
||||
array(
|
||||
'team_ids' => array( $away_id ),
|
||||
)
|
||||
);
|
||||
|
||||
$this->assertCount( 1, $events );
|
||||
$this->assertSame( 'Away Team', $events[0]['team_name'] );
|
||||
$this->assertSame( 'Home Team', $events[0]['title'] );
|
||||
$this->assertSame( 'at', $events[0]['separator'] );
|
||||
$this->assertSame( 'Home Team', $events[0]['opponent_name'] );
|
||||
$this->assertSame( 'Away', $events[0]['location_flag'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Matchup title format should ignore selected-team perspective.
|
||||
*/
|
||||
public function test_team_export_matchup_title_format_uses_away_at_home() {
|
||||
$home_id = $this->create_team( 'Home Team' );
|
||||
$away_id = $this->create_team( 'Away Team' );
|
||||
$this->create_event( $home_id, $away_id );
|
||||
|
||||
$events = tse_sp_event_export_get_events(
|
||||
array(
|
||||
'team_ids' => array( $home_id ),
|
||||
'title_format' => 'matchup',
|
||||
)
|
||||
);
|
||||
|
||||
$this->assertCount( 1, $events );
|
||||
$this->assertSame( 'Away Team at Home Team', $events[0]['title'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Multi-team printable entries should use selected-team-first titles by default.
|
||||
*/
|
||||
public function test_printable_multi_team_entries_use_selected_first_titles() {
|
||||
$home_id = $this->create_team( 'Home Team' );
|
||||
$away_id = $this->create_team( 'Away Team' );
|
||||
$venue_id = self::factory()->term->create( array( 'taxonomy' => 'sp_venue', 'name' => 'North Field' ) );
|
||||
$this->create_event( $home_id, $away_id, array( $venue_id ) );
|
||||
|
||||
$printable = Tony_Sportspress_Printable_Calendars::instance();
|
||||
$method = new ReflectionMethod( $printable, 'get_schedule_entries' );
|
||||
$method->setAccessible( true );
|
||||
|
||||
$entries = $method->invoke( $printable, array( $home_id, $away_id ), 0, 0, array() );
|
||||
|
||||
$this->assertCount( 1, $entries );
|
||||
$this->assertFalse( $entries[0]['is_matchup'] );
|
||||
$this->assertSame( 'Away Team', $entries[0]['title_team_name'] );
|
||||
$this->assertSame( 'at', $entries[0]['title_separator'] );
|
||||
$this->assertSame( 'Home Team', $entries[0]['title_opponent_name'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Exactly one selected field should suppress per-event venue labels.
|
||||
*/
|
||||
public function test_render_month_grid_can_suppress_event_venue_label() {
|
||||
$printable = Tony_Sportspress_Printable_Calendars::instance();
|
||||
$method = new ReflectionMethod( $printable, 'render_month_grid' );
|
||||
$method->setAccessible( true );
|
||||
|
||||
$entries = array(
|
||||
'2026-05-20' => array(
|
||||
array(
|
||||
'day_key' => '2026-05-20',
|
||||
'month_key' => '2026-05',
|
||||
'timestamp' => strtotime( '2026-05-20 18:00:00' ),
|
||||
'is_matchup' => true,
|
||||
'away_team_name' => 'Away',
|
||||
'home_team_name' => 'Home',
|
||||
'event_time' => '6:00 PM',
|
||||
'venue_label' => 'North',
|
||||
'venue_name' => 'North Field',
|
||||
'venue_key' => 'v:1',
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
ob_start();
|
||||
$method->invoke(
|
||||
$printable,
|
||||
'2026-05',
|
||||
$entries,
|
||||
array( 'v:1' => array( 'name' => 'North Field', 'color' => '#1D4ED8' ) ),
|
||||
array(
|
||||
'primary' => '#1D4ED8',
|
||||
'secondary' => '#DC2626',
|
||||
'accent' => '#1D4ED8',
|
||||
'ink' => '#111827',
|
||||
'muted_ink' => '#334155',
|
||||
),
|
||||
true,
|
||||
true
|
||||
);
|
||||
$output = ob_get_clean();
|
||||
|
||||
$this->assertStringContainsString( 'Away', $output );
|
||||
$this->assertStringContainsString( 'Home', $output );
|
||||
$this->assertStringNotContainsString( 'class="event-venue"', $output );
|
||||
}
|
||||
}
|
||||
408
tests/test-sp-webhooks.php
Normal file
408
tests/test-sp-webhooks.php
Normal file
@@ -0,0 +1,408 @@
|
||||
<?php
|
||||
/**
|
||||
* Tests for configurable SportsPress webhooks.
|
||||
*
|
||||
* @package Tonys_Sportspress_Enhancements
|
||||
*/
|
||||
|
||||
/**
|
||||
* Webhook feature tests.
|
||||
*/
|
||||
class Test_SP_Webhooks extends WP_UnitTestCase {
|
||||
|
||||
/**
|
||||
* Template placeholders should resolve nested values and JSON serialization.
|
||||
*/
|
||||
public function test_render_template_supports_dot_paths_and_tojson() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = 'Trigger={{ trigger.key }} Team={{ event.teams.0.name }} Image={{ event.image }} Payload={{ event|tojson }}';
|
||||
$context = array(
|
||||
'trigger' => array(
|
||||
'key' => 'event_results_updated',
|
||||
),
|
||||
'event' => array(
|
||||
'id' => 55,
|
||||
'image' => 'https://example.com/head-to-head?post=55',
|
||||
'teams' => array(
|
||||
array(
|
||||
'name' => 'Blue Team',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertStringContainsString( 'Trigger=event_results_updated', $rendered );
|
||||
$this->assertStringContainsString( 'Team=Blue Team', $rendered );
|
||||
$this->assertStringContainsString( 'Image=https://example.com/head-to-head?post=55', $rendered );
|
||||
$this->assertStringContainsString( '"id":55', $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Venue aliases and split schedule fields should render from the context.
|
||||
*/
|
||||
public function test_render_template_supports_field_alias_and_schedule_parts() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = 'Field={{ event.field.short_name }} Venue={{ event.venue.abbreviation }} Time={{ event.scheduled.time }} {{ event.scheduled.timezone }}';
|
||||
$context = array(
|
||||
'event' => array(
|
||||
'field' => array(
|
||||
'short_name' => 'North',
|
||||
),
|
||||
'venue' => array(
|
||||
'abbreviation' => 'NF',
|
||||
),
|
||||
'scheduled' => array(
|
||||
'time' => '7:30 PM',
|
||||
'timezone' => 'CDT',
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertSame( 'Field=North Venue=NF Time=7:30 PM CDT', $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Event context should expose home and away team aliases.
|
||||
*/
|
||||
public function test_render_template_supports_event_team_aliases() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = 'Home={{ event.home_team.name }} Away={{ event.away_team.name }}';
|
||||
$context = array(
|
||||
'event' => array(
|
||||
'home_team' => array(
|
||||
'name' => 'Home Team',
|
||||
),
|
||||
'away_team' => array(
|
||||
'name' => 'Away Team',
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertSame( 'Home=Home Team Away=Away Team', $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Event context should expose the current SportsPress schedule status.
|
||||
*/
|
||||
public function test_render_template_supports_event_status_alias() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = 'Status={{ event.status }}';
|
||||
$context = array(
|
||||
'event' => array(
|
||||
'status' => 'Postponed',
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertSame( 'Status=Postponed', $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Date filter should accept PHP date format strings for schedule values.
|
||||
*/
|
||||
public function test_render_template_supports_date_filter() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = 'Time={{ event.scheduled.timestamp|date("g:i A") }} ISO={{ event.scheduled.local_iso|date("m/d g:i A") }}';
|
||||
$context = array(
|
||||
'event' => array(
|
||||
'scheduled' => array(
|
||||
'timestamp' => 1714005000,
|
||||
'local_iso' => '2024-04-24T19:30:00-05:00',
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertSame( 'Time=7:30 PM ISO=04/24 7:30 PM', $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Change notifications should expose before and after venue/time values.
|
||||
*/
|
||||
public function test_render_template_supports_before_after_venue_and_time() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = 'Venue {{ changes.previous.venue.name }} -> {{ changes.current.venue.name }} Time {{ changes.previous.time }} -> {{ changes.current.time }}';
|
||||
$context = array(
|
||||
'changes' => array(
|
||||
'previous' => array(
|
||||
'time' => '6:00 PM',
|
||||
'venue' => array(
|
||||
'name' => 'North Field',
|
||||
),
|
||||
),
|
||||
'current' => array(
|
||||
'time' => '7:30 PM',
|
||||
'venue' => array(
|
||||
'name' => 'South Field',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertSame( 'Venue North Field -> South Field Time 6:00 PM -> 7:30 PM', $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Change notifications should expose before and after home/away team values.
|
||||
*/
|
||||
public function test_render_template_supports_before_after_teams() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = 'Home {{ changes.previous.home_team.name }} -> {{ changes.current.home_team.name }} Away {{ changes.previous.away_team.name }} -> {{ changes.current.away_team.name }}';
|
||||
$context = array(
|
||||
'changes' => array(
|
||||
'previous' => array(
|
||||
'home_team' => array(
|
||||
'name' => 'Old Home',
|
||||
),
|
||||
'away_team' => array(
|
||||
'name' => 'Old Away',
|
||||
),
|
||||
),
|
||||
'current' => array(
|
||||
'home_team' => array(
|
||||
'name' => 'New Home',
|
||||
),
|
||||
'away_team' => array(
|
||||
'name' => 'New Away',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertSame( 'Home Old Home -> New Home Away Old Away -> New Away', $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Change notifications should expose before and after status values.
|
||||
*/
|
||||
public function test_render_template_supports_before_after_status() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = 'Status {{ changes.previous.status }} -> {{ changes.current.status }}';
|
||||
$context = array(
|
||||
'changes' => array(
|
||||
'previous' => array(
|
||||
'status' => 'On time',
|
||||
),
|
||||
'current' => array(
|
||||
'status' => 'Postponed',
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertSame( 'Status On time -> Postponed', $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Schedule snapshots should treat status changes as meaningful changes.
|
||||
*/
|
||||
public function test_schedule_snapshot_signature_changes_when_status_changes() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$method = new ReflectionMethod( $webhooks, 'schedule_snapshots_match' );
|
||||
$method->setAccessible( true );
|
||||
|
||||
$left = array(
|
||||
'local_iso' => '2026-05-02T10:30:00-05:00',
|
||||
'gmt_iso' => '2026-05-02T15:30:00+00:00',
|
||||
'status' => 'On time',
|
||||
'venue' => array(
|
||||
'name' => 'Winnemac Park',
|
||||
),
|
||||
'teams' => array(
|
||||
array( 'name' => 'Hawks' ),
|
||||
array( 'name' => 'Electrons' ),
|
||||
),
|
||||
);
|
||||
$right = $left;
|
||||
$right['status'] = 'Canceled';
|
||||
|
||||
$this->assertFalse( $method->invoke( $webhooks, $left, $right ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Status snapshots should expose a display label and keep the raw SportsPress key.
|
||||
*/
|
||||
public function test_build_change_snapshot_normalizes_status_label_and_slug() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$method = new ReflectionMethod( $webhooks, 'build_change_snapshot' );
|
||||
$method->setAccessible( true );
|
||||
|
||||
$snapshot = $method->invoke( $webhooks, array(), array(), array(), 'cancelled' );
|
||||
|
||||
$this->assertSame( 'Canceled', $snapshot['status'] );
|
||||
$this->assertSame( 'cancelled', $snapshot['sp_status'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Conditionals should support simple comparisons and else branches.
|
||||
*/
|
||||
public function test_render_template_supports_conditionals() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = "{% if changes.previous.time != changes.current.time %}Time changed: {{ changes.previous.time }} -> {{ changes.current.time }}\n{% endif %}{% if changes.previous.field.name == changes.current.field.name %}Field unchanged{% else %}Field changed: {{ changes.previous.field.name }} -> {{ changes.current.field.name }}{% endif %}";
|
||||
$context = array(
|
||||
'changes' => array(
|
||||
'previous' => array(
|
||||
'time' => '6:00 PM',
|
||||
'field' => array(
|
||||
'name' => 'North Field',
|
||||
),
|
||||
),
|
||||
'current' => array(
|
||||
'time' => '7:30 PM',
|
||||
'field' => array(
|
||||
'name' => 'South Field',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertSame( "Time changed: 6:00 PM -> 7:30 PM\nField changed: North Field -> South Field", $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Truthy conditionals should render when the referenced value exists.
|
||||
*/
|
||||
public function test_render_template_supports_truthy_conditionals() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = '{% if changes.current.home_team.name %}Home: {{ changes.current.home_team.name }}{% endif %}';
|
||||
$context = array(
|
||||
'changes' => array(
|
||||
'current' => array(
|
||||
'home_team' => array(
|
||||
'name' => 'Home Team',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertSame( 'Home: Home Team', $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Team change conditionals should stay false when only schedule fields changed.
|
||||
*/
|
||||
public function test_team_conditionals_do_not_fire_for_schedule_only_changes() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = '{% if changes.previous.away_team.name != changes.current.away_team.name %}Away changed{% else %}Away unchanged{% endif %}';
|
||||
$context = array(
|
||||
'changes' => array(
|
||||
'previous' => array(
|
||||
'time' => '6:00 PM',
|
||||
'away_team' => array(
|
||||
'name' => 'Away Team',
|
||||
),
|
||||
),
|
||||
'current' => array(
|
||||
'time' => '7:30 PM',
|
||||
'away_team' => array(
|
||||
'name' => 'Away Team',
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertSame( 'Away unchanged', $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Conditionals should support simple or expressions with else branches.
|
||||
*/
|
||||
public function test_render_template_supports_or_conditionals() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$template = '{% if event.status == "Postponed" or event.status == "Canceled" %}Delayed{% else %}Normal{% endif %}';
|
||||
$context = array(
|
||||
'event' => array(
|
||||
'status' => 'Canceled',
|
||||
),
|
||||
);
|
||||
|
||||
$rendered = $webhooks->render_template( $template, $context );
|
||||
|
||||
$this->assertSame( 'Delayed', $rendered );
|
||||
}
|
||||
|
||||
/**
|
||||
* Test webhook AJAX should honor the submitted row index.
|
||||
*/
|
||||
public function test_get_submitted_test_webhook_row_uses_matching_index() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$method = new ReflectionMethod( $webhooks, 'get_submitted_test_webhook_row' );
|
||||
$method->setAccessible( true );
|
||||
|
||||
$result = $method->invoke(
|
||||
$webhooks,
|
||||
array(
|
||||
'2' => array(
|
||||
'name' => 'Second Row',
|
||||
),
|
||||
),
|
||||
array(
|
||||
'2' => '123',
|
||||
)
|
||||
);
|
||||
|
||||
$this->assertSame( 'Second Row', $result['row']['name'] );
|
||||
$this->assertSame( 123, $result['event_id'] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Sanitization should keep only complete provider-specific webhook rows.
|
||||
*/
|
||||
public function test_sanitize_settings_keeps_only_valid_webhooks() {
|
||||
$webhooks = Tony_Sportspress_Webhooks::instance();
|
||||
$sanitized = $webhooks->sanitize_settings(
|
||||
array(
|
||||
'webhooks' => array(
|
||||
array(
|
||||
'name' => 'Results',
|
||||
'enabled' => '1',
|
||||
'provider' => 'google_chat',
|
||||
'url' => 'https://chat.googleapis.com/v1/spaces/AAA/messages?key=test&token=test',
|
||||
'triggers' => array( 'event_results_updated' ),
|
||||
'template' => '{"summary":"{{ results.summary }}"}',
|
||||
),
|
||||
array(
|
||||
'name' => 'Invalid',
|
||||
'enabled' => '1',
|
||||
'provider' => 'groupme_bot',
|
||||
'url' => 'invalid bot id',
|
||||
'triggers' => array( 'event_datetime_changed' ),
|
||||
'template' => 'ignored',
|
||||
),
|
||||
array(
|
||||
'name' => 'Missing trigger',
|
||||
'enabled' => '1',
|
||||
'provider' => 'generic_json',
|
||||
'url' => 'https://example.com/missing-trigger',
|
||||
'template' => 'ignored',
|
||||
),
|
||||
),
|
||||
)
|
||||
);
|
||||
|
||||
$this->assertCount( 1, $sanitized['webhooks'] );
|
||||
$this->assertSame( 'Results', $sanitized['webhooks'][0]['name'] );
|
||||
$this->assertSame( 'google_chat', $sanitized['webhooks'][0]['provider'] );
|
||||
$this->assertSame( 'https://chat.googleapis.com/v1/spaces/AAA/messages?key=test&token=test', $sanitized['webhooks'][0]['url'] );
|
||||
$this->assertSame( array( 'event_results_updated' ), $sanitized['webhooks'][0]['triggers'] );
|
||||
}
|
||||
}
|
||||
@@ -7,13 +7,14 @@
|
||||
* Author URI: https://github.com/anthonyscorrea/
|
||||
* Text Domain: tonys-sportspress-enhancements
|
||||
* Domain Path: /languages
|
||||
* Version: 0.1.6
|
||||
* Update URI: https://github.com/anthonyscorrea/tonys-sportspress-enhancements
|
||||
* Version: 0.1.15
|
||||
*
|
||||
* @package Tonys_Sportspress_Enhancements
|
||||
*/
|
||||
|
||||
if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_VERSION' ) ) {
|
||||
define( 'TONY_SPORTSPRESS_ENHANCEMENTS_VERSION', '0.1.6' );
|
||||
define( 'TONY_SPORTSPRESS_ENHANCEMENTS_VERSION', '0.1.15' );
|
||||
}
|
||||
|
||||
if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_FILE' ) ) {
|
||||
@@ -28,14 +29,25 @@ if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_URL' ) ) {
|
||||
define( 'TONY_SPORTSPRESS_ENHANCEMENTS_URL', plugin_dir_url( __FILE__ ) );
|
||||
}
|
||||
|
||||
if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_PLUGIN_BASENAME' ) ) {
|
||||
define( 'TONY_SPORTSPRESS_ENHANCEMENTS_PLUGIN_BASENAME', plugin_basename( __FILE__ ) );
|
||||
}
|
||||
|
||||
// Include other files here
|
||||
require_once plugin_dir_path(__FILE__) . 'includes/sp-github-updater.php';
|
||||
require_once plugin_dir_path(__FILE__) . 'includes/sp-officials-manager-role.php';
|
||||
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-webhooks.php';
|
||||
require_once plugin_dir_path(__FILE__) . 'includes/sp-schedule-exporter.php';
|
||||
require_once plugin_dir_path(__FILE__) . 'includes/sp-venue-meta.php';
|
||||
|
||||
register_activation_hook( __FILE__, 'tony_sportspress_sync_officials_manager_roles' );
|
||||
|
||||
Reference in New Issue
Block a user