appointments = array( $appointment ); return $this->get_ics( $this->appointments, $template ); } /** * Get .ics feed for appointments. * * @param array $appointments an array of appointments objects. * @param string $template the ics template to use. * * @return array|null an array containing the appointment data and headers if there are appointments. Null if no appointments. */ public function get_ics_feed( $appointments, $template = 'customer' ) { if ( empty( $appointments ) ) { return null; } if ( ! is_array( $appointments ) ) { $appointments = array( $appointments ); } $this->appointments = $appointments; $sitename = get_bloginfo( 'name' ); return $this->get_ics( $this->appointments, $template, $sitename ); } /** * Get .ics for appointments. * * @param array $appointments Array with SSA_Appointment_Object objects. * @param string $template the ics template to use. * @param string $filename .ics filename. * * @return string .ics path */ public function get_ics( $appointments, $template = 'customer', $filename = '' ) { $this->appointments = $appointments; $this->template = $template; if ( '' === $filename ) { $filename = 'appointment-' . time() . '-' . wp_hash( wp_json_encode( $this->appointments['0']->id ) . $this->template ); } // Create the .ics. $ics = $this->generate(); return array( 'data' => $ics, 'headers' => array( 'Content-Disposition' => 'attachment; filename="' . $filename . '.ics"', 'Content-Type' => 'text/calendar; charset=utf-8', ), ); } /** * Format the date * * @param int $timestamp Timestamp to format. * @param SSA_Appointment_Object $appointment Booking object. * * @return string Formatted date for ICS. */ protected function format_date( $timestamp, $appointment = null ) { $pattern = 'Ymd\THis'; if ( $appointment ) { $pattern = ( $appointment->is_all_day() ) ? 'Ymd' : $pattern; } $formatted_date = gmdate( $pattern, $timestamp ); $formatted_date .= 'Z'; // Zulu (UTC). return $formatted_date; } /** * Sanitize strings for .ics * * @param string $string String to sanitize. * * @return string */ protected function sanitize_string( $string ) { $clear_map = array( '

' => "

\r\n", '
' => "\r\n", '
' => "\r\n", '
' => "\r\n", ); $string = str_replace( array_keys( $clear_map ), array_values( $clear_map ), $string ); $string = wp_strip_all_tags( $string ); $string = preg_replace( '/([\,;])/', '\\\$1', $string ); $string = str_replace( "\n\n\n", "\n\n", $string ); $string = str_replace( "\n", '\n', $string ); $string = sanitize_text_field( $string ); $string = html_entity_decode( $string, ENT_QUOTES | ENT_XML1, 'UTF-8' ); return $string; } /** * Fold a content line per RFC 5545 (max 75 octets, fold with CRLF + space) * * @param string $line The full line including property name (e.g., "DESCRIPTION:text...") * @return string The folded line(s) */ protected function fold_line( $line ) { $max_length = 75; // If line is already short enough, return as-is if ( strlen( $line ) <= $max_length ) { return $line; } $result = ''; $remaining = $line; $first_line = true; while ( strlen( $remaining ) > 0 ) { if ( $first_line ) { // First line: take up to 75 chars $chunk_length = $max_length; $first_line = false; } else { // Continuation lines: account for the leading space (74 chars of content + 1 space = 75) $chunk_length = $max_length - 1; } // Get chunk respecting UTF-8 boundaries $chunk = mb_strcut( $remaining, 0, $chunk_length, 'UTF-8' ); $remaining = substr( $remaining, strlen( $chunk ) ); if ( $result !== '' ) { // Add CRLF + space before continuation $result .= "\r\n "; } $result .= $chunk; } return $result; } /** * Generate the .ics content * * @return string */ protected function generate() { $settings = ssa()->settings->get(); $sitename = $settings['global']['company_name']; // Set the ics data. $ics = 'BEGIN:VCALENDAR' . $this->eol; $ics .= 'VERSION:2.0' . $this->eol; $ics .= 'PRODID:-//SSA//Simply Schedule Appointments ' . Simply_Schedule_Appointments::VERSION . '//EN' . $this->eol; $ics .= 'CALSCALE:GREGORIAN' . $this->eol; $ics .= 'X-ORIGINAL-URL:' . $this->sanitize_string( home_url( '/' ) ) . $this->eol; $ics .= 'X-WR-CALDESC:' . $this->sanitize_string( sprintf( __( 'Appointments from %s', 'simply-schedule-appointments' ), $sitename ) ) . $this->eol; $ics .= 'TRANSP:' . 'OPAQUE' . $this->eol; // Create recipient and URL once — avoids N×3 object creations inside the // per-appointment get_calendar_event_title/description/location calls. if ( 'staff' === $this->template ) { $recipient = SSA_Recipient_Staff::create(); $url = ssa()->wp_admin->url( 'ssa/appointments' ); } else { $recipient = SSA_Recipient_Customer::create(); $url = ''; } // Enable batch-mode caching on the template engine for the duration of this loop. // Each appointment needs title, description, and location — all three trigger the // same heavy filter chain (DB queries + meta lookups) on get_template_vars(). // Batch mode caches the result after the first call per appointment so the 2nd // and 3rd calls are instant. Disabled immediately after the loop; zero side effects // on any other part of the application. ssa()->templates->enable_batch_mode(); foreach ( $this->appointments as $appointment ) { if ( in_array( $this->template, array( 'customer', 'staff' ) ) ) { $summary = $appointment->get_calendar_event_title( $recipient ); $description = $appointment->get_calendar_event_description( $recipient ); $location = $appointment->get_calendar_event_location( $recipient ); $date_prefix = ( $appointment->is_all_day() ) ? ';VALUE=DATE:' : ':'; } $ics .= 'BEGIN:VEVENT' . $this->eol; $ics .= 'UID:' . $this->uid_prefix . $appointment->id . $this->template . $this->eol; $ics .= 'DTSTAMP:' . $this->format_date( time() ) . $this->eol; if ( ! empty( $location ) ) { $ics .= $this->fold_line( 'LOCATION:' . $this->sanitize_string( $location ) ) . $this->eol; } else { $ics .= 'LOCATION:' . $this->eol; } $ics .= $this->fold_line( 'DESCRIPTION:' . $this->sanitize_string( $description ) ) . $this->eol; $ics .= $this->fold_line( 'URL;VALUE=URI:' . $this->sanitize_string( $url ) ) . $this->eol; $ics .= $this->fold_line( 'SUMMARY:' . $this->sanitize_string( $summary ) ) . $this->eol; $ics .= 'DTSTART' . $date_prefix . $this->format_date( $appointment->start_date_timestamp, $appointment ) . $this->eol; $ics .= 'DTEND' . $date_prefix . $this->format_date( $appointment->end_date_timestamp, $appointment ) . $this->eol; $ics .= 'END:VEVENT' . $this->eol; } ssa()->templates->disable_batch_mode(); $ics .= 'END:VCALENDAR'; return $ics; } }