Compare commits
9 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
b2d4c240ad
|
|||
|
6837907aa9
|
|||
|
bfc74fcab6
|
|||
|
894109ae6c
|
|||
|
569d11ac1a
|
|||
|
e019bf2061
|
|||
|
d2ff863ca5
|
|||
|
785f617134
|
|||
|
910cc8905e
|
503
assets/print-calendar.css
Normal file
503
assets/print-calendar.css
Normal file
@@ -0,0 +1,503 @@
|
|||||||
|
@import url("https://fonts.googleapis.com/css2?family=Nunito+Sans:opsz,wdth,wght@6..12,75..125,600..900&family=Open+Sans:wght@400..800&display=swap");
|
||||||
|
|
||||||
|
:root {
|
||||||
|
--pc-font-body: "Open Sans", Arial, sans-serif;
|
||||||
|
--pc-font-display: "Nunito Sans", "Open Sans", Arial, sans-serif;
|
||||||
|
--pc-page-padding: 0.45in;
|
||||||
|
--pc-gap: 8px;
|
||||||
|
--pc-border: #d5dde6;
|
||||||
|
--pc-preview-bg: #d9dce3;
|
||||||
|
--pc-team-logo-size: clamp(40px, 5vw, 56px);
|
||||||
|
--pc-brand-logo-height: clamp(40px, 5vw, 56px);
|
||||||
|
--pc-brand-logo-max-width: 34%;
|
||||||
|
--pc-muted-day-bg: rgba(243, 245, 248, 0.72);
|
||||||
|
--pc-empty-day-bg: rgba(233, 237, 242, 0.72);
|
||||||
|
--pc-event-logo-height: clamp(28px, 100%, 74px);
|
||||||
|
--pc-qr-size: 50px;
|
||||||
|
--pc-qr-offset: calc(var(--pc-qr-size) + var(--pc-gap));
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
margin: 0;
|
||||||
|
padding: 20px;
|
||||||
|
font-family: var(--pc-font-body);
|
||||||
|
color: #111;
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.print-shell {
|
||||||
|
margin: 0;
|
||||||
|
width: calc(100% / var(--sheet-scale));
|
||||||
|
background: #fff;
|
||||||
|
transform-origin: top left;
|
||||||
|
transform: scale(var(--sheet-scale));
|
||||||
|
}
|
||||||
|
|
||||||
|
.print-page {
|
||||||
|
padding: var(--pc-page-padding);
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen {
|
||||||
|
body.print-preview {
|
||||||
|
min-height: 100vh;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: flex-start;
|
||||||
|
padding: 24px;
|
||||||
|
background: var(--pc-preview-bg);
|
||||||
|
}
|
||||||
|
|
||||||
|
body.print-preview .print-shell {
|
||||||
|
box-shadow: 0 10px 30px rgba(17, 24, 39, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
body.print-preview .print-shell.letter {
|
||||||
|
width: 8.5in;
|
||||||
|
min-height: 11in;
|
||||||
|
}
|
||||||
|
|
||||||
|
body.print-preview .print-shell.ledger {
|
||||||
|
width: 11in;
|
||||||
|
min-height: 17in;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
gap: 12px;
|
||||||
|
margin: 0 0 12px;
|
||||||
|
padding-bottom: 8px;
|
||||||
|
border-bottom: 2px solid var(--team-accent, #b61f0f);
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-brand {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 10px;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.team-logo,
|
||||||
|
.league-logo {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
flex: 0 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.team-logo {
|
||||||
|
width: var(--pc-team-logo-size);
|
||||||
|
height: var(--pc-team-logo-size);
|
||||||
|
}
|
||||||
|
|
||||||
|
.team-logo img,
|
||||||
|
.team-logo-img {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
display: block;
|
||||||
|
object-fit: contain;
|
||||||
|
}
|
||||||
|
|
||||||
|
.league-logo {
|
||||||
|
height: var(--pc-brand-logo-height);
|
||||||
|
max-width: var(--pc-brand-logo-max-width);
|
||||||
|
}
|
||||||
|
|
||||||
|
.league-logo a,
|
||||||
|
.league-logo .custom-logo-link,
|
||||||
|
.league-logo .wp-block-site-logo__link {
|
||||||
|
height: 100%;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.league-logo img,
|
||||||
|
.league-logo-img,
|
||||||
|
.league-logo .custom-logo,
|
||||||
|
.league-logo .wp-block-site-logo__image {
|
||||||
|
width: auto;
|
||||||
|
height: 100%;
|
||||||
|
max-width: 100%;
|
||||||
|
display: block;
|
||||||
|
object-fit: contain;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-copy {
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
margin: 0;
|
||||||
|
font-family: var(--pc-font-display);
|
||||||
|
font-size: 28px;
|
||||||
|
font-weight: 800;
|
||||||
|
font-variation-settings: "wdth" 92, "wght" 800;
|
||||||
|
line-height: 1.1;
|
||||||
|
color: var(--team-ink, #111);
|
||||||
|
}
|
||||||
|
|
||||||
|
.meta {
|
||||||
|
margin: 0;
|
||||||
|
font-size: 13px;
|
||||||
|
color: var(--team-muted-ink, #333);
|
||||||
|
}
|
||||||
|
|
||||||
|
.legend {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: var(--pc-gap);
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.legend-item {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
font-family: var(--pc-font-display);
|
||||||
|
font-variation-settings: "wdth" 70, "wght" 700;
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sheet-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(var(--month-columns), minmax(0, 1fr));
|
||||||
|
gap: var(--pc-gap);
|
||||||
|
align-items: start;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sheet-grid > .month:nth-child(3):last-child {
|
||||||
|
grid-column: 1 / -1;
|
||||||
|
width: calc((100% - var(--pc-gap)) / 2);
|
||||||
|
max-width: calc((100% - var(--pc-gap)) / 2);
|
||||||
|
justify-self: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.month {
|
||||||
|
margin: 0;
|
||||||
|
break-inside: avoid;
|
||||||
|
page-break-inside: avoid;
|
||||||
|
}
|
||||||
|
|
||||||
|
.month-title {
|
||||||
|
margin: 0;
|
||||||
|
padding: 2px;
|
||||||
|
font-family: var(--pc-font-display);
|
||||||
|
font-size: calc(20px * var(--month-font-scale));
|
||||||
|
font-weight: 800;
|
||||||
|
font-variation-settings: "wdth" 80, "wght" 800;
|
||||||
|
line-height: 1.1;
|
||||||
|
letter-spacing: 0.01em;
|
||||||
|
text-align: center;
|
||||||
|
text-transform: uppercase;
|
||||||
|
background: var(--team-primary);
|
||||||
|
color: var(--team-on-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.dow,
|
||||||
|
.grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(7, minmax(0, 1fr));
|
||||||
|
}
|
||||||
|
|
||||||
|
.dow span {
|
||||||
|
display: block;
|
||||||
|
padding: 2px 1px;
|
||||||
|
font-size: calc(10px * var(--month-font-scale));
|
||||||
|
font-weight: 700;
|
||||||
|
text-align: center;
|
||||||
|
text-transform: uppercase;
|
||||||
|
border-bottom: 1px solid var(--pc-border);
|
||||||
|
background: var(--team-link-color);
|
||||||
|
color: var(--team-on-link-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.day {
|
||||||
|
--corner-badge-size: calc(15px * var(--month-font-scale));
|
||||||
|
--corner-badge-offset: 0px;
|
||||||
|
position: relative;
|
||||||
|
overflow: hidden;
|
||||||
|
aspect-ratio: var(--day-aspect);
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.day.muted {
|
||||||
|
background: var(--pc-muted-day-bg);
|
||||||
|
color: #9aa5b1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.day.no-events {
|
||||||
|
background: var(--pc-empty-day-bg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.day-num {
|
||||||
|
position: absolute;
|
||||||
|
top: var(--corner-badge-offset);
|
||||||
|
left: var(--corner-badge-offset);
|
||||||
|
z-index: 4;
|
||||||
|
width: var(--corner-badge-size);
|
||||||
|
height: var(--corner-badge-size);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
font-family: var(--pc-font-display);
|
||||||
|
font-size: calc(12px * var(--month-font-scale));
|
||||||
|
font-weight: 800;
|
||||||
|
font-variation-settings: "wdth" 86, "wght" 800;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.day.has-events .day-num {
|
||||||
|
color: var(--day-num-color, #fff);
|
||||||
|
}
|
||||||
|
|
||||||
|
.events-stack {
|
||||||
|
height: 100%;
|
||||||
|
display: grid;
|
||||||
|
grid-template-rows: repeat(var(--event-count), minmax(0, 1fr));
|
||||||
|
}
|
||||||
|
|
||||||
|
.event {
|
||||||
|
--event-top-band: calc(var(--corner-badge-size, 11px) + var(--corner-badge-offset, 2px));
|
||||||
|
--event-bottom-band: 26px;
|
||||||
|
--event-logo-height: var(--pc-event-logo-height);
|
||||||
|
position: relative;
|
||||||
|
background: var(--event-bg, var(--team-primary, #1b76d1));
|
||||||
|
color: var(--event-fg, #fff);
|
||||||
|
}
|
||||||
|
|
||||||
|
.event.h {
|
||||||
|
--event-bg: var(--team-primary, #1b76d1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.event.a {
|
||||||
|
--event-bg: var(--team-link-color, var(--team-secondary, #8b3f1f));
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-center {
|
||||||
|
position: absolute;
|
||||||
|
top: var(--event-top-band);
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: var(--event-bottom-band);
|
||||||
|
z-index: 1;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
overflow: hidden;
|
||||||
|
width: 100%;
|
||||||
|
font-size: calc(10px * var(--month-font-scale));
|
||||||
|
font-weight: 700;
|
||||||
|
line-height: 1;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-center img {
|
||||||
|
width: auto;
|
||||||
|
height: var(--event-logo-height);
|
||||||
|
max-width: 100%;
|
||||||
|
max-height: 100%;
|
||||||
|
display: block;
|
||||||
|
object-fit: contain;
|
||||||
|
object-position: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-name {
|
||||||
|
width: 100%;
|
||||||
|
max-width: 100%;
|
||||||
|
max-height: 100%;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: clip;
|
||||||
|
white-space: normal;
|
||||||
|
word-break: normal;
|
||||||
|
overflow-wrap: normal;
|
||||||
|
hyphens: none;
|
||||||
|
line-height: 1.05;
|
||||||
|
font-weight: 700;
|
||||||
|
opacity: 0.85;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event.no-logo .event-name {
|
||||||
|
font-family: var(--pc-font-display);
|
||||||
|
font-weight: 700;
|
||||||
|
font-variation-settings: "wdth" 30, "wght" 700;
|
||||||
|
letter-spacing: -0.01em;
|
||||||
|
text-transform: uppercase;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ha-flag {
|
||||||
|
position: absolute;
|
||||||
|
top: var(--corner-badge-offset, 2px);
|
||||||
|
right: var(--corner-badge-offset, 2px);
|
||||||
|
z-index: 3;
|
||||||
|
width: var(--corner-badge-size, 11px);
|
||||||
|
height: var(--corner-badge-size, 11px);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
font-family: var(--pc-font-display);
|
||||||
|
font-size: calc(10px * var(--month-font-scale));
|
||||||
|
font-weight: 900;
|
||||||
|
font-variation-settings: "wdth" 84, "wght" 900;
|
||||||
|
line-height: 1;
|
||||||
|
letter-spacing: -0.01em;
|
||||||
|
background: #111;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event.a .ha-flag {
|
||||||
|
background: #fff;
|
||||||
|
color: #111;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-meta {
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 2px;
|
||||||
|
z-index: 3;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-end;
|
||||||
|
gap: 1px;
|
||||||
|
padding: 0 2px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-time {
|
||||||
|
order: 1;
|
||||||
|
max-width: 100%;
|
||||||
|
font-size: calc(12px * var(--month-font-scale));
|
||||||
|
font-weight: 800;
|
||||||
|
line-height: 1;
|
||||||
|
text-transform: uppercase;
|
||||||
|
text-align: center;
|
||||||
|
color: currentColor;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: clip;
|
||||||
|
opacity: 0.95;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-venue {
|
||||||
|
order: 2;
|
||||||
|
max-width: 100%;
|
||||||
|
font-size: calc(8px * var(--month-font-scale));
|
||||||
|
font-weight: 700;
|
||||||
|
line-height: 1;
|
||||||
|
text-align: center;
|
||||||
|
text-transform: uppercase;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: clip;
|
||||||
|
opacity: 0.88;
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty {
|
||||||
|
padding: 16px;
|
||||||
|
border: 2px dashed #c8d2de;
|
||||||
|
background: #f8fafc;
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-meta {
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-start;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: var(--pc-gap);
|
||||||
|
margin-top: 10px;
|
||||||
|
padding-top: 8px;
|
||||||
|
border-top: 1px solid var(--pc-border);
|
||||||
|
}
|
||||||
|
|
||||||
|
.legend-bottom {
|
||||||
|
--month-width: calc((100% - ((var(--month-columns) - 1) * var(--pc-gap))) / var(--month-columns));
|
||||||
|
flex: 0 0 var(--month-width);
|
||||||
|
width: var(--month-width);
|
||||||
|
max-width: var(--month-width);
|
||||||
|
align-content: flex-start;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.legend-bottom .legend-item {
|
||||||
|
padding: 3px 6px;
|
||||||
|
font-size: 10px;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-qr {
|
||||||
|
position: relative;
|
||||||
|
flex: 0 0 auto;
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-end;
|
||||||
|
justify-content: flex-end;
|
||||||
|
gap: var(--pc-gap);
|
||||||
|
min-height: var(--pc-qr-size);
|
||||||
|
padding-right: var(--pc-qr-offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-qr-copy {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: flex-end;
|
||||||
|
gap: 2px;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-qr-label {
|
||||||
|
font-size: 9px;
|
||||||
|
font-weight: 700;
|
||||||
|
letter-spacing: 0.02em;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: var(--team-muted-ink, #333);
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-qr-link {
|
||||||
|
font-size: 10px;
|
||||||
|
font-weight: 700;
|
||||||
|
text-decoration: none;
|
||||||
|
color: var(--team-ink, #111);
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-qr-image {
|
||||||
|
position: absolute;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
width: var(--pc-qr-size);
|
||||||
|
height: var(--pc-qr-size);
|
||||||
|
display: block;
|
||||||
|
border: 1px solid var(--pc-border);
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media print {
|
||||||
|
body,
|
||||||
|
body.print-preview {
|
||||||
|
padding: 0;
|
||||||
|
display: block;
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.print-shell,
|
||||||
|
body.print-preview .print-shell,
|
||||||
|
body.print-preview .print-shell.letter,
|
||||||
|
body.print-preview .print-shell.ledger {
|
||||||
|
width: auto;
|
||||||
|
min-height: auto;
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.print-page {
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header {
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
font-size: calc(26px * var(--month-font-scale));
|
||||||
|
}
|
||||||
|
}
|
||||||
27
assets/schedule-exporter-block.js
Normal file
27
assets/schedule-exporter-block.js
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
( function ( blocks, blockEditor, element, i18n ) {
|
||||||
|
var el = element.createElement;
|
||||||
|
var useBlockProps = blockEditor.useBlockProps;
|
||||||
|
var __ = i18n.__;
|
||||||
|
|
||||||
|
blocks.registerBlockType( 'tse/schedule-exporter', {
|
||||||
|
edit: function () {
|
||||||
|
var blockProps = useBlockProps( {
|
||||||
|
className: 'tse-schedule-exporter-block-placeholder',
|
||||||
|
} );
|
||||||
|
|
||||||
|
return el(
|
||||||
|
'div',
|
||||||
|
blockProps,
|
||||||
|
el( 'strong', null, __( 'Schedule Exporter', 'tonys-sportspress-enhancements' ) ),
|
||||||
|
el(
|
||||||
|
'p',
|
||||||
|
null,
|
||||||
|
__( 'This block renders the public schedule exporter on the frontend.', 'tonys-sportspress-enhancements' )
|
||||||
|
)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
save: function () {
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
} );
|
||||||
|
} )( window.wp.blocks, window.wp.blockEditor, window.wp.element, window.wp.i18n );
|
||||||
@@ -10,256 +10,8 @@ if ( ! defined( 'ABSPATH' ) ) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Register the SportsPress calendar CSV feed endpoint.
|
* SportsPress event CSV importer tools.
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
*/
|
*/
|
||||||
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.
|
* CSV headers recognized by this importer.
|
||||||
|
|||||||
922
includes/sp-event-export.php
Normal file
922
includes/sp-event-export.php
Normal file
@@ -0,0 +1,922 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* Standalone SportsPress schedule feeds.
|
||||||
|
*
|
||||||
|
* @package Tonys_Sportspress_Enhancements
|
||||||
|
*/
|
||||||
|
|
||||||
|
if ( ! defined( 'ABSPATH' ) ) {
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register the standalone SportsPress CSV feed endpoint.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_register_feed() {
|
||||||
|
add_feed( 'sp-csv', 'tse_sp_event_export_render_feed' );
|
||||||
|
add_feed( 'sp-ics', 'tse_sp_event_export_render_ical_feed' );
|
||||||
|
}
|
||||||
|
add_action( 'init', 'tse_sp_event_export_register_feed', 20 );
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get available export formats.
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_formats() {
|
||||||
|
return array(
|
||||||
|
'matchup' => array(
|
||||||
|
'label' => __( 'Matchup', 'tonys-sportspress-enhancements' ),
|
||||||
|
'description' => __( 'Date, time, away team, home team, and field columns.', 'tonys-sportspress-enhancements' ),
|
||||||
|
),
|
||||||
|
'team' => array(
|
||||||
|
'label' => __( 'Team', 'tonys-sportspress-enhancements' ),
|
||||||
|
'description' => __( 'Team-centric schedule rows with opponent and home/away columns.', 'tonys-sportspress-enhancements' ),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get available export columns grouped by format.
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_column_definitions() {
|
||||||
|
return array(
|
||||||
|
'matchup' => array(
|
||||||
|
'date' => __( 'Date', 'tonys-sportspress-enhancements' ),
|
||||||
|
'time' => __( 'Time', 'tonys-sportspress-enhancements' ),
|
||||||
|
'season' => __( 'Season', 'tonys-sportspress-enhancements' ),
|
||||||
|
'league' => __( 'League', 'tonys-sportspress-enhancements' ),
|
||||||
|
'away_team' => __( 'Away Team', 'tonys-sportspress-enhancements' ),
|
||||||
|
'home_team' => __( 'Home Team', 'tonys-sportspress-enhancements' ),
|
||||||
|
'field_name' => __( 'Field Name', 'tonys-sportspress-enhancements' ),
|
||||||
|
'officials' => __( 'Officials', 'tonys-sportspress-enhancements' ),
|
||||||
|
),
|
||||||
|
'team' => array(
|
||||||
|
'label' => __( 'Extra Label', 'tonys-sportspress-enhancements' ),
|
||||||
|
'date' => __( 'Date', 'tonys-sportspress-enhancements' ),
|
||||||
|
'time' => __( 'Time', 'tonys-sportspress-enhancements' ),
|
||||||
|
'season' => __( 'Season', 'tonys-sportspress-enhancements' ),
|
||||||
|
'league' => __( 'League', 'tonys-sportspress-enhancements' ),
|
||||||
|
'team_name' => __( 'Team', 'tonys-sportspress-enhancements' ),
|
||||||
|
'opponent_name' => __( 'Opponent', 'tonys-sportspress-enhancements' ),
|
||||||
|
'location_flag' => __( 'Home/Away', 'tonys-sportspress-enhancements' ),
|
||||||
|
'field_name' => __( 'Field Name', 'tonys-sportspress-enhancements' ),
|
||||||
|
'field_abbreviation' => __( 'Field Abbreviation', 'tonys-sportspress-enhancements' ),
|
||||||
|
'field_short_name' => __( 'Field Short Name', 'tonys-sportspress-enhancements' ),
|
||||||
|
'officials' => __( 'Officials', 'tonys-sportspress-enhancements' ),
|
||||||
|
'home_team' => __( 'Home Team', 'tonys-sportspress-enhancements' ),
|
||||||
|
'away_team' => __( 'Away Team', 'tonys-sportspress-enhancements' ),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get default columns for an export format.
|
||||||
|
*
|
||||||
|
* @param string $format Export format.
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_default_columns( $format ) {
|
||||||
|
$defaults = array(
|
||||||
|
'matchup' => array( 'date', 'time', 'season', 'league', 'away_team', 'home_team', 'field_name' ),
|
||||||
|
'team' => array( 'label', 'date', 'time', 'season', 'league', 'opponent_name', 'location_flag', 'field_name' ),
|
||||||
|
);
|
||||||
|
|
||||||
|
return isset( $defaults[ $format ] ) ? $defaults[ $format ] : $defaults['matchup'];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sanitize an export format.
|
||||||
|
*
|
||||||
|
* @param string $format Raw format.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_sanitize_format( $format ) {
|
||||||
|
$format = sanitize_key( (string) $format );
|
||||||
|
$formats = tse_sp_event_export_get_formats();
|
||||||
|
|
||||||
|
return isset( $formats[ $format ] ) ? $format : 'matchup';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sanitize requested columns for an export format.
|
||||||
|
*
|
||||||
|
* @param string $format Export format.
|
||||||
|
* @param string|array $columns Requested columns.
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_sanitize_columns( $format, $columns ) {
|
||||||
|
$definitions = tse_sp_event_export_get_column_definitions();
|
||||||
|
$available = isset( $definitions[ $format ] ) ? $definitions[ $format ] : array();
|
||||||
|
|
||||||
|
if ( ! is_array( $columns ) ) {
|
||||||
|
$columns = explode( ',', (string) $columns );
|
||||||
|
}
|
||||||
|
|
||||||
|
$sanitized = array();
|
||||||
|
|
||||||
|
foreach ( $columns as $column ) {
|
||||||
|
$key = sanitize_key( (string) $column );
|
||||||
|
if ( '' === $key || ! isset( $available[ $key ] ) || in_array( $key, $sanitized, true ) ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$sanitized[] = $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( empty( $sanitized ) ) {
|
||||||
|
return tse_sp_event_export_get_default_columns( $format );
|
||||||
|
}
|
||||||
|
|
||||||
|
return $sanitized;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normalize one or more numeric IDs from a request value.
|
||||||
|
*
|
||||||
|
* Accepts scalars, arrays, and comma-delimited strings.
|
||||||
|
*
|
||||||
|
* @param mixed $value Raw request value.
|
||||||
|
* @return int[]
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_parse_id_list( $value ) {
|
||||||
|
if ( is_array( $value ) ) {
|
||||||
|
$raw_values = $value;
|
||||||
|
} else {
|
||||||
|
$raw_values = explode( ',', (string) $value );
|
||||||
|
}
|
||||||
|
|
||||||
|
$ids = array();
|
||||||
|
|
||||||
|
foreach ( $raw_values as $raw_value ) {
|
||||||
|
$id = absint( trim( (string) $raw_value ) );
|
||||||
|
if ( $id > 0 ) {
|
||||||
|
$ids[] = $id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$ids = array_values( array_unique( $ids ) );
|
||||||
|
|
||||||
|
return $ids;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normalize export request arguments.
|
||||||
|
*
|
||||||
|
* @param array|null $source Optional input source.
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_normalize_request_args( $source = null ) {
|
||||||
|
$source = is_array( $source ) ? $source : $_GET;
|
||||||
|
$format = isset( $source['format'] ) ? tse_sp_event_export_sanitize_format( wp_unslash( $source['format'] ) ) : 'matchup';
|
||||||
|
$team_ids = isset( $source['team_id'] ) ? tse_sp_event_export_parse_id_list( wp_unslash( $source['team_id'] ) ) : array();
|
||||||
|
$season_ids = isset( $source['season_id'] ) ? tse_sp_event_export_parse_id_list( wp_unslash( $source['season_id'] ) ) : array();
|
||||||
|
$league_ids = isset( $source['league_id'] ) ? tse_sp_event_export_parse_id_list( wp_unslash( $source['league_id'] ) ) : array();
|
||||||
|
$field_ids = isset( $source['field_id'] ) ? tse_sp_event_export_parse_id_list( wp_unslash( $source['field_id'] ) ) : array();
|
||||||
|
|
||||||
|
return array(
|
||||||
|
'team_id' => isset( $team_ids[0] ) ? $team_ids[0] : 0,
|
||||||
|
'team_ids' => $team_ids,
|
||||||
|
'season_id' => isset( $season_ids[0] ) ? $season_ids[0] : 0,
|
||||||
|
'season_ids'=> $season_ids,
|
||||||
|
'league_id' => isset( $league_ids[0] ) ? $league_ids[0] : 0,
|
||||||
|
'league_ids'=> $league_ids,
|
||||||
|
'field_id' => isset( $field_ids[0] ) ? $field_ids[0] : 0,
|
||||||
|
'field_ids' => $field_ids,
|
||||||
|
'format' => $format,
|
||||||
|
'columns' => isset( $source['columns'] ) ? tse_sp_event_export_sanitize_columns( $format, wp_unslash( $source['columns'] ) ) : tse_sp_event_export_get_default_columns( $format ),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validate export filters for the requested format.
|
||||||
|
*
|
||||||
|
* @param array $filters Export filters.
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_validate_filters( $filters ) {
|
||||||
|
$format = tse_sp_event_export_sanitize_format( isset( $filters['format'] ) ? $filters['format'] : 'matchup' );
|
||||||
|
$team_ids = isset( $filters['team_ids'] ) && is_array( $filters['team_ids'] ) ? $filters['team_ids'] : array();
|
||||||
|
|
||||||
|
if ( 'team' !== $format ) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( empty( $team_ids ) ) {
|
||||||
|
wp_die( esc_html__( 'Team format requires a team filter.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 400 ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( count( $team_ids ) > 1 ) {
|
||||||
|
wp_die( esc_html__( 'Team format does not support multiple teams.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 400 ) );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Query matching event posts for export.
|
||||||
|
*
|
||||||
|
* @param array $filters Export filters.
|
||||||
|
* @return WP_Post[]
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_query_posts( $filters ) {
|
||||||
|
$team_ids = isset( $filters['team_ids'] ) && is_array( $filters['team_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['team_ids'] ) ) ) : array();
|
||||||
|
$season_ids = isset( $filters['season_ids'] ) && is_array( $filters['season_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['season_ids'] ) ) ) : array();
|
||||||
|
$league_ids = isset( $filters['league_ids'] ) && is_array( $filters['league_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['league_ids'] ) ) ) : array();
|
||||||
|
$field_ids = isset( $filters['field_ids'] ) && is_array( $filters['field_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['field_ids'] ) ) ) : array();
|
||||||
|
|
||||||
|
$args = array(
|
||||||
|
'post_type' => 'sp_event',
|
||||||
|
'post_status' => array( 'publish', 'future' ),
|
||||||
|
'posts_per_page' => -1,
|
||||||
|
'orderby' => 'date',
|
||||||
|
'order' => 'ASC',
|
||||||
|
'no_found_rows' => true,
|
||||||
|
);
|
||||||
|
|
||||||
|
if ( ! empty( $team_ids ) ) {
|
||||||
|
$args['meta_query'] = array(
|
||||||
|
array(
|
||||||
|
'key' => 'sp_team',
|
||||||
|
'value' => array_map( 'strval', $team_ids ),
|
||||||
|
'compare' => 'IN',
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$tax_query = array();
|
||||||
|
|
||||||
|
if ( ! empty( $season_ids ) ) {
|
||||||
|
$tax_query[] = array(
|
||||||
|
'taxonomy' => 'sp_season',
|
||||||
|
'field' => 'term_id',
|
||||||
|
'terms' => $season_ids,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ! empty( $league_ids ) ) {
|
||||||
|
$tax_query[] = array(
|
||||||
|
'taxonomy' => 'sp_league',
|
||||||
|
'field' => 'term_id',
|
||||||
|
'terms' => $league_ids,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ! empty( $field_ids ) ) {
|
||||||
|
$tax_query[] = array(
|
||||||
|
'taxonomy' => 'sp_venue',
|
||||||
|
'field' => 'term_id',
|
||||||
|
'terms' => $field_ids,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ! empty( $tax_query ) ) {
|
||||||
|
if ( count( $tax_query ) > 1 ) {
|
||||||
|
$tax_query['relation'] = 'AND';
|
||||||
|
}
|
||||||
|
|
||||||
|
$args['tax_query'] = $tax_query;
|
||||||
|
}
|
||||||
|
|
||||||
|
$query = new WP_Query( $args );
|
||||||
|
|
||||||
|
return is_array( $query->posts ) ? $query->posts : array();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Query matching schedule events for export.
|
||||||
|
*
|
||||||
|
* @param array $filters Export filters.
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_events( $filters ) {
|
||||||
|
$team_id = isset( $filters['team_id'] ) ? absint( $filters['team_id'] ) : 0;
|
||||||
|
$selected_ids = isset( $filters['team_ids'] ) && is_array( $filters['team_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['team_ids'] ) ) ) : array();
|
||||||
|
$query_posts = tse_sp_event_export_query_posts( $filters );
|
||||||
|
$events = array();
|
||||||
|
$team_name = $team_id > 0 ? get_the_title( $team_id ) : '';
|
||||||
|
|
||||||
|
foreach ( $query_posts as $event ) {
|
||||||
|
$event_id = $event instanceof WP_Post ? (int) $event->ID : 0;
|
||||||
|
if ( $event_id <= 0 ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$teams = array_values( array_unique( array_map( 'intval', get_post_meta( $event_id, 'sp_team', false ) ) ) );
|
||||||
|
if ( ! empty( $selected_ids ) && empty( array_intersect( $selected_ids, $teams ) ) ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$home_id = isset( $teams[0] ) ? (int) $teams[0] : 0;
|
||||||
|
$away_id = isset( $teams[1] ) ? (int) $teams[1] : 0;
|
||||||
|
$venue = tse_sp_event_export_get_primary_field( $event_id );
|
||||||
|
|
||||||
|
if ( $team_id > 0 ) {
|
||||||
|
$location_flag = $home_id === $team_id ? 'Home' : 'Away';
|
||||||
|
$opponent_id = $home_id === $team_id ? $away_id : $home_id;
|
||||||
|
} else {
|
||||||
|
$location_flag = '';
|
||||||
|
$opponent_id = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
$events[] = array(
|
||||||
|
'event_id' => $event_id,
|
||||||
|
'label' => '',
|
||||||
|
'date' => get_post_time( 'm/d/Y', false, $event_id, true ),
|
||||||
|
'time' => strtoupper( (string) ( function_exists( 'sp_get_time' ) ? sp_get_time( $event_id ) : get_post_time( get_option( 'time_format' ), false, $event_id, true ) ) ),
|
||||||
|
'team_name' => is_string( $team_name ) ? $team_name : '',
|
||||||
|
'opponent_name' => $opponent_id > 0 ? get_the_title( $opponent_id ) : '',
|
||||||
|
'location_flag' => $location_flag,
|
||||||
|
'home_team' => $home_id > 0 ? get_the_title( $home_id ) : '',
|
||||||
|
'away_team' => $away_id > 0 ? get_the_title( $away_id ) : '',
|
||||||
|
'field_name' => isset( $venue['name'] ) ? $venue['name'] : '',
|
||||||
|
'field_abbreviation' => isset( $venue['abbreviation'] ) ? $venue['abbreviation'] : '',
|
||||||
|
'field_short_name' => isset( $venue['short_name'] ) ? $venue['short_name'] : '',
|
||||||
|
'season' => tse_sp_event_export_get_event_term_names( $event_id, 'sp_season' ),
|
||||||
|
'league' => tse_sp_event_export_get_event_term_names( $event_id, 'sp_league' ),
|
||||||
|
'officials' => tse_sp_event_export_get_officials_value( $event_id ),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ( $events as $index => $event ) {
|
||||||
|
$events[ $index ]['label'] = sprintf( 'G#%02d', $index + 1 );
|
||||||
|
}
|
||||||
|
|
||||||
|
wp_reset_postdata();
|
||||||
|
|
||||||
|
return $events;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get event term names as a semicolon-delimited string.
|
||||||
|
*
|
||||||
|
* @param int $event_id Event ID.
|
||||||
|
* @param string $taxonomy Taxonomy name.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_event_term_names( $event_id, $taxonomy ) {
|
||||||
|
$terms = get_the_terms( $event_id, $taxonomy );
|
||||||
|
|
||||||
|
if ( ! is_array( $terms ) || empty( $terms ) ) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
$names = array_values( array_filter( array_map( 'strval', wp_list_pluck( $terms, 'name' ) ) ) );
|
||||||
|
|
||||||
|
return implode( '; ', array_unique( $names ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get primary field metadata for an event.
|
||||||
|
*
|
||||||
|
* @param int $event_id Event ID.
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_primary_field( $event_id ) {
|
||||||
|
$venues = get_the_terms( $event_id, 'sp_venue' );
|
||||||
|
|
||||||
|
if ( ! is_array( $venues ) || ! isset( $venues[0] ) || ! $venues[0] instanceof WP_Term ) {
|
||||||
|
return array(
|
||||||
|
'name' => '',
|
||||||
|
'abbreviation' => '',
|
||||||
|
'short_name' => '',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$venue = $venues[0];
|
||||||
|
|
||||||
|
return array(
|
||||||
|
'name' => isset( $venue->name ) ? (string) $venue->name : '',
|
||||||
|
'abbreviation' => trim( (string) get_term_meta( $venue->term_id, 'tse_abbreviation', true ) ),
|
||||||
|
'short_name' => trim( (string) get_term_meta( $venue->term_id, 'tse_short_name', true ) ),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get event officials as a semicolon-delimited string.
|
||||||
|
*
|
||||||
|
* @param int $event_id Event ID.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_officials_value( $event_id ) {
|
||||||
|
$official_groups = get_post_meta( $event_id, 'sp_officials', true );
|
||||||
|
|
||||||
|
if ( ! is_array( $official_groups ) || empty( $official_groups ) ) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
$official_names = array();
|
||||||
|
|
||||||
|
foreach ( $official_groups as $official_ids ) {
|
||||||
|
if ( ! is_array( $official_ids ) ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ( $official_ids as $official_id ) {
|
||||||
|
$official_id = absint( $official_id );
|
||||||
|
if ( $official_id <= 0 || 'sp_official' !== get_post_type( $official_id ) ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$name = get_the_title( $official_id );
|
||||||
|
if ( '' === $name ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$official_names[ $official_id ] = $name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( empty( $official_names ) ) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return implode( '; ', array_values( $official_names ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Escape iCalendar text content.
|
||||||
|
*
|
||||||
|
* @param string $value Raw value.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_escape_ical_text( $value ) {
|
||||||
|
$value = html_entity_decode( wp_strip_all_tags( (string) $value ), ENT_QUOTES, get_bloginfo( 'charset' ) );
|
||||||
|
$value = str_replace( array( '\\', "\r\n", "\r", "\n", ',', ';' ), array( '\\\\', '\n', '\n', '\n', '\,', '\;' ), $value );
|
||||||
|
|
||||||
|
return $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fold an iCalendar content line.
|
||||||
|
*
|
||||||
|
* @param string $line Raw line.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_fold_ical_line( $line ) {
|
||||||
|
return wordwrap( (string) $line, 60, "\r\n\t", true );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the ICS summary for an event.
|
||||||
|
*
|
||||||
|
* For matchup format, this mirrors the SportsPress result/title behavior.
|
||||||
|
* For team format, this becomes a team-centric opponent summary.
|
||||||
|
*
|
||||||
|
* @param WP_Post $event Event post.
|
||||||
|
* @param array $filters Export filters.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_ical_summary( $event, $filters ) {
|
||||||
|
$format = tse_sp_event_export_sanitize_format( isset( $filters['format'] ) ? $filters['format'] : 'matchup' );
|
||||||
|
$team_ids = isset( $filters['team_ids'] ) && is_array( $filters['team_ids'] ) ? array_values( array_filter( array_map( 'absint', $filters['team_ids'] ) ) ) : array();
|
||||||
|
$team_id = isset( $team_ids[0] ) ? $team_ids[0] : 0;
|
||||||
|
|
||||||
|
if ( 'team' === $format && $team_id > 0 ) {
|
||||||
|
$teams = array_values( array_unique( array_map( 'intval', get_post_meta( $event->ID, 'sp_team', false ) ) ) );
|
||||||
|
$home_id = isset( $teams[0] ) ? (int) $teams[0] : 0;
|
||||||
|
$away_id = isset( $teams[1] ) ? (int) $teams[1] : 0;
|
||||||
|
|
||||||
|
if ( in_array( $team_id, $teams, true ) ) {
|
||||||
|
$is_home = $home_id === $team_id;
|
||||||
|
$opponent_id = $is_home ? $away_id : $home_id;
|
||||||
|
$opponent = $opponent_id > 0 ? get_the_title( $opponent_id ) : __( 'TBD', 'tonys-sportspress-enhancements' );
|
||||||
|
$summary = sprintf(
|
||||||
|
/* translators: 1: preposition, 2: opponent name. */
|
||||||
|
__( '%1$s %2$s', 'tonys-sportspress-enhancements' ),
|
||||||
|
$is_home ? 'vs' : 'at',
|
||||||
|
$opponent
|
||||||
|
);
|
||||||
|
|
||||||
|
return apply_filters( 'sportspress_ical_feed_summary', $summary, $event );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$main_result = get_option( 'sportspress_primary_result', null );
|
||||||
|
$results = array();
|
||||||
|
$teams = (array) get_post_meta( $event->ID, 'sp_team', false );
|
||||||
|
$teams = array_filter( array_unique( $teams ) );
|
||||||
|
|
||||||
|
if ( ! empty( $teams ) ) {
|
||||||
|
$event_results = get_post_meta( $event->ID, 'sp_results', true );
|
||||||
|
|
||||||
|
foreach ( $teams as $team_id ) {
|
||||||
|
$team_id = absint( $team_id );
|
||||||
|
if ( $team_id <= 0 ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$team = get_post( $team_id );
|
||||||
|
if ( ! $team instanceof WP_Post ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$team_results = is_array( $event_results ) && isset( $event_results[ $team_id ] ) ? $event_results[ $team_id ] : null;
|
||||||
|
|
||||||
|
if ( $main_result ) {
|
||||||
|
$team_result = is_array( $team_results ) && isset( $team_results[ $main_result ] ) ? $team_results[ $main_result ] : null;
|
||||||
|
} else {
|
||||||
|
if ( is_array( $team_results ) ) {
|
||||||
|
end( $team_results );
|
||||||
|
$team_result = prev( $team_results );
|
||||||
|
} else {
|
||||||
|
$team_result = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( null !== $team_result && '' !== (string) $team_result ) {
|
||||||
|
$results[] = get_the_title( $team_id ) . ' ' . $team_result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$summary = ! empty( $results ) ? implode( ' ', $results ) : $event->post_title;
|
||||||
|
|
||||||
|
$summary = preg_replace_callback(
|
||||||
|
'/(&#[0-9]+;)/',
|
||||||
|
static function( $matches ) {
|
||||||
|
return mb_convert_encoding( $matches[1], 'UTF-8', 'HTML-ENTITIES' );
|
||||||
|
},
|
||||||
|
$summary
|
||||||
|
);
|
||||||
|
|
||||||
|
return apply_filters( 'sportspress_ical_feed_summary', $summary, $event );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the ICS location payload for an event.
|
||||||
|
*
|
||||||
|
* @param WP_Post $event Event post.
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_ical_location_data( $event ) {
|
||||||
|
$location = '';
|
||||||
|
$geo = false;
|
||||||
|
$venues = get_the_terms( $event->ID, 'sp_venue' );
|
||||||
|
|
||||||
|
if ( ! is_array( $venues ) || empty( $venues ) ) {
|
||||||
|
return array(
|
||||||
|
'location' => $location,
|
||||||
|
'geo' => $geo,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$venue = reset( $venues );
|
||||||
|
$location = $venue->name;
|
||||||
|
$meta = get_option( 'taxonomy_' . $venue->term_id );
|
||||||
|
$address = is_array( $meta ) && isset( $meta['sp_address'] ) ? $meta['sp_address'] : false;
|
||||||
|
|
||||||
|
if ( false !== $address && '' !== (string) $address ) {
|
||||||
|
$location = $venue->name . ', ' . $address;
|
||||||
|
}
|
||||||
|
|
||||||
|
$latitude = is_array( $meta ) && isset( $meta['sp_latitude'] ) ? $meta['sp_latitude'] : false;
|
||||||
|
$longitude = is_array( $meta ) && isset( $meta['sp_longitude'] ) ? $meta['sp_longitude'] : false;
|
||||||
|
|
||||||
|
if ( false !== $latitude && false !== $longitude && '' !== (string) $latitude && '' !== (string) $longitude ) {
|
||||||
|
$geo = $latitude . ';' . $longitude;
|
||||||
|
}
|
||||||
|
|
||||||
|
return array(
|
||||||
|
'location' => (string) $location,
|
||||||
|
'geo' => $geo,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build iCalendar output for the requested filters.
|
||||||
|
*
|
||||||
|
* @param array $filters Export filters.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_build_ical_output( $filters ) {
|
||||||
|
$query_posts = tse_sp_event_export_query_posts( $filters );
|
||||||
|
$locale = substr( get_locale(), 0, 2 );
|
||||||
|
$timezone = sanitize_option( 'timezone_string', get_option( 'timezone_string' ) );
|
||||||
|
$url = tse_sp_event_export_get_feed_url( $filters, 'ics' );
|
||||||
|
$calendar = tse_sp_event_export_get_feed_title( $filters );
|
||||||
|
$output =
|
||||||
|
"BEGIN:VCALENDAR\r\n" .
|
||||||
|
"VERSION:2.0\r\n" .
|
||||||
|
'PRODID:-//ThemeBoy//SportsPress//' . strtoupper( $locale ) . "\r\n" .
|
||||||
|
"CALSCALE:GREGORIAN\r\n" .
|
||||||
|
"METHOD:PUBLISH\r\n" .
|
||||||
|
'URL:' . tse_sp_event_export_fold_ical_line( $url ) . "\r\n" .
|
||||||
|
'X-FROM-URL:' . tse_sp_event_export_fold_ical_line( $url ) . "\r\n" .
|
||||||
|
'NAME:' . tse_sp_event_export_escape_ical_text( $calendar ) . "\r\n" .
|
||||||
|
'X-WR-CALNAME:' . tse_sp_event_export_escape_ical_text( $calendar ) . "\r\n" .
|
||||||
|
'DESCRIPTION:' . tse_sp_event_export_escape_ical_text( $calendar ) . "\r\n" .
|
||||||
|
'X-WR-CALDESC:' . tse_sp_event_export_escape_ical_text( $calendar ) . "\r\n" .
|
||||||
|
"REFRESH-INTERVAL;VALUE=DURATION:PT2M\r\n" .
|
||||||
|
"X-PUBLISHED-TTL:PT2M\r\n" .
|
||||||
|
'TZID:' . $timezone . "\r\n" .
|
||||||
|
'X-WR-TIMEZONE:' . $timezone . "\r\n";
|
||||||
|
|
||||||
|
foreach ( $query_posts as $event ) {
|
||||||
|
if ( ! $event instanceof WP_Post ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$date_format = 'Ymd\THis';
|
||||||
|
$description = tse_sp_event_export_escape_ical_text( $event->post_content );
|
||||||
|
$summary = tse_sp_event_export_get_ical_summary( $event, $filters );
|
||||||
|
$minutes = get_post_meta( $event->ID, 'sp_minutes', true );
|
||||||
|
$minutes = '' === $minutes ? get_option( 'sportspress_event_minutes', 90 ) : $minutes;
|
||||||
|
$end = new DateTime( $event->post_date );
|
||||||
|
$end->add( new DateInterval( 'PT' . absint( $minutes ) . 'M' ) );
|
||||||
|
$location = tse_sp_event_export_get_ical_location_data( $event );
|
||||||
|
$event_url = get_permalink( $event );
|
||||||
|
|
||||||
|
$output .= "BEGIN:VEVENT\r\n";
|
||||||
|
$output .= tse_sp_event_export_fold_ical_line( 'SUMMARY:' . tse_sp_event_export_escape_ical_text( $summary ) ) . "\r\n";
|
||||||
|
$output .= 'UID:' . $event->ID . "\r\n";
|
||||||
|
$output .= "STATUS:CONFIRMED\r\n";
|
||||||
|
$output .= "DTSTAMP:19700101T000000\r\n";
|
||||||
|
$output .= 'DTSTART:' . mysql2date( $date_format, $event->post_date ) . "\r\n";
|
||||||
|
$output .= 'DTEND:' . $end->format( $date_format ) . "\r\n";
|
||||||
|
$output .= 'LAST-MODIFIED:' . mysql2date( $date_format, $event->post_modified_gmt ) . "\r\n";
|
||||||
|
|
||||||
|
if ( '' !== $description ) {
|
||||||
|
$output .= tse_sp_event_export_fold_ical_line( 'DESCRIPTION:' . $description ) . "\r\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( '' !== $location['location'] ) {
|
||||||
|
$output .= tse_sp_event_export_fold_ical_line( 'LOCATION:' . tse_sp_event_export_escape_ical_text( $location['location'] ) ) . "\r\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ! empty( $location['geo'] ) ) {
|
||||||
|
$output .= 'GEO:' . $location['geo'] . "\r\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( is_string( $event_url ) && '' !== $event_url ) {
|
||||||
|
$output .= tse_sp_event_export_fold_ical_line( 'URL:' . esc_url_raw( $event_url ) ) . "\r\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
$output .= "END:VEVENT\r\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
$output .= 'END:VCALENDAR';
|
||||||
|
|
||||||
|
return $output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stream ICS output.
|
||||||
|
*
|
||||||
|
* @param array $filters Export filters.
|
||||||
|
* @param array $args Optional render args.
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_stream_ical( $filters, $args = array() ) {
|
||||||
|
tse_sp_event_export_validate_filters( $filters );
|
||||||
|
|
||||||
|
$disposition = isset( $args['disposition'] ) ? sanitize_key( $args['disposition'] ) : 'inline';
|
||||||
|
$disposition = in_array( $disposition, array( 'inline', 'attachment' ), true ) ? $disposition : 'inline';
|
||||||
|
$output = tse_sp_event_export_build_ical_output( $filters );
|
||||||
|
$filename = tse_sp_event_export_build_filename( $filters ) . '.ics';
|
||||||
|
$etag = md5( $output );
|
||||||
|
|
||||||
|
header( 'Content-type: text/calendar; charset=utf-8' );
|
||||||
|
header( 'Etag:' . '"' . $etag . '"' );
|
||||||
|
header( 'Content-Disposition: ' . $disposition . '; filename=' . $filename );
|
||||||
|
|
||||||
|
echo $output; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a CSV row value for a logical column.
|
||||||
|
*
|
||||||
|
* @param array $event Normalized event row.
|
||||||
|
* @param string $column Column key.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_row_value( $event, $column ) {
|
||||||
|
return isset( $event[ $column ] ) ? (string) $event[ $column ] : '';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a CSV filename for the export.
|
||||||
|
*
|
||||||
|
* @param array $filters Export filters.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_build_filename( $filters ) {
|
||||||
|
$parts = array( 'schedule' );
|
||||||
|
|
||||||
|
$parts = array_merge( $parts, tse_sp_event_export_get_post_slugs( isset( $filters['team_ids'] ) ? $filters['team_ids'] : array() ) );
|
||||||
|
$parts = array_merge( $parts, tse_sp_event_export_get_term_slugs_by_ids( isset( $filters['season_ids'] ) ? $filters['season_ids'] : array(), 'sp_season' ) );
|
||||||
|
$parts = array_merge( $parts, tse_sp_event_export_get_term_slugs_by_ids( isset( $filters['league_ids'] ) ? $filters['league_ids'] : array(), 'sp_league' ) );
|
||||||
|
$parts = array_merge( $parts, tse_sp_event_export_get_term_slugs_by_ids( isset( $filters['field_ids'] ) ? $filters['field_ids'] : array(), 'sp_venue' ) );
|
||||||
|
|
||||||
|
$parts[] = tse_sp_event_export_sanitize_format( isset( $filters['format'] ) ? $filters['format'] : 'matchup' );
|
||||||
|
$parts = array_values( array_filter( $parts ) );
|
||||||
|
|
||||||
|
return implode( '-', $parts );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a human-readable feed title from filters.
|
||||||
|
*
|
||||||
|
* @param array $filters Export filters.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_feed_title( $filters ) {
|
||||||
|
$site_title = trim( (string) get_bloginfo( 'name' ) );
|
||||||
|
$team_name = tse_sp_event_export_get_post_titles( isset( $filters['team_ids'] ) ? $filters['team_ids'] : array() );
|
||||||
|
$season_name = tse_sp_event_export_get_term_names_by_ids( isset( $filters['season_ids'] ) ? $filters['season_ids'] : array(), 'sp_season' );
|
||||||
|
$league_name = tse_sp_event_export_get_term_names_by_ids( isset( $filters['league_ids'] ) ? $filters['league_ids'] : array(), 'sp_league' );
|
||||||
|
$field_name = tse_sp_event_export_get_term_names_by_ids( isset( $filters['field_ids'] ) ? $filters['field_ids'] : array(), 'sp_venue' );
|
||||||
|
|
||||||
|
$title = '' !== $site_title ? $site_title . ' ' . __( 'Event Feed', 'tonys-sportspress-enhancements' ) : __( 'Event Feed', 'tonys-sportspress-enhancements' );
|
||||||
|
$details = array_filter( array( $team_name, $season_name, $league_name, $field_name ) );
|
||||||
|
|
||||||
|
if ( ! empty( $details ) ) {
|
||||||
|
$title .= ' (' . implode( ' • ', $details ) . ')';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $title;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get term names for a list of term IDs.
|
||||||
|
*
|
||||||
|
* @param array $ids Term IDs.
|
||||||
|
* @param string $taxonomy Taxonomy name.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_term_names_by_ids( $ids, $taxonomy ) {
|
||||||
|
$names = array();
|
||||||
|
|
||||||
|
foreach ( (array) $ids as $id ) {
|
||||||
|
$term = get_term( absint( $id ), $taxonomy );
|
||||||
|
if ( $term instanceof WP_Term ) {
|
||||||
|
$names[] = $term->name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$names = array_values( array_unique( array_filter( $names ) ) );
|
||||||
|
|
||||||
|
return implode( '; ', $names );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get post titles for a list of post IDs.
|
||||||
|
*
|
||||||
|
* @param array $ids Post IDs.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_post_titles( $ids ) {
|
||||||
|
$titles = array();
|
||||||
|
|
||||||
|
foreach ( (array) $ids as $id ) {
|
||||||
|
$post = get_post( absint( $id ) );
|
||||||
|
if ( $post instanceof WP_Post ) {
|
||||||
|
$titles[] = $post->post_title;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$titles = array_values( array_unique( array_filter( $titles ) ) );
|
||||||
|
|
||||||
|
return implode( '; ', $titles );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get term slugs for a list of term IDs.
|
||||||
|
*
|
||||||
|
* @param array $ids Term IDs.
|
||||||
|
* @param string $taxonomy Taxonomy name.
|
||||||
|
* @return string[]
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_term_slugs_by_ids( $ids, $taxonomy ) {
|
||||||
|
$slugs = array();
|
||||||
|
|
||||||
|
foreach ( (array) $ids as $id ) {
|
||||||
|
$term = get_term( absint( $id ), $taxonomy );
|
||||||
|
if ( $term instanceof WP_Term && ! empty( $term->slug ) ) {
|
||||||
|
$slugs[] = sanitize_title( $term->slug );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_values( array_unique( array_filter( $slugs ) ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get post slugs for a list of post IDs.
|
||||||
|
*
|
||||||
|
* @param array $ids Post IDs.
|
||||||
|
* @return string[]
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_post_slugs( $ids ) {
|
||||||
|
$slugs = array();
|
||||||
|
|
||||||
|
foreach ( (array) $ids as $id ) {
|
||||||
|
$post = get_post( absint( $id ) );
|
||||||
|
if ( $post instanceof WP_Post ) {
|
||||||
|
$slugs[] = sanitize_title( $post->post_name ? $post->post_name : $post->post_title );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_values( array_unique( array_filter( $slugs ) ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stream CSV output for the requested export.
|
||||||
|
*
|
||||||
|
* @param array $filters Export filters.
|
||||||
|
* @param array $args Optional render args.
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_stream_csv( $filters, $args = array() ) {
|
||||||
|
$filters['format'] = tse_sp_event_export_sanitize_format( isset( $filters['format'] ) ? $filters['format'] : 'matchup' );
|
||||||
|
$filters['columns'] = tse_sp_event_export_sanitize_columns( $filters['format'], isset( $filters['columns'] ) ? $filters['columns'] : array() );
|
||||||
|
tse_sp_event_export_validate_filters( $filters );
|
||||||
|
|
||||||
|
$definitions = tse_sp_event_export_get_column_definitions();
|
||||||
|
$headers = array();
|
||||||
|
|
||||||
|
foreach ( $filters['columns'] as $column ) {
|
||||||
|
$headers[] = $definitions[ $filters['format'] ][ $column ];
|
||||||
|
}
|
||||||
|
|
||||||
|
$events = tse_sp_event_export_get_events( $filters );
|
||||||
|
$disposition = isset( $args['disposition'] ) ? sanitize_key( $args['disposition'] ) : 'inline';
|
||||||
|
$disposition = in_array( $disposition, array( 'inline', 'attachment' ), true ) ? $disposition : 'inline';
|
||||||
|
$filename = tse_sp_event_export_build_filename( $filters ) . '.csv';
|
||||||
|
|
||||||
|
header( 'Content-Type: text/csv; charset=utf-8' );
|
||||||
|
header( 'Content-Disposition: ' . $disposition . '; filename=' . $filename );
|
||||||
|
|
||||||
|
$output = fopen( 'php://output', 'w' );
|
||||||
|
|
||||||
|
if ( false === $output ) {
|
||||||
|
wp_die( esc_html__( 'Unable to generate the CSV export.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 500 ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
fwrite( $output, "\xEF\xBB\xBF" );
|
||||||
|
fputcsv( $output, $headers );
|
||||||
|
|
||||||
|
foreach ( $events as $event ) {
|
||||||
|
$row = array();
|
||||||
|
|
||||||
|
foreach ( $filters['columns'] as $column ) {
|
||||||
|
$row[] = tse_sp_event_export_get_row_value( $event, $column );
|
||||||
|
}
|
||||||
|
|
||||||
|
fputcsv( $output, $row );
|
||||||
|
}
|
||||||
|
|
||||||
|
fclose( $output );
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render the standalone CSV feed.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_render_feed() {
|
||||||
|
if ( ! class_exists( 'SP_Calendar' ) && ! post_type_exists( 'sp_event' ) ) {
|
||||||
|
wp_die( esc_html__( 'SportsPress is required for this feed.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 500 ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
tse_sp_event_export_stream_csv( tse_sp_event_export_normalize_request_args() );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render the standalone ICS feed.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_render_ical_feed() {
|
||||||
|
if ( ! class_exists( 'SP_Calendar' ) && ! post_type_exists( 'sp_event' ) ) {
|
||||||
|
wp_die( esc_html__( 'SportsPress is required for this feed.', 'tonys-sportspress-enhancements' ), '', array( 'response' => 500 ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
tse_sp_event_export_stream_ical( tse_sp_event_export_normalize_request_args() );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a shareable feed URL.
|
||||||
|
*
|
||||||
|
* @param array $args Export args.
|
||||||
|
* @param string $feed_type Feed type.
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function tse_sp_event_export_get_feed_url( $args = array(), $feed_type = 'csv' ) {
|
||||||
|
$filters = tse_sp_event_export_normalize_request_args( $args );
|
||||||
|
$feed = 'ics' === sanitize_key( $feed_type ) ? 'sp-ics' : 'sp-csv';
|
||||||
|
$query = array(
|
||||||
|
'feed' => $feed,
|
||||||
|
'format' => $filters['format'],
|
||||||
|
'team_id' => ! empty( $filters['team_ids'] ) ? implode( ',', $filters['team_ids'] ) : '',
|
||||||
|
'season_id' => ! empty( $filters['season_ids'] ) ? implode( ',', $filters['season_ids'] ) : '',
|
||||||
|
'league_id' => ! empty( $filters['league_ids'] ) ? implode( ',', $filters['league_ids'] ) : '',
|
||||||
|
'field_id' => ! empty( $filters['field_ids'] ) ? implode( ',', $filters['field_ids'] ) : '',
|
||||||
|
'columns' => implode( ',', $filters['columns'] ),
|
||||||
|
);
|
||||||
|
|
||||||
|
return add_query_arg( array_filter( $query, 'strlen' ), home_url( '/' ) );
|
||||||
|
}
|
||||||
@@ -75,15 +75,27 @@ function custom_event_rewrite_flush() {
|
|||||||
register_activation_hook(__FILE__, 'custom_event_rewrite_flush');
|
register_activation_hook(__FILE__, 'custom_event_rewrite_flush');
|
||||||
register_deactivation_hook(__FILE__, 'flush_rewrite_rules');
|
register_deactivation_hook(__FILE__, 'flush_rewrite_rules');
|
||||||
|
|
||||||
// Modify the query to handle custom permalinks and include future posts
|
// Modify the front-end single event query to allow scheduled events to resolve.
|
||||||
function custom_event_parse_request($query) {
|
function custom_event_parse_request( $query ) {
|
||||||
$post_type = sp_array_value( $query->query, 'post_type', null );
|
if ( ! $query instanceof WP_Query ) {
|
||||||
if (isset($query->query_vars['post_type']) && $query->query_vars['post_type'] === 'sp_event') {
|
return;
|
||||||
if (isset($query->query_vars['p'])) {
|
|
||||||
$query->set('post_type', 'sp_event');
|
|
||||||
$query->set('p', $query->query_vars['p']);
|
|
||||||
$query->set('post_status', array('publish', 'future'));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ( is_admin() || ! $query->is_main_query() ) {
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ( 'sp_event' !== $query->get( 'post_type' ) ) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$post_id = absint( $query->get( 'p' ) );
|
||||||
|
if ( $post_id <= 0 ) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$query->set( 'post_type', 'sp_event' );
|
||||||
|
$query->set( 'p', $post_id );
|
||||||
|
$query->set( 'post_status', array( 'publish', 'future' ) );
|
||||||
}
|
}
|
||||||
add_action('pre_get_posts', 'custom_event_parse_request');
|
add_action('pre_get_posts', 'custom_event_parse_request');
|
||||||
|
|||||||
1889
includes/sp-printable-calendars.php
Normal file
1889
includes/sp-printable-calendars.php
Normal file
File diff suppressed because it is too large
Load Diff
1155
includes/sp-schedule-exporter.php
Normal file
1155
includes/sp-schedule-exporter.php
Normal file
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
|
||||||
|
}
|
||||||
116
includes/sp-venue-meta.php
Normal file
116
includes/sp-venue-meta.php
Normal file
@@ -0,0 +1,116 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* Venue term metadata support.
|
||||||
|
*
|
||||||
|
* Adds short name and abbreviation fields to SportsPress venues.
|
||||||
|
*/
|
||||||
|
|
||||||
|
if ( ! defined( 'ABSPATH' ) ) {
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register venue term meta.
|
||||||
|
*/
|
||||||
|
function tony_sportspress_register_venue_term_meta() {
|
||||||
|
register_term_meta(
|
||||||
|
'sp_venue',
|
||||||
|
'tse_short_name',
|
||||||
|
array(
|
||||||
|
'type' => 'string',
|
||||||
|
'single' => true,
|
||||||
|
'sanitize_callback' => 'sanitize_text_field',
|
||||||
|
'show_in_rest' => true,
|
||||||
|
'auth_callback' => static function() {
|
||||||
|
return current_user_can( 'manage_categories' );
|
||||||
|
},
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
register_term_meta(
|
||||||
|
'sp_venue',
|
||||||
|
'tse_abbreviation',
|
||||||
|
array(
|
||||||
|
'type' => 'string',
|
||||||
|
'single' => true,
|
||||||
|
'sanitize_callback' => 'sanitize_text_field',
|
||||||
|
'show_in_rest' => true,
|
||||||
|
'auth_callback' => static function() {
|
||||||
|
return current_user_can( 'manage_categories' );
|
||||||
|
},
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
add_action( 'init', 'tony_sportspress_register_venue_term_meta' );
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render add-form fields for venue metadata.
|
||||||
|
*/
|
||||||
|
function tony_sportspress_add_venue_meta_fields() {
|
||||||
|
?>
|
||||||
|
<div class="form-field term-short-name-wrap">
|
||||||
|
<label for="tse_short_name"><?php esc_html_e( 'Short Name', 'tonys-sportspress-enhancements' ); ?></label>
|
||||||
|
<input name="tse_short_name" id="tse_short_name" type="text" value="" maxlength="100" />
|
||||||
|
<p><?php esc_html_e( 'Optional shorter label for this field or venue.', 'tonys-sportspress-enhancements' ); ?></p>
|
||||||
|
</div>
|
||||||
|
<div class="form-field term-abbreviation-wrap">
|
||||||
|
<label for="tse_abbreviation"><?php esc_html_e( 'Abbreviation', 'tonys-sportspress-enhancements' ); ?></label>
|
||||||
|
<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>
|
||||||
|
<?php
|
||||||
|
}
|
||||||
|
add_action( 'sp_venue_add_form_fields', 'tony_sportspress_add_venue_meta_fields' );
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render edit-form fields for venue metadata.
|
||||||
|
*
|
||||||
|
* @param WP_Term $term Venue term.
|
||||||
|
*/
|
||||||
|
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 );
|
||||||
|
?>
|
||||||
|
<tr class="form-field term-short-name-wrap">
|
||||||
|
<th scope="row">
|
||||||
|
<label for="tse_short_name"><?php esc_html_e( 'Short Name', 'tonys-sportspress-enhancements' ); ?></label>
|
||||||
|
</th>
|
||||||
|
<td>
|
||||||
|
<input name="tse_short_name" id="tse_short_name" type="text" value="<?php echo esc_attr( $short_name ); ?>" maxlength="100" />
|
||||||
|
<p class="description"><?php esc_html_e( 'Optional shorter label for this field or venue.', 'tonys-sportspress-enhancements' ); ?></p>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr class="form-field term-abbreviation-wrap">
|
||||||
|
<th scope="row">
|
||||||
|
<label for="tse_abbreviation"><?php esc_html_e( 'Abbreviation', 'tonys-sportspress-enhancements' ); ?></label>
|
||||||
|
</th>
|
||||||
|
<td>
|
||||||
|
<input name="tse_abbreviation" id="tse_abbreviation" type="text" value="<?php echo esc_attr( $abbreviation ); ?>" maxlength="20" />
|
||||||
|
<p class="description"><?php esc_html_e( 'Optional abbreviation such as CC East or Field 1.', 'tonys-sportspress-enhancements' ); ?></p>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<?php
|
||||||
|
}
|
||||||
|
add_action( 'sp_venue_edit_form_fields', 'tony_sportspress_edit_venue_meta_fields' );
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Save venue metadata fields.
|
||||||
|
*
|
||||||
|
* @param int $term_id Venue term ID.
|
||||||
|
*/
|
||||||
|
function tony_sportspress_save_venue_meta_fields( $term_id ) {
|
||||||
|
if ( ! current_user_can( 'manage_categories' ) ) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$short_name = isset( $_POST['tse_short_name'] ) ? sanitize_text_field( wp_unslash( $_POST['tse_short_name'] ) ) : '';
|
||||||
|
$short_name = is_string( $short_name ) ? trim( $short_name ) : '';
|
||||||
|
|
||||||
|
$abbreviation = isset( $_POST['tse_abbreviation'] ) ? sanitize_text_field( wp_unslash( $_POST['tse_abbreviation'] ) ) : '';
|
||||||
|
$abbreviation = is_string( $abbreviation ) ? trim( $abbreviation ) : '';
|
||||||
|
|
||||||
|
update_term_meta( $term_id, 'tse_short_name', $short_name );
|
||||||
|
update_term_meta( $term_id, 'tse_abbreviation', $abbreviation );
|
||||||
|
}
|
||||||
|
add_action( 'created_sp_venue', 'tony_sportspress_save_venue_meta_fields' );
|
||||||
|
add_action( 'edited_sp_venue', 'tony_sportspress_save_venue_meta_fields' );
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
|
|
||||||
{
|
{
|
||||||
"name": "tonys-sportspress-enhancements",
|
"name": "tonys-sportspress-enhancements",
|
||||||
"version": "0.1.0",
|
"version": "0.1.5",
|
||||||
"main": "Gruntfile.js",
|
"main": "Gruntfile.js",
|
||||||
"author": "YOUR NAME HERE",
|
"author": "YOUR NAME HERE",
|
||||||
"scripts" : {
|
"scripts" : {
|
||||||
|
|||||||
@@ -9,12 +9,14 @@ Tony's SportsPress Enhancements is a collection of add-ons for the [SportsPress]
|
|||||||
- **Custom event permalinks** for `sp_event` post types, including season and team slugs.
|
- **Custom event permalinks** for `sp_event` post types, including season and team slugs.
|
||||||
- **Open Graph meta tags** for events, with dynamic titles, descriptions, and images.
|
- **Open Graph meta tags** for events, with dynamic titles, descriptions, and images.
|
||||||
- **Automatic featured image generation** for events, combining team colors and logos into a shareable image.
|
- **Automatic featured image generation** for events, combining team colors and logos into a shareable image.
|
||||||
|
- **Printable team schedules** with season-aware venue colors and a print-friendly calendar layout.
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
- Custom rewrite rules and permalinks for SportsPress events.
|
- Custom rewrite rules and permalinks for SportsPress events.
|
||||||
- Open Graph integration for better social sharing.
|
- Open Graph integration for better social sharing.
|
||||||
- Dynamic, cached event images based on team data.
|
- Dynamic, cached event images based on team data.
|
||||||
|
- Printable schedule pages linked from team profiles.
|
||||||
- Compatible with WordPress 4.5+ and PHP 5.6+.
|
- Compatible with WordPress 4.5+ and PHP 5.6+.
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|||||||
81
tests/test-sp-event-permalink.php
Normal file
81
tests/test-sp-event-permalink.php
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* Tests for custom event permalink query behavior.
|
||||||
|
*
|
||||||
|
* @package Tonys_Sportspress_Enhancements
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Event permalink query tests.
|
||||||
|
*/
|
||||||
|
class Test_SP_Event_Permalink extends WP_UnitTestCase {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Preserve global main query references for each test.
|
||||||
|
*
|
||||||
|
* @var WP_Query
|
||||||
|
*/
|
||||||
|
private $original_wp_query;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Preserve global main query references for each test.
|
||||||
|
*
|
||||||
|
* @var WP_Query
|
||||||
|
*/
|
||||||
|
private $original_wp_the_query;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set up test case state.
|
||||||
|
*/
|
||||||
|
public function set_up(): void {
|
||||||
|
parent::set_up();
|
||||||
|
$this->original_wp_query = $GLOBALS['wp_query'];
|
||||||
|
$this->original_wp_the_query = $GLOBALS['wp_the_query'];
|
||||||
|
set_current_screen( 'front' );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Restore global query references.
|
||||||
|
*/
|
||||||
|
public function tear_down(): void {
|
||||||
|
$GLOBALS['wp_query'] = $this->original_wp_query;
|
||||||
|
$GLOBALS['wp_the_query'] = $this->original_wp_the_query;
|
||||||
|
set_current_screen( 'front' );
|
||||||
|
parent::tear_down();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The admin event list query should not be altered by permalink handling.
|
||||||
|
*/
|
||||||
|
public function test_admin_event_queries_are_not_modified() {
|
||||||
|
set_current_screen( 'edit-sp_event' );
|
||||||
|
|
||||||
|
$query = new WP_Query();
|
||||||
|
$query->set( 'post_type', 'sp_event' );
|
||||||
|
$query->set( 'p', 123 );
|
||||||
|
$query->set( 'post_status', 'future' );
|
||||||
|
|
||||||
|
custom_event_parse_request( $query );
|
||||||
|
|
||||||
|
$this->assertSame( 'future', $query->get( 'post_status' ) );
|
||||||
|
$this->assertSame( 123, $query->get( 'p' ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Front-end single event requests should include future posts.
|
||||||
|
*/
|
||||||
|
public function test_frontend_single_event_queries_include_future_posts() {
|
||||||
|
$query = new WP_Query();
|
||||||
|
$query->set( 'post_type', 'sp_event' );
|
||||||
|
$query->set( 'p', 456 );
|
||||||
|
|
||||||
|
$GLOBALS['wp_query'] = $query;
|
||||||
|
$GLOBALS['wp_the_query'] = $query;
|
||||||
|
|
||||||
|
custom_event_parse_request( $query );
|
||||||
|
|
||||||
|
$this->assertSame( 'sp_event', $query->get( 'post_type' ) );
|
||||||
|
$this->assertSame( 456, $query->get( 'p' ) );
|
||||||
|
$this->assertSame( array( 'publish', 'future' ), $query->get( 'post_status' ) );
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -7,17 +7,37 @@
|
|||||||
* Author URI: https://github.com/anthonyscorrea/
|
* Author URI: https://github.com/anthonyscorrea/
|
||||||
* Text Domain: tonys-sportspress-enhancements
|
* Text Domain: tonys-sportspress-enhancements
|
||||||
* Domain Path: /languages
|
* Domain Path: /languages
|
||||||
* Version: 0.1.4
|
* Version: 0.1.7
|
||||||
*
|
*
|
||||||
* @package Tonys_Sportspress_Enhancements
|
* @package Tonys_Sportspress_Enhancements
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_VERSION' ) ) {
|
||||||
|
define( 'TONY_SPORTSPRESS_ENHANCEMENTS_VERSION', '0.1.7' );
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_FILE' ) ) {
|
||||||
|
define( 'TONY_SPORTSPRESS_ENHANCEMENTS_FILE', __FILE__ );
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_DIR' ) ) {
|
||||||
|
define( 'TONY_SPORTSPRESS_ENHANCEMENTS_DIR', plugin_dir_path( __FILE__ ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ! defined( 'TONY_SPORTSPRESS_ENHANCEMENTS_URL' ) ) {
|
||||||
|
define( 'TONY_SPORTSPRESS_ENHANCEMENTS_URL', plugin_dir_url( __FILE__ ) );
|
||||||
|
}
|
||||||
|
|
||||||
// Include other files here
|
// Include other files here
|
||||||
require_once plugin_dir_path(__FILE__) . 'includes/open-graph-tags.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/featured-image-generator.php';
|
||||||
require_once plugin_dir_path(__FILE__) . 'includes/sp-event-permalink.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-csv.php';
|
||||||
require_once plugin_dir_path(__FILE__) . 'includes/sp-event-admin-week-filter.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-quick-edit-officials.php';
|
||||||
require_once plugin_dir_path(__FILE__) . 'includes/sp-event-team-ordering.php';
|
require_once plugin_dir_path(__FILE__) . 'includes/sp-event-team-ordering.php';
|
||||||
|
require_once plugin_dir_path(__FILE__) . 'includes/sp-printable-calendars.php';
|
||||||
|
require_once plugin_dir_path(__FILE__) . 'includes/sp-url-builder.php';
|
||||||
|
require_once plugin_dir_path(__FILE__) . 'includes/sp-schedule-exporter.php';
|
||||||
|
require_once plugin_dir_path(__FILE__) . 'includes/sp-venue-meta.php';
|
||||||
|
|||||||
Reference in New Issue
Block a user