post_types() as $post_type ) { add_action( 'save_post_' . $post_type, array( $this, 'save_image_header' ) ); add_action( 'save_post_' . $post_type, array( $this, 'save_global_stock' ) ); } add_action( 'admin_menu', array( $this, 'attendees_page_register' ) ); add_filter( 'post_row_actions', array( $this, 'attendees_row_action' ) ); add_filter( 'page_row_actions', array( $this, 'attendees_row_action' ) ); add_action( 'tribe_tickets_attendees_event_details_list_top', array( $this, 'event_details_top' ), 20 ); add_action( 'tribe_tickets_plus_report_event_details_list_top', array( $this, 'event_details_top' ), 20 ); add_action( 'tribe_tickets_attendees_event_details_list_top', array( $this, 'event_action_links' ), 25 ); add_action( 'tribe_tickets_plus_report_event_details_list_top', array( $this, 'event_action_links' ), 25 ); add_action( 'tribe_events_tickets_attendees_totals_top', array( $this, 'print_checkedin_totals' ), 0 ); $this->path = trailingslashit( dirname( dirname( dirname( __FILE__ ) ) ) ); } /** * Injects event post type * * @param int $event_id */ public function event_details_top( $event_id ) { $pto = get_post_type_object( get_post_type( $event_id ) ); echo '
%s
', esc_html__( 'Columns', 'event-tickets' ), esc_html__( 'You can use Screen Options to select which columns you want to see. The selection works in the table below, in the email, for print and for the CSV export.', 'event-tickets' ) ), 'position' => array( 'edge' => 'top', 'align' => 'right' ), ), ); wp_enqueue_script( 'wp-pointer' ); wp_enqueue_style( 'wp-pointer' ); } wp_localize_script( self::$attendees_slug, 'AttendeesPointer', $pointer ); } /** * Setups the Attendees screen data. */ public function attendees_page_screen_setup() { /* There's no reason for attendee screen setup to happen twice, but because * of a fix for bug #46198 it can indeed be called twice in the same request. * This flag variable is used to workaround that. * * @see Tribe__Tickets__Tickets_Handler::attendees_page_register() (and related @todo inside that method) * @see https://central.tri.be/issues/46198 * * @todo remove the has_run check once the above workaround is dispensed with */ static $has_run = false; if ( $has_run || ( is_admin() && ( empty( $_GET['page'] ) || self::$attendees_slug !== $_GET['page'] ) ) ) { return; } $has_run = true; /** * This is a workaround to fix the problem * * @see https://central.tri.be/issues/46198 * @todo remove this */ if ( current_filter() === 'admin_init' ) { $this->attendees_page_load_css_js( $this->attendees_page ); $GLOBALS['current_screen'] = WP_Screen::get( $this->attendees_page ); } if ( ! empty( $_GET['action'] ) && in_array( $_GET['action'], array( 'email' ) ) ) { define( 'IFRAME_REQUEST', true ); // Use iFrame Header -- WP Method iframe_header(); // Check if we need to send an Email! if ( isset( $_POST['tribe-send-email'] ) && $_POST['tribe-send-email'] ) { $status = $this->send_attendee_mail_list(); } else { $status = false; } $which_tmpl = sanitize_file_name( $_GET['action'] ); include $this->path . 'src/admin-views/attendees-' . $which_tmpl . '.php'; // Use iFrame Footer -- WP Method iframe_footer(); // We need nothing else here exit; } else { $this->attendees_table = new Tribe__Tickets__Attendees_Table(); $this->maybe_generate_attendees_csv(); add_filter( 'admin_title', array( $this, 'attendees_admin_title' ), 10, 2 ); add_filter( 'admin_body_class', array( $this, 'attendees_admin_body_class' ) ); } } public function attendees_admin_body_class( $body_classes ) { return $body_classes . ' plugins-php'; } /** * Sets the browser title for the Attendees admin page. * Uses the event title. * * @param $admin_title * @param $title * * @return string */ public function attendees_admin_title( $admin_title, $title ) { if ( ! empty( $_GET['event_id'] ) ) { $event = get_post( $_GET['event_id'] ); $admin_title = sprintf( '%s - Attendee list', $event->post_title ); } return $admin_title; } /** * Renders the Attendees page */ public function attendees_page_inside() { /** * Fires immediately before the content of the attendees screen * is rendered. * * @param $this Tribe__Tickets__Tickets_Handler The current ticket handler instance. */ do_action( 'tribe_tickets_attendees_page_inside', $this ); include $this->path . 'src/admin-views/attendees.php'; } /** * Generates a list of attendees taking into account the Screen Options. * It's used both for the Email functionality, as for the CSV export. * * @param $event_id * * @return array */ private function generate_filtered_attendees_list( $event_id ) { /** * Fire immediately prior to the generation of a filtered (exportable) attendee list. * * @param int $event_id */ do_action( 'tribe_events_tickets_generate_filtered_attendees_list', $event_id ); if ( empty( $this->attendees_page ) ) { $this->attendees_page = 'tribe_events_page_tickets-attendees'; } //Add in Columns or get_column_headers() returns nothing $filter_name = "manage_{$this->attendees_page}_columns"; add_filter( $filter_name, array( $this->attendees_table, 'get_columns' ), 15 ); $items = Tribe__Tickets__Tickets::get_event_attendees( $event_id ); //Add Handler for Community Tickets to Prevent Notices in Exports if ( ! is_admin() ) { $columns = apply_filters( $filter_name, array() ); } else { $columns = array_map( 'wp_strip_all_tags', get_column_headers( get_current_screen() ) ); } // We dont want HTML inputs, private data or other columns that are superfluous in a CSV export $hidden = array_merge( get_hidden_columns( $this->attendees_page ), array( 'cb', 'meta_details', 'provider', 'purchaser', 'status', ) ); $hidden = array_flip( $hidden ); $export_columns = array_diff_key( $columns, $hidden ); // Add additional expected columns $export_columns['order_id'] = esc_html_x( 'Order ID', 'attendee export', 'event-tickets' ); $export_columns['order_status_label'] = esc_html_x( 'Order Status', 'attendee export', 'event-tickets' ); $export_columns['attendee_id'] = esc_html_x( 'Ticket #', 'attendee export', 'event-tickets' ); $export_columns['purchaser_name'] = esc_html_x( 'Customer Name', 'attendee export', 'event-tickets' ); $export_columns['purchaser_email'] = esc_html_x( 'Customer Email Address', 'attendee export', 'event-tickets' ); /** * Used to modify what columns should be shown on the CSV export * The column name should be the Array Index and the Header is the array Value * * @var array Columns, associative array * @var array Items to be exported * @var int Event ID */ $export_columns = apply_filters( 'tribe_events_tickets_attendees_csv_export_columns', $export_columns, $items, $event_id ); // Add the export column headers as the first row $rows = array( array_values( $export_columns ), ); foreach ( $items as $single_item ) { // Fresh row! $row = array(); foreach ( $export_columns as $column_id => $column_name ) { // If additional columns have been added to the attendee list table we can obtain the // values by calling the table object's column_default() method - any other values // should simply be passed back unmodified $row[ $column_id ] = $this->attendees_table->column_default( $single_item, $column_id ); // Special handling for the check_in column if ( 'check_in' === $column_id && 1 == $single_item[ $column_id ] ) { $row[ $column_id ] = esc_html__( 'Yes', 'event-tickets' ); } // Special handling for new human readable id if ( 'attendee_id' === $column_id ) { if ( isset( $single_item[ $column_id ] ) ) { $ticket_unique_id = get_post_meta( $single_item[ $column_id ], '_unique_id', true ); $ticket_unique_id = $ticket_unique_id === '' ? $single_item[ $column_id ] : $ticket_unique_id; $row[ $column_id ] = esc_html( $ticket_unique_id ); } } // Handle custom columns that might have names containing HTML tags $row[ $column_id ] = wp_strip_all_tags( $row[ $column_id ] ); } $rows[] = array_values( $row ); } return array_filter( $rows ); } /** * Checks if the user requested a CSV export from the attendees list. * If so, generates the download and finishes the execution. */ public function maybe_generate_attendees_csv() { if ( empty( $_GET['attendees_csv'] ) || empty( $_GET['attendees_csv_nonce'] ) || empty( $_GET['event_id'] ) ) { return; } if ( ! wp_verify_nonce( $_GET['attendees_csv_nonce'], 'attendees_csv_nonce' ) || ! $this->user_can( 'edit_posts', $_GET['event_id'] ) ) { return; } $items = apply_filters( 'tribe_events_tickets_attendees_csv_items', $this->generate_filtered_attendees_list( $_GET['event_id'] ) );; $event = get_post( $_GET['event_id'] ); if ( ! empty( $items ) ) { $charset = get_option( 'blog_charset' ); $filename = sanitize_file_name( $event->post_title . '-' . __( 'attendees', 'event-tickets' ) ); // output headers so that the file is downloaded rather than displayed header( "Content-Type: text/csv; charset=$charset" ); header( "Content-Disposition: attachment; filename=$filename.csv" ); // create a file pointer connected to the output stream $output = fopen( 'php://output', 'w' ); //And echo the data foreach ( $items as $item ) { fputcsv( $output, $item ); } fclose( $output ); exit; } } /** * Handles the "send to email" action for the attendees list. */ public function send_attendee_mail_list() { $error = new WP_Error(); if ( empty( $_GET['event_id'] ) ) { $error->add( 'no-event-id', esc_html__( 'Invalid Event ID', 'event-tickets' ), array( 'type' => 'general' ) ); return $error; } $cap = 'edit_posts'; $event_id = absint( ! empty( $_GET['event_id'] ) && is_numeric( $_GET['event_id'] ) ? $_GET['event_id'] : 0 ); if ( ! current_user_can( 'edit_posts' ) && $event_id ) { $event = get_post( $event_id ); if ( $event instanceof WP_Post && get_current_user_id() === (int) $event->post_author ) { $cap = 'read'; } } if ( empty( $_POST['_wpnonce'] ) || ! wp_verify_nonce( $_POST['_wpnonce'], 'email-attendees-list' ) || ! $this->user_can( $cap, $_GET['event_id'] ) ) { $error->add( 'nonce-fail', esc_html__( 'Cheatin Huh?', 'event-tickets' ), array( 'type' => 'general' ) ); return $error; } if ( empty( $_POST['email_to_address'] ) && ( empty( $_POST['email_to_user'] ) || 0 >= (int) $_POST['email_to_user'] ) ) { $error->add( 'empty-fields', esc_html__( 'Empty user and email', 'event-tickets' ), array( 'type' => 'general' ) ); return $error; } if ( ! empty( $_POST['email_to_address'] ) ) { $type = 'email'; } else { $type = 'user'; } if ( 'email' === $type && ! is_email( $_POST['email_to_address'] ) ) { $error->add( 'invalid-email', esc_html__( 'Invalid Email', 'event-tickets' ), array( 'type' => $type ) ); return $error; } if ( 'user' === $type && ! is_numeric( $_POST['email_to_user'] ) ) { $error->add( 'invalid-user', esc_html__( 'Invalid User ID', 'event-tickets' ), array( 'type' => $type ) ); return $error; } /** * Now we know we have valid data */ if ( 'email' === $type ) { // We already check this variable so, no harm here $email = $_POST['email_to_address']; } else { $user = get_user_by( 'id', $_POST['email_to_user'] ); if ( ! is_object( $user ) ) { $error->add( 'invalid-user', esc_html__( 'Invalid User ID', 'event-tickets' ), array( 'type' => $type ) ); return $error; } $email = $user->data->user_email; } $this->attendees_table = new Tribe__Tickets__Attendees_Table(); $items = $this->generate_filtered_attendees_list( $_GET['event_id'] ); $event = get_post( $_GET['event_id'] ); ob_start(); $attendee_tpl = Tribe__Tickets__Templates::get_template_hierarchy( 'tickets/attendees-email.php', array( 'disable_view_check' => true ) ); include $attendee_tpl; $content = ob_get_clean(); add_filter( 'wp_mail_content_type', array( $this, 'set_contenttype' ) ); if ( ! wp_mail( $email, sprintf( esc_html__( 'Attendee List for: %s', 'event-tickets' ), $event->post_title ), $content ) ) { $error->add( 'email-error', esc_html__( 'Error when sending the email', 'event-tickets' ), array( 'type' => 'general' ) ); return $error; } return esc_html__( 'Email sent successfully!', 'event-tickets' ); } /** * Sets the content type for the attendees to email functionality. * Allows for sending an HTML email. * * @param $content_type * * @return string */ public function set_contenttype( $content_type ) { return 'text/html'; } /** * Tests if the user has the specified capability in relation to whatever post type * the ticket relates to. * * For example, if tickets are created for the banana post type, the generic capability * "edit_posts" will be mapped to "edit_bananas" or whatever is appropriate. * * @internal for internal plugin use only (in spite of having public visibility) * * @param string $generic_cap * @param int $event_id * @return boolean */ public function user_can( $generic_cap, $event_id ) { $type = get_post_type( $event_id ); // It's possible we'll get null back if ( null === $type ) { return false; } $type_config = get_post_type_object( $type ); if ( ! empty( $type_config->cap->{$generic_cap} ) ) { return current_user_can( $type_config->cap->{$generic_cap} ); } return false; } /* Tickets Metabox */ /** * Includes the tickets metabox inside the Event edit screen * * @param WP_Post $post */ public function do_meta_box( $post ) { $startMinuteOptions = Tribe__View_Helpers::getMinuteOptions( null ); $endMinuteOptions = Tribe__View_Helpers::getMinuteOptions( null ); $startHourOptions = Tribe__View_Helpers::getHourOptions( null, true ); $endHourOptions = Tribe__View_Helpers::getHourOptions( null, false ); $startMeridianOptions = Tribe__View_Helpers::getMeridianOptions( null, true ); $endMeridianOptions = Tribe__View_Helpers::getMeridianOptions( null ); $show_global_stock = Tribe__Tickets__Tickets::global_stock_available(); $tickets = Tribe__Tickets__Tickets::get_event_tickets( $post->ID ); $global_stock = new Tribe__Tickets__Global_Stock( $post->ID ); include $this->path . 'src/admin-views/meta-box.php'; } /** * Echoes the markup for the tickets list in the tickets metabox * * @param array $tickets */ public function ticket_list_markup( $tickets = array() ) { if ( ! empty( $tickets ) ) { include $this->path . 'src/admin-views/list.php'; } } /** * Returns the markup for the tickets list in the tickets metabox * * @param array $tickets * * @return string */ public function get_ticket_list_markup( $tickets = array() ) { ob_start(); $this->ticket_list_markup( $tickets ); $return = ob_get_contents(); ob_end_clean(); return $return; } /** * Returns the attachment ID for the header image for a event. * * @param $event_id * * @return mixed */ public function get_header_image_id( $event_id ) { return get_post_meta( $event_id, $this->image_header_field, true ); } /** * Save or delete the image header for tickets on an event * * @param int $post_id */ public function save_image_header( $post_id ) { if ( ! ( isset($_POST[ 'tribe-tickets-post-settings' ]) && wp_verify_nonce( $_POST[ 'tribe-tickets-post-settings' ], 'tribe-tickets-meta-box' ) ) ) { return; } // don't do anything on autosave or auto-draft either or massupdates if ( wp_is_post_autosave( $post_id ) || wp_is_post_revision( $post_id ) ) { return; } if ( empty( $_POST['tribe_ticket_header_image_id'] ) ) { delete_post_meta( $post_id, $this->image_header_field ); } else { update_post_meta( $post_id, $this->image_header_field, $_POST['tribe_ticket_header_image_id'] ); } return; } /** * Save the current global stock properties for this event. * * @param int $post_id */ public function save_global_stock( $post_id ) { if ( ! ( isset( $_POST[ 'tribe-tickets-post-settings' ] ) && wp_verify_nonce( $_POST[ 'tribe-tickets-post-settings' ], 'tribe-tickets-meta-box' ) ) ) { return; } // Bail on autosaves/bulk updates if ( wp_is_post_autosave( $post_id ) || wp_is_post_revision( $post_id ) ) { return; } $enable = ! empty( $_POST[ 'tribe-tickets-enable-global-stock' ] ); $stock = (int) @$_POST[ 'tribe-tickets-global-stock' ]; $post_global_stock = new Tribe__Tickets__Global_Stock( $post_id ); $post_global_stock->enable( $enable ); $post_global_stock->set_stock_level( $stock ); } /** * Static Singleton Factory Method * * @return Tribe__Tickets__Tickets_Handler */ public static function instance() { return tribe( 'tickets.handler' ); } /** * Returns the current post being handled. * * @return array|bool|null|WP_Post */ public function get_post() { return $this->attendees_table->event; } /** * Whether the ticket handler should render the title in the attendees report. * * @param bool $should_render_title */ public function should_render_title( $should_render_title ) { $this->should_render_title = $should_render_title; } /** * Returns the full URL to the attendees report page. * * @param WP_Post $post * * @return string */ public function get_attendee_report_link( $post ) { $url = add_query_arg( array( 'post_type' => $post->post_type, 'page' => self::$attendees_slug, 'event_id' => $post->ID, ), admin_url( 'edit.php' ) ); return $url; } }