familyHub/tabs/calendar.php

209 lines
10 KiB
PHP

<?php
require_once __DIR__ . '/../includes/family_settings.php';
require_once __DIR__ . '/../includes/calendar_helpers.php';
require_once __DIR__ . '/../includes/utils.php';
$rawEmbed = defined('GOOGLE_CALENDAR_EMBED_CODE') ? trim((string) GOOGLE_CALENDAR_EMBED_CODE) : '';
$calId = defined('GOOGLE_CALENDAR_ID') ? trim((string) GOOGLE_CALENDAR_ID) : '';
/**
* @return string|null HTTPS embed URL restricted to Google Calendar hosts
*/
function familyHub_google_calendar_embed_url(string $raw): ?string {
$raw = trim($raw);
if ($raw === '' || strcasecmp($raw, 'your_embed_code_here') === 0) {
return null;
}
$allowed = static function (string $u): bool {
$p = parse_url($u);
if ($p === false || empty($p['scheme']) || empty($p['host'])) {
return false;
}
if (strtolower($p['scheme']) !== 'https') {
return false;
}
$h = strtolower($p['host']);
if ($h === 'calendar.google.com') {
return true;
}
if ($h === 'www.google.com' && isset($p['path']) && str_starts_with($p['path'], '/calendar')) {
return true;
}
return false;
};
if ($allowed($raw)) {
return $raw;
}
// Match iframe with src first or after other attributes (previous pattern required \s before src and failed on <iframe src="...">).
if (preg_match('#<iframe\s[^>]*\s*src\s*=\s*["\']([^"\']+)["\']#i', $raw, $m)) {
$u = html_entity_decode($m[1], ENT_QUOTES | ENT_HTML5, 'UTF-8');
return $allowed($u) ? $u : null;
}
return null;
}
$embedUrl = familyHub_google_calendar_embed_url($rawEmbed);
$fromIdOnly = false;
if ($embedUrl === null && $calId !== '' && $calId !== 'your_calendar_id_here') {
$embedUrl = 'https://calendar.google.com/calendar/embed?src=' . rawurlencode($calId);
$fromIdOnly = true;
}
[$rangeStart, $rangeEnd] = hubCalendarDefaultAgendaRange($familySettings);
$events = hubCalendarAgendaEvents($rangeStart, $rangeEnd, $people, $familySettings);
[$calendarTzId, $tzLocal] = familyHubCalendarContext($familySettings);
$todayYmd = familyHubTodayYmdInTz($tzLocal);
$twoWayMerge = !empty($familySettings['calendar_two_way_google']);
$isTvView = isset($_GET['view']) && (string) $_GET['view'] === 'tv';
$calendarTvUrl = '?tab=calendar&view=tv';
$eventsByDate = [];
foreach ($events as $ev) {
$d = (string) ($ev['date'] ?? '');
if ($d === '') {
continue;
}
if (!isset($eventsByDate[$d])) {
$eventsByDate[$d] = [];
}
$eventsByDate[$d][] = $ev;
}
ksort($eventsByDate);
$iconByType = [
'chore' => 'fa-tasks',
'meal' => 'fa-utensils',
'grocery_due' => 'fa-cart-shopping',
'expense' => 'fa-coins',
'bill_day' => 'fa-file-invoice-dollar',
'birthday_reminder' => 'fa-cake-candles',
];
?>
<div id="calendar" class="tab-content<?= $isTvView ? ' calendar-tv-view' : '' ?>">
<?php if (!$isTvView): ?>
<div class="d-flex flex-wrap justify-content-between align-items-center gap-2 mb-2">
<h2 class="mb-0">Calendar</h2>
<a href="<?= htmlspecialchars($calendarTvUrl, ENT_QUOTES, 'UTF-8') ?>" class="btn btn-outline-secondary btn-sm" title="Open TV view dashboard">
<i class="fa fa-tv me-1" aria-hidden="true"></i>
Open TV view
</a>
</div>
<p class="text-muted small mb-3">Open TV view for a full-width dashboard without site chrome.</p>
<div class="alert alert-secondary mb-3" role="status">
<strong>Google Calendar is not synced into Family Hub data.</strong>
The Google panel below is an embedded view from Google (live in your browser). The <strong>Family Hub agenda</strong> lists chores, meals, groceries, expenses, bill reminders, and birthdays from this app only—Google events never appear there until a future API connection exists.
<?php if ($twoWayMerge): ?>
<span class="d-block mt-1 small">You turned on “two-way sync” in settings; OAuth with Google is <strong>not implemented yet</strong>, so that option does not change behavior today.</span>
<?php endif; ?>
</div>
<?php endif; ?>
<div class="row <?= $isTvView ? 'g-3' : 'g-4' ?> calendar-tab-layout<?= $embedUrl !== null ? ' calendar-tab-layout-split' : '' ?>">
<div class="col-12 calendar-tab-google">
<h3 class="h5 mb-3"><?= $isTvView ? 'Calendar' : 'Google Calendar' ?></h3>
<?php if ($embedUrl === null): ?>
<?php if (!$isTvView): ?>
<p class="text-muted">Connect a shared family calendar by setting <strong>GOOGLE_CALENDAR_EMBED_CODE</strong> (embed URL or full <code>&lt;iframe&gt;</code> from Google Calendar) or <strong>GOOGLE_CALENDAR_ID</strong> in <code>.env</code>. See <a href="https://calendar.google.com/">Google Calendar</a> → calendar menu → <strong>Settings and sharing</strong> → <strong>Integrate calendar</strong>.</p>
<?php else: ?>
<p class="text-muted small mb-0">Google Calendar not configured (set <code class="small">.env</code>).</p>
<?php endif; ?>
<?php else: ?>
<?php if ($fromIdOnly && !$isTvView): ?>
<p class="small text-muted mb-2">Showing calendar from <code>GOOGLE_CALENDAR_ID</code>. For more control (view, height), paste the embed URL or iframe into <code>GOOGLE_CALENDAR_EMBED_CODE</code> instead.</p>
<?php endif; ?>
<div class="calendar-embed rounded border overflow-hidden bg-white">
<iframe
class="w-100 border-0 d-block"
style="height: 70vh; min-height: 360px;"
src="<?= htmlspecialchars($embedUrl, ENT_QUOTES, 'UTF-8') ?>"
loading="lazy"
title="Family calendar"
allowfullscreen
></iframe>
</div>
<?php endif; ?>
</div>
<div class="col-12 calendar-tab-agenda">
<h3 class="h5 mb-3"><?= $isTvView ? 'Weekly hub' : 'Family Hub agenda' ?></h3>
<?php if ($isTvView): ?>
<div class="calendar-tv-clock mb-3" aria-hidden="true">
<time id="calendarTvClock" class="calendar-tv-clock-time"></time>
</div>
<?php endif; ?>
<?php if (!$isTvView): ?>
<p class="text-muted small mb-2">Hub data only (not Google Calendar).</p>
<?php endif; ?>
<p class="text-muted small mb-3 hide hidden" style="display: none;">
<?= htmlspecialchars($rangeStart, ENT_QUOTES, 'UTF-8') ?> to <?= htmlspecialchars($rangeEnd, ENT_QUOTES, 'UTF-8') ?>
· Times use <strong><?= htmlspecialchars(str_replace('_', ' ', (string) ($familySettings['timezone'] ?? '')), ENT_QUOTES, 'UTF-8') ?></strong>
</p>
<?php if ($eventsByDate === []): ?>
<p class="text-muted mb-0">Nothing scheduled in this range. Add chores, meal plan slots, expenses, or bill reminders.</p>
<?php else: ?>
<div class="list-group calendar-agenda-list">
<?php foreach ($eventsByDate as $dateStr => $dayEvents): ?>
<div class="list-group-item calendar-agenda-day px-0 py-2<?= $dateStr === $todayYmd ? ' calendar-agenda-today border-primary border' : '' ?>">
<div class="px-3 pb-1 small fw-semibold text-muted calendar-agenda-date-label">
<?php if ($dateStr === $todayYmd): ?>
<span class="text-primary">Today</span> ·
<?php endif; ?>
<?= htmlspecialchars($dateStr, ENT_QUOTES, 'UTF-8') ?>
<?php if ($dateStr < $todayYmd): ?>
<span class="badge bg-secondary ms-1">Past</span>
<?php endif; ?>
</div>
<?php foreach ($dayEvents as $ev): ?>
<?php
$t = (string) ($ev['type'] ?? '');
$ic = $iconByType[$t] ?? 'fa-circle';
$href = '?' . (string) ($ev['hrefQuery'] ?? 'tab=calendar');
$det = trim((string) ($ev['detail'] ?? ''));
?>
<a href="<?= htmlspecialchars($href, ENT_QUOTES, 'UTF-8') ?>" class="list-group-item list-group-item-action d-flex align-items-start gap-3 py-3 border-0 border-top rounded-0">
<span class="fa <?= htmlspecialchars($ic, ENT_QUOTES, 'UTF-8') ?> fa-fw text-muted mt-1" aria-hidden="true"></span>
<span class="flex-grow-1 min-w-0">
<span class="d-block fw-medium"><?= sanitizeInput((string) ($ev['title'] ?? '')) ?></span>
<?php if ($det !== ''): ?>
<span class="d-block small text-muted text-break"><?= sanitizeInput($det) ?></span>
<?php endif; ?>
</span>
</a>
<?php endforeach; ?>
</div>
<?php endforeach; ?>
</div>
<?php endif; ?>
</div>
</div>
</div>
<?php if ($isTvView): ?>
<script>
(function () {
var tz = <?= json_encode($calendarTzId, JSON_UNESCAPED_UNICODE) ?>;
var el = document.getElementById('calendarTvClock');
function tick() {
if (!el) {
return;
}
var now = new Date();
var opts = {
timeZone: tz,
hour: 'numeric',
minute: '2-digit',
second: '2-digit',
hour12: true
};
el.textContent = new Intl.DateTimeFormat(undefined, opts).format(now);
el.setAttribute('datetime', now.toISOString());
}
tick();
window.setInterval(tick, 1000);
window.setTimeout(function () {
window.location.reload();
}, 10 * 60 * 1000);
})();
</script>
<?php endif; ?>