@@ -21,2045 +21,2045 @@ |
||
| 21 | 21 | class EEH_Line_Item |
| 22 | 22 | { |
| 23 | 23 | |
| 24 | - /** |
|
| 25 | - * Adds a simple item (unrelated to any other model object) to the provided PARENT line item. |
|
| 26 | - * Does NOT automatically re-calculate the line item totals or update the related transaction. |
|
| 27 | - * You should call recalculate_total_including_taxes() on the grant total line item after this |
|
| 28 | - * to update the subtotals, and EE_Registration_Processor::calculate_reg_final_prices_per_line_item() |
|
| 29 | - * to keep the registration final prices in-sync with the transaction's total. |
|
| 30 | - * |
|
| 31 | - * @param EE_Line_Item $parent_line_item |
|
| 32 | - * @param string $name |
|
| 33 | - * @param float $unit_price |
|
| 34 | - * @param string $description |
|
| 35 | - * @param int $quantity |
|
| 36 | - * @param boolean $taxable |
|
| 37 | - * @param boolean $code if set to a value, ensures there is only one line item with that code |
|
| 38 | - * @return boolean success |
|
| 39 | - * @throws EE_Error |
|
| 40 | - * @throws InvalidArgumentException |
|
| 41 | - * @throws InvalidDataTypeException |
|
| 42 | - * @throws InvalidInterfaceException |
|
| 43 | - * @throws ReflectionException |
|
| 44 | - */ |
|
| 45 | - public static function add_unrelated_item( |
|
| 46 | - EE_Line_Item $parent_line_item, |
|
| 47 | - $name, |
|
| 48 | - $unit_price, |
|
| 49 | - $description = '', |
|
| 50 | - $quantity = 1, |
|
| 51 | - $taxable = false, |
|
| 52 | - $code = null |
|
| 53 | - ) { |
|
| 54 | - $items_subtotal = self::get_pre_tax_subtotal($parent_line_item); |
|
| 55 | - $line_item = EE_Line_Item::new_instance(array( |
|
| 56 | - 'LIN_name' => $name, |
|
| 57 | - 'LIN_desc' => $description, |
|
| 58 | - 'LIN_unit_price' => $unit_price, |
|
| 59 | - 'LIN_quantity' => $quantity, |
|
| 60 | - 'LIN_percent' => null, |
|
| 61 | - 'LIN_is_taxable' => $taxable, |
|
| 62 | - 'LIN_order' => $items_subtotal instanceof EE_Line_Item ? count($items_subtotal->children()) : 0, |
|
| 63 | - 'LIN_total' => (float) $unit_price * (int) $quantity, |
|
| 64 | - 'LIN_type' => EEM_Line_Item::type_line_item, |
|
| 65 | - 'LIN_code' => $code, |
|
| 66 | - )); |
|
| 67 | - $line_item = apply_filters( |
|
| 68 | - 'FHEE__EEH_Line_Item__add_unrelated_item__line_item', |
|
| 69 | - $line_item, |
|
| 70 | - $parent_line_item |
|
| 71 | - ); |
|
| 72 | - return self::add_item($parent_line_item, $line_item); |
|
| 73 | - } |
|
| 74 | - |
|
| 75 | - |
|
| 76 | - /** |
|
| 77 | - * Adds a simple item ( unrelated to any other model object) to the total line item, |
|
| 78 | - * in the correct spot in the line item tree. Does not automatically |
|
| 79 | - * re-calculate the line item totals, nor update the related transaction, nor upgrade the transaction's |
|
| 80 | - * registrations' final prices (which should probably change because of this). |
|
| 81 | - * You should call recalculate_total_including_taxes() on the grand total line item, then |
|
| 82 | - * update the transaction's total, and EE_Registration_Processor::update_registration_final_prices() |
|
| 83 | - * after using this, to keep the registration final prices in-sync with the transaction's total. |
|
| 84 | - * |
|
| 85 | - * @param EE_Line_Item $parent_line_item |
|
| 86 | - * @param string $name |
|
| 87 | - * @param float $percentage_amount |
|
| 88 | - * @param string $description |
|
| 89 | - * @param boolean $taxable |
|
| 90 | - * @return boolean success |
|
| 91 | - * @throws EE_Error |
|
| 92 | - */ |
|
| 93 | - public static function add_percentage_based_item( |
|
| 94 | - EE_Line_Item $parent_line_item, |
|
| 95 | - $name, |
|
| 96 | - $percentage_amount, |
|
| 97 | - $description = '', |
|
| 98 | - $taxable = false |
|
| 99 | - ) { |
|
| 100 | - $line_item = EE_Line_Item::new_instance(array( |
|
| 101 | - 'LIN_name' => $name, |
|
| 102 | - 'LIN_desc' => $description, |
|
| 103 | - 'LIN_unit_price' => 0, |
|
| 104 | - 'LIN_percent' => $percentage_amount, |
|
| 105 | - 'LIN_quantity' => 1, |
|
| 106 | - 'LIN_is_taxable' => $taxable, |
|
| 107 | - 'LIN_total' => (float) ($percentage_amount * ($parent_line_item->total() / 100)), |
|
| 108 | - 'LIN_type' => EEM_Line_Item::type_line_item, |
|
| 109 | - 'LIN_parent' => $parent_line_item->ID(), |
|
| 110 | - )); |
|
| 111 | - $line_item = apply_filters( |
|
| 112 | - 'FHEE__EEH_Line_Item__add_percentage_based_item__line_item', |
|
| 113 | - $line_item |
|
| 114 | - ); |
|
| 115 | - return $parent_line_item->add_child_line_item($line_item, false); |
|
| 116 | - } |
|
| 117 | - |
|
| 118 | - |
|
| 119 | - /** |
|
| 120 | - * Returns the new line item created by adding a purchase of the ticket |
|
| 121 | - * ensures that ticket line item is saved, and that cart total has been recalculated. |
|
| 122 | - * If this ticket has already been purchased, just increments its count. |
|
| 123 | - * Automatically re-calculates the line item totals and updates the related transaction. But |
|
| 124 | - * DOES NOT automatically upgrade the transaction's registrations' final prices (which |
|
| 125 | - * should probably change because of this). |
|
| 126 | - * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item() |
|
| 127 | - * after using this, to keep the registration final prices in-sync with the transaction's total. |
|
| 128 | - * |
|
| 129 | - * @param EE_Line_Item $total_line_item grand total line item of type EEM_Line_Item::type_total |
|
| 130 | - * @param EE_Ticket $ticket |
|
| 131 | - * @param int $qty |
|
| 132 | - * @return EE_Line_Item |
|
| 133 | - * @throws EE_Error |
|
| 134 | - * @throws InvalidArgumentException |
|
| 135 | - * @throws InvalidDataTypeException |
|
| 136 | - * @throws InvalidInterfaceException |
|
| 137 | - * @throws ReflectionException |
|
| 138 | - */ |
|
| 139 | - public static function add_ticket_purchase(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1) |
|
| 140 | - { |
|
| 141 | - if (! $total_line_item instanceof EE_Line_Item || ! $total_line_item->is_total()) { |
|
| 142 | - throw new EE_Error( |
|
| 143 | - sprintf( |
|
| 144 | - esc_html__( |
|
| 145 | - 'A valid line item total is required in order to add tickets. A line item of type "%s" was passed.', |
|
| 146 | - 'event_espresso' |
|
| 147 | - ), |
|
| 148 | - $ticket->ID(), |
|
| 149 | - $total_line_item->ID() |
|
| 150 | - ) |
|
| 151 | - ); |
|
| 152 | - } |
|
| 153 | - // either increment the qty for an existing ticket |
|
| 154 | - $line_item = self::increment_ticket_qty_if_already_in_cart($total_line_item, $ticket, $qty); |
|
| 155 | - // or add a new one |
|
| 156 | - if (! $line_item instanceof EE_Line_Item) { |
|
| 157 | - $line_item = self::create_ticket_line_item($total_line_item, $ticket, $qty); |
|
| 158 | - } |
|
| 159 | - $total_line_item->recalculate_total_including_taxes(); |
|
| 160 | - return $line_item; |
|
| 161 | - } |
|
| 162 | - |
|
| 163 | - |
|
| 164 | - /** |
|
| 165 | - * Returns the new line item created by adding a purchase of the ticket |
|
| 166 | - * |
|
| 167 | - * @param EE_Line_Item $total_line_item |
|
| 168 | - * @param EE_Ticket $ticket |
|
| 169 | - * @param int $qty |
|
| 170 | - * @return EE_Line_Item |
|
| 171 | - * @throws EE_Error |
|
| 172 | - * @throws InvalidArgumentException |
|
| 173 | - * @throws InvalidDataTypeException |
|
| 174 | - * @throws InvalidInterfaceException |
|
| 175 | - * @throws ReflectionException |
|
| 176 | - */ |
|
| 177 | - public static function increment_ticket_qty_if_already_in_cart( |
|
| 178 | - EE_Line_Item $total_line_item, |
|
| 179 | - EE_Ticket $ticket, |
|
| 180 | - $qty = 1 |
|
| 181 | - ) { |
|
| 182 | - $line_item = null; |
|
| 183 | - if ($total_line_item instanceof EE_Line_Item && $total_line_item->is_total()) { |
|
| 184 | - $ticket_line_items = EEH_Line_Item::get_ticket_line_items($total_line_item); |
|
| 185 | - foreach ((array) $ticket_line_items as $ticket_line_item) { |
|
| 186 | - if ($ticket_line_item instanceof EE_Line_Item |
|
| 187 | - && (int) $ticket_line_item->OBJ_ID() === (int) $ticket->ID() |
|
| 188 | - ) { |
|
| 189 | - $line_item = $ticket_line_item; |
|
| 190 | - break; |
|
| 191 | - } |
|
| 192 | - } |
|
| 193 | - } |
|
| 194 | - if ($line_item instanceof EE_Line_Item) { |
|
| 195 | - EEH_Line_Item::increment_quantity($line_item, $qty); |
|
| 196 | - return $line_item; |
|
| 197 | - } |
|
| 198 | - return null; |
|
| 199 | - } |
|
| 200 | - |
|
| 201 | - |
|
| 202 | - /** |
|
| 203 | - * Increments the line item and all its children's quantity by $qty (but percent line items are unaffected). |
|
| 204 | - * Does NOT save or recalculate other line items totals |
|
| 205 | - * |
|
| 206 | - * @param EE_Line_Item $line_item |
|
| 207 | - * @param int $qty |
|
| 208 | - * @return void |
|
| 209 | - * @throws EE_Error |
|
| 210 | - * @throws InvalidArgumentException |
|
| 211 | - * @throws InvalidDataTypeException |
|
| 212 | - * @throws InvalidInterfaceException |
|
| 213 | - * @throws ReflectionException |
|
| 214 | - */ |
|
| 215 | - public static function increment_quantity(EE_Line_Item $line_item, $qty = 1) |
|
| 216 | - { |
|
| 217 | - if (! $line_item->is_percent()) { |
|
| 218 | - $qty += $line_item->quantity(); |
|
| 219 | - $line_item->set_quantity($qty); |
|
| 220 | - $line_item->set_total($line_item->unit_price() * $qty); |
|
| 221 | - $line_item->save(); |
|
| 222 | - } |
|
| 223 | - foreach ($line_item->children() as $child) { |
|
| 224 | - if ($child->is_sub_line_item()) { |
|
| 225 | - EEH_Line_Item::update_quantity($child, $qty); |
|
| 226 | - } |
|
| 227 | - } |
|
| 228 | - } |
|
| 229 | - |
|
| 230 | - |
|
| 231 | - /** |
|
| 232 | - * Decrements the line item and all its children's quantity by $qty (but percent line items are unaffected). |
|
| 233 | - * Does NOT save or recalculate other line items totals |
|
| 234 | - * |
|
| 235 | - * @param EE_Line_Item $line_item |
|
| 236 | - * @param int $qty |
|
| 237 | - * @return void |
|
| 238 | - * @throws EE_Error |
|
| 239 | - * @throws InvalidArgumentException |
|
| 240 | - * @throws InvalidDataTypeException |
|
| 241 | - * @throws InvalidInterfaceException |
|
| 242 | - * @throws ReflectionException |
|
| 243 | - */ |
|
| 244 | - public static function decrement_quantity(EE_Line_Item $line_item, $qty = 1) |
|
| 245 | - { |
|
| 246 | - if (! $line_item->is_percent()) { |
|
| 247 | - $qty = $line_item->quantity() - $qty; |
|
| 248 | - $qty = max($qty, 0); |
|
| 249 | - $line_item->set_quantity($qty); |
|
| 250 | - $line_item->set_total($line_item->unit_price() * $qty); |
|
| 251 | - $line_item->save(); |
|
| 252 | - } |
|
| 253 | - foreach ($line_item->children() as $child) { |
|
| 254 | - if ($child->is_sub_line_item()) { |
|
| 255 | - EEH_Line_Item::update_quantity($child, $qty); |
|
| 256 | - } |
|
| 257 | - } |
|
| 258 | - } |
|
| 259 | - |
|
| 260 | - |
|
| 261 | - /** |
|
| 262 | - * Updates the line item and its children's quantities to the specified number. |
|
| 263 | - * Does NOT save them or recalculate totals. |
|
| 264 | - * |
|
| 265 | - * @param EE_Line_Item $line_item |
|
| 266 | - * @param int $new_quantity |
|
| 267 | - * @throws EE_Error |
|
| 268 | - * @throws InvalidArgumentException |
|
| 269 | - * @throws InvalidDataTypeException |
|
| 270 | - * @throws InvalidInterfaceException |
|
| 271 | - * @throws ReflectionException |
|
| 272 | - */ |
|
| 273 | - public static function update_quantity(EE_Line_Item $line_item, $new_quantity) |
|
| 274 | - { |
|
| 275 | - if (! $line_item->is_percent()) { |
|
| 276 | - $line_item->set_quantity($new_quantity); |
|
| 277 | - $line_item->set_total($line_item->unit_price() * $new_quantity); |
|
| 278 | - $line_item->save(); |
|
| 279 | - } |
|
| 280 | - foreach ($line_item->children() as $child) { |
|
| 281 | - if ($child->is_sub_line_item()) { |
|
| 282 | - EEH_Line_Item::update_quantity($child, $new_quantity); |
|
| 283 | - } |
|
| 284 | - } |
|
| 285 | - } |
|
| 286 | - |
|
| 287 | - |
|
| 288 | - /** |
|
| 289 | - * Returns the new line item created by adding a purchase of the ticket |
|
| 290 | - * |
|
| 291 | - * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 292 | - * @param EE_Ticket $ticket |
|
| 293 | - * @param int $qty |
|
| 294 | - * @return EE_Line_Item |
|
| 295 | - * @throws EE_Error |
|
| 296 | - * @throws InvalidArgumentException |
|
| 297 | - * @throws InvalidDataTypeException |
|
| 298 | - * @throws InvalidInterfaceException |
|
| 299 | - * @throws ReflectionException |
|
| 300 | - */ |
|
| 301 | - public static function create_ticket_line_item(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1) |
|
| 302 | - { |
|
| 303 | - $datetimes = $ticket->datetimes(); |
|
| 304 | - $first_datetime = reset($datetimes); |
|
| 305 | - $first_datetime_name = esc_html__('Event', 'event_espresso'); |
|
| 306 | - if ($first_datetime instanceof EE_Datetime && $first_datetime->event() instanceof EE_Event) { |
|
| 307 | - $first_datetime_name = $first_datetime->event()->name(); |
|
| 308 | - } |
|
| 309 | - $event = sprintf(_x('(For %1$s)', '(For Event Name)', 'event_espresso'), $first_datetime_name); |
|
| 310 | - // get event subtotal line |
|
| 311 | - $events_sub_total = self::get_event_line_item_for_ticket($total_line_item, $ticket); |
|
| 312 | - // add $ticket to cart |
|
| 313 | - $line_item = EE_Line_Item::new_instance(array( |
|
| 314 | - 'LIN_name' => $ticket->name(), |
|
| 315 | - 'LIN_desc' => $ticket->description() !== '' ? $ticket->description() . ' ' . $event : $event, |
|
| 316 | - 'LIN_unit_price' => $ticket->price(), |
|
| 317 | - 'LIN_quantity' => $qty, |
|
| 318 | - 'LIN_is_taxable' => $ticket->taxable(), |
|
| 319 | - 'LIN_order' => count($events_sub_total->children()), |
|
| 320 | - 'LIN_total' => $ticket->price() * $qty, |
|
| 321 | - 'LIN_type' => EEM_Line_Item::type_line_item, |
|
| 322 | - 'OBJ_ID' => $ticket->ID(), |
|
| 323 | - 'OBJ_type' => EEM_Line_Item::OBJ_TYPE_TICKET, |
|
| 324 | - )); |
|
| 325 | - $line_item = apply_filters( |
|
| 326 | - 'FHEE__EEH_Line_Item__create_ticket_line_item__line_item', |
|
| 327 | - $line_item |
|
| 328 | - ); |
|
| 329 | - $events_sub_total->add_child_line_item($line_item); |
|
| 330 | - // now add the sub-line items |
|
| 331 | - $running_total_for_ticket = 0; |
|
| 332 | - foreach ($ticket->prices(array('order_by' => array('PRC_order' => 'ASC'))) as $price) { |
|
| 333 | - $sign = $price->is_discount() ? -1 : 1; |
|
| 334 | - $price_total = $price->is_percent() |
|
| 335 | - ? $running_total_for_ticket * $price->amount() / 100 |
|
| 336 | - : $price->amount() * $qty; |
|
| 337 | - $sub_line_item = EE_Line_Item::new_instance(array( |
|
| 338 | - 'LIN_name' => $price->name(), |
|
| 339 | - 'LIN_desc' => $price->desc(), |
|
| 340 | - 'LIN_quantity' => $price->is_percent() ? null : $qty, |
|
| 341 | - 'LIN_is_taxable' => false, |
|
| 342 | - 'LIN_order' => $price->order(), |
|
| 343 | - 'LIN_total' => $sign * $price_total, |
|
| 344 | - 'LIN_type' => EEM_Line_Item::type_sub_line_item, |
|
| 345 | - 'OBJ_ID' => $price->ID(), |
|
| 346 | - 'OBJ_type' => EEM_Line_Item::OBJ_TYPE_PRICE, |
|
| 347 | - )); |
|
| 348 | - $sub_line_item = apply_filters( |
|
| 349 | - 'FHEE__EEH_Line_Item__create_ticket_line_item__sub_line_item', |
|
| 350 | - $sub_line_item |
|
| 351 | - ); |
|
| 352 | - if ($price->is_percent()) { |
|
| 353 | - $sub_line_item->set_percent($sign * $price->amount()); |
|
| 354 | - } else { |
|
| 355 | - $sub_line_item->set_unit_price($sign * $price->amount()); |
|
| 356 | - } |
|
| 357 | - $running_total_for_ticket += $price_total; |
|
| 358 | - $line_item->add_child_line_item($sub_line_item); |
|
| 359 | - } |
|
| 360 | - return $line_item; |
|
| 361 | - } |
|
| 362 | - |
|
| 363 | - |
|
| 364 | - /** |
|
| 365 | - * Adds the specified item under the pre-tax-sub-total line item. Automatically |
|
| 366 | - * re-calculates the line item totals and updates the related transaction. But |
|
| 367 | - * DOES NOT automatically upgrade the transaction's registrations' final prices (which |
|
| 368 | - * should probably change because of this). |
|
| 369 | - * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item() |
|
| 370 | - * after using this, to keep the registration final prices in-sync with the transaction's total. |
|
| 371 | - * |
|
| 372 | - * @param EE_Line_Item $total_line_item |
|
| 373 | - * @param EE_Line_Item $item to be added |
|
| 374 | - * @return boolean |
|
| 375 | - * @throws EE_Error |
|
| 376 | - * @throws InvalidArgumentException |
|
| 377 | - * @throws InvalidDataTypeException |
|
| 378 | - * @throws InvalidInterfaceException |
|
| 379 | - * @throws ReflectionException |
|
| 380 | - */ |
|
| 381 | - public static function add_item(EE_Line_Item $total_line_item, EE_Line_Item $item) |
|
| 382 | - { |
|
| 383 | - $pre_tax_subtotal = self::get_pre_tax_subtotal($total_line_item); |
|
| 384 | - if ($pre_tax_subtotal instanceof EE_Line_Item) { |
|
| 385 | - $success = $pre_tax_subtotal->add_child_line_item($item); |
|
| 386 | - } else { |
|
| 387 | - return false; |
|
| 388 | - } |
|
| 389 | - $total_line_item->recalculate_total_including_taxes(); |
|
| 390 | - return $success; |
|
| 391 | - } |
|
| 392 | - |
|
| 393 | - |
|
| 394 | - /** |
|
| 395 | - * cancels an existing ticket line item, |
|
| 396 | - * by decrementing it's quantity by 1 and adding a new "type_cancellation" sub-line-item. |
|
| 397 | - * ALL totals and subtotals will NEED TO BE UPDATED after performing this action |
|
| 398 | - * |
|
| 399 | - * @param EE_Line_Item $ticket_line_item |
|
| 400 | - * @param int $qty |
|
| 401 | - * @return bool success |
|
| 402 | - * @throws EE_Error |
|
| 403 | - * @throws InvalidArgumentException |
|
| 404 | - * @throws InvalidDataTypeException |
|
| 405 | - * @throws InvalidInterfaceException |
|
| 406 | - * @throws ReflectionException |
|
| 407 | - */ |
|
| 408 | - public static function cancel_ticket_line_item(EE_Line_Item $ticket_line_item, $qty = 1) |
|
| 409 | - { |
|
| 410 | - // validate incoming line_item |
|
| 411 | - if ($ticket_line_item->OBJ_type() !== EEM_Line_Item::OBJ_TYPE_TICKET) { |
|
| 412 | - throw new EE_Error( |
|
| 413 | - sprintf( |
|
| 414 | - esc_html__( |
|
| 415 | - 'The supplied line item must have an Object Type of "Ticket", not %1$s.', |
|
| 416 | - 'event_espresso' |
|
| 417 | - ), |
|
| 418 | - $ticket_line_item->type() |
|
| 419 | - ) |
|
| 420 | - ); |
|
| 421 | - } |
|
| 422 | - if ($ticket_line_item->quantity() < $qty) { |
|
| 423 | - throw new EE_Error( |
|
| 424 | - sprintf( |
|
| 425 | - esc_html__( |
|
| 426 | - 'Can not cancel %1$d ticket(s) because the supplied line item has a quantity of %2$d.', |
|
| 427 | - 'event_espresso' |
|
| 428 | - ), |
|
| 429 | - $qty, |
|
| 430 | - $ticket_line_item->quantity() |
|
| 431 | - ) |
|
| 432 | - ); |
|
| 433 | - } |
|
| 434 | - // decrement ticket quantity; don't rely on auto-fixing when recalculating totals to do this |
|
| 435 | - $ticket_line_item->set_quantity($ticket_line_item->quantity() - $qty); |
|
| 436 | - foreach ($ticket_line_item->children() as $child_line_item) { |
|
| 437 | - if ($child_line_item->is_sub_line_item() |
|
| 438 | - && ! $child_line_item->is_percent() |
|
| 439 | - && ! $child_line_item->is_cancellation() |
|
| 440 | - ) { |
|
| 441 | - $child_line_item->set_quantity($child_line_item->quantity() - $qty); |
|
| 442 | - } |
|
| 443 | - } |
|
| 444 | - // get cancellation sub line item |
|
| 445 | - $cancellation_line_item = EEH_Line_Item::get_descendants_of_type( |
|
| 446 | - $ticket_line_item, |
|
| 447 | - EEM_Line_Item::type_cancellation |
|
| 448 | - ); |
|
| 449 | - $cancellation_line_item = reset($cancellation_line_item); |
|
| 450 | - // verify that this ticket was indeed previously cancelled |
|
| 451 | - if ($cancellation_line_item instanceof EE_Line_Item) { |
|
| 452 | - // increment cancelled quantity |
|
| 453 | - $cancellation_line_item->set_quantity($cancellation_line_item->quantity() + $qty); |
|
| 454 | - } else { |
|
| 455 | - // create cancellation sub line item |
|
| 456 | - $cancellation_line_item = EE_Line_Item::new_instance(array( |
|
| 457 | - 'LIN_name' => esc_html__('Cancellation', 'event_espresso'), |
|
| 458 | - 'LIN_desc' => sprintf( |
|
| 459 | - esc_html_x( |
|
| 460 | - 'Cancelled %1$s : %2$s', |
|
| 461 | - 'Cancelled Ticket Name : 2015-01-01 11:11', |
|
| 462 | - 'event_espresso' |
|
| 463 | - ), |
|
| 464 | - $ticket_line_item->name(), |
|
| 465 | - current_time(get_option('date_format') . ' ' . get_option('time_format')) |
|
| 466 | - ), |
|
| 467 | - 'LIN_unit_price' => 0, // $ticket_line_item->unit_price() |
|
| 468 | - 'LIN_quantity' => $qty, |
|
| 469 | - 'LIN_is_taxable' => $ticket_line_item->is_taxable(), |
|
| 470 | - 'LIN_order' => count($ticket_line_item->children()), |
|
| 471 | - 'LIN_total' => 0, // $ticket_line_item->unit_price() |
|
| 472 | - 'LIN_type' => EEM_Line_Item::type_cancellation, |
|
| 473 | - )); |
|
| 474 | - $ticket_line_item->add_child_line_item($cancellation_line_item); |
|
| 475 | - } |
|
| 476 | - if ($ticket_line_item->save_this_and_descendants() > 0) { |
|
| 477 | - // decrement parent line item quantity |
|
| 478 | - $event_line_item = $ticket_line_item->parent(); |
|
| 479 | - if ($event_line_item instanceof EE_Line_Item |
|
| 480 | - && $event_line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_EVENT |
|
| 481 | - ) { |
|
| 482 | - $event_line_item->set_quantity($event_line_item->quantity() - $qty); |
|
| 483 | - $event_line_item->save(); |
|
| 484 | - } |
|
| 485 | - EEH_Line_Item::get_grand_total_and_recalculate_everything($ticket_line_item); |
|
| 486 | - return true; |
|
| 487 | - } |
|
| 488 | - return false; |
|
| 489 | - } |
|
| 490 | - |
|
| 491 | - |
|
| 492 | - /** |
|
| 493 | - * reinstates (un-cancels?) a previously canceled ticket line item, |
|
| 494 | - * by incrementing it's quantity by 1, and decrementing it's "type_cancellation" sub-line-item. |
|
| 495 | - * ALL totals and subtotals will NEED TO BE UPDATED after performing this action |
|
| 496 | - * |
|
| 497 | - * @param EE_Line_Item $ticket_line_item |
|
| 498 | - * @param int $qty |
|
| 499 | - * @return bool success |
|
| 500 | - * @throws EE_Error |
|
| 501 | - * @throws InvalidArgumentException |
|
| 502 | - * @throws InvalidDataTypeException |
|
| 503 | - * @throws InvalidInterfaceException |
|
| 504 | - * @throws ReflectionException |
|
| 505 | - */ |
|
| 506 | - public static function reinstate_canceled_ticket_line_item(EE_Line_Item $ticket_line_item, $qty = 1) |
|
| 507 | - { |
|
| 508 | - // validate incoming line_item |
|
| 509 | - if ($ticket_line_item->OBJ_type() !== EEM_Line_Item::OBJ_TYPE_TICKET) { |
|
| 510 | - throw new EE_Error( |
|
| 511 | - sprintf( |
|
| 512 | - esc_html__( |
|
| 513 | - 'The supplied line item must have an Object Type of "Ticket", not %1$s.', |
|
| 514 | - 'event_espresso' |
|
| 515 | - ), |
|
| 516 | - $ticket_line_item->type() |
|
| 517 | - ) |
|
| 518 | - ); |
|
| 519 | - } |
|
| 520 | - // get cancellation sub line item |
|
| 521 | - $cancellation_line_item = EEH_Line_Item::get_descendants_of_type( |
|
| 522 | - $ticket_line_item, |
|
| 523 | - EEM_Line_Item::type_cancellation |
|
| 524 | - ); |
|
| 525 | - $cancellation_line_item = reset($cancellation_line_item); |
|
| 526 | - // verify that this ticket was indeed previously cancelled |
|
| 527 | - if (! $cancellation_line_item instanceof EE_Line_Item) { |
|
| 528 | - return false; |
|
| 529 | - } |
|
| 530 | - if ($cancellation_line_item->quantity() > $qty) { |
|
| 531 | - // decrement cancelled quantity |
|
| 532 | - $cancellation_line_item->set_quantity($cancellation_line_item->quantity() - $qty); |
|
| 533 | - } elseif ($cancellation_line_item->quantity() === $qty) { |
|
| 534 | - // decrement cancelled quantity in case anyone still has the object kicking around |
|
| 535 | - $cancellation_line_item->set_quantity($cancellation_line_item->quantity() - $qty); |
|
| 536 | - // delete because quantity will end up as 0 |
|
| 537 | - $cancellation_line_item->delete(); |
|
| 538 | - // and attempt to destroy the object, |
|
| 539 | - // even though PHP won't actually destroy it until it needs the memory |
|
| 540 | - unset($cancellation_line_item); |
|
| 541 | - } else { |
|
| 542 | - // what ?!?! negative quantity ?!?! |
|
| 543 | - throw new EE_Error( |
|
| 544 | - sprintf( |
|
| 545 | - esc_html__( |
|
| 546 | - 'Can not reinstate %1$d cancelled ticket(s) because the cancelled ticket quantity is only %2$d.', |
|
| 547 | - 'event_espresso' |
|
| 548 | - ), |
|
| 549 | - $qty, |
|
| 550 | - $cancellation_line_item->quantity() |
|
| 551 | - ) |
|
| 552 | - ); |
|
| 553 | - } |
|
| 554 | - // increment ticket quantity |
|
| 555 | - $ticket_line_item->set_quantity($ticket_line_item->quantity() + $qty); |
|
| 556 | - if ($ticket_line_item->save_this_and_descendants() > 0) { |
|
| 557 | - // increment parent line item quantity |
|
| 558 | - $event_line_item = $ticket_line_item->parent(); |
|
| 559 | - if ($event_line_item instanceof EE_Line_Item |
|
| 560 | - && $event_line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_EVENT |
|
| 561 | - ) { |
|
| 562 | - $event_line_item->set_quantity($event_line_item->quantity() + $qty); |
|
| 563 | - } |
|
| 564 | - EEH_Line_Item::get_grand_total_and_recalculate_everything($ticket_line_item); |
|
| 565 | - return true; |
|
| 566 | - } |
|
| 567 | - return false; |
|
| 568 | - } |
|
| 569 | - |
|
| 570 | - |
|
| 571 | - /** |
|
| 572 | - * calls EEH_Line_Item::find_transaction_grand_total_for_line_item() |
|
| 573 | - * then EE_Line_Item::recalculate_total_including_taxes() on the result |
|
| 574 | - * |
|
| 575 | - * @param EE_Line_Item $line_item |
|
| 576 | - * @return float |
|
| 577 | - * @throws EE_Error |
|
| 578 | - * @throws InvalidArgumentException |
|
| 579 | - * @throws InvalidDataTypeException |
|
| 580 | - * @throws InvalidInterfaceException |
|
| 581 | - * @throws ReflectionException |
|
| 582 | - */ |
|
| 583 | - public static function get_grand_total_and_recalculate_everything(EE_Line_Item $line_item) |
|
| 584 | - { |
|
| 585 | - $grand_total_line_item = EEH_Line_Item::find_transaction_grand_total_for_line_item($line_item); |
|
| 586 | - return $grand_total_line_item->recalculate_total_including_taxes(); |
|
| 587 | - } |
|
| 588 | - |
|
| 589 | - |
|
| 590 | - /** |
|
| 591 | - * Gets the line item which contains the subtotal of all the items |
|
| 592 | - * |
|
| 593 | - * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 594 | - * @return EE_Line_Item |
|
| 595 | - * @throws EE_Error |
|
| 596 | - * @throws InvalidArgumentException |
|
| 597 | - * @throws InvalidDataTypeException |
|
| 598 | - * @throws InvalidInterfaceException |
|
| 599 | - * @throws ReflectionException |
|
| 600 | - */ |
|
| 601 | - public static function get_pre_tax_subtotal(EE_Line_Item $total_line_item) |
|
| 602 | - { |
|
| 603 | - $pre_tax_subtotal = $total_line_item->get_child_line_item('pre-tax-subtotal'); |
|
| 604 | - return $pre_tax_subtotal instanceof EE_Line_Item |
|
| 605 | - ? $pre_tax_subtotal |
|
| 606 | - : self::create_pre_tax_subtotal($total_line_item); |
|
| 607 | - } |
|
| 608 | - |
|
| 609 | - |
|
| 610 | - /** |
|
| 611 | - * Gets the line item for the taxes subtotal |
|
| 612 | - * |
|
| 613 | - * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 614 | - * @return EE_Line_Item |
|
| 615 | - * @throws EE_Error |
|
| 616 | - * @throws InvalidArgumentException |
|
| 617 | - * @throws InvalidDataTypeException |
|
| 618 | - * @throws InvalidInterfaceException |
|
| 619 | - * @throws ReflectionException |
|
| 620 | - */ |
|
| 621 | - public static function get_taxes_subtotal(EE_Line_Item $total_line_item) |
|
| 622 | - { |
|
| 623 | - $taxes = $total_line_item->get_child_line_item('taxes'); |
|
| 624 | - return $taxes ? $taxes : self::create_taxes_subtotal($total_line_item); |
|
| 625 | - } |
|
| 626 | - |
|
| 627 | - |
|
| 628 | - /** |
|
| 629 | - * sets the TXN ID on an EE_Line_Item if passed a valid EE_Transaction object |
|
| 630 | - * |
|
| 631 | - * @param EE_Line_Item $line_item |
|
| 632 | - * @param EE_Transaction $transaction |
|
| 633 | - * @return void |
|
| 634 | - * @throws EE_Error |
|
| 635 | - * @throws InvalidArgumentException |
|
| 636 | - * @throws InvalidDataTypeException |
|
| 637 | - * @throws InvalidInterfaceException |
|
| 638 | - * @throws ReflectionException |
|
| 639 | - */ |
|
| 640 | - public static function set_TXN_ID(EE_Line_Item $line_item, $transaction = null) |
|
| 641 | - { |
|
| 642 | - if ($transaction) { |
|
| 643 | - /** @type EEM_Transaction $EEM_Transaction */ |
|
| 644 | - $EEM_Transaction = EE_Registry::instance()->load_model('Transaction'); |
|
| 645 | - $TXN_ID = $EEM_Transaction->ensure_is_ID($transaction); |
|
| 646 | - $line_item->set_TXN_ID($TXN_ID); |
|
| 647 | - } |
|
| 648 | - } |
|
| 649 | - |
|
| 650 | - |
|
| 651 | - /** |
|
| 652 | - * Creates a new default total line item for the transaction, |
|
| 653 | - * and its tickets subtotal and taxes subtotal line items (and adds the |
|
| 654 | - * existing taxes as children of the taxes subtotal line item) |
|
| 655 | - * |
|
| 656 | - * @param EE_Transaction $transaction |
|
| 657 | - * @return EE_Line_Item of type total |
|
| 658 | - * @throws EE_Error |
|
| 659 | - * @throws InvalidArgumentException |
|
| 660 | - * @throws InvalidDataTypeException |
|
| 661 | - * @throws InvalidInterfaceException |
|
| 662 | - * @throws ReflectionException |
|
| 663 | - */ |
|
| 664 | - public static function create_total_line_item($transaction = null) |
|
| 665 | - { |
|
| 666 | - $total_line_item = EE_Line_Item::new_instance(array( |
|
| 667 | - 'LIN_code' => 'total', |
|
| 668 | - 'LIN_name' => esc_html__('Grand Total', 'event_espresso'), |
|
| 669 | - 'LIN_type' => EEM_Line_Item::type_total, |
|
| 670 | - 'OBJ_type' => EEM_Line_Item::OBJ_TYPE_TRANSACTION, |
|
| 671 | - )); |
|
| 672 | - $total_line_item = apply_filters( |
|
| 673 | - 'FHEE__EEH_Line_Item__create_total_line_item__total_line_item', |
|
| 674 | - $total_line_item |
|
| 675 | - ); |
|
| 676 | - self::set_TXN_ID($total_line_item, $transaction); |
|
| 677 | - self::create_pre_tax_subtotal($total_line_item, $transaction); |
|
| 678 | - self::create_taxes_subtotal($total_line_item, $transaction); |
|
| 679 | - return $total_line_item; |
|
| 680 | - } |
|
| 681 | - |
|
| 682 | - |
|
| 683 | - /** |
|
| 684 | - * Creates a default items subtotal line item |
|
| 685 | - * |
|
| 686 | - * @param EE_Line_Item $total_line_item |
|
| 687 | - * @param EE_Transaction $transaction |
|
| 688 | - * @return EE_Line_Item |
|
| 689 | - * @throws EE_Error |
|
| 690 | - * @throws InvalidArgumentException |
|
| 691 | - * @throws InvalidDataTypeException |
|
| 692 | - * @throws InvalidInterfaceException |
|
| 693 | - * @throws ReflectionException |
|
| 694 | - */ |
|
| 695 | - protected static function create_pre_tax_subtotal(EE_Line_Item $total_line_item, $transaction = null) |
|
| 696 | - { |
|
| 697 | - $pre_tax_line_item = EE_Line_Item::new_instance(array( |
|
| 698 | - 'LIN_code' => 'pre-tax-subtotal', |
|
| 699 | - 'LIN_name' => esc_html__('Pre-Tax Subtotal', 'event_espresso'), |
|
| 700 | - 'LIN_type' => EEM_Line_Item::type_sub_total, |
|
| 701 | - )); |
|
| 702 | - $pre_tax_line_item = apply_filters( |
|
| 703 | - 'FHEE__EEH_Line_Item__create_pre_tax_subtotal__pre_tax_line_item', |
|
| 704 | - $pre_tax_line_item |
|
| 705 | - ); |
|
| 706 | - self::set_TXN_ID($pre_tax_line_item, $transaction); |
|
| 707 | - $total_line_item->add_child_line_item($pre_tax_line_item); |
|
| 708 | - self::create_event_subtotal($pre_tax_line_item, $transaction); |
|
| 709 | - return $pre_tax_line_item; |
|
| 710 | - } |
|
| 711 | - |
|
| 712 | - |
|
| 713 | - /** |
|
| 714 | - * Creates a line item for the taxes subtotal and finds all the tax prices |
|
| 715 | - * and applies taxes to it |
|
| 716 | - * |
|
| 717 | - * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 718 | - * @param EE_Transaction $transaction |
|
| 719 | - * @return EE_Line_Item |
|
| 720 | - * @throws EE_Error |
|
| 721 | - * @throws InvalidArgumentException |
|
| 722 | - * @throws InvalidDataTypeException |
|
| 723 | - * @throws InvalidInterfaceException |
|
| 724 | - * @throws ReflectionException |
|
| 725 | - */ |
|
| 726 | - protected static function create_taxes_subtotal(EE_Line_Item $total_line_item, $transaction = null) |
|
| 727 | - { |
|
| 728 | - $tax_line_item = EE_Line_Item::new_instance(array( |
|
| 729 | - 'LIN_code' => 'taxes', |
|
| 730 | - 'LIN_name' => esc_html__('Taxes', 'event_espresso'), |
|
| 731 | - 'LIN_type' => EEM_Line_Item::type_tax_sub_total, |
|
| 732 | - 'LIN_order' => 1000,// this should always come last |
|
| 733 | - )); |
|
| 734 | - $tax_line_item = apply_filters( |
|
| 735 | - 'FHEE__EEH_Line_Item__create_taxes_subtotal__tax_line_item', |
|
| 736 | - $tax_line_item |
|
| 737 | - ); |
|
| 738 | - self::set_TXN_ID($tax_line_item, $transaction); |
|
| 739 | - $total_line_item->add_child_line_item($tax_line_item); |
|
| 740 | - // and lastly, add the actual taxes |
|
| 741 | - self::apply_taxes($total_line_item); |
|
| 742 | - return $tax_line_item; |
|
| 743 | - } |
|
| 744 | - |
|
| 745 | - |
|
| 746 | - /** |
|
| 747 | - * Creates a default items subtotal line item |
|
| 748 | - * |
|
| 749 | - * @param EE_Line_Item $pre_tax_line_item |
|
| 750 | - * @param EE_Transaction $transaction |
|
| 751 | - * @param EE_Event $event |
|
| 752 | - * @return EE_Line_Item |
|
| 753 | - * @throws EE_Error |
|
| 754 | - * @throws InvalidArgumentException |
|
| 755 | - * @throws InvalidDataTypeException |
|
| 756 | - * @throws InvalidInterfaceException |
|
| 757 | - * @throws ReflectionException |
|
| 758 | - */ |
|
| 759 | - public static function create_event_subtotal(EE_Line_Item $pre_tax_line_item, $transaction = null, $event = null) |
|
| 760 | - { |
|
| 761 | - $event_line_item = EE_Line_Item::new_instance(array( |
|
| 762 | - 'LIN_code' => self::get_event_code($event), |
|
| 763 | - 'LIN_name' => self::get_event_name($event), |
|
| 764 | - 'LIN_desc' => self::get_event_desc($event), |
|
| 765 | - 'LIN_type' => EEM_Line_Item::type_sub_total, |
|
| 766 | - 'OBJ_type' => EEM_Line_Item::OBJ_TYPE_EVENT, |
|
| 767 | - 'OBJ_ID' => $event instanceof EE_Event ? $event->ID() : 0, |
|
| 768 | - )); |
|
| 769 | - $event_line_item = apply_filters( |
|
| 770 | - 'FHEE__EEH_Line_Item__create_event_subtotal__event_line_item', |
|
| 771 | - $event_line_item |
|
| 772 | - ); |
|
| 773 | - self::set_TXN_ID($event_line_item, $transaction); |
|
| 774 | - $pre_tax_line_item->add_child_line_item($event_line_item); |
|
| 775 | - return $event_line_item; |
|
| 776 | - } |
|
| 777 | - |
|
| 778 | - |
|
| 779 | - /** |
|
| 780 | - * Gets what the event ticket's code SHOULD be |
|
| 781 | - * |
|
| 782 | - * @param EE_Event $event |
|
| 783 | - * @return string |
|
| 784 | - * @throws EE_Error |
|
| 785 | - */ |
|
| 786 | - public static function get_event_code($event) |
|
| 787 | - { |
|
| 788 | - return 'event-' . ($event instanceof EE_Event ? $event->ID() : '0'); |
|
| 789 | - } |
|
| 790 | - |
|
| 791 | - |
|
| 792 | - /** |
|
| 793 | - * Gets the event name |
|
| 794 | - * |
|
| 795 | - * @param EE_Event $event |
|
| 796 | - * @return string |
|
| 797 | - * @throws EE_Error |
|
| 798 | - */ |
|
| 799 | - public static function get_event_name($event) |
|
| 800 | - { |
|
| 801 | - return $event instanceof EE_Event |
|
| 802 | - ? mb_substr($event->name(), 0, 245) |
|
| 803 | - : esc_html__('Event', 'event_espresso'); |
|
| 804 | - } |
|
| 805 | - |
|
| 806 | - |
|
| 807 | - /** |
|
| 808 | - * Gets the event excerpt |
|
| 809 | - * |
|
| 810 | - * @param EE_Event $event |
|
| 811 | - * @return string |
|
| 812 | - * @throws EE_Error |
|
| 813 | - */ |
|
| 814 | - public static function get_event_desc($event) |
|
| 815 | - { |
|
| 816 | - return $event instanceof EE_Event ? $event->short_description() : ''; |
|
| 817 | - } |
|
| 818 | - |
|
| 819 | - |
|
| 820 | - /** |
|
| 821 | - * Given the grand total line item and a ticket, finds the event sub-total |
|
| 822 | - * line item the ticket's purchase should be added onto |
|
| 823 | - * |
|
| 824 | - * @access public |
|
| 825 | - * @param EE_Line_Item $grand_total the grand total line item |
|
| 826 | - * @param EE_Ticket $ticket |
|
| 827 | - * @return EE_Line_Item |
|
| 828 | - * @throws EE_Error |
|
| 829 | - * @throws InvalidArgumentException |
|
| 830 | - * @throws InvalidDataTypeException |
|
| 831 | - * @throws InvalidInterfaceException |
|
| 832 | - * @throws ReflectionException |
|
| 833 | - */ |
|
| 834 | - public static function get_event_line_item_for_ticket(EE_Line_Item $grand_total, EE_Ticket $ticket) |
|
| 835 | - { |
|
| 836 | - $first_datetime = $ticket->first_datetime(); |
|
| 837 | - if (! $first_datetime instanceof EE_Datetime) { |
|
| 838 | - throw new EE_Error( |
|
| 839 | - sprintf( |
|
| 840 | - __('The supplied ticket (ID %d) has no datetimes', 'event_espresso'), |
|
| 841 | - $ticket->ID() |
|
| 842 | - ) |
|
| 843 | - ); |
|
| 844 | - } |
|
| 845 | - $event = $first_datetime->event(); |
|
| 846 | - if (! $event instanceof EE_Event) { |
|
| 847 | - throw new EE_Error( |
|
| 848 | - sprintf( |
|
| 849 | - esc_html__( |
|
| 850 | - 'The supplied ticket (ID %d) has no event data associated with it.', |
|
| 851 | - 'event_espresso' |
|
| 852 | - ), |
|
| 853 | - $ticket->ID() |
|
| 854 | - ) |
|
| 855 | - ); |
|
| 856 | - } |
|
| 857 | - $events_sub_total = EEH_Line_Item::get_event_line_item($grand_total, $event); |
|
| 858 | - if (! $events_sub_total instanceof EE_Line_Item) { |
|
| 859 | - throw new EE_Error( |
|
| 860 | - sprintf( |
|
| 861 | - esc_html__( |
|
| 862 | - 'There is no events sub-total for ticket %s on total line item %d', |
|
| 863 | - 'event_espresso' |
|
| 864 | - ), |
|
| 865 | - $ticket->ID(), |
|
| 866 | - $grand_total->ID() |
|
| 867 | - ) |
|
| 868 | - ); |
|
| 869 | - } |
|
| 870 | - return $events_sub_total; |
|
| 871 | - } |
|
| 872 | - |
|
| 873 | - |
|
| 874 | - /** |
|
| 875 | - * Gets the event line item |
|
| 876 | - * |
|
| 877 | - * @param EE_Line_Item $grand_total |
|
| 878 | - * @param EE_Event $event |
|
| 879 | - * @return EE_Line_Item for the event subtotal which is a child of $grand_total |
|
| 880 | - * @throws EE_Error |
|
| 881 | - * @throws InvalidArgumentException |
|
| 882 | - * @throws InvalidDataTypeException |
|
| 883 | - * @throws InvalidInterfaceException |
|
| 884 | - * @throws ReflectionException |
|
| 885 | - */ |
|
| 886 | - public static function get_event_line_item(EE_Line_Item $grand_total, $event) |
|
| 887 | - { |
|
| 888 | - /** @type EE_Event $event */ |
|
| 889 | - $event = EEM_Event::instance()->ensure_is_obj($event, true); |
|
| 890 | - $event_line_item = null; |
|
| 891 | - $found = false; |
|
| 892 | - foreach (EEH_Line_Item::get_event_subtotals($grand_total) as $event_line_item) { |
|
| 893 | - // default event subtotal, we should only ever find this the first time this method is called |
|
| 894 | - if (! $event_line_item->OBJ_ID()) { |
|
| 895 | - // let's use this! but first... set the event details |
|
| 896 | - EEH_Line_Item::set_event_subtotal_details($event_line_item, $event); |
|
| 897 | - $found = true; |
|
| 898 | - break; |
|
| 899 | - } |
|
| 900 | - if ($event_line_item->OBJ_ID() === $event->ID()) { |
|
| 901 | - // found existing line item for this event in the cart, so break out of loop and use this one |
|
| 902 | - $found = true; |
|
| 903 | - break; |
|
| 904 | - } |
|
| 905 | - } |
|
| 906 | - if (! $found) { |
|
| 907 | - // there is no event sub-total yet, so add it |
|
| 908 | - $pre_tax_subtotal = EEH_Line_Item::get_pre_tax_subtotal($grand_total); |
|
| 909 | - // create a new "event" subtotal below that |
|
| 910 | - $event_line_item = EEH_Line_Item::create_event_subtotal($pre_tax_subtotal, null, $event); |
|
| 911 | - // and set the event details |
|
| 912 | - EEH_Line_Item::set_event_subtotal_details($event_line_item, $event); |
|
| 913 | - } |
|
| 914 | - return $event_line_item; |
|
| 915 | - } |
|
| 916 | - |
|
| 917 | - |
|
| 918 | - /** |
|
| 919 | - * Creates a default items subtotal line item |
|
| 920 | - * |
|
| 921 | - * @param EE_Line_Item $event_line_item |
|
| 922 | - * @param EE_Event $event |
|
| 923 | - * @param EE_Transaction $transaction |
|
| 924 | - * @return void |
|
| 925 | - * @throws EE_Error |
|
| 926 | - * @throws InvalidArgumentException |
|
| 927 | - * @throws InvalidDataTypeException |
|
| 928 | - * @throws InvalidInterfaceException |
|
| 929 | - * @throws ReflectionException |
|
| 930 | - */ |
|
| 931 | - public static function set_event_subtotal_details( |
|
| 932 | - EE_Line_Item $event_line_item, |
|
| 933 | - EE_Event $event, |
|
| 934 | - $transaction = null |
|
| 935 | - ) { |
|
| 936 | - if ($event instanceof EE_Event) { |
|
| 937 | - $event_line_item->set_code(self::get_event_code($event)); |
|
| 938 | - $event_line_item->set_name(self::get_event_name($event)); |
|
| 939 | - $event_line_item->set_desc(self::get_event_desc($event)); |
|
| 940 | - $event_line_item->set_OBJ_ID($event->ID()); |
|
| 941 | - } |
|
| 942 | - self::set_TXN_ID($event_line_item, $transaction); |
|
| 943 | - } |
|
| 944 | - |
|
| 945 | - |
|
| 946 | - /** |
|
| 947 | - * Finds what taxes should apply, adds them as tax line items under the taxes sub-total, |
|
| 948 | - * and recalculates the taxes sub-total and the grand total. Resets the taxes, so |
|
| 949 | - * any old taxes are removed |
|
| 950 | - * |
|
| 951 | - * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 952 | - * @param bool $update_txn_status |
|
| 953 | - * @return bool |
|
| 954 | - * @throws EE_Error |
|
| 955 | - * @throws InvalidArgumentException |
|
| 956 | - * @throws InvalidDataTypeException |
|
| 957 | - * @throws InvalidInterfaceException |
|
| 958 | - * @throws ReflectionException |
|
| 959 | - * @throws RuntimeException |
|
| 960 | - */ |
|
| 961 | - public static function apply_taxes(EE_Line_Item $total_line_item, $update_txn_status = false) |
|
| 962 | - { |
|
| 963 | - /** @type EEM_Price $EEM_Price */ |
|
| 964 | - $EEM_Price = EE_Registry::instance()->load_model('Price'); |
|
| 965 | - // get array of taxes via Price Model |
|
| 966 | - $ordered_taxes = $EEM_Price->get_all_prices_that_are_taxes(); |
|
| 967 | - ksort($ordered_taxes); |
|
| 968 | - $taxes_line_item = self::get_taxes_subtotal($total_line_item); |
|
| 969 | - // just to be safe, remove its old tax line items |
|
| 970 | - $deleted = $taxes_line_item->delete_children_line_items(); |
|
| 971 | - $updates = false; |
|
| 972 | - // loop thru taxes |
|
| 973 | - foreach ($ordered_taxes as $order => $taxes) { |
|
| 974 | - foreach ($taxes as $tax) { |
|
| 975 | - if ($tax instanceof EE_Price) { |
|
| 976 | - $tax_line_item = EE_Line_Item::new_instance( |
|
| 977 | - array( |
|
| 978 | - 'LIN_name' => $tax->name(), |
|
| 979 | - 'LIN_desc' => $tax->desc(), |
|
| 980 | - 'LIN_percent' => $tax->amount(), |
|
| 981 | - 'LIN_is_taxable' => false, |
|
| 982 | - 'LIN_order' => $order, |
|
| 983 | - 'LIN_total' => 0, |
|
| 984 | - 'LIN_type' => EEM_Line_Item::type_tax, |
|
| 985 | - 'OBJ_type' => EEM_Line_Item::OBJ_TYPE_PRICE, |
|
| 986 | - 'OBJ_ID' => $tax->ID(), |
|
| 987 | - ) |
|
| 988 | - ); |
|
| 989 | - $tax_line_item = apply_filters( |
|
| 990 | - 'FHEE__EEH_Line_Item__apply_taxes__tax_line_item', |
|
| 991 | - $tax_line_item |
|
| 992 | - ); |
|
| 993 | - $updates = $taxes_line_item->add_child_line_item($tax_line_item) ? |
|
| 994 | - true : |
|
| 995 | - $updates; |
|
| 996 | - } |
|
| 997 | - } |
|
| 998 | - } |
|
| 999 | - // only recalculate totals if something changed |
|
| 1000 | - if ($deleted || $updates) { |
|
| 1001 | - $total_line_item->recalculate_total_including_taxes($update_txn_status); |
|
| 1002 | - return true; |
|
| 1003 | - } |
|
| 1004 | - return false; |
|
| 1005 | - } |
|
| 1006 | - |
|
| 1007 | - |
|
| 1008 | - /** |
|
| 1009 | - * Ensures that taxes have been applied to the order, if not applies them. |
|
| 1010 | - * Returns the total amount of tax |
|
| 1011 | - * |
|
| 1012 | - * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 1013 | - * @return float |
|
| 1014 | - * @throws EE_Error |
|
| 1015 | - * @throws InvalidArgumentException |
|
| 1016 | - * @throws InvalidDataTypeException |
|
| 1017 | - * @throws InvalidInterfaceException |
|
| 1018 | - * @throws ReflectionException |
|
| 1019 | - */ |
|
| 1020 | - public static function ensure_taxes_applied($total_line_item) |
|
| 1021 | - { |
|
| 1022 | - $taxes_subtotal = self::get_taxes_subtotal($total_line_item); |
|
| 1023 | - if (! $taxes_subtotal->children()) { |
|
| 1024 | - self::apply_taxes($total_line_item); |
|
| 1025 | - } |
|
| 1026 | - return $taxes_subtotal->total(); |
|
| 1027 | - } |
|
| 1028 | - |
|
| 1029 | - |
|
| 1030 | - /** |
|
| 1031 | - * Deletes ALL children of the passed line item |
|
| 1032 | - * |
|
| 1033 | - * @param EE_Line_Item $parent_line_item |
|
| 1034 | - * @return bool |
|
| 1035 | - * @throws EE_Error |
|
| 1036 | - * @throws InvalidArgumentException |
|
| 1037 | - * @throws InvalidDataTypeException |
|
| 1038 | - * @throws InvalidInterfaceException |
|
| 1039 | - * @throws ReflectionException |
|
| 1040 | - */ |
|
| 1041 | - public static function delete_all_child_items(EE_Line_Item $parent_line_item) |
|
| 1042 | - { |
|
| 1043 | - $deleted = 0; |
|
| 1044 | - foreach ($parent_line_item->children() as $child_line_item) { |
|
| 1045 | - if ($child_line_item instanceof EE_Line_Item) { |
|
| 1046 | - $deleted += EEH_Line_Item::delete_all_child_items($child_line_item); |
|
| 1047 | - if ($child_line_item->ID()) { |
|
| 1048 | - $child_line_item->delete(); |
|
| 1049 | - unset($child_line_item); |
|
| 1050 | - } else { |
|
| 1051 | - $parent_line_item->delete_child_line_item($child_line_item->code()); |
|
| 1052 | - } |
|
| 1053 | - $deleted++; |
|
| 1054 | - } |
|
| 1055 | - } |
|
| 1056 | - return $deleted; |
|
| 1057 | - } |
|
| 1058 | - |
|
| 1059 | - |
|
| 1060 | - /** |
|
| 1061 | - * Deletes the line items as indicated by the line item code(s) provided, |
|
| 1062 | - * regardless of where they're found in the line item tree. Automatically |
|
| 1063 | - * re-calculates the line item totals and updates the related transaction. But |
|
| 1064 | - * DOES NOT automatically upgrade the transaction's registrations' final prices (which |
|
| 1065 | - * should probably change because of this). |
|
| 1066 | - * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item() |
|
| 1067 | - * after using this, to keep the registration final prices in-sync with the transaction's total. |
|
| 1068 | - * |
|
| 1069 | - * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 1070 | - * @param array|bool|string $line_item_codes |
|
| 1071 | - * @return int number of items successfully removed |
|
| 1072 | - * @throws EE_Error |
|
| 1073 | - */ |
|
| 1074 | - public static function delete_items(EE_Line_Item $total_line_item, $line_item_codes = false) |
|
| 1075 | - { |
|
| 1076 | - |
|
| 1077 | - if ($total_line_item->type() !== EEM_Line_Item::type_total) { |
|
| 1078 | - EE_Error::doing_it_wrong( |
|
| 1079 | - 'EEH_Line_Item::delete_items', |
|
| 1080 | - esc_html__( |
|
| 1081 | - 'This static method should only be called with a TOTAL line item, otherwise we won\'t recalculate the totals correctly', |
|
| 1082 | - 'event_espresso' |
|
| 1083 | - ), |
|
| 1084 | - '4.6.18' |
|
| 1085 | - ); |
|
| 1086 | - } |
|
| 1087 | - do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
| 1088 | - |
|
| 1089 | - // check if only a single line_item_id was passed |
|
| 1090 | - if (! empty($line_item_codes) && ! is_array($line_item_codes)) { |
|
| 1091 | - // place single line_item_id in an array to appear as multiple line_item_ids |
|
| 1092 | - $line_item_codes = array($line_item_codes); |
|
| 1093 | - } |
|
| 1094 | - $removals = 0; |
|
| 1095 | - // cycle thru line_item_ids |
|
| 1096 | - foreach ($line_item_codes as $line_item_id) { |
|
| 1097 | - $removals += $total_line_item->delete_child_line_item($line_item_id); |
|
| 1098 | - } |
|
| 1099 | - |
|
| 1100 | - if ($removals > 0) { |
|
| 1101 | - $total_line_item->recalculate_taxes_and_tax_total(); |
|
| 1102 | - return $removals; |
|
| 1103 | - } else { |
|
| 1104 | - return false; |
|
| 1105 | - } |
|
| 1106 | - } |
|
| 1107 | - |
|
| 1108 | - |
|
| 1109 | - /** |
|
| 1110 | - * Overwrites the previous tax by clearing out the old taxes, and creates a new |
|
| 1111 | - * tax and updates the total line item accordingly |
|
| 1112 | - * |
|
| 1113 | - * @param EE_Line_Item $total_line_item |
|
| 1114 | - * @param float $amount |
|
| 1115 | - * @param string $name |
|
| 1116 | - * @param string $description |
|
| 1117 | - * @param string $code |
|
| 1118 | - * @param boolean $add_to_existing_line_item |
|
| 1119 | - * if true, and a duplicate line item with the same code is found, |
|
| 1120 | - * $amount will be added onto it; otherwise will simply set the taxes to match $amount |
|
| 1121 | - * @return EE_Line_Item the new tax line item created |
|
| 1122 | - * @throws EE_Error |
|
| 1123 | - * @throws InvalidArgumentException |
|
| 1124 | - * @throws InvalidDataTypeException |
|
| 1125 | - * @throws InvalidInterfaceException |
|
| 1126 | - * @throws ReflectionException |
|
| 1127 | - */ |
|
| 1128 | - public static function set_total_tax_to( |
|
| 1129 | - EE_Line_Item $total_line_item, |
|
| 1130 | - $amount, |
|
| 1131 | - $name = null, |
|
| 1132 | - $description = null, |
|
| 1133 | - $code = null, |
|
| 1134 | - $add_to_existing_line_item = false |
|
| 1135 | - ) { |
|
| 1136 | - $tax_subtotal = self::get_taxes_subtotal($total_line_item); |
|
| 1137 | - $taxable_total = $total_line_item->taxable_total(); |
|
| 1138 | - |
|
| 1139 | - if ($add_to_existing_line_item) { |
|
| 1140 | - $new_tax = $tax_subtotal->get_child_line_item($code); |
|
| 1141 | - EEM_Line_Item::instance()->delete( |
|
| 1142 | - array(array('LIN_code' => array('!=', $code), 'LIN_parent' => $tax_subtotal->ID())) |
|
| 1143 | - ); |
|
| 1144 | - } else { |
|
| 1145 | - $new_tax = null; |
|
| 1146 | - $tax_subtotal->delete_children_line_items(); |
|
| 1147 | - } |
|
| 1148 | - if ($new_tax) { |
|
| 1149 | - $new_tax->set_total($new_tax->total() + $amount); |
|
| 1150 | - $new_tax->set_percent($taxable_total ? $new_tax->total() / $taxable_total * 100 : 0); |
|
| 1151 | - } else { |
|
| 1152 | - // no existing tax item. Create it |
|
| 1153 | - $new_tax = EE_Line_Item::new_instance(array( |
|
| 1154 | - 'TXN_ID' => $total_line_item->TXN_ID(), |
|
| 1155 | - 'LIN_name' => $name ? $name : esc_html__('Tax', 'event_espresso'), |
|
| 1156 | - 'LIN_desc' => $description ? $description : '', |
|
| 1157 | - 'LIN_percent' => $taxable_total ? ($amount / $taxable_total * 100) : 0, |
|
| 1158 | - 'LIN_total' => $amount, |
|
| 1159 | - 'LIN_parent' => $tax_subtotal->ID(), |
|
| 1160 | - 'LIN_type' => EEM_Line_Item::type_tax, |
|
| 1161 | - 'LIN_code' => $code, |
|
| 1162 | - )); |
|
| 1163 | - } |
|
| 1164 | - |
|
| 1165 | - $new_tax = apply_filters( |
|
| 1166 | - 'FHEE__EEH_Line_Item__set_total_tax_to__new_tax_subtotal', |
|
| 1167 | - $new_tax, |
|
| 1168 | - $total_line_item |
|
| 1169 | - ); |
|
| 1170 | - $new_tax->save(); |
|
| 1171 | - $tax_subtotal->set_total($new_tax->total()); |
|
| 1172 | - $tax_subtotal->save(); |
|
| 1173 | - $total_line_item->recalculate_total_including_taxes(); |
|
| 1174 | - return $new_tax; |
|
| 1175 | - } |
|
| 1176 | - |
|
| 1177 | - |
|
| 1178 | - /** |
|
| 1179 | - * Makes all the line items which are children of $line_item taxable (or not). |
|
| 1180 | - * Does NOT save the line items |
|
| 1181 | - * |
|
| 1182 | - * @param EE_Line_Item $line_item |
|
| 1183 | - * @param boolean $taxable |
|
| 1184 | - * @param string $code_substring_for_whitelist if this string is part of the line item's code |
|
| 1185 | - * it will be whitelisted (ie, except from becoming taxable) |
|
| 1186 | - * @throws EE_Error |
|
| 1187 | - */ |
|
| 1188 | - public static function set_line_items_taxable( |
|
| 1189 | - EE_Line_Item $line_item, |
|
| 1190 | - $taxable = true, |
|
| 1191 | - $code_substring_for_whitelist = null |
|
| 1192 | - ) { |
|
| 1193 | - $whitelisted = false; |
|
| 1194 | - if ($code_substring_for_whitelist !== null) { |
|
| 1195 | - $whitelisted = strpos($line_item->code(), $code_substring_for_whitelist) !== false; |
|
| 1196 | - } |
|
| 1197 | - if (! $whitelisted && $line_item->is_line_item()) { |
|
| 1198 | - $line_item->set_is_taxable($taxable); |
|
| 1199 | - } |
|
| 1200 | - foreach ($line_item->children() as $child_line_item) { |
|
| 1201 | - EEH_Line_Item::set_line_items_taxable( |
|
| 1202 | - $child_line_item, |
|
| 1203 | - $taxable, |
|
| 1204 | - $code_substring_for_whitelist |
|
| 1205 | - ); |
|
| 1206 | - } |
|
| 1207 | - } |
|
| 1208 | - |
|
| 1209 | - |
|
| 1210 | - /** |
|
| 1211 | - * Gets all descendants that are event subtotals |
|
| 1212 | - * |
|
| 1213 | - * @uses EEH_Line_Item::get_subtotals_of_object_type() |
|
| 1214 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1215 | - * @return EE_Line_Item[] |
|
| 1216 | - * @throws EE_Error |
|
| 1217 | - */ |
|
| 1218 | - public static function get_event_subtotals(EE_Line_Item $parent_line_item) |
|
| 1219 | - { |
|
| 1220 | - return self::get_subtotals_of_object_type($parent_line_item, EEM_Line_Item::OBJ_TYPE_EVENT); |
|
| 1221 | - } |
|
| 1222 | - |
|
| 1223 | - |
|
| 1224 | - /** |
|
| 1225 | - * Gets all descendants subtotals that match the supplied object type |
|
| 1226 | - * |
|
| 1227 | - * @uses EEH_Line_Item::_get_descendants_by_type_and_object_type() |
|
| 1228 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1229 | - * @param string $obj_type |
|
| 1230 | - * @return EE_Line_Item[] |
|
| 1231 | - * @throws EE_Error |
|
| 1232 | - */ |
|
| 1233 | - public static function get_subtotals_of_object_type(EE_Line_Item $parent_line_item, $obj_type = '') |
|
| 1234 | - { |
|
| 1235 | - return self::_get_descendants_by_type_and_object_type( |
|
| 1236 | - $parent_line_item, |
|
| 1237 | - EEM_Line_Item::type_sub_total, |
|
| 1238 | - $obj_type |
|
| 1239 | - ); |
|
| 1240 | - } |
|
| 1241 | - |
|
| 1242 | - |
|
| 1243 | - /** |
|
| 1244 | - * Gets all descendants that are tickets |
|
| 1245 | - * |
|
| 1246 | - * @uses EEH_Line_Item::get_line_items_of_object_type() |
|
| 1247 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1248 | - * @return EE_Line_Item[] |
|
| 1249 | - * @throws EE_Error |
|
| 1250 | - */ |
|
| 1251 | - public static function get_ticket_line_items(EE_Line_Item $parent_line_item) |
|
| 1252 | - { |
|
| 1253 | - return self::get_line_items_of_object_type( |
|
| 1254 | - $parent_line_item, |
|
| 1255 | - EEM_Line_Item::OBJ_TYPE_TICKET |
|
| 1256 | - ); |
|
| 1257 | - } |
|
| 1258 | - |
|
| 1259 | - |
|
| 1260 | - /** |
|
| 1261 | - * Gets all descendants subtotals that match the supplied object type |
|
| 1262 | - * |
|
| 1263 | - * @uses EEH_Line_Item::_get_descendants_by_type_and_object_type() |
|
| 1264 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1265 | - * @param string $obj_type |
|
| 1266 | - * @return EE_Line_Item[] |
|
| 1267 | - * @throws EE_Error |
|
| 1268 | - */ |
|
| 1269 | - public static function get_line_items_of_object_type(EE_Line_Item $parent_line_item, $obj_type = '') |
|
| 1270 | - { |
|
| 1271 | - return self::_get_descendants_by_type_and_object_type( |
|
| 1272 | - $parent_line_item, |
|
| 1273 | - EEM_Line_Item::type_line_item, |
|
| 1274 | - $obj_type |
|
| 1275 | - ); |
|
| 1276 | - } |
|
| 1277 | - |
|
| 1278 | - |
|
| 1279 | - /** |
|
| 1280 | - * Gets all the descendants (ie, children or children of children etc) that are of the type 'tax' |
|
| 1281 | - * |
|
| 1282 | - * @uses EEH_Line_Item::get_descendants_of_type() |
|
| 1283 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1284 | - * @return EE_Line_Item[] |
|
| 1285 | - * @throws EE_Error |
|
| 1286 | - */ |
|
| 1287 | - public static function get_tax_descendants(EE_Line_Item $parent_line_item) |
|
| 1288 | - { |
|
| 1289 | - return EEH_Line_Item::get_descendants_of_type( |
|
| 1290 | - $parent_line_item, |
|
| 1291 | - EEM_Line_Item::type_tax |
|
| 1292 | - ); |
|
| 1293 | - } |
|
| 1294 | - |
|
| 1295 | - |
|
| 1296 | - /** |
|
| 1297 | - * Gets all the real items purchased which are children of this item |
|
| 1298 | - * |
|
| 1299 | - * @uses EEH_Line_Item::get_descendants_of_type() |
|
| 1300 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1301 | - * @return EE_Line_Item[] |
|
| 1302 | - * @throws EE_Error |
|
| 1303 | - */ |
|
| 1304 | - public static function get_line_item_descendants(EE_Line_Item $parent_line_item) |
|
| 1305 | - { |
|
| 1306 | - return EEH_Line_Item::get_descendants_of_type( |
|
| 1307 | - $parent_line_item, |
|
| 1308 | - EEM_Line_Item::type_line_item |
|
| 1309 | - ); |
|
| 1310 | - } |
|
| 1311 | - |
|
| 1312 | - |
|
| 1313 | - /** |
|
| 1314 | - * Gets all descendants of supplied line item that match the supplied line item type |
|
| 1315 | - * |
|
| 1316 | - * @uses EEH_Line_Item::_get_descendants_by_type_and_object_type() |
|
| 1317 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1318 | - * @param string $line_item_type one of the EEM_Line_Item constants |
|
| 1319 | - * @return EE_Line_Item[] |
|
| 1320 | - * @throws EE_Error |
|
| 1321 | - */ |
|
| 1322 | - public static function get_descendants_of_type(EE_Line_Item $parent_line_item, $line_item_type) |
|
| 1323 | - { |
|
| 1324 | - return self::_get_descendants_by_type_and_object_type( |
|
| 1325 | - $parent_line_item, |
|
| 1326 | - $line_item_type, |
|
| 1327 | - null |
|
| 1328 | - ); |
|
| 1329 | - } |
|
| 1330 | - |
|
| 1331 | - |
|
| 1332 | - /** |
|
| 1333 | - * Gets all descendants of supplied line item that match the supplied line item type and possibly the object type |
|
| 1334 | - * as well |
|
| 1335 | - * |
|
| 1336 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1337 | - * @param string $line_item_type one of the EEM_Line_Item constants |
|
| 1338 | - * @param string | NULL $obj_type object model class name (minus prefix) or NULL to ignore object type when |
|
| 1339 | - * searching |
|
| 1340 | - * @return EE_Line_Item[] |
|
| 1341 | - * @throws EE_Error |
|
| 1342 | - */ |
|
| 1343 | - protected static function _get_descendants_by_type_and_object_type( |
|
| 1344 | - EE_Line_Item $parent_line_item, |
|
| 1345 | - $line_item_type, |
|
| 1346 | - $obj_type = null |
|
| 1347 | - ) { |
|
| 1348 | - $objects = array(); |
|
| 1349 | - foreach ($parent_line_item->children() as $child_line_item) { |
|
| 1350 | - if ($child_line_item instanceof EE_Line_Item) { |
|
| 1351 | - if ($child_line_item->type() === $line_item_type |
|
| 1352 | - && ( |
|
| 1353 | - $child_line_item->OBJ_type() === $obj_type || $obj_type === null |
|
| 1354 | - ) |
|
| 1355 | - ) { |
|
| 1356 | - $objects[] = $child_line_item; |
|
| 1357 | - } else { |
|
| 1358 | - // go-through-all-its children looking for more matches |
|
| 1359 | - $objects = array_merge( |
|
| 1360 | - $objects, |
|
| 1361 | - self::_get_descendants_by_type_and_object_type( |
|
| 1362 | - $child_line_item, |
|
| 1363 | - $line_item_type, |
|
| 1364 | - $obj_type |
|
| 1365 | - ) |
|
| 1366 | - ); |
|
| 1367 | - } |
|
| 1368 | - } |
|
| 1369 | - } |
|
| 1370 | - return $objects; |
|
| 1371 | - } |
|
| 1372 | - |
|
| 1373 | - |
|
| 1374 | - /** |
|
| 1375 | - * Gets all descendants subtotals that match the supplied object type |
|
| 1376 | - * |
|
| 1377 | - * @uses EEH_Line_Item::_get_descendants_by_type_and_object_type() |
|
| 1378 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1379 | - * @param string $OBJ_type object type (like Event) |
|
| 1380 | - * @param array $OBJ_IDs array of OBJ_IDs |
|
| 1381 | - * @return EE_Line_Item[] |
|
| 1382 | - * @throws EE_Error |
|
| 1383 | - */ |
|
| 1384 | - public static function get_line_items_by_object_type_and_IDs( |
|
| 1385 | - EE_Line_Item $parent_line_item, |
|
| 1386 | - $OBJ_type = '', |
|
| 1387 | - $OBJ_IDs = array() |
|
| 1388 | - ) { |
|
| 1389 | - return self::_get_descendants_by_object_type_and_object_ID( |
|
| 1390 | - $parent_line_item, |
|
| 1391 | - $OBJ_type, |
|
| 1392 | - $OBJ_IDs |
|
| 1393 | - ); |
|
| 1394 | - } |
|
| 1395 | - |
|
| 1396 | - |
|
| 1397 | - /** |
|
| 1398 | - * Gets all descendants of supplied line item that match the supplied line item type and possibly the object type |
|
| 1399 | - * as well |
|
| 1400 | - * |
|
| 1401 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1402 | - * @param string $OBJ_type object type (like Event) |
|
| 1403 | - * @param array $OBJ_IDs array of OBJ_IDs |
|
| 1404 | - * @return EE_Line_Item[] |
|
| 1405 | - * @throws EE_Error |
|
| 1406 | - */ |
|
| 1407 | - protected static function _get_descendants_by_object_type_and_object_ID( |
|
| 1408 | - EE_Line_Item $parent_line_item, |
|
| 1409 | - $OBJ_type, |
|
| 1410 | - $OBJ_IDs |
|
| 1411 | - ) { |
|
| 1412 | - $objects = array(); |
|
| 1413 | - foreach ($parent_line_item->children() as $child_line_item) { |
|
| 1414 | - if ($child_line_item instanceof EE_Line_Item) { |
|
| 1415 | - if ($child_line_item->OBJ_type() === $OBJ_type |
|
| 1416 | - && is_array($OBJ_IDs) |
|
| 1417 | - && in_array($child_line_item->OBJ_ID(), $OBJ_IDs) |
|
| 1418 | - ) { |
|
| 1419 | - $objects[] = $child_line_item; |
|
| 1420 | - } else { |
|
| 1421 | - // go-through-all-its children looking for more matches |
|
| 1422 | - $objects = array_merge( |
|
| 1423 | - $objects, |
|
| 1424 | - self::_get_descendants_by_object_type_and_object_ID( |
|
| 1425 | - $child_line_item, |
|
| 1426 | - $OBJ_type, |
|
| 1427 | - $OBJ_IDs |
|
| 1428 | - ) |
|
| 1429 | - ); |
|
| 1430 | - } |
|
| 1431 | - } |
|
| 1432 | - } |
|
| 1433 | - return $objects; |
|
| 1434 | - } |
|
| 1435 | - |
|
| 1436 | - |
|
| 1437 | - /** |
|
| 1438 | - * Uses a breadth-first-search in order to find the nearest descendant of |
|
| 1439 | - * the specified type and returns it, else NULL |
|
| 1440 | - * |
|
| 1441 | - * @uses EEH_Line_Item::_get_nearest_descendant() |
|
| 1442 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1443 | - * @param string $type like one of the EEM_Line_Item::type_* |
|
| 1444 | - * @return EE_Line_Item |
|
| 1445 | - * @throws EE_Error |
|
| 1446 | - * @throws InvalidArgumentException |
|
| 1447 | - * @throws InvalidDataTypeException |
|
| 1448 | - * @throws InvalidInterfaceException |
|
| 1449 | - * @throws ReflectionException |
|
| 1450 | - */ |
|
| 1451 | - public static function get_nearest_descendant_of_type(EE_Line_Item $parent_line_item, $type) |
|
| 1452 | - { |
|
| 1453 | - return self::_get_nearest_descendant($parent_line_item, 'LIN_type', $type); |
|
| 1454 | - } |
|
| 1455 | - |
|
| 1456 | - |
|
| 1457 | - /** |
|
| 1458 | - * Uses a breadth-first-search in order to find the nearest descendant |
|
| 1459 | - * having the specified LIN_code and returns it, else NULL |
|
| 1460 | - * |
|
| 1461 | - * @uses EEH_Line_Item::_get_nearest_descendant() |
|
| 1462 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1463 | - * @param string $code any value used for LIN_code |
|
| 1464 | - * @return EE_Line_Item |
|
| 1465 | - * @throws EE_Error |
|
| 1466 | - * @throws InvalidArgumentException |
|
| 1467 | - * @throws InvalidDataTypeException |
|
| 1468 | - * @throws InvalidInterfaceException |
|
| 1469 | - * @throws ReflectionException |
|
| 1470 | - */ |
|
| 1471 | - public static function get_nearest_descendant_having_code(EE_Line_Item $parent_line_item, $code) |
|
| 1472 | - { |
|
| 1473 | - return self::_get_nearest_descendant($parent_line_item, 'LIN_code', $code); |
|
| 1474 | - } |
|
| 1475 | - |
|
| 1476 | - |
|
| 1477 | - /** |
|
| 1478 | - * Uses a breadth-first-search in order to find the nearest descendant |
|
| 1479 | - * having the specified LIN_code and returns it, else NULL |
|
| 1480 | - * |
|
| 1481 | - * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1482 | - * @param string $search_field name of EE_Line_Item property |
|
| 1483 | - * @param string $value any value stored in $search_field |
|
| 1484 | - * @return EE_Line_Item |
|
| 1485 | - * @throws EE_Error |
|
| 1486 | - * @throws InvalidArgumentException |
|
| 1487 | - * @throws InvalidDataTypeException |
|
| 1488 | - * @throws InvalidInterfaceException |
|
| 1489 | - * @throws ReflectionException |
|
| 1490 | - */ |
|
| 1491 | - protected static function _get_nearest_descendant(EE_Line_Item $parent_line_item, $search_field, $value) |
|
| 1492 | - { |
|
| 1493 | - foreach ($parent_line_item->children() as $child) { |
|
| 1494 | - if ($child->get($search_field) == $value) { |
|
| 1495 | - return $child; |
|
| 1496 | - } |
|
| 1497 | - } |
|
| 1498 | - foreach ($parent_line_item->children() as $child) { |
|
| 1499 | - $descendant_found = self::_get_nearest_descendant( |
|
| 1500 | - $child, |
|
| 1501 | - $search_field, |
|
| 1502 | - $value |
|
| 1503 | - ); |
|
| 1504 | - if ($descendant_found) { |
|
| 1505 | - return $descendant_found; |
|
| 1506 | - } |
|
| 1507 | - } |
|
| 1508 | - return null; |
|
| 1509 | - } |
|
| 1510 | - |
|
| 1511 | - |
|
| 1512 | - /** |
|
| 1513 | - * if passed line item has a TXN ID, uses that to jump directly to the grand total line item for the transaction, |
|
| 1514 | - * else recursively walks up the line item tree until a parent of type total is found, |
|
| 1515 | - * |
|
| 1516 | - * @param EE_Line_Item $line_item |
|
| 1517 | - * @return EE_Line_Item |
|
| 1518 | - * @throws EE_Error |
|
| 1519 | - */ |
|
| 1520 | - public static function find_transaction_grand_total_for_line_item(EE_Line_Item $line_item) |
|
| 1521 | - { |
|
| 1522 | - if ($line_item->TXN_ID()) { |
|
| 1523 | - $total_line_item = $line_item->transaction()->total_line_item(false); |
|
| 1524 | - if ($total_line_item instanceof EE_Line_Item) { |
|
| 1525 | - return $total_line_item; |
|
| 1526 | - } |
|
| 1527 | - } else { |
|
| 1528 | - $line_item_parent = $line_item->parent(); |
|
| 1529 | - if ($line_item_parent instanceof EE_Line_Item) { |
|
| 1530 | - if ($line_item_parent->is_total()) { |
|
| 1531 | - return $line_item_parent; |
|
| 1532 | - } |
|
| 1533 | - return EEH_Line_Item::find_transaction_grand_total_for_line_item($line_item_parent); |
|
| 1534 | - } |
|
| 1535 | - } |
|
| 1536 | - throw new EE_Error( |
|
| 1537 | - sprintf( |
|
| 1538 | - esc_html__( |
|
| 1539 | - 'A valid grand total for line item %1$d was not found.', |
|
| 1540 | - 'event_espresso' |
|
| 1541 | - ), |
|
| 1542 | - $line_item->ID() |
|
| 1543 | - ) |
|
| 1544 | - ); |
|
| 1545 | - } |
|
| 1546 | - |
|
| 1547 | - |
|
| 1548 | - /** |
|
| 1549 | - * Prints out a representation of the line item tree |
|
| 1550 | - * |
|
| 1551 | - * @param EE_Line_Item $line_item |
|
| 1552 | - * @param int $indentation |
|
| 1553 | - * @return void |
|
| 1554 | - * @throws EE_Error |
|
| 1555 | - */ |
|
| 1556 | - public static function visualize(EE_Line_Item $line_item, $indentation = 0) |
|
| 1557 | - { |
|
| 1558 | - echo defined('EE_TESTS_DIR') ? "\n" : '<br />'; |
|
| 1559 | - if (! $indentation) { |
|
| 1560 | - echo defined('EE_TESTS_DIR') ? "\n" : '<br />'; |
|
| 1561 | - } |
|
| 1562 | - for ($i = 0; $i < $indentation; $i++) { |
|
| 1563 | - echo '. '; |
|
| 1564 | - } |
|
| 1565 | - $breakdown = ''; |
|
| 1566 | - if ($line_item->is_line_item()) { |
|
| 1567 | - if ($line_item->is_percent()) { |
|
| 1568 | - $breakdown = "{$line_item->percent()}%"; |
|
| 1569 | - } else { |
|
| 1570 | - $breakdown = '$' . "{$line_item->unit_price()} x {$line_item->quantity()}"; |
|
| 1571 | - } |
|
| 1572 | - } |
|
| 1573 | - echo $line_item->name(); |
|
| 1574 | - echo " [ ID:{$line_item->ID()} | qty:{$line_item->quantity()} ] {$line_item->type()} : "; |
|
| 1575 | - echo '$' . (string) $line_item->total(); |
|
| 1576 | - if ($breakdown) { |
|
| 1577 | - echo " ( {$breakdown} )"; |
|
| 1578 | - } |
|
| 1579 | - if ($line_item->is_taxable()) { |
|
| 1580 | - echo ' * taxable'; |
|
| 1581 | - } |
|
| 1582 | - if ($line_item->children()) { |
|
| 1583 | - foreach ($line_item->children() as $child) { |
|
| 1584 | - self::visualize($child, $indentation + 1); |
|
| 1585 | - } |
|
| 1586 | - } |
|
| 1587 | - } |
|
| 1588 | - |
|
| 1589 | - |
|
| 1590 | - /** |
|
| 1591 | - * Calculates the registration's final price, taking into account that they |
|
| 1592 | - * need to not only help pay for their OWN ticket, but also any transaction-wide surcharges and taxes, |
|
| 1593 | - * and receive a portion of any transaction-wide discounts. |
|
| 1594 | - * eg1, if I buy a $1 ticket and brent buys a $9 ticket, and we receive a $5 discount |
|
| 1595 | - * then I'll get 1/10 of that $5 discount, which is $0.50, and brent will get |
|
| 1596 | - * 9/10ths of that $5 discount, which is $4.50. So my final price should be $0.50 |
|
| 1597 | - * and brent's final price should be $5.50. |
|
| 1598 | - * In order to do this, we basically need to traverse the line item tree calculating |
|
| 1599 | - * the running totals (just as if we were recalculating the total), but when we identify |
|
| 1600 | - * regular line items, we need to keep track of their share of the grand total. |
|
| 1601 | - * Also, we need to keep track of the TAXABLE total for each ticket purchase, so |
|
| 1602 | - * we can know how to apply taxes to it. (Note: "taxable total" does not equal the "pretax total" |
|
| 1603 | - * when there are non-taxable items; otherwise they would be the same) |
|
| 1604 | - * |
|
| 1605 | - * @param EE_Line_Item $line_item |
|
| 1606 | - * @param array $billable_ticket_quantities array of EE_Ticket IDs and their corresponding quantity that |
|
| 1607 | - * can be included in price calculations at this moment |
|
| 1608 | - * @return array keys are line items for tickets IDs and values are their share of the running total, |
|
| 1609 | - * plus the key 'total', and 'taxable' which also has keys of all |
|
| 1610 | - * the ticket IDs. |
|
| 1611 | - * Eg array( |
|
| 1612 | - * 12 => 4.3 |
|
| 1613 | - * 23 => 8.0 |
|
| 1614 | - * 'total' => 16.6, |
|
| 1615 | - * 'taxable' => array( |
|
| 1616 | - * 12 => 10, |
|
| 1617 | - * 23 => 4 |
|
| 1618 | - * ). |
|
| 1619 | - * So to find which registrations have which final price, we need |
|
| 1620 | - * to find which line item is theirs, which can be done with |
|
| 1621 | - * `EEM_Line_Item::instance()->get_line_item_for_registration( |
|
| 1622 | - * $registration );` |
|
| 1623 | - * @throws EE_Error |
|
| 1624 | - * @throws InvalidArgumentException |
|
| 1625 | - * @throws InvalidDataTypeException |
|
| 1626 | - * @throws InvalidInterfaceException |
|
| 1627 | - * @throws ReflectionException |
|
| 1628 | - */ |
|
| 1629 | - public static function calculate_reg_final_prices_per_line_item( |
|
| 1630 | - EE_Line_Item $line_item, |
|
| 1631 | - $billable_ticket_quantities = array() |
|
| 1632 | - ) { |
|
| 1633 | - $running_totals = [ |
|
| 1634 | - 'total' => 0, |
|
| 1635 | - 'taxable' => ['total' => 0] |
|
| 1636 | - ]; |
|
| 1637 | - foreach ($line_item->children() as $child_line_item) { |
|
| 1638 | - switch ($child_line_item->type()) { |
|
| 1639 | - case EEM_Line_Item::type_sub_total: |
|
| 1640 | - $running_totals_from_subtotal = EEH_Line_Item::calculate_reg_final_prices_per_line_item( |
|
| 1641 | - $child_line_item, |
|
| 1642 | - $billable_ticket_quantities |
|
| 1643 | - ); |
|
| 1644 | - // combine arrays but preserve numeric keys |
|
| 1645 | - $running_totals = array_replace_recursive($running_totals_from_subtotal, $running_totals); |
|
| 1646 | - $running_totals['total'] += $running_totals_from_subtotal['total']; |
|
| 1647 | - $running_totals['taxable']['total'] += $running_totals_from_subtotal['taxable']['total']; |
|
| 1648 | - break; |
|
| 1649 | - |
|
| 1650 | - case EEM_Line_Item::type_tax_sub_total: |
|
| 1651 | - // find how much the taxes percentage is |
|
| 1652 | - if ($child_line_item->percent() !== 0) { |
|
| 1653 | - $tax_percent_decimal = $child_line_item->percent() / 100; |
|
| 1654 | - } else { |
|
| 1655 | - $tax_percent_decimal = EE_Taxes::get_total_taxes_percentage() / 100; |
|
| 1656 | - } |
|
| 1657 | - // and apply to all the taxable totals, and add to the pretax totals |
|
| 1658 | - foreach ($running_totals as $line_item_id => $this_running_total) { |
|
| 1659 | - // "total" and "taxable" array key is an exception |
|
| 1660 | - if ($line_item_id === 'taxable') { |
|
| 1661 | - continue; |
|
| 1662 | - } |
|
| 1663 | - $taxable_total = $running_totals['taxable'][ $line_item_id ]; |
|
| 1664 | - $running_totals[ $line_item_id ] += ($taxable_total * $tax_percent_decimal); |
|
| 1665 | - } |
|
| 1666 | - break; |
|
| 1667 | - |
|
| 1668 | - case EEM_Line_Item::type_line_item: |
|
| 1669 | - // ticket line items or ???? |
|
| 1670 | - if ($child_line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_TICKET) { |
|
| 1671 | - // kk it's a ticket |
|
| 1672 | - if (isset($running_totals[ $child_line_item->ID() ])) { |
|
| 1673 | - // huh? that shouldn't happen. |
|
| 1674 | - $running_totals['total'] += $child_line_item->total(); |
|
| 1675 | - } else { |
|
| 1676 | - // its not in our running totals yet. great. |
|
| 1677 | - if ($child_line_item->is_taxable()) { |
|
| 1678 | - $taxable_amount = $child_line_item->unit_price(); |
|
| 1679 | - } else { |
|
| 1680 | - $taxable_amount = 0; |
|
| 1681 | - } |
|
| 1682 | - // are we only calculating totals for some tickets? |
|
| 1683 | - if (isset($billable_ticket_quantities[ $child_line_item->OBJ_ID() ])) { |
|
| 1684 | - $quantity = $billable_ticket_quantities[ $child_line_item->OBJ_ID() ]; |
|
| 1685 | - $running_totals[ $child_line_item->ID() ] = $quantity |
|
| 1686 | - ? $child_line_item->unit_price() |
|
| 1687 | - : 0; |
|
| 1688 | - $running_totals['taxable'][ $child_line_item->ID() ] = $quantity |
|
| 1689 | - ? $taxable_amount |
|
| 1690 | - : 0; |
|
| 1691 | - } else { |
|
| 1692 | - $quantity = $child_line_item->quantity(); |
|
| 1693 | - $running_totals[ $child_line_item->ID() ] = $child_line_item->unit_price(); |
|
| 1694 | - $running_totals['taxable'][ $child_line_item->ID() ] = $taxable_amount; |
|
| 1695 | - } |
|
| 1696 | - $running_totals['taxable']['total'] += $taxable_amount * $quantity; |
|
| 1697 | - $running_totals['total'] += $child_line_item->unit_price() * $quantity; |
|
| 1698 | - } |
|
| 1699 | - } else { |
|
| 1700 | - // it's some other type of item added to the cart |
|
| 1701 | - // it should affect the running totals |
|
| 1702 | - // basically we want to convert it into a PERCENT modifier. Because |
|
| 1703 | - // more clearly affect all registration's final price equally |
|
| 1704 | - $line_items_percent_of_running_total = $running_totals['total'] > 0 |
|
| 1705 | - ? ($child_line_item->total() / $running_totals['total']) + 1 |
|
| 1706 | - : 1; |
|
| 1707 | - foreach ($running_totals as $line_item_id => $this_running_total) { |
|
| 1708 | - // the "taxable" array key is an exception |
|
| 1709 | - if ($line_item_id === 'taxable') { |
|
| 1710 | - continue; |
|
| 1711 | - } |
|
| 1712 | - // update the running totals |
|
| 1713 | - // yes this actually even works for the running grand total! |
|
| 1714 | - $running_totals[ $line_item_id ] = |
|
| 1715 | - $line_items_percent_of_running_total * $this_running_total; |
|
| 1716 | - |
|
| 1717 | - if ($child_line_item->is_taxable()) { |
|
| 1718 | - $running_totals['taxable'][ $line_item_id ] = |
|
| 1719 | - $line_items_percent_of_running_total * $running_totals['taxable'][ $line_item_id ]; |
|
| 1720 | - } |
|
| 1721 | - } |
|
| 1722 | - } |
|
| 1723 | - break; |
|
| 1724 | - } |
|
| 1725 | - } |
|
| 1726 | - return $running_totals; |
|
| 1727 | - } |
|
| 1728 | - |
|
| 1729 | - |
|
| 1730 | - /** |
|
| 1731 | - * @param EE_Line_Item $total_line_item |
|
| 1732 | - * @param EE_Line_Item $ticket_line_item |
|
| 1733 | - * @return float | null |
|
| 1734 | - * @throws EE_Error |
|
| 1735 | - * @throws InvalidArgumentException |
|
| 1736 | - * @throws InvalidDataTypeException |
|
| 1737 | - * @throws InvalidInterfaceException |
|
| 1738 | - * @throws OutOfRangeException |
|
| 1739 | - * @throws ReflectionException |
|
| 1740 | - */ |
|
| 1741 | - public static function calculate_final_price_for_ticket_line_item( |
|
| 1742 | - EE_Line_Item $total_line_item, |
|
| 1743 | - EE_Line_Item $ticket_line_item |
|
| 1744 | - ) { |
|
| 1745 | - static $final_prices_per_ticket_line_item = array(); |
|
| 1746 | - if (empty($final_prices_per_ticket_line_item) || empty($final_prices_per_ticket_line_item[ $total_line_item->ID() ])) { |
|
| 1747 | - $final_prices_per_ticket_line_item[ $total_line_item->ID() ] = EEH_Line_Item::calculate_reg_final_prices_per_line_item( |
|
| 1748 | - $total_line_item |
|
| 1749 | - ); |
|
| 1750 | - } |
|
| 1751 | - // ok now find this new registration's final price |
|
| 1752 | - if (isset($final_prices_per_ticket_line_item[ $total_line_item->ID() ][ $ticket_line_item->ID() ])) { |
|
| 1753 | - return $final_prices_per_ticket_line_item[ $total_line_item->ID() ][ $ticket_line_item->ID() ]; |
|
| 1754 | - } |
|
| 1755 | - $message = sprintf( |
|
| 1756 | - esc_html__( |
|
| 1757 | - 'The final price for the ticket line item (ID:%1$d) could not be calculated.', |
|
| 1758 | - 'event_espresso' |
|
| 1759 | - ), |
|
| 1760 | - $ticket_line_item->ID() |
|
| 1761 | - ); |
|
| 1762 | - if (WP_DEBUG) { |
|
| 1763 | - $message .= '<br>' . print_r($final_prices_per_ticket_line_item, true); |
|
| 1764 | - throw new OutOfRangeException($message); |
|
| 1765 | - } |
|
| 1766 | - EE_Log::instance()->log(__CLASS__, __FUNCTION__, $message); |
|
| 1767 | - return null; |
|
| 1768 | - } |
|
| 1769 | - |
|
| 1770 | - |
|
| 1771 | - /** |
|
| 1772 | - * Creates a duplicate of the line item tree, except only includes billable items |
|
| 1773 | - * and the portion of line items attributed to billable things |
|
| 1774 | - * |
|
| 1775 | - * @param EE_Line_Item $line_item |
|
| 1776 | - * @param EE_Registration[] $registrations |
|
| 1777 | - * @return EE_Line_Item |
|
| 1778 | - * @throws EE_Error |
|
| 1779 | - * @throws InvalidArgumentException |
|
| 1780 | - * @throws InvalidDataTypeException |
|
| 1781 | - * @throws InvalidInterfaceException |
|
| 1782 | - * @throws ReflectionException |
|
| 1783 | - */ |
|
| 1784 | - public static function billable_line_item_tree(EE_Line_Item $line_item, $registrations) |
|
| 1785 | - { |
|
| 1786 | - $copy_li = EEH_Line_Item::billable_line_item($line_item, $registrations); |
|
| 1787 | - foreach ($line_item->children() as $child_li) { |
|
| 1788 | - $copy_li->add_child_line_item( |
|
| 1789 | - EEH_Line_Item::billable_line_item_tree($child_li, $registrations) |
|
| 1790 | - ); |
|
| 1791 | - } |
|
| 1792 | - // if this is the grand total line item, make sure the totals all add up |
|
| 1793 | - // (we could have duplicated this logic AS we copied the line items, but |
|
| 1794 | - // it seems DRYer this way) |
|
| 1795 | - if ($copy_li->type() === EEM_Line_Item::type_total) { |
|
| 1796 | - $copy_li->recalculate_total_including_taxes(); |
|
| 1797 | - } |
|
| 1798 | - return $copy_li; |
|
| 1799 | - } |
|
| 1800 | - |
|
| 1801 | - |
|
| 1802 | - /** |
|
| 1803 | - * Creates a new, unsaved line item from $line_item that factors in the |
|
| 1804 | - * number of billable registrations on $registrations. |
|
| 1805 | - * |
|
| 1806 | - * @param EE_Line_Item $line_item |
|
| 1807 | - * @param EE_Registration[] $registrations |
|
| 1808 | - * @return EE_Line_Item |
|
| 1809 | - * @throws EE_Error |
|
| 1810 | - * @throws InvalidArgumentException |
|
| 1811 | - * @throws InvalidDataTypeException |
|
| 1812 | - * @throws InvalidInterfaceException |
|
| 1813 | - * @throws ReflectionException |
|
| 1814 | - */ |
|
| 1815 | - public static function billable_line_item(EE_Line_Item $line_item, $registrations) |
|
| 1816 | - { |
|
| 1817 | - $new_li_fields = $line_item->model_field_array(); |
|
| 1818 | - if ($line_item->type() === EEM_Line_Item::type_line_item && |
|
| 1819 | - $line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_TICKET |
|
| 1820 | - ) { |
|
| 1821 | - $count = 0; |
|
| 1822 | - foreach ($registrations as $registration) { |
|
| 1823 | - if ($line_item->OBJ_ID() === $registration->ticket_ID() && |
|
| 1824 | - in_array( |
|
| 1825 | - $registration->status_ID(), |
|
| 1826 | - EEM_Registration::reg_statuses_that_allow_payment(), |
|
| 1827 | - true |
|
| 1828 | - ) |
|
| 1829 | - ) { |
|
| 1830 | - $count++; |
|
| 1831 | - } |
|
| 1832 | - } |
|
| 1833 | - $new_li_fields['LIN_quantity'] = $count; |
|
| 1834 | - } |
|
| 1835 | - // don't set the total. We'll leave that up to the code that calculates it |
|
| 1836 | - unset($new_li_fields['LIN_ID'], $new_li_fields['LIN_parent'], $new_li_fields['LIN_total']); |
|
| 1837 | - return EE_Line_Item::new_instance($new_li_fields); |
|
| 1838 | - } |
|
| 1839 | - |
|
| 1840 | - |
|
| 1841 | - /** |
|
| 1842 | - * Returns a modified line item tree where all the subtotals which have a total of 0 |
|
| 1843 | - * are removed, and line items with a quantity of 0 |
|
| 1844 | - * |
|
| 1845 | - * @param EE_Line_Item $line_item |null |
|
| 1846 | - * @return EE_Line_Item|null |
|
| 1847 | - * @throws EE_Error |
|
| 1848 | - * @throws InvalidArgumentException |
|
| 1849 | - * @throws InvalidDataTypeException |
|
| 1850 | - * @throws InvalidInterfaceException |
|
| 1851 | - * @throws ReflectionException |
|
| 1852 | - */ |
|
| 1853 | - public static function non_empty_line_items(EE_Line_Item $line_item) |
|
| 1854 | - { |
|
| 1855 | - $copied_li = EEH_Line_Item::non_empty_line_item($line_item); |
|
| 1856 | - if ($copied_li === null) { |
|
| 1857 | - return null; |
|
| 1858 | - } |
|
| 1859 | - // if this is an event subtotal, we want to only include it if it |
|
| 1860 | - // has a non-zero total and at least one ticket line item child |
|
| 1861 | - $ticket_children = 0; |
|
| 1862 | - foreach ($line_item->children() as $child_li) { |
|
| 1863 | - $child_li_copy = EEH_Line_Item::non_empty_line_items($child_li); |
|
| 1864 | - if ($child_li_copy !== null) { |
|
| 1865 | - $copied_li->add_child_line_item($child_li_copy); |
|
| 1866 | - if ($child_li_copy->type() === EEM_Line_Item::type_line_item && |
|
| 1867 | - $child_li_copy->OBJ_type() === EEM_Line_Item::OBJ_TYPE_TICKET |
|
| 1868 | - ) { |
|
| 1869 | - $ticket_children++; |
|
| 1870 | - } |
|
| 1871 | - } |
|
| 1872 | - } |
|
| 1873 | - // if this is an event subtotal with NO ticket children |
|
| 1874 | - // we basically want to ignore it |
|
| 1875 | - if ($ticket_children === 0 |
|
| 1876 | - && $line_item->type() === EEM_Line_Item::type_sub_total |
|
| 1877 | - && $line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_EVENT |
|
| 1878 | - && $line_item->total() === 0 |
|
| 1879 | - ) { |
|
| 1880 | - return null; |
|
| 1881 | - } |
|
| 1882 | - return $copied_li; |
|
| 1883 | - } |
|
| 1884 | - |
|
| 1885 | - |
|
| 1886 | - /** |
|
| 1887 | - * Creates a new, unsaved line item, but if it's a ticket line item |
|
| 1888 | - * with a total of 0, or a subtotal of 0, returns null instead |
|
| 1889 | - * |
|
| 1890 | - * @param EE_Line_Item $line_item |
|
| 1891 | - * @return EE_Line_Item |
|
| 1892 | - * @throws EE_Error |
|
| 1893 | - * @throws InvalidArgumentException |
|
| 1894 | - * @throws InvalidDataTypeException |
|
| 1895 | - * @throws InvalidInterfaceException |
|
| 1896 | - * @throws ReflectionException |
|
| 1897 | - */ |
|
| 1898 | - public static function non_empty_line_item(EE_Line_Item $line_item) |
|
| 1899 | - { |
|
| 1900 | - if ($line_item->type() === EEM_Line_Item::type_line_item |
|
| 1901 | - && $line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_TICKET |
|
| 1902 | - && $line_item->quantity() === 0 |
|
| 1903 | - ) { |
|
| 1904 | - return null; |
|
| 1905 | - } |
|
| 1906 | - $new_li_fields = $line_item->model_field_array(); |
|
| 1907 | - // don't set the total. We'll leave that up to the code that calculates it |
|
| 1908 | - unset($new_li_fields['LIN_ID'], $new_li_fields['LIN_parent']); |
|
| 1909 | - return EE_Line_Item::new_instance($new_li_fields); |
|
| 1910 | - } |
|
| 1911 | - |
|
| 1912 | - |
|
| 1913 | - /** |
|
| 1914 | - * Cycles through all of the ticket line items for the supplied total line item |
|
| 1915 | - * and ensures that the line item's "is_taxable" field matches that of its corresponding ticket |
|
| 1916 | - * |
|
| 1917 | - * @param EE_Line_Item $total_line_item |
|
| 1918 | - * @since 4.9.79.p |
|
| 1919 | - * @throws EE_Error |
|
| 1920 | - * @throws InvalidArgumentException |
|
| 1921 | - * @throws InvalidDataTypeException |
|
| 1922 | - * @throws InvalidInterfaceException |
|
| 1923 | - * @throws ReflectionException |
|
| 1924 | - */ |
|
| 1925 | - public static function resetIsTaxableForTickets(EE_Line_Item $total_line_item) |
|
| 1926 | - { |
|
| 1927 | - $ticket_line_items = self::get_ticket_line_items($total_line_item); |
|
| 1928 | - foreach ($ticket_line_items as $ticket_line_item) { |
|
| 1929 | - if ($ticket_line_item instanceof EE_Line_Item |
|
| 1930 | - && $ticket_line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_TICKET |
|
| 1931 | - ) { |
|
| 1932 | - $ticket = $ticket_line_item->ticket(); |
|
| 1933 | - if ($ticket instanceof EE_Ticket && $ticket->taxable() !== $ticket_line_item->is_taxable()) { |
|
| 1934 | - $ticket_line_item->set_is_taxable($ticket->taxable()); |
|
| 1935 | - $ticket_line_item->save(); |
|
| 1936 | - } |
|
| 1937 | - } |
|
| 1938 | - } |
|
| 1939 | - } |
|
| 1940 | - |
|
| 1941 | - |
|
| 1942 | - |
|
| 1943 | - /**************************************** @DEPRECATED METHODS *************************************** */ |
|
| 1944 | - /** |
|
| 1945 | - * @deprecated |
|
| 1946 | - * @param EE_Line_Item $total_line_item |
|
| 1947 | - * @return EE_Line_Item |
|
| 1948 | - * @throws EE_Error |
|
| 1949 | - * @throws InvalidArgumentException |
|
| 1950 | - * @throws InvalidDataTypeException |
|
| 1951 | - * @throws InvalidInterfaceException |
|
| 1952 | - * @throws ReflectionException |
|
| 1953 | - */ |
|
| 1954 | - public static function get_items_subtotal(EE_Line_Item $total_line_item) |
|
| 1955 | - { |
|
| 1956 | - EE_Error::doing_it_wrong( |
|
| 1957 | - 'EEH_Line_Item::get_items_subtotal()', |
|
| 1958 | - sprintf( |
|
| 1959 | - esc_html__('Method replaced with %1$s', 'event_espresso'), |
|
| 1960 | - 'EEH_Line_Item::get_pre_tax_subtotal()' |
|
| 1961 | - ), |
|
| 1962 | - '4.6.0' |
|
| 1963 | - ); |
|
| 1964 | - return self::get_pre_tax_subtotal($total_line_item); |
|
| 1965 | - } |
|
| 1966 | - |
|
| 1967 | - |
|
| 1968 | - /** |
|
| 1969 | - * @deprecated |
|
| 1970 | - * @param EE_Transaction $transaction |
|
| 1971 | - * @return EE_Line_Item |
|
| 1972 | - * @throws EE_Error |
|
| 1973 | - * @throws InvalidArgumentException |
|
| 1974 | - * @throws InvalidDataTypeException |
|
| 1975 | - * @throws InvalidInterfaceException |
|
| 1976 | - * @throws ReflectionException |
|
| 1977 | - */ |
|
| 1978 | - public static function create_default_total_line_item($transaction = null) |
|
| 1979 | - { |
|
| 1980 | - EE_Error::doing_it_wrong( |
|
| 1981 | - 'EEH_Line_Item::create_default_total_line_item()', |
|
| 1982 | - sprintf( |
|
| 1983 | - esc_html__('Method replaced with %1$s', 'event_espresso'), |
|
| 1984 | - 'EEH_Line_Item::create_total_line_item()' |
|
| 1985 | - ), |
|
| 1986 | - '4.6.0' |
|
| 1987 | - ); |
|
| 1988 | - return self::create_total_line_item($transaction); |
|
| 1989 | - } |
|
| 1990 | - |
|
| 1991 | - |
|
| 1992 | - /** |
|
| 1993 | - * @deprecated |
|
| 1994 | - * @param EE_Line_Item $total_line_item |
|
| 1995 | - * @param EE_Transaction $transaction |
|
| 1996 | - * @return EE_Line_Item |
|
| 1997 | - * @throws EE_Error |
|
| 1998 | - * @throws InvalidArgumentException |
|
| 1999 | - * @throws InvalidDataTypeException |
|
| 2000 | - * @throws InvalidInterfaceException |
|
| 2001 | - * @throws ReflectionException |
|
| 2002 | - */ |
|
| 2003 | - public static function create_default_tickets_subtotal(EE_Line_Item $total_line_item, $transaction = null) |
|
| 2004 | - { |
|
| 2005 | - EE_Error::doing_it_wrong( |
|
| 2006 | - 'EEH_Line_Item::create_default_tickets_subtotal()', |
|
| 2007 | - sprintf( |
|
| 2008 | - esc_html__('Method replaced with %1$s', 'event_espresso'), |
|
| 2009 | - 'EEH_Line_Item::create_pre_tax_subtotal()' |
|
| 2010 | - ), |
|
| 2011 | - '4.6.0' |
|
| 2012 | - ); |
|
| 2013 | - return self::create_pre_tax_subtotal($total_line_item, $transaction); |
|
| 2014 | - } |
|
| 2015 | - |
|
| 2016 | - |
|
| 2017 | - /** |
|
| 2018 | - * @deprecated |
|
| 2019 | - * @param EE_Line_Item $total_line_item |
|
| 2020 | - * @param EE_Transaction $transaction |
|
| 2021 | - * @return EE_Line_Item |
|
| 2022 | - * @throws EE_Error |
|
| 2023 | - * @throws InvalidArgumentException |
|
| 2024 | - * @throws InvalidDataTypeException |
|
| 2025 | - * @throws InvalidInterfaceException |
|
| 2026 | - * @throws ReflectionException |
|
| 2027 | - */ |
|
| 2028 | - public static function create_default_taxes_subtotal(EE_Line_Item $total_line_item, $transaction = null) |
|
| 2029 | - { |
|
| 2030 | - EE_Error::doing_it_wrong( |
|
| 2031 | - 'EEH_Line_Item::create_default_taxes_subtotal()', |
|
| 2032 | - sprintf( |
|
| 2033 | - esc_html__('Method replaced with %1$s', 'event_espresso'), |
|
| 2034 | - 'EEH_Line_Item::create_taxes_subtotal()' |
|
| 2035 | - ), |
|
| 2036 | - '4.6.0' |
|
| 2037 | - ); |
|
| 2038 | - return self::create_taxes_subtotal($total_line_item, $transaction); |
|
| 2039 | - } |
|
| 2040 | - |
|
| 2041 | - |
|
| 2042 | - /** |
|
| 2043 | - * @deprecated |
|
| 2044 | - * @param EE_Line_Item $total_line_item |
|
| 2045 | - * @param EE_Transaction $transaction |
|
| 2046 | - * @return EE_Line_Item |
|
| 2047 | - * @throws EE_Error |
|
| 2048 | - * @throws InvalidArgumentException |
|
| 2049 | - * @throws InvalidDataTypeException |
|
| 2050 | - * @throws InvalidInterfaceException |
|
| 2051 | - * @throws ReflectionException |
|
| 2052 | - */ |
|
| 2053 | - public static function create_default_event_subtotal(EE_Line_Item $total_line_item, $transaction = null) |
|
| 2054 | - { |
|
| 2055 | - EE_Error::doing_it_wrong( |
|
| 2056 | - 'EEH_Line_Item::create_default_event_subtotal()', |
|
| 2057 | - sprintf( |
|
| 2058 | - esc_html__('Method replaced with %1$s', 'event_espresso'), |
|
| 2059 | - 'EEH_Line_Item::create_event_subtotal()' |
|
| 2060 | - ), |
|
| 2061 | - '4.6.0' |
|
| 2062 | - ); |
|
| 2063 | - return self::create_event_subtotal($total_line_item, $transaction); |
|
| 2064 | - } |
|
| 24 | + /** |
|
| 25 | + * Adds a simple item (unrelated to any other model object) to the provided PARENT line item. |
|
| 26 | + * Does NOT automatically re-calculate the line item totals or update the related transaction. |
|
| 27 | + * You should call recalculate_total_including_taxes() on the grant total line item after this |
|
| 28 | + * to update the subtotals, and EE_Registration_Processor::calculate_reg_final_prices_per_line_item() |
|
| 29 | + * to keep the registration final prices in-sync with the transaction's total. |
|
| 30 | + * |
|
| 31 | + * @param EE_Line_Item $parent_line_item |
|
| 32 | + * @param string $name |
|
| 33 | + * @param float $unit_price |
|
| 34 | + * @param string $description |
|
| 35 | + * @param int $quantity |
|
| 36 | + * @param boolean $taxable |
|
| 37 | + * @param boolean $code if set to a value, ensures there is only one line item with that code |
|
| 38 | + * @return boolean success |
|
| 39 | + * @throws EE_Error |
|
| 40 | + * @throws InvalidArgumentException |
|
| 41 | + * @throws InvalidDataTypeException |
|
| 42 | + * @throws InvalidInterfaceException |
|
| 43 | + * @throws ReflectionException |
|
| 44 | + */ |
|
| 45 | + public static function add_unrelated_item( |
|
| 46 | + EE_Line_Item $parent_line_item, |
|
| 47 | + $name, |
|
| 48 | + $unit_price, |
|
| 49 | + $description = '', |
|
| 50 | + $quantity = 1, |
|
| 51 | + $taxable = false, |
|
| 52 | + $code = null |
|
| 53 | + ) { |
|
| 54 | + $items_subtotal = self::get_pre_tax_subtotal($parent_line_item); |
|
| 55 | + $line_item = EE_Line_Item::new_instance(array( |
|
| 56 | + 'LIN_name' => $name, |
|
| 57 | + 'LIN_desc' => $description, |
|
| 58 | + 'LIN_unit_price' => $unit_price, |
|
| 59 | + 'LIN_quantity' => $quantity, |
|
| 60 | + 'LIN_percent' => null, |
|
| 61 | + 'LIN_is_taxable' => $taxable, |
|
| 62 | + 'LIN_order' => $items_subtotal instanceof EE_Line_Item ? count($items_subtotal->children()) : 0, |
|
| 63 | + 'LIN_total' => (float) $unit_price * (int) $quantity, |
|
| 64 | + 'LIN_type' => EEM_Line_Item::type_line_item, |
|
| 65 | + 'LIN_code' => $code, |
|
| 66 | + )); |
|
| 67 | + $line_item = apply_filters( |
|
| 68 | + 'FHEE__EEH_Line_Item__add_unrelated_item__line_item', |
|
| 69 | + $line_item, |
|
| 70 | + $parent_line_item |
|
| 71 | + ); |
|
| 72 | + return self::add_item($parent_line_item, $line_item); |
|
| 73 | + } |
|
| 74 | + |
|
| 75 | + |
|
| 76 | + /** |
|
| 77 | + * Adds a simple item ( unrelated to any other model object) to the total line item, |
|
| 78 | + * in the correct spot in the line item tree. Does not automatically |
|
| 79 | + * re-calculate the line item totals, nor update the related transaction, nor upgrade the transaction's |
|
| 80 | + * registrations' final prices (which should probably change because of this). |
|
| 81 | + * You should call recalculate_total_including_taxes() on the grand total line item, then |
|
| 82 | + * update the transaction's total, and EE_Registration_Processor::update_registration_final_prices() |
|
| 83 | + * after using this, to keep the registration final prices in-sync with the transaction's total. |
|
| 84 | + * |
|
| 85 | + * @param EE_Line_Item $parent_line_item |
|
| 86 | + * @param string $name |
|
| 87 | + * @param float $percentage_amount |
|
| 88 | + * @param string $description |
|
| 89 | + * @param boolean $taxable |
|
| 90 | + * @return boolean success |
|
| 91 | + * @throws EE_Error |
|
| 92 | + */ |
|
| 93 | + public static function add_percentage_based_item( |
|
| 94 | + EE_Line_Item $parent_line_item, |
|
| 95 | + $name, |
|
| 96 | + $percentage_amount, |
|
| 97 | + $description = '', |
|
| 98 | + $taxable = false |
|
| 99 | + ) { |
|
| 100 | + $line_item = EE_Line_Item::new_instance(array( |
|
| 101 | + 'LIN_name' => $name, |
|
| 102 | + 'LIN_desc' => $description, |
|
| 103 | + 'LIN_unit_price' => 0, |
|
| 104 | + 'LIN_percent' => $percentage_amount, |
|
| 105 | + 'LIN_quantity' => 1, |
|
| 106 | + 'LIN_is_taxable' => $taxable, |
|
| 107 | + 'LIN_total' => (float) ($percentage_amount * ($parent_line_item->total() / 100)), |
|
| 108 | + 'LIN_type' => EEM_Line_Item::type_line_item, |
|
| 109 | + 'LIN_parent' => $parent_line_item->ID(), |
|
| 110 | + )); |
|
| 111 | + $line_item = apply_filters( |
|
| 112 | + 'FHEE__EEH_Line_Item__add_percentage_based_item__line_item', |
|
| 113 | + $line_item |
|
| 114 | + ); |
|
| 115 | + return $parent_line_item->add_child_line_item($line_item, false); |
|
| 116 | + } |
|
| 117 | + |
|
| 118 | + |
|
| 119 | + /** |
|
| 120 | + * Returns the new line item created by adding a purchase of the ticket |
|
| 121 | + * ensures that ticket line item is saved, and that cart total has been recalculated. |
|
| 122 | + * If this ticket has already been purchased, just increments its count. |
|
| 123 | + * Automatically re-calculates the line item totals and updates the related transaction. But |
|
| 124 | + * DOES NOT automatically upgrade the transaction's registrations' final prices (which |
|
| 125 | + * should probably change because of this). |
|
| 126 | + * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item() |
|
| 127 | + * after using this, to keep the registration final prices in-sync with the transaction's total. |
|
| 128 | + * |
|
| 129 | + * @param EE_Line_Item $total_line_item grand total line item of type EEM_Line_Item::type_total |
|
| 130 | + * @param EE_Ticket $ticket |
|
| 131 | + * @param int $qty |
|
| 132 | + * @return EE_Line_Item |
|
| 133 | + * @throws EE_Error |
|
| 134 | + * @throws InvalidArgumentException |
|
| 135 | + * @throws InvalidDataTypeException |
|
| 136 | + * @throws InvalidInterfaceException |
|
| 137 | + * @throws ReflectionException |
|
| 138 | + */ |
|
| 139 | + public static function add_ticket_purchase(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1) |
|
| 140 | + { |
|
| 141 | + if (! $total_line_item instanceof EE_Line_Item || ! $total_line_item->is_total()) { |
|
| 142 | + throw new EE_Error( |
|
| 143 | + sprintf( |
|
| 144 | + esc_html__( |
|
| 145 | + 'A valid line item total is required in order to add tickets. A line item of type "%s" was passed.', |
|
| 146 | + 'event_espresso' |
|
| 147 | + ), |
|
| 148 | + $ticket->ID(), |
|
| 149 | + $total_line_item->ID() |
|
| 150 | + ) |
|
| 151 | + ); |
|
| 152 | + } |
|
| 153 | + // either increment the qty for an existing ticket |
|
| 154 | + $line_item = self::increment_ticket_qty_if_already_in_cart($total_line_item, $ticket, $qty); |
|
| 155 | + // or add a new one |
|
| 156 | + if (! $line_item instanceof EE_Line_Item) { |
|
| 157 | + $line_item = self::create_ticket_line_item($total_line_item, $ticket, $qty); |
|
| 158 | + } |
|
| 159 | + $total_line_item->recalculate_total_including_taxes(); |
|
| 160 | + return $line_item; |
|
| 161 | + } |
|
| 162 | + |
|
| 163 | + |
|
| 164 | + /** |
|
| 165 | + * Returns the new line item created by adding a purchase of the ticket |
|
| 166 | + * |
|
| 167 | + * @param EE_Line_Item $total_line_item |
|
| 168 | + * @param EE_Ticket $ticket |
|
| 169 | + * @param int $qty |
|
| 170 | + * @return EE_Line_Item |
|
| 171 | + * @throws EE_Error |
|
| 172 | + * @throws InvalidArgumentException |
|
| 173 | + * @throws InvalidDataTypeException |
|
| 174 | + * @throws InvalidInterfaceException |
|
| 175 | + * @throws ReflectionException |
|
| 176 | + */ |
|
| 177 | + public static function increment_ticket_qty_if_already_in_cart( |
|
| 178 | + EE_Line_Item $total_line_item, |
|
| 179 | + EE_Ticket $ticket, |
|
| 180 | + $qty = 1 |
|
| 181 | + ) { |
|
| 182 | + $line_item = null; |
|
| 183 | + if ($total_line_item instanceof EE_Line_Item && $total_line_item->is_total()) { |
|
| 184 | + $ticket_line_items = EEH_Line_Item::get_ticket_line_items($total_line_item); |
|
| 185 | + foreach ((array) $ticket_line_items as $ticket_line_item) { |
|
| 186 | + if ($ticket_line_item instanceof EE_Line_Item |
|
| 187 | + && (int) $ticket_line_item->OBJ_ID() === (int) $ticket->ID() |
|
| 188 | + ) { |
|
| 189 | + $line_item = $ticket_line_item; |
|
| 190 | + break; |
|
| 191 | + } |
|
| 192 | + } |
|
| 193 | + } |
|
| 194 | + if ($line_item instanceof EE_Line_Item) { |
|
| 195 | + EEH_Line_Item::increment_quantity($line_item, $qty); |
|
| 196 | + return $line_item; |
|
| 197 | + } |
|
| 198 | + return null; |
|
| 199 | + } |
|
| 200 | + |
|
| 201 | + |
|
| 202 | + /** |
|
| 203 | + * Increments the line item and all its children's quantity by $qty (but percent line items are unaffected). |
|
| 204 | + * Does NOT save or recalculate other line items totals |
|
| 205 | + * |
|
| 206 | + * @param EE_Line_Item $line_item |
|
| 207 | + * @param int $qty |
|
| 208 | + * @return void |
|
| 209 | + * @throws EE_Error |
|
| 210 | + * @throws InvalidArgumentException |
|
| 211 | + * @throws InvalidDataTypeException |
|
| 212 | + * @throws InvalidInterfaceException |
|
| 213 | + * @throws ReflectionException |
|
| 214 | + */ |
|
| 215 | + public static function increment_quantity(EE_Line_Item $line_item, $qty = 1) |
|
| 216 | + { |
|
| 217 | + if (! $line_item->is_percent()) { |
|
| 218 | + $qty += $line_item->quantity(); |
|
| 219 | + $line_item->set_quantity($qty); |
|
| 220 | + $line_item->set_total($line_item->unit_price() * $qty); |
|
| 221 | + $line_item->save(); |
|
| 222 | + } |
|
| 223 | + foreach ($line_item->children() as $child) { |
|
| 224 | + if ($child->is_sub_line_item()) { |
|
| 225 | + EEH_Line_Item::update_quantity($child, $qty); |
|
| 226 | + } |
|
| 227 | + } |
|
| 228 | + } |
|
| 229 | + |
|
| 230 | + |
|
| 231 | + /** |
|
| 232 | + * Decrements the line item and all its children's quantity by $qty (but percent line items are unaffected). |
|
| 233 | + * Does NOT save or recalculate other line items totals |
|
| 234 | + * |
|
| 235 | + * @param EE_Line_Item $line_item |
|
| 236 | + * @param int $qty |
|
| 237 | + * @return void |
|
| 238 | + * @throws EE_Error |
|
| 239 | + * @throws InvalidArgumentException |
|
| 240 | + * @throws InvalidDataTypeException |
|
| 241 | + * @throws InvalidInterfaceException |
|
| 242 | + * @throws ReflectionException |
|
| 243 | + */ |
|
| 244 | + public static function decrement_quantity(EE_Line_Item $line_item, $qty = 1) |
|
| 245 | + { |
|
| 246 | + if (! $line_item->is_percent()) { |
|
| 247 | + $qty = $line_item->quantity() - $qty; |
|
| 248 | + $qty = max($qty, 0); |
|
| 249 | + $line_item->set_quantity($qty); |
|
| 250 | + $line_item->set_total($line_item->unit_price() * $qty); |
|
| 251 | + $line_item->save(); |
|
| 252 | + } |
|
| 253 | + foreach ($line_item->children() as $child) { |
|
| 254 | + if ($child->is_sub_line_item()) { |
|
| 255 | + EEH_Line_Item::update_quantity($child, $qty); |
|
| 256 | + } |
|
| 257 | + } |
|
| 258 | + } |
|
| 259 | + |
|
| 260 | + |
|
| 261 | + /** |
|
| 262 | + * Updates the line item and its children's quantities to the specified number. |
|
| 263 | + * Does NOT save them or recalculate totals. |
|
| 264 | + * |
|
| 265 | + * @param EE_Line_Item $line_item |
|
| 266 | + * @param int $new_quantity |
|
| 267 | + * @throws EE_Error |
|
| 268 | + * @throws InvalidArgumentException |
|
| 269 | + * @throws InvalidDataTypeException |
|
| 270 | + * @throws InvalidInterfaceException |
|
| 271 | + * @throws ReflectionException |
|
| 272 | + */ |
|
| 273 | + public static function update_quantity(EE_Line_Item $line_item, $new_quantity) |
|
| 274 | + { |
|
| 275 | + if (! $line_item->is_percent()) { |
|
| 276 | + $line_item->set_quantity($new_quantity); |
|
| 277 | + $line_item->set_total($line_item->unit_price() * $new_quantity); |
|
| 278 | + $line_item->save(); |
|
| 279 | + } |
|
| 280 | + foreach ($line_item->children() as $child) { |
|
| 281 | + if ($child->is_sub_line_item()) { |
|
| 282 | + EEH_Line_Item::update_quantity($child, $new_quantity); |
|
| 283 | + } |
|
| 284 | + } |
|
| 285 | + } |
|
| 286 | + |
|
| 287 | + |
|
| 288 | + /** |
|
| 289 | + * Returns the new line item created by adding a purchase of the ticket |
|
| 290 | + * |
|
| 291 | + * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 292 | + * @param EE_Ticket $ticket |
|
| 293 | + * @param int $qty |
|
| 294 | + * @return EE_Line_Item |
|
| 295 | + * @throws EE_Error |
|
| 296 | + * @throws InvalidArgumentException |
|
| 297 | + * @throws InvalidDataTypeException |
|
| 298 | + * @throws InvalidInterfaceException |
|
| 299 | + * @throws ReflectionException |
|
| 300 | + */ |
|
| 301 | + public static function create_ticket_line_item(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1) |
|
| 302 | + { |
|
| 303 | + $datetimes = $ticket->datetimes(); |
|
| 304 | + $first_datetime = reset($datetimes); |
|
| 305 | + $first_datetime_name = esc_html__('Event', 'event_espresso'); |
|
| 306 | + if ($first_datetime instanceof EE_Datetime && $first_datetime->event() instanceof EE_Event) { |
|
| 307 | + $first_datetime_name = $first_datetime->event()->name(); |
|
| 308 | + } |
|
| 309 | + $event = sprintf(_x('(For %1$s)', '(For Event Name)', 'event_espresso'), $first_datetime_name); |
|
| 310 | + // get event subtotal line |
|
| 311 | + $events_sub_total = self::get_event_line_item_for_ticket($total_line_item, $ticket); |
|
| 312 | + // add $ticket to cart |
|
| 313 | + $line_item = EE_Line_Item::new_instance(array( |
|
| 314 | + 'LIN_name' => $ticket->name(), |
|
| 315 | + 'LIN_desc' => $ticket->description() !== '' ? $ticket->description() . ' ' . $event : $event, |
|
| 316 | + 'LIN_unit_price' => $ticket->price(), |
|
| 317 | + 'LIN_quantity' => $qty, |
|
| 318 | + 'LIN_is_taxable' => $ticket->taxable(), |
|
| 319 | + 'LIN_order' => count($events_sub_total->children()), |
|
| 320 | + 'LIN_total' => $ticket->price() * $qty, |
|
| 321 | + 'LIN_type' => EEM_Line_Item::type_line_item, |
|
| 322 | + 'OBJ_ID' => $ticket->ID(), |
|
| 323 | + 'OBJ_type' => EEM_Line_Item::OBJ_TYPE_TICKET, |
|
| 324 | + )); |
|
| 325 | + $line_item = apply_filters( |
|
| 326 | + 'FHEE__EEH_Line_Item__create_ticket_line_item__line_item', |
|
| 327 | + $line_item |
|
| 328 | + ); |
|
| 329 | + $events_sub_total->add_child_line_item($line_item); |
|
| 330 | + // now add the sub-line items |
|
| 331 | + $running_total_for_ticket = 0; |
|
| 332 | + foreach ($ticket->prices(array('order_by' => array('PRC_order' => 'ASC'))) as $price) { |
|
| 333 | + $sign = $price->is_discount() ? -1 : 1; |
|
| 334 | + $price_total = $price->is_percent() |
|
| 335 | + ? $running_total_for_ticket * $price->amount() / 100 |
|
| 336 | + : $price->amount() * $qty; |
|
| 337 | + $sub_line_item = EE_Line_Item::new_instance(array( |
|
| 338 | + 'LIN_name' => $price->name(), |
|
| 339 | + 'LIN_desc' => $price->desc(), |
|
| 340 | + 'LIN_quantity' => $price->is_percent() ? null : $qty, |
|
| 341 | + 'LIN_is_taxable' => false, |
|
| 342 | + 'LIN_order' => $price->order(), |
|
| 343 | + 'LIN_total' => $sign * $price_total, |
|
| 344 | + 'LIN_type' => EEM_Line_Item::type_sub_line_item, |
|
| 345 | + 'OBJ_ID' => $price->ID(), |
|
| 346 | + 'OBJ_type' => EEM_Line_Item::OBJ_TYPE_PRICE, |
|
| 347 | + )); |
|
| 348 | + $sub_line_item = apply_filters( |
|
| 349 | + 'FHEE__EEH_Line_Item__create_ticket_line_item__sub_line_item', |
|
| 350 | + $sub_line_item |
|
| 351 | + ); |
|
| 352 | + if ($price->is_percent()) { |
|
| 353 | + $sub_line_item->set_percent($sign * $price->amount()); |
|
| 354 | + } else { |
|
| 355 | + $sub_line_item->set_unit_price($sign * $price->amount()); |
|
| 356 | + } |
|
| 357 | + $running_total_for_ticket += $price_total; |
|
| 358 | + $line_item->add_child_line_item($sub_line_item); |
|
| 359 | + } |
|
| 360 | + return $line_item; |
|
| 361 | + } |
|
| 362 | + |
|
| 363 | + |
|
| 364 | + /** |
|
| 365 | + * Adds the specified item under the pre-tax-sub-total line item. Automatically |
|
| 366 | + * re-calculates the line item totals and updates the related transaction. But |
|
| 367 | + * DOES NOT automatically upgrade the transaction's registrations' final prices (which |
|
| 368 | + * should probably change because of this). |
|
| 369 | + * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item() |
|
| 370 | + * after using this, to keep the registration final prices in-sync with the transaction's total. |
|
| 371 | + * |
|
| 372 | + * @param EE_Line_Item $total_line_item |
|
| 373 | + * @param EE_Line_Item $item to be added |
|
| 374 | + * @return boolean |
|
| 375 | + * @throws EE_Error |
|
| 376 | + * @throws InvalidArgumentException |
|
| 377 | + * @throws InvalidDataTypeException |
|
| 378 | + * @throws InvalidInterfaceException |
|
| 379 | + * @throws ReflectionException |
|
| 380 | + */ |
|
| 381 | + public static function add_item(EE_Line_Item $total_line_item, EE_Line_Item $item) |
|
| 382 | + { |
|
| 383 | + $pre_tax_subtotal = self::get_pre_tax_subtotal($total_line_item); |
|
| 384 | + if ($pre_tax_subtotal instanceof EE_Line_Item) { |
|
| 385 | + $success = $pre_tax_subtotal->add_child_line_item($item); |
|
| 386 | + } else { |
|
| 387 | + return false; |
|
| 388 | + } |
|
| 389 | + $total_line_item->recalculate_total_including_taxes(); |
|
| 390 | + return $success; |
|
| 391 | + } |
|
| 392 | + |
|
| 393 | + |
|
| 394 | + /** |
|
| 395 | + * cancels an existing ticket line item, |
|
| 396 | + * by decrementing it's quantity by 1 and adding a new "type_cancellation" sub-line-item. |
|
| 397 | + * ALL totals and subtotals will NEED TO BE UPDATED after performing this action |
|
| 398 | + * |
|
| 399 | + * @param EE_Line_Item $ticket_line_item |
|
| 400 | + * @param int $qty |
|
| 401 | + * @return bool success |
|
| 402 | + * @throws EE_Error |
|
| 403 | + * @throws InvalidArgumentException |
|
| 404 | + * @throws InvalidDataTypeException |
|
| 405 | + * @throws InvalidInterfaceException |
|
| 406 | + * @throws ReflectionException |
|
| 407 | + */ |
|
| 408 | + public static function cancel_ticket_line_item(EE_Line_Item $ticket_line_item, $qty = 1) |
|
| 409 | + { |
|
| 410 | + // validate incoming line_item |
|
| 411 | + if ($ticket_line_item->OBJ_type() !== EEM_Line_Item::OBJ_TYPE_TICKET) { |
|
| 412 | + throw new EE_Error( |
|
| 413 | + sprintf( |
|
| 414 | + esc_html__( |
|
| 415 | + 'The supplied line item must have an Object Type of "Ticket", not %1$s.', |
|
| 416 | + 'event_espresso' |
|
| 417 | + ), |
|
| 418 | + $ticket_line_item->type() |
|
| 419 | + ) |
|
| 420 | + ); |
|
| 421 | + } |
|
| 422 | + if ($ticket_line_item->quantity() < $qty) { |
|
| 423 | + throw new EE_Error( |
|
| 424 | + sprintf( |
|
| 425 | + esc_html__( |
|
| 426 | + 'Can not cancel %1$d ticket(s) because the supplied line item has a quantity of %2$d.', |
|
| 427 | + 'event_espresso' |
|
| 428 | + ), |
|
| 429 | + $qty, |
|
| 430 | + $ticket_line_item->quantity() |
|
| 431 | + ) |
|
| 432 | + ); |
|
| 433 | + } |
|
| 434 | + // decrement ticket quantity; don't rely on auto-fixing when recalculating totals to do this |
|
| 435 | + $ticket_line_item->set_quantity($ticket_line_item->quantity() - $qty); |
|
| 436 | + foreach ($ticket_line_item->children() as $child_line_item) { |
|
| 437 | + if ($child_line_item->is_sub_line_item() |
|
| 438 | + && ! $child_line_item->is_percent() |
|
| 439 | + && ! $child_line_item->is_cancellation() |
|
| 440 | + ) { |
|
| 441 | + $child_line_item->set_quantity($child_line_item->quantity() - $qty); |
|
| 442 | + } |
|
| 443 | + } |
|
| 444 | + // get cancellation sub line item |
|
| 445 | + $cancellation_line_item = EEH_Line_Item::get_descendants_of_type( |
|
| 446 | + $ticket_line_item, |
|
| 447 | + EEM_Line_Item::type_cancellation |
|
| 448 | + ); |
|
| 449 | + $cancellation_line_item = reset($cancellation_line_item); |
|
| 450 | + // verify that this ticket was indeed previously cancelled |
|
| 451 | + if ($cancellation_line_item instanceof EE_Line_Item) { |
|
| 452 | + // increment cancelled quantity |
|
| 453 | + $cancellation_line_item->set_quantity($cancellation_line_item->quantity() + $qty); |
|
| 454 | + } else { |
|
| 455 | + // create cancellation sub line item |
|
| 456 | + $cancellation_line_item = EE_Line_Item::new_instance(array( |
|
| 457 | + 'LIN_name' => esc_html__('Cancellation', 'event_espresso'), |
|
| 458 | + 'LIN_desc' => sprintf( |
|
| 459 | + esc_html_x( |
|
| 460 | + 'Cancelled %1$s : %2$s', |
|
| 461 | + 'Cancelled Ticket Name : 2015-01-01 11:11', |
|
| 462 | + 'event_espresso' |
|
| 463 | + ), |
|
| 464 | + $ticket_line_item->name(), |
|
| 465 | + current_time(get_option('date_format') . ' ' . get_option('time_format')) |
|
| 466 | + ), |
|
| 467 | + 'LIN_unit_price' => 0, // $ticket_line_item->unit_price() |
|
| 468 | + 'LIN_quantity' => $qty, |
|
| 469 | + 'LIN_is_taxable' => $ticket_line_item->is_taxable(), |
|
| 470 | + 'LIN_order' => count($ticket_line_item->children()), |
|
| 471 | + 'LIN_total' => 0, // $ticket_line_item->unit_price() |
|
| 472 | + 'LIN_type' => EEM_Line_Item::type_cancellation, |
|
| 473 | + )); |
|
| 474 | + $ticket_line_item->add_child_line_item($cancellation_line_item); |
|
| 475 | + } |
|
| 476 | + if ($ticket_line_item->save_this_and_descendants() > 0) { |
|
| 477 | + // decrement parent line item quantity |
|
| 478 | + $event_line_item = $ticket_line_item->parent(); |
|
| 479 | + if ($event_line_item instanceof EE_Line_Item |
|
| 480 | + && $event_line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_EVENT |
|
| 481 | + ) { |
|
| 482 | + $event_line_item->set_quantity($event_line_item->quantity() - $qty); |
|
| 483 | + $event_line_item->save(); |
|
| 484 | + } |
|
| 485 | + EEH_Line_Item::get_grand_total_and_recalculate_everything($ticket_line_item); |
|
| 486 | + return true; |
|
| 487 | + } |
|
| 488 | + return false; |
|
| 489 | + } |
|
| 490 | + |
|
| 491 | + |
|
| 492 | + /** |
|
| 493 | + * reinstates (un-cancels?) a previously canceled ticket line item, |
|
| 494 | + * by incrementing it's quantity by 1, and decrementing it's "type_cancellation" sub-line-item. |
|
| 495 | + * ALL totals and subtotals will NEED TO BE UPDATED after performing this action |
|
| 496 | + * |
|
| 497 | + * @param EE_Line_Item $ticket_line_item |
|
| 498 | + * @param int $qty |
|
| 499 | + * @return bool success |
|
| 500 | + * @throws EE_Error |
|
| 501 | + * @throws InvalidArgumentException |
|
| 502 | + * @throws InvalidDataTypeException |
|
| 503 | + * @throws InvalidInterfaceException |
|
| 504 | + * @throws ReflectionException |
|
| 505 | + */ |
|
| 506 | + public static function reinstate_canceled_ticket_line_item(EE_Line_Item $ticket_line_item, $qty = 1) |
|
| 507 | + { |
|
| 508 | + // validate incoming line_item |
|
| 509 | + if ($ticket_line_item->OBJ_type() !== EEM_Line_Item::OBJ_TYPE_TICKET) { |
|
| 510 | + throw new EE_Error( |
|
| 511 | + sprintf( |
|
| 512 | + esc_html__( |
|
| 513 | + 'The supplied line item must have an Object Type of "Ticket", not %1$s.', |
|
| 514 | + 'event_espresso' |
|
| 515 | + ), |
|
| 516 | + $ticket_line_item->type() |
|
| 517 | + ) |
|
| 518 | + ); |
|
| 519 | + } |
|
| 520 | + // get cancellation sub line item |
|
| 521 | + $cancellation_line_item = EEH_Line_Item::get_descendants_of_type( |
|
| 522 | + $ticket_line_item, |
|
| 523 | + EEM_Line_Item::type_cancellation |
|
| 524 | + ); |
|
| 525 | + $cancellation_line_item = reset($cancellation_line_item); |
|
| 526 | + // verify that this ticket was indeed previously cancelled |
|
| 527 | + if (! $cancellation_line_item instanceof EE_Line_Item) { |
|
| 528 | + return false; |
|
| 529 | + } |
|
| 530 | + if ($cancellation_line_item->quantity() > $qty) { |
|
| 531 | + // decrement cancelled quantity |
|
| 532 | + $cancellation_line_item->set_quantity($cancellation_line_item->quantity() - $qty); |
|
| 533 | + } elseif ($cancellation_line_item->quantity() === $qty) { |
|
| 534 | + // decrement cancelled quantity in case anyone still has the object kicking around |
|
| 535 | + $cancellation_line_item->set_quantity($cancellation_line_item->quantity() - $qty); |
|
| 536 | + // delete because quantity will end up as 0 |
|
| 537 | + $cancellation_line_item->delete(); |
|
| 538 | + // and attempt to destroy the object, |
|
| 539 | + // even though PHP won't actually destroy it until it needs the memory |
|
| 540 | + unset($cancellation_line_item); |
|
| 541 | + } else { |
|
| 542 | + // what ?!?! negative quantity ?!?! |
|
| 543 | + throw new EE_Error( |
|
| 544 | + sprintf( |
|
| 545 | + esc_html__( |
|
| 546 | + 'Can not reinstate %1$d cancelled ticket(s) because the cancelled ticket quantity is only %2$d.', |
|
| 547 | + 'event_espresso' |
|
| 548 | + ), |
|
| 549 | + $qty, |
|
| 550 | + $cancellation_line_item->quantity() |
|
| 551 | + ) |
|
| 552 | + ); |
|
| 553 | + } |
|
| 554 | + // increment ticket quantity |
|
| 555 | + $ticket_line_item->set_quantity($ticket_line_item->quantity() + $qty); |
|
| 556 | + if ($ticket_line_item->save_this_and_descendants() > 0) { |
|
| 557 | + // increment parent line item quantity |
|
| 558 | + $event_line_item = $ticket_line_item->parent(); |
|
| 559 | + if ($event_line_item instanceof EE_Line_Item |
|
| 560 | + && $event_line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_EVENT |
|
| 561 | + ) { |
|
| 562 | + $event_line_item->set_quantity($event_line_item->quantity() + $qty); |
|
| 563 | + } |
|
| 564 | + EEH_Line_Item::get_grand_total_and_recalculate_everything($ticket_line_item); |
|
| 565 | + return true; |
|
| 566 | + } |
|
| 567 | + return false; |
|
| 568 | + } |
|
| 569 | + |
|
| 570 | + |
|
| 571 | + /** |
|
| 572 | + * calls EEH_Line_Item::find_transaction_grand_total_for_line_item() |
|
| 573 | + * then EE_Line_Item::recalculate_total_including_taxes() on the result |
|
| 574 | + * |
|
| 575 | + * @param EE_Line_Item $line_item |
|
| 576 | + * @return float |
|
| 577 | + * @throws EE_Error |
|
| 578 | + * @throws InvalidArgumentException |
|
| 579 | + * @throws InvalidDataTypeException |
|
| 580 | + * @throws InvalidInterfaceException |
|
| 581 | + * @throws ReflectionException |
|
| 582 | + */ |
|
| 583 | + public static function get_grand_total_and_recalculate_everything(EE_Line_Item $line_item) |
|
| 584 | + { |
|
| 585 | + $grand_total_line_item = EEH_Line_Item::find_transaction_grand_total_for_line_item($line_item); |
|
| 586 | + return $grand_total_line_item->recalculate_total_including_taxes(); |
|
| 587 | + } |
|
| 588 | + |
|
| 589 | + |
|
| 590 | + /** |
|
| 591 | + * Gets the line item which contains the subtotal of all the items |
|
| 592 | + * |
|
| 593 | + * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 594 | + * @return EE_Line_Item |
|
| 595 | + * @throws EE_Error |
|
| 596 | + * @throws InvalidArgumentException |
|
| 597 | + * @throws InvalidDataTypeException |
|
| 598 | + * @throws InvalidInterfaceException |
|
| 599 | + * @throws ReflectionException |
|
| 600 | + */ |
|
| 601 | + public static function get_pre_tax_subtotal(EE_Line_Item $total_line_item) |
|
| 602 | + { |
|
| 603 | + $pre_tax_subtotal = $total_line_item->get_child_line_item('pre-tax-subtotal'); |
|
| 604 | + return $pre_tax_subtotal instanceof EE_Line_Item |
|
| 605 | + ? $pre_tax_subtotal |
|
| 606 | + : self::create_pre_tax_subtotal($total_line_item); |
|
| 607 | + } |
|
| 608 | + |
|
| 609 | + |
|
| 610 | + /** |
|
| 611 | + * Gets the line item for the taxes subtotal |
|
| 612 | + * |
|
| 613 | + * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 614 | + * @return EE_Line_Item |
|
| 615 | + * @throws EE_Error |
|
| 616 | + * @throws InvalidArgumentException |
|
| 617 | + * @throws InvalidDataTypeException |
|
| 618 | + * @throws InvalidInterfaceException |
|
| 619 | + * @throws ReflectionException |
|
| 620 | + */ |
|
| 621 | + public static function get_taxes_subtotal(EE_Line_Item $total_line_item) |
|
| 622 | + { |
|
| 623 | + $taxes = $total_line_item->get_child_line_item('taxes'); |
|
| 624 | + return $taxes ? $taxes : self::create_taxes_subtotal($total_line_item); |
|
| 625 | + } |
|
| 626 | + |
|
| 627 | + |
|
| 628 | + /** |
|
| 629 | + * sets the TXN ID on an EE_Line_Item if passed a valid EE_Transaction object |
|
| 630 | + * |
|
| 631 | + * @param EE_Line_Item $line_item |
|
| 632 | + * @param EE_Transaction $transaction |
|
| 633 | + * @return void |
|
| 634 | + * @throws EE_Error |
|
| 635 | + * @throws InvalidArgumentException |
|
| 636 | + * @throws InvalidDataTypeException |
|
| 637 | + * @throws InvalidInterfaceException |
|
| 638 | + * @throws ReflectionException |
|
| 639 | + */ |
|
| 640 | + public static function set_TXN_ID(EE_Line_Item $line_item, $transaction = null) |
|
| 641 | + { |
|
| 642 | + if ($transaction) { |
|
| 643 | + /** @type EEM_Transaction $EEM_Transaction */ |
|
| 644 | + $EEM_Transaction = EE_Registry::instance()->load_model('Transaction'); |
|
| 645 | + $TXN_ID = $EEM_Transaction->ensure_is_ID($transaction); |
|
| 646 | + $line_item->set_TXN_ID($TXN_ID); |
|
| 647 | + } |
|
| 648 | + } |
|
| 649 | + |
|
| 650 | + |
|
| 651 | + /** |
|
| 652 | + * Creates a new default total line item for the transaction, |
|
| 653 | + * and its tickets subtotal and taxes subtotal line items (and adds the |
|
| 654 | + * existing taxes as children of the taxes subtotal line item) |
|
| 655 | + * |
|
| 656 | + * @param EE_Transaction $transaction |
|
| 657 | + * @return EE_Line_Item of type total |
|
| 658 | + * @throws EE_Error |
|
| 659 | + * @throws InvalidArgumentException |
|
| 660 | + * @throws InvalidDataTypeException |
|
| 661 | + * @throws InvalidInterfaceException |
|
| 662 | + * @throws ReflectionException |
|
| 663 | + */ |
|
| 664 | + public static function create_total_line_item($transaction = null) |
|
| 665 | + { |
|
| 666 | + $total_line_item = EE_Line_Item::new_instance(array( |
|
| 667 | + 'LIN_code' => 'total', |
|
| 668 | + 'LIN_name' => esc_html__('Grand Total', 'event_espresso'), |
|
| 669 | + 'LIN_type' => EEM_Line_Item::type_total, |
|
| 670 | + 'OBJ_type' => EEM_Line_Item::OBJ_TYPE_TRANSACTION, |
|
| 671 | + )); |
|
| 672 | + $total_line_item = apply_filters( |
|
| 673 | + 'FHEE__EEH_Line_Item__create_total_line_item__total_line_item', |
|
| 674 | + $total_line_item |
|
| 675 | + ); |
|
| 676 | + self::set_TXN_ID($total_line_item, $transaction); |
|
| 677 | + self::create_pre_tax_subtotal($total_line_item, $transaction); |
|
| 678 | + self::create_taxes_subtotal($total_line_item, $transaction); |
|
| 679 | + return $total_line_item; |
|
| 680 | + } |
|
| 681 | + |
|
| 682 | + |
|
| 683 | + /** |
|
| 684 | + * Creates a default items subtotal line item |
|
| 685 | + * |
|
| 686 | + * @param EE_Line_Item $total_line_item |
|
| 687 | + * @param EE_Transaction $transaction |
|
| 688 | + * @return EE_Line_Item |
|
| 689 | + * @throws EE_Error |
|
| 690 | + * @throws InvalidArgumentException |
|
| 691 | + * @throws InvalidDataTypeException |
|
| 692 | + * @throws InvalidInterfaceException |
|
| 693 | + * @throws ReflectionException |
|
| 694 | + */ |
|
| 695 | + protected static function create_pre_tax_subtotal(EE_Line_Item $total_line_item, $transaction = null) |
|
| 696 | + { |
|
| 697 | + $pre_tax_line_item = EE_Line_Item::new_instance(array( |
|
| 698 | + 'LIN_code' => 'pre-tax-subtotal', |
|
| 699 | + 'LIN_name' => esc_html__('Pre-Tax Subtotal', 'event_espresso'), |
|
| 700 | + 'LIN_type' => EEM_Line_Item::type_sub_total, |
|
| 701 | + )); |
|
| 702 | + $pre_tax_line_item = apply_filters( |
|
| 703 | + 'FHEE__EEH_Line_Item__create_pre_tax_subtotal__pre_tax_line_item', |
|
| 704 | + $pre_tax_line_item |
|
| 705 | + ); |
|
| 706 | + self::set_TXN_ID($pre_tax_line_item, $transaction); |
|
| 707 | + $total_line_item->add_child_line_item($pre_tax_line_item); |
|
| 708 | + self::create_event_subtotal($pre_tax_line_item, $transaction); |
|
| 709 | + return $pre_tax_line_item; |
|
| 710 | + } |
|
| 711 | + |
|
| 712 | + |
|
| 713 | + /** |
|
| 714 | + * Creates a line item for the taxes subtotal and finds all the tax prices |
|
| 715 | + * and applies taxes to it |
|
| 716 | + * |
|
| 717 | + * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 718 | + * @param EE_Transaction $transaction |
|
| 719 | + * @return EE_Line_Item |
|
| 720 | + * @throws EE_Error |
|
| 721 | + * @throws InvalidArgumentException |
|
| 722 | + * @throws InvalidDataTypeException |
|
| 723 | + * @throws InvalidInterfaceException |
|
| 724 | + * @throws ReflectionException |
|
| 725 | + */ |
|
| 726 | + protected static function create_taxes_subtotal(EE_Line_Item $total_line_item, $transaction = null) |
|
| 727 | + { |
|
| 728 | + $tax_line_item = EE_Line_Item::new_instance(array( |
|
| 729 | + 'LIN_code' => 'taxes', |
|
| 730 | + 'LIN_name' => esc_html__('Taxes', 'event_espresso'), |
|
| 731 | + 'LIN_type' => EEM_Line_Item::type_tax_sub_total, |
|
| 732 | + 'LIN_order' => 1000,// this should always come last |
|
| 733 | + )); |
|
| 734 | + $tax_line_item = apply_filters( |
|
| 735 | + 'FHEE__EEH_Line_Item__create_taxes_subtotal__tax_line_item', |
|
| 736 | + $tax_line_item |
|
| 737 | + ); |
|
| 738 | + self::set_TXN_ID($tax_line_item, $transaction); |
|
| 739 | + $total_line_item->add_child_line_item($tax_line_item); |
|
| 740 | + // and lastly, add the actual taxes |
|
| 741 | + self::apply_taxes($total_line_item); |
|
| 742 | + return $tax_line_item; |
|
| 743 | + } |
|
| 744 | + |
|
| 745 | + |
|
| 746 | + /** |
|
| 747 | + * Creates a default items subtotal line item |
|
| 748 | + * |
|
| 749 | + * @param EE_Line_Item $pre_tax_line_item |
|
| 750 | + * @param EE_Transaction $transaction |
|
| 751 | + * @param EE_Event $event |
|
| 752 | + * @return EE_Line_Item |
|
| 753 | + * @throws EE_Error |
|
| 754 | + * @throws InvalidArgumentException |
|
| 755 | + * @throws InvalidDataTypeException |
|
| 756 | + * @throws InvalidInterfaceException |
|
| 757 | + * @throws ReflectionException |
|
| 758 | + */ |
|
| 759 | + public static function create_event_subtotal(EE_Line_Item $pre_tax_line_item, $transaction = null, $event = null) |
|
| 760 | + { |
|
| 761 | + $event_line_item = EE_Line_Item::new_instance(array( |
|
| 762 | + 'LIN_code' => self::get_event_code($event), |
|
| 763 | + 'LIN_name' => self::get_event_name($event), |
|
| 764 | + 'LIN_desc' => self::get_event_desc($event), |
|
| 765 | + 'LIN_type' => EEM_Line_Item::type_sub_total, |
|
| 766 | + 'OBJ_type' => EEM_Line_Item::OBJ_TYPE_EVENT, |
|
| 767 | + 'OBJ_ID' => $event instanceof EE_Event ? $event->ID() : 0, |
|
| 768 | + )); |
|
| 769 | + $event_line_item = apply_filters( |
|
| 770 | + 'FHEE__EEH_Line_Item__create_event_subtotal__event_line_item', |
|
| 771 | + $event_line_item |
|
| 772 | + ); |
|
| 773 | + self::set_TXN_ID($event_line_item, $transaction); |
|
| 774 | + $pre_tax_line_item->add_child_line_item($event_line_item); |
|
| 775 | + return $event_line_item; |
|
| 776 | + } |
|
| 777 | + |
|
| 778 | + |
|
| 779 | + /** |
|
| 780 | + * Gets what the event ticket's code SHOULD be |
|
| 781 | + * |
|
| 782 | + * @param EE_Event $event |
|
| 783 | + * @return string |
|
| 784 | + * @throws EE_Error |
|
| 785 | + */ |
|
| 786 | + public static function get_event_code($event) |
|
| 787 | + { |
|
| 788 | + return 'event-' . ($event instanceof EE_Event ? $event->ID() : '0'); |
|
| 789 | + } |
|
| 790 | + |
|
| 791 | + |
|
| 792 | + /** |
|
| 793 | + * Gets the event name |
|
| 794 | + * |
|
| 795 | + * @param EE_Event $event |
|
| 796 | + * @return string |
|
| 797 | + * @throws EE_Error |
|
| 798 | + */ |
|
| 799 | + public static function get_event_name($event) |
|
| 800 | + { |
|
| 801 | + return $event instanceof EE_Event |
|
| 802 | + ? mb_substr($event->name(), 0, 245) |
|
| 803 | + : esc_html__('Event', 'event_espresso'); |
|
| 804 | + } |
|
| 805 | + |
|
| 806 | + |
|
| 807 | + /** |
|
| 808 | + * Gets the event excerpt |
|
| 809 | + * |
|
| 810 | + * @param EE_Event $event |
|
| 811 | + * @return string |
|
| 812 | + * @throws EE_Error |
|
| 813 | + */ |
|
| 814 | + public static function get_event_desc($event) |
|
| 815 | + { |
|
| 816 | + return $event instanceof EE_Event ? $event->short_description() : ''; |
|
| 817 | + } |
|
| 818 | + |
|
| 819 | + |
|
| 820 | + /** |
|
| 821 | + * Given the grand total line item and a ticket, finds the event sub-total |
|
| 822 | + * line item the ticket's purchase should be added onto |
|
| 823 | + * |
|
| 824 | + * @access public |
|
| 825 | + * @param EE_Line_Item $grand_total the grand total line item |
|
| 826 | + * @param EE_Ticket $ticket |
|
| 827 | + * @return EE_Line_Item |
|
| 828 | + * @throws EE_Error |
|
| 829 | + * @throws InvalidArgumentException |
|
| 830 | + * @throws InvalidDataTypeException |
|
| 831 | + * @throws InvalidInterfaceException |
|
| 832 | + * @throws ReflectionException |
|
| 833 | + */ |
|
| 834 | + public static function get_event_line_item_for_ticket(EE_Line_Item $grand_total, EE_Ticket $ticket) |
|
| 835 | + { |
|
| 836 | + $first_datetime = $ticket->first_datetime(); |
|
| 837 | + if (! $first_datetime instanceof EE_Datetime) { |
|
| 838 | + throw new EE_Error( |
|
| 839 | + sprintf( |
|
| 840 | + __('The supplied ticket (ID %d) has no datetimes', 'event_espresso'), |
|
| 841 | + $ticket->ID() |
|
| 842 | + ) |
|
| 843 | + ); |
|
| 844 | + } |
|
| 845 | + $event = $first_datetime->event(); |
|
| 846 | + if (! $event instanceof EE_Event) { |
|
| 847 | + throw new EE_Error( |
|
| 848 | + sprintf( |
|
| 849 | + esc_html__( |
|
| 850 | + 'The supplied ticket (ID %d) has no event data associated with it.', |
|
| 851 | + 'event_espresso' |
|
| 852 | + ), |
|
| 853 | + $ticket->ID() |
|
| 854 | + ) |
|
| 855 | + ); |
|
| 856 | + } |
|
| 857 | + $events_sub_total = EEH_Line_Item::get_event_line_item($grand_total, $event); |
|
| 858 | + if (! $events_sub_total instanceof EE_Line_Item) { |
|
| 859 | + throw new EE_Error( |
|
| 860 | + sprintf( |
|
| 861 | + esc_html__( |
|
| 862 | + 'There is no events sub-total for ticket %s on total line item %d', |
|
| 863 | + 'event_espresso' |
|
| 864 | + ), |
|
| 865 | + $ticket->ID(), |
|
| 866 | + $grand_total->ID() |
|
| 867 | + ) |
|
| 868 | + ); |
|
| 869 | + } |
|
| 870 | + return $events_sub_total; |
|
| 871 | + } |
|
| 872 | + |
|
| 873 | + |
|
| 874 | + /** |
|
| 875 | + * Gets the event line item |
|
| 876 | + * |
|
| 877 | + * @param EE_Line_Item $grand_total |
|
| 878 | + * @param EE_Event $event |
|
| 879 | + * @return EE_Line_Item for the event subtotal which is a child of $grand_total |
|
| 880 | + * @throws EE_Error |
|
| 881 | + * @throws InvalidArgumentException |
|
| 882 | + * @throws InvalidDataTypeException |
|
| 883 | + * @throws InvalidInterfaceException |
|
| 884 | + * @throws ReflectionException |
|
| 885 | + */ |
|
| 886 | + public static function get_event_line_item(EE_Line_Item $grand_total, $event) |
|
| 887 | + { |
|
| 888 | + /** @type EE_Event $event */ |
|
| 889 | + $event = EEM_Event::instance()->ensure_is_obj($event, true); |
|
| 890 | + $event_line_item = null; |
|
| 891 | + $found = false; |
|
| 892 | + foreach (EEH_Line_Item::get_event_subtotals($grand_total) as $event_line_item) { |
|
| 893 | + // default event subtotal, we should only ever find this the first time this method is called |
|
| 894 | + if (! $event_line_item->OBJ_ID()) { |
|
| 895 | + // let's use this! but first... set the event details |
|
| 896 | + EEH_Line_Item::set_event_subtotal_details($event_line_item, $event); |
|
| 897 | + $found = true; |
|
| 898 | + break; |
|
| 899 | + } |
|
| 900 | + if ($event_line_item->OBJ_ID() === $event->ID()) { |
|
| 901 | + // found existing line item for this event in the cart, so break out of loop and use this one |
|
| 902 | + $found = true; |
|
| 903 | + break; |
|
| 904 | + } |
|
| 905 | + } |
|
| 906 | + if (! $found) { |
|
| 907 | + // there is no event sub-total yet, so add it |
|
| 908 | + $pre_tax_subtotal = EEH_Line_Item::get_pre_tax_subtotal($grand_total); |
|
| 909 | + // create a new "event" subtotal below that |
|
| 910 | + $event_line_item = EEH_Line_Item::create_event_subtotal($pre_tax_subtotal, null, $event); |
|
| 911 | + // and set the event details |
|
| 912 | + EEH_Line_Item::set_event_subtotal_details($event_line_item, $event); |
|
| 913 | + } |
|
| 914 | + return $event_line_item; |
|
| 915 | + } |
|
| 916 | + |
|
| 917 | + |
|
| 918 | + /** |
|
| 919 | + * Creates a default items subtotal line item |
|
| 920 | + * |
|
| 921 | + * @param EE_Line_Item $event_line_item |
|
| 922 | + * @param EE_Event $event |
|
| 923 | + * @param EE_Transaction $transaction |
|
| 924 | + * @return void |
|
| 925 | + * @throws EE_Error |
|
| 926 | + * @throws InvalidArgumentException |
|
| 927 | + * @throws InvalidDataTypeException |
|
| 928 | + * @throws InvalidInterfaceException |
|
| 929 | + * @throws ReflectionException |
|
| 930 | + */ |
|
| 931 | + public static function set_event_subtotal_details( |
|
| 932 | + EE_Line_Item $event_line_item, |
|
| 933 | + EE_Event $event, |
|
| 934 | + $transaction = null |
|
| 935 | + ) { |
|
| 936 | + if ($event instanceof EE_Event) { |
|
| 937 | + $event_line_item->set_code(self::get_event_code($event)); |
|
| 938 | + $event_line_item->set_name(self::get_event_name($event)); |
|
| 939 | + $event_line_item->set_desc(self::get_event_desc($event)); |
|
| 940 | + $event_line_item->set_OBJ_ID($event->ID()); |
|
| 941 | + } |
|
| 942 | + self::set_TXN_ID($event_line_item, $transaction); |
|
| 943 | + } |
|
| 944 | + |
|
| 945 | + |
|
| 946 | + /** |
|
| 947 | + * Finds what taxes should apply, adds them as tax line items under the taxes sub-total, |
|
| 948 | + * and recalculates the taxes sub-total and the grand total. Resets the taxes, so |
|
| 949 | + * any old taxes are removed |
|
| 950 | + * |
|
| 951 | + * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 952 | + * @param bool $update_txn_status |
|
| 953 | + * @return bool |
|
| 954 | + * @throws EE_Error |
|
| 955 | + * @throws InvalidArgumentException |
|
| 956 | + * @throws InvalidDataTypeException |
|
| 957 | + * @throws InvalidInterfaceException |
|
| 958 | + * @throws ReflectionException |
|
| 959 | + * @throws RuntimeException |
|
| 960 | + */ |
|
| 961 | + public static function apply_taxes(EE_Line_Item $total_line_item, $update_txn_status = false) |
|
| 962 | + { |
|
| 963 | + /** @type EEM_Price $EEM_Price */ |
|
| 964 | + $EEM_Price = EE_Registry::instance()->load_model('Price'); |
|
| 965 | + // get array of taxes via Price Model |
|
| 966 | + $ordered_taxes = $EEM_Price->get_all_prices_that_are_taxes(); |
|
| 967 | + ksort($ordered_taxes); |
|
| 968 | + $taxes_line_item = self::get_taxes_subtotal($total_line_item); |
|
| 969 | + // just to be safe, remove its old tax line items |
|
| 970 | + $deleted = $taxes_line_item->delete_children_line_items(); |
|
| 971 | + $updates = false; |
|
| 972 | + // loop thru taxes |
|
| 973 | + foreach ($ordered_taxes as $order => $taxes) { |
|
| 974 | + foreach ($taxes as $tax) { |
|
| 975 | + if ($tax instanceof EE_Price) { |
|
| 976 | + $tax_line_item = EE_Line_Item::new_instance( |
|
| 977 | + array( |
|
| 978 | + 'LIN_name' => $tax->name(), |
|
| 979 | + 'LIN_desc' => $tax->desc(), |
|
| 980 | + 'LIN_percent' => $tax->amount(), |
|
| 981 | + 'LIN_is_taxable' => false, |
|
| 982 | + 'LIN_order' => $order, |
|
| 983 | + 'LIN_total' => 0, |
|
| 984 | + 'LIN_type' => EEM_Line_Item::type_tax, |
|
| 985 | + 'OBJ_type' => EEM_Line_Item::OBJ_TYPE_PRICE, |
|
| 986 | + 'OBJ_ID' => $tax->ID(), |
|
| 987 | + ) |
|
| 988 | + ); |
|
| 989 | + $tax_line_item = apply_filters( |
|
| 990 | + 'FHEE__EEH_Line_Item__apply_taxes__tax_line_item', |
|
| 991 | + $tax_line_item |
|
| 992 | + ); |
|
| 993 | + $updates = $taxes_line_item->add_child_line_item($tax_line_item) ? |
|
| 994 | + true : |
|
| 995 | + $updates; |
|
| 996 | + } |
|
| 997 | + } |
|
| 998 | + } |
|
| 999 | + // only recalculate totals if something changed |
|
| 1000 | + if ($deleted || $updates) { |
|
| 1001 | + $total_line_item->recalculate_total_including_taxes($update_txn_status); |
|
| 1002 | + return true; |
|
| 1003 | + } |
|
| 1004 | + return false; |
|
| 1005 | + } |
|
| 1006 | + |
|
| 1007 | + |
|
| 1008 | + /** |
|
| 1009 | + * Ensures that taxes have been applied to the order, if not applies them. |
|
| 1010 | + * Returns the total amount of tax |
|
| 1011 | + * |
|
| 1012 | + * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 1013 | + * @return float |
|
| 1014 | + * @throws EE_Error |
|
| 1015 | + * @throws InvalidArgumentException |
|
| 1016 | + * @throws InvalidDataTypeException |
|
| 1017 | + * @throws InvalidInterfaceException |
|
| 1018 | + * @throws ReflectionException |
|
| 1019 | + */ |
|
| 1020 | + public static function ensure_taxes_applied($total_line_item) |
|
| 1021 | + { |
|
| 1022 | + $taxes_subtotal = self::get_taxes_subtotal($total_line_item); |
|
| 1023 | + if (! $taxes_subtotal->children()) { |
|
| 1024 | + self::apply_taxes($total_line_item); |
|
| 1025 | + } |
|
| 1026 | + return $taxes_subtotal->total(); |
|
| 1027 | + } |
|
| 1028 | + |
|
| 1029 | + |
|
| 1030 | + /** |
|
| 1031 | + * Deletes ALL children of the passed line item |
|
| 1032 | + * |
|
| 1033 | + * @param EE_Line_Item $parent_line_item |
|
| 1034 | + * @return bool |
|
| 1035 | + * @throws EE_Error |
|
| 1036 | + * @throws InvalidArgumentException |
|
| 1037 | + * @throws InvalidDataTypeException |
|
| 1038 | + * @throws InvalidInterfaceException |
|
| 1039 | + * @throws ReflectionException |
|
| 1040 | + */ |
|
| 1041 | + public static function delete_all_child_items(EE_Line_Item $parent_line_item) |
|
| 1042 | + { |
|
| 1043 | + $deleted = 0; |
|
| 1044 | + foreach ($parent_line_item->children() as $child_line_item) { |
|
| 1045 | + if ($child_line_item instanceof EE_Line_Item) { |
|
| 1046 | + $deleted += EEH_Line_Item::delete_all_child_items($child_line_item); |
|
| 1047 | + if ($child_line_item->ID()) { |
|
| 1048 | + $child_line_item->delete(); |
|
| 1049 | + unset($child_line_item); |
|
| 1050 | + } else { |
|
| 1051 | + $parent_line_item->delete_child_line_item($child_line_item->code()); |
|
| 1052 | + } |
|
| 1053 | + $deleted++; |
|
| 1054 | + } |
|
| 1055 | + } |
|
| 1056 | + return $deleted; |
|
| 1057 | + } |
|
| 1058 | + |
|
| 1059 | + |
|
| 1060 | + /** |
|
| 1061 | + * Deletes the line items as indicated by the line item code(s) provided, |
|
| 1062 | + * regardless of where they're found in the line item tree. Automatically |
|
| 1063 | + * re-calculates the line item totals and updates the related transaction. But |
|
| 1064 | + * DOES NOT automatically upgrade the transaction's registrations' final prices (which |
|
| 1065 | + * should probably change because of this). |
|
| 1066 | + * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item() |
|
| 1067 | + * after using this, to keep the registration final prices in-sync with the transaction's total. |
|
| 1068 | + * |
|
| 1069 | + * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total |
|
| 1070 | + * @param array|bool|string $line_item_codes |
|
| 1071 | + * @return int number of items successfully removed |
|
| 1072 | + * @throws EE_Error |
|
| 1073 | + */ |
|
| 1074 | + public static function delete_items(EE_Line_Item $total_line_item, $line_item_codes = false) |
|
| 1075 | + { |
|
| 1076 | + |
|
| 1077 | + if ($total_line_item->type() !== EEM_Line_Item::type_total) { |
|
| 1078 | + EE_Error::doing_it_wrong( |
|
| 1079 | + 'EEH_Line_Item::delete_items', |
|
| 1080 | + esc_html__( |
|
| 1081 | + 'This static method should only be called with a TOTAL line item, otherwise we won\'t recalculate the totals correctly', |
|
| 1082 | + 'event_espresso' |
|
| 1083 | + ), |
|
| 1084 | + '4.6.18' |
|
| 1085 | + ); |
|
| 1086 | + } |
|
| 1087 | + do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
| 1088 | + |
|
| 1089 | + // check if only a single line_item_id was passed |
|
| 1090 | + if (! empty($line_item_codes) && ! is_array($line_item_codes)) { |
|
| 1091 | + // place single line_item_id in an array to appear as multiple line_item_ids |
|
| 1092 | + $line_item_codes = array($line_item_codes); |
|
| 1093 | + } |
|
| 1094 | + $removals = 0; |
|
| 1095 | + // cycle thru line_item_ids |
|
| 1096 | + foreach ($line_item_codes as $line_item_id) { |
|
| 1097 | + $removals += $total_line_item->delete_child_line_item($line_item_id); |
|
| 1098 | + } |
|
| 1099 | + |
|
| 1100 | + if ($removals > 0) { |
|
| 1101 | + $total_line_item->recalculate_taxes_and_tax_total(); |
|
| 1102 | + return $removals; |
|
| 1103 | + } else { |
|
| 1104 | + return false; |
|
| 1105 | + } |
|
| 1106 | + } |
|
| 1107 | + |
|
| 1108 | + |
|
| 1109 | + /** |
|
| 1110 | + * Overwrites the previous tax by clearing out the old taxes, and creates a new |
|
| 1111 | + * tax and updates the total line item accordingly |
|
| 1112 | + * |
|
| 1113 | + * @param EE_Line_Item $total_line_item |
|
| 1114 | + * @param float $amount |
|
| 1115 | + * @param string $name |
|
| 1116 | + * @param string $description |
|
| 1117 | + * @param string $code |
|
| 1118 | + * @param boolean $add_to_existing_line_item |
|
| 1119 | + * if true, and a duplicate line item with the same code is found, |
|
| 1120 | + * $amount will be added onto it; otherwise will simply set the taxes to match $amount |
|
| 1121 | + * @return EE_Line_Item the new tax line item created |
|
| 1122 | + * @throws EE_Error |
|
| 1123 | + * @throws InvalidArgumentException |
|
| 1124 | + * @throws InvalidDataTypeException |
|
| 1125 | + * @throws InvalidInterfaceException |
|
| 1126 | + * @throws ReflectionException |
|
| 1127 | + */ |
|
| 1128 | + public static function set_total_tax_to( |
|
| 1129 | + EE_Line_Item $total_line_item, |
|
| 1130 | + $amount, |
|
| 1131 | + $name = null, |
|
| 1132 | + $description = null, |
|
| 1133 | + $code = null, |
|
| 1134 | + $add_to_existing_line_item = false |
|
| 1135 | + ) { |
|
| 1136 | + $tax_subtotal = self::get_taxes_subtotal($total_line_item); |
|
| 1137 | + $taxable_total = $total_line_item->taxable_total(); |
|
| 1138 | + |
|
| 1139 | + if ($add_to_existing_line_item) { |
|
| 1140 | + $new_tax = $tax_subtotal->get_child_line_item($code); |
|
| 1141 | + EEM_Line_Item::instance()->delete( |
|
| 1142 | + array(array('LIN_code' => array('!=', $code), 'LIN_parent' => $tax_subtotal->ID())) |
|
| 1143 | + ); |
|
| 1144 | + } else { |
|
| 1145 | + $new_tax = null; |
|
| 1146 | + $tax_subtotal->delete_children_line_items(); |
|
| 1147 | + } |
|
| 1148 | + if ($new_tax) { |
|
| 1149 | + $new_tax->set_total($new_tax->total() + $amount); |
|
| 1150 | + $new_tax->set_percent($taxable_total ? $new_tax->total() / $taxable_total * 100 : 0); |
|
| 1151 | + } else { |
|
| 1152 | + // no existing tax item. Create it |
|
| 1153 | + $new_tax = EE_Line_Item::new_instance(array( |
|
| 1154 | + 'TXN_ID' => $total_line_item->TXN_ID(), |
|
| 1155 | + 'LIN_name' => $name ? $name : esc_html__('Tax', 'event_espresso'), |
|
| 1156 | + 'LIN_desc' => $description ? $description : '', |
|
| 1157 | + 'LIN_percent' => $taxable_total ? ($amount / $taxable_total * 100) : 0, |
|
| 1158 | + 'LIN_total' => $amount, |
|
| 1159 | + 'LIN_parent' => $tax_subtotal->ID(), |
|
| 1160 | + 'LIN_type' => EEM_Line_Item::type_tax, |
|
| 1161 | + 'LIN_code' => $code, |
|
| 1162 | + )); |
|
| 1163 | + } |
|
| 1164 | + |
|
| 1165 | + $new_tax = apply_filters( |
|
| 1166 | + 'FHEE__EEH_Line_Item__set_total_tax_to__new_tax_subtotal', |
|
| 1167 | + $new_tax, |
|
| 1168 | + $total_line_item |
|
| 1169 | + ); |
|
| 1170 | + $new_tax->save(); |
|
| 1171 | + $tax_subtotal->set_total($new_tax->total()); |
|
| 1172 | + $tax_subtotal->save(); |
|
| 1173 | + $total_line_item->recalculate_total_including_taxes(); |
|
| 1174 | + return $new_tax; |
|
| 1175 | + } |
|
| 1176 | + |
|
| 1177 | + |
|
| 1178 | + /** |
|
| 1179 | + * Makes all the line items which are children of $line_item taxable (or not). |
|
| 1180 | + * Does NOT save the line items |
|
| 1181 | + * |
|
| 1182 | + * @param EE_Line_Item $line_item |
|
| 1183 | + * @param boolean $taxable |
|
| 1184 | + * @param string $code_substring_for_whitelist if this string is part of the line item's code |
|
| 1185 | + * it will be whitelisted (ie, except from becoming taxable) |
|
| 1186 | + * @throws EE_Error |
|
| 1187 | + */ |
|
| 1188 | + public static function set_line_items_taxable( |
|
| 1189 | + EE_Line_Item $line_item, |
|
| 1190 | + $taxable = true, |
|
| 1191 | + $code_substring_for_whitelist = null |
|
| 1192 | + ) { |
|
| 1193 | + $whitelisted = false; |
|
| 1194 | + if ($code_substring_for_whitelist !== null) { |
|
| 1195 | + $whitelisted = strpos($line_item->code(), $code_substring_for_whitelist) !== false; |
|
| 1196 | + } |
|
| 1197 | + if (! $whitelisted && $line_item->is_line_item()) { |
|
| 1198 | + $line_item->set_is_taxable($taxable); |
|
| 1199 | + } |
|
| 1200 | + foreach ($line_item->children() as $child_line_item) { |
|
| 1201 | + EEH_Line_Item::set_line_items_taxable( |
|
| 1202 | + $child_line_item, |
|
| 1203 | + $taxable, |
|
| 1204 | + $code_substring_for_whitelist |
|
| 1205 | + ); |
|
| 1206 | + } |
|
| 1207 | + } |
|
| 1208 | + |
|
| 1209 | + |
|
| 1210 | + /** |
|
| 1211 | + * Gets all descendants that are event subtotals |
|
| 1212 | + * |
|
| 1213 | + * @uses EEH_Line_Item::get_subtotals_of_object_type() |
|
| 1214 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1215 | + * @return EE_Line_Item[] |
|
| 1216 | + * @throws EE_Error |
|
| 1217 | + */ |
|
| 1218 | + public static function get_event_subtotals(EE_Line_Item $parent_line_item) |
|
| 1219 | + { |
|
| 1220 | + return self::get_subtotals_of_object_type($parent_line_item, EEM_Line_Item::OBJ_TYPE_EVENT); |
|
| 1221 | + } |
|
| 1222 | + |
|
| 1223 | + |
|
| 1224 | + /** |
|
| 1225 | + * Gets all descendants subtotals that match the supplied object type |
|
| 1226 | + * |
|
| 1227 | + * @uses EEH_Line_Item::_get_descendants_by_type_and_object_type() |
|
| 1228 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1229 | + * @param string $obj_type |
|
| 1230 | + * @return EE_Line_Item[] |
|
| 1231 | + * @throws EE_Error |
|
| 1232 | + */ |
|
| 1233 | + public static function get_subtotals_of_object_type(EE_Line_Item $parent_line_item, $obj_type = '') |
|
| 1234 | + { |
|
| 1235 | + return self::_get_descendants_by_type_and_object_type( |
|
| 1236 | + $parent_line_item, |
|
| 1237 | + EEM_Line_Item::type_sub_total, |
|
| 1238 | + $obj_type |
|
| 1239 | + ); |
|
| 1240 | + } |
|
| 1241 | + |
|
| 1242 | + |
|
| 1243 | + /** |
|
| 1244 | + * Gets all descendants that are tickets |
|
| 1245 | + * |
|
| 1246 | + * @uses EEH_Line_Item::get_line_items_of_object_type() |
|
| 1247 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1248 | + * @return EE_Line_Item[] |
|
| 1249 | + * @throws EE_Error |
|
| 1250 | + */ |
|
| 1251 | + public static function get_ticket_line_items(EE_Line_Item $parent_line_item) |
|
| 1252 | + { |
|
| 1253 | + return self::get_line_items_of_object_type( |
|
| 1254 | + $parent_line_item, |
|
| 1255 | + EEM_Line_Item::OBJ_TYPE_TICKET |
|
| 1256 | + ); |
|
| 1257 | + } |
|
| 1258 | + |
|
| 1259 | + |
|
| 1260 | + /** |
|
| 1261 | + * Gets all descendants subtotals that match the supplied object type |
|
| 1262 | + * |
|
| 1263 | + * @uses EEH_Line_Item::_get_descendants_by_type_and_object_type() |
|
| 1264 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1265 | + * @param string $obj_type |
|
| 1266 | + * @return EE_Line_Item[] |
|
| 1267 | + * @throws EE_Error |
|
| 1268 | + */ |
|
| 1269 | + public static function get_line_items_of_object_type(EE_Line_Item $parent_line_item, $obj_type = '') |
|
| 1270 | + { |
|
| 1271 | + return self::_get_descendants_by_type_and_object_type( |
|
| 1272 | + $parent_line_item, |
|
| 1273 | + EEM_Line_Item::type_line_item, |
|
| 1274 | + $obj_type |
|
| 1275 | + ); |
|
| 1276 | + } |
|
| 1277 | + |
|
| 1278 | + |
|
| 1279 | + /** |
|
| 1280 | + * Gets all the descendants (ie, children or children of children etc) that are of the type 'tax' |
|
| 1281 | + * |
|
| 1282 | + * @uses EEH_Line_Item::get_descendants_of_type() |
|
| 1283 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1284 | + * @return EE_Line_Item[] |
|
| 1285 | + * @throws EE_Error |
|
| 1286 | + */ |
|
| 1287 | + public static function get_tax_descendants(EE_Line_Item $parent_line_item) |
|
| 1288 | + { |
|
| 1289 | + return EEH_Line_Item::get_descendants_of_type( |
|
| 1290 | + $parent_line_item, |
|
| 1291 | + EEM_Line_Item::type_tax |
|
| 1292 | + ); |
|
| 1293 | + } |
|
| 1294 | + |
|
| 1295 | + |
|
| 1296 | + /** |
|
| 1297 | + * Gets all the real items purchased which are children of this item |
|
| 1298 | + * |
|
| 1299 | + * @uses EEH_Line_Item::get_descendants_of_type() |
|
| 1300 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1301 | + * @return EE_Line_Item[] |
|
| 1302 | + * @throws EE_Error |
|
| 1303 | + */ |
|
| 1304 | + public static function get_line_item_descendants(EE_Line_Item $parent_line_item) |
|
| 1305 | + { |
|
| 1306 | + return EEH_Line_Item::get_descendants_of_type( |
|
| 1307 | + $parent_line_item, |
|
| 1308 | + EEM_Line_Item::type_line_item |
|
| 1309 | + ); |
|
| 1310 | + } |
|
| 1311 | + |
|
| 1312 | + |
|
| 1313 | + /** |
|
| 1314 | + * Gets all descendants of supplied line item that match the supplied line item type |
|
| 1315 | + * |
|
| 1316 | + * @uses EEH_Line_Item::_get_descendants_by_type_and_object_type() |
|
| 1317 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1318 | + * @param string $line_item_type one of the EEM_Line_Item constants |
|
| 1319 | + * @return EE_Line_Item[] |
|
| 1320 | + * @throws EE_Error |
|
| 1321 | + */ |
|
| 1322 | + public static function get_descendants_of_type(EE_Line_Item $parent_line_item, $line_item_type) |
|
| 1323 | + { |
|
| 1324 | + return self::_get_descendants_by_type_and_object_type( |
|
| 1325 | + $parent_line_item, |
|
| 1326 | + $line_item_type, |
|
| 1327 | + null |
|
| 1328 | + ); |
|
| 1329 | + } |
|
| 1330 | + |
|
| 1331 | + |
|
| 1332 | + /** |
|
| 1333 | + * Gets all descendants of supplied line item that match the supplied line item type and possibly the object type |
|
| 1334 | + * as well |
|
| 1335 | + * |
|
| 1336 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1337 | + * @param string $line_item_type one of the EEM_Line_Item constants |
|
| 1338 | + * @param string | NULL $obj_type object model class name (minus prefix) or NULL to ignore object type when |
|
| 1339 | + * searching |
|
| 1340 | + * @return EE_Line_Item[] |
|
| 1341 | + * @throws EE_Error |
|
| 1342 | + */ |
|
| 1343 | + protected static function _get_descendants_by_type_and_object_type( |
|
| 1344 | + EE_Line_Item $parent_line_item, |
|
| 1345 | + $line_item_type, |
|
| 1346 | + $obj_type = null |
|
| 1347 | + ) { |
|
| 1348 | + $objects = array(); |
|
| 1349 | + foreach ($parent_line_item->children() as $child_line_item) { |
|
| 1350 | + if ($child_line_item instanceof EE_Line_Item) { |
|
| 1351 | + if ($child_line_item->type() === $line_item_type |
|
| 1352 | + && ( |
|
| 1353 | + $child_line_item->OBJ_type() === $obj_type || $obj_type === null |
|
| 1354 | + ) |
|
| 1355 | + ) { |
|
| 1356 | + $objects[] = $child_line_item; |
|
| 1357 | + } else { |
|
| 1358 | + // go-through-all-its children looking for more matches |
|
| 1359 | + $objects = array_merge( |
|
| 1360 | + $objects, |
|
| 1361 | + self::_get_descendants_by_type_and_object_type( |
|
| 1362 | + $child_line_item, |
|
| 1363 | + $line_item_type, |
|
| 1364 | + $obj_type |
|
| 1365 | + ) |
|
| 1366 | + ); |
|
| 1367 | + } |
|
| 1368 | + } |
|
| 1369 | + } |
|
| 1370 | + return $objects; |
|
| 1371 | + } |
|
| 1372 | + |
|
| 1373 | + |
|
| 1374 | + /** |
|
| 1375 | + * Gets all descendants subtotals that match the supplied object type |
|
| 1376 | + * |
|
| 1377 | + * @uses EEH_Line_Item::_get_descendants_by_type_and_object_type() |
|
| 1378 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1379 | + * @param string $OBJ_type object type (like Event) |
|
| 1380 | + * @param array $OBJ_IDs array of OBJ_IDs |
|
| 1381 | + * @return EE_Line_Item[] |
|
| 1382 | + * @throws EE_Error |
|
| 1383 | + */ |
|
| 1384 | + public static function get_line_items_by_object_type_and_IDs( |
|
| 1385 | + EE_Line_Item $parent_line_item, |
|
| 1386 | + $OBJ_type = '', |
|
| 1387 | + $OBJ_IDs = array() |
|
| 1388 | + ) { |
|
| 1389 | + return self::_get_descendants_by_object_type_and_object_ID( |
|
| 1390 | + $parent_line_item, |
|
| 1391 | + $OBJ_type, |
|
| 1392 | + $OBJ_IDs |
|
| 1393 | + ); |
|
| 1394 | + } |
|
| 1395 | + |
|
| 1396 | + |
|
| 1397 | + /** |
|
| 1398 | + * Gets all descendants of supplied line item that match the supplied line item type and possibly the object type |
|
| 1399 | + * as well |
|
| 1400 | + * |
|
| 1401 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1402 | + * @param string $OBJ_type object type (like Event) |
|
| 1403 | + * @param array $OBJ_IDs array of OBJ_IDs |
|
| 1404 | + * @return EE_Line_Item[] |
|
| 1405 | + * @throws EE_Error |
|
| 1406 | + */ |
|
| 1407 | + protected static function _get_descendants_by_object_type_and_object_ID( |
|
| 1408 | + EE_Line_Item $parent_line_item, |
|
| 1409 | + $OBJ_type, |
|
| 1410 | + $OBJ_IDs |
|
| 1411 | + ) { |
|
| 1412 | + $objects = array(); |
|
| 1413 | + foreach ($parent_line_item->children() as $child_line_item) { |
|
| 1414 | + if ($child_line_item instanceof EE_Line_Item) { |
|
| 1415 | + if ($child_line_item->OBJ_type() === $OBJ_type |
|
| 1416 | + && is_array($OBJ_IDs) |
|
| 1417 | + && in_array($child_line_item->OBJ_ID(), $OBJ_IDs) |
|
| 1418 | + ) { |
|
| 1419 | + $objects[] = $child_line_item; |
|
| 1420 | + } else { |
|
| 1421 | + // go-through-all-its children looking for more matches |
|
| 1422 | + $objects = array_merge( |
|
| 1423 | + $objects, |
|
| 1424 | + self::_get_descendants_by_object_type_and_object_ID( |
|
| 1425 | + $child_line_item, |
|
| 1426 | + $OBJ_type, |
|
| 1427 | + $OBJ_IDs |
|
| 1428 | + ) |
|
| 1429 | + ); |
|
| 1430 | + } |
|
| 1431 | + } |
|
| 1432 | + } |
|
| 1433 | + return $objects; |
|
| 1434 | + } |
|
| 1435 | + |
|
| 1436 | + |
|
| 1437 | + /** |
|
| 1438 | + * Uses a breadth-first-search in order to find the nearest descendant of |
|
| 1439 | + * the specified type and returns it, else NULL |
|
| 1440 | + * |
|
| 1441 | + * @uses EEH_Line_Item::_get_nearest_descendant() |
|
| 1442 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1443 | + * @param string $type like one of the EEM_Line_Item::type_* |
|
| 1444 | + * @return EE_Line_Item |
|
| 1445 | + * @throws EE_Error |
|
| 1446 | + * @throws InvalidArgumentException |
|
| 1447 | + * @throws InvalidDataTypeException |
|
| 1448 | + * @throws InvalidInterfaceException |
|
| 1449 | + * @throws ReflectionException |
|
| 1450 | + */ |
|
| 1451 | + public static function get_nearest_descendant_of_type(EE_Line_Item $parent_line_item, $type) |
|
| 1452 | + { |
|
| 1453 | + return self::_get_nearest_descendant($parent_line_item, 'LIN_type', $type); |
|
| 1454 | + } |
|
| 1455 | + |
|
| 1456 | + |
|
| 1457 | + /** |
|
| 1458 | + * Uses a breadth-first-search in order to find the nearest descendant |
|
| 1459 | + * having the specified LIN_code and returns it, else NULL |
|
| 1460 | + * |
|
| 1461 | + * @uses EEH_Line_Item::_get_nearest_descendant() |
|
| 1462 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1463 | + * @param string $code any value used for LIN_code |
|
| 1464 | + * @return EE_Line_Item |
|
| 1465 | + * @throws EE_Error |
|
| 1466 | + * @throws InvalidArgumentException |
|
| 1467 | + * @throws InvalidDataTypeException |
|
| 1468 | + * @throws InvalidInterfaceException |
|
| 1469 | + * @throws ReflectionException |
|
| 1470 | + */ |
|
| 1471 | + public static function get_nearest_descendant_having_code(EE_Line_Item $parent_line_item, $code) |
|
| 1472 | + { |
|
| 1473 | + return self::_get_nearest_descendant($parent_line_item, 'LIN_code', $code); |
|
| 1474 | + } |
|
| 1475 | + |
|
| 1476 | + |
|
| 1477 | + /** |
|
| 1478 | + * Uses a breadth-first-search in order to find the nearest descendant |
|
| 1479 | + * having the specified LIN_code and returns it, else NULL |
|
| 1480 | + * |
|
| 1481 | + * @param EE_Line_Item $parent_line_item - the line item to find descendants of |
|
| 1482 | + * @param string $search_field name of EE_Line_Item property |
|
| 1483 | + * @param string $value any value stored in $search_field |
|
| 1484 | + * @return EE_Line_Item |
|
| 1485 | + * @throws EE_Error |
|
| 1486 | + * @throws InvalidArgumentException |
|
| 1487 | + * @throws InvalidDataTypeException |
|
| 1488 | + * @throws InvalidInterfaceException |
|
| 1489 | + * @throws ReflectionException |
|
| 1490 | + */ |
|
| 1491 | + protected static function _get_nearest_descendant(EE_Line_Item $parent_line_item, $search_field, $value) |
|
| 1492 | + { |
|
| 1493 | + foreach ($parent_line_item->children() as $child) { |
|
| 1494 | + if ($child->get($search_field) == $value) { |
|
| 1495 | + return $child; |
|
| 1496 | + } |
|
| 1497 | + } |
|
| 1498 | + foreach ($parent_line_item->children() as $child) { |
|
| 1499 | + $descendant_found = self::_get_nearest_descendant( |
|
| 1500 | + $child, |
|
| 1501 | + $search_field, |
|
| 1502 | + $value |
|
| 1503 | + ); |
|
| 1504 | + if ($descendant_found) { |
|
| 1505 | + return $descendant_found; |
|
| 1506 | + } |
|
| 1507 | + } |
|
| 1508 | + return null; |
|
| 1509 | + } |
|
| 1510 | + |
|
| 1511 | + |
|
| 1512 | + /** |
|
| 1513 | + * if passed line item has a TXN ID, uses that to jump directly to the grand total line item for the transaction, |
|
| 1514 | + * else recursively walks up the line item tree until a parent of type total is found, |
|
| 1515 | + * |
|
| 1516 | + * @param EE_Line_Item $line_item |
|
| 1517 | + * @return EE_Line_Item |
|
| 1518 | + * @throws EE_Error |
|
| 1519 | + */ |
|
| 1520 | + public static function find_transaction_grand_total_for_line_item(EE_Line_Item $line_item) |
|
| 1521 | + { |
|
| 1522 | + if ($line_item->TXN_ID()) { |
|
| 1523 | + $total_line_item = $line_item->transaction()->total_line_item(false); |
|
| 1524 | + if ($total_line_item instanceof EE_Line_Item) { |
|
| 1525 | + return $total_line_item; |
|
| 1526 | + } |
|
| 1527 | + } else { |
|
| 1528 | + $line_item_parent = $line_item->parent(); |
|
| 1529 | + if ($line_item_parent instanceof EE_Line_Item) { |
|
| 1530 | + if ($line_item_parent->is_total()) { |
|
| 1531 | + return $line_item_parent; |
|
| 1532 | + } |
|
| 1533 | + return EEH_Line_Item::find_transaction_grand_total_for_line_item($line_item_parent); |
|
| 1534 | + } |
|
| 1535 | + } |
|
| 1536 | + throw new EE_Error( |
|
| 1537 | + sprintf( |
|
| 1538 | + esc_html__( |
|
| 1539 | + 'A valid grand total for line item %1$d was not found.', |
|
| 1540 | + 'event_espresso' |
|
| 1541 | + ), |
|
| 1542 | + $line_item->ID() |
|
| 1543 | + ) |
|
| 1544 | + ); |
|
| 1545 | + } |
|
| 1546 | + |
|
| 1547 | + |
|
| 1548 | + /** |
|
| 1549 | + * Prints out a representation of the line item tree |
|
| 1550 | + * |
|
| 1551 | + * @param EE_Line_Item $line_item |
|
| 1552 | + * @param int $indentation |
|
| 1553 | + * @return void |
|
| 1554 | + * @throws EE_Error |
|
| 1555 | + */ |
|
| 1556 | + public static function visualize(EE_Line_Item $line_item, $indentation = 0) |
|
| 1557 | + { |
|
| 1558 | + echo defined('EE_TESTS_DIR') ? "\n" : '<br />'; |
|
| 1559 | + if (! $indentation) { |
|
| 1560 | + echo defined('EE_TESTS_DIR') ? "\n" : '<br />'; |
|
| 1561 | + } |
|
| 1562 | + for ($i = 0; $i < $indentation; $i++) { |
|
| 1563 | + echo '. '; |
|
| 1564 | + } |
|
| 1565 | + $breakdown = ''; |
|
| 1566 | + if ($line_item->is_line_item()) { |
|
| 1567 | + if ($line_item->is_percent()) { |
|
| 1568 | + $breakdown = "{$line_item->percent()}%"; |
|
| 1569 | + } else { |
|
| 1570 | + $breakdown = '$' . "{$line_item->unit_price()} x {$line_item->quantity()}"; |
|
| 1571 | + } |
|
| 1572 | + } |
|
| 1573 | + echo $line_item->name(); |
|
| 1574 | + echo " [ ID:{$line_item->ID()} | qty:{$line_item->quantity()} ] {$line_item->type()} : "; |
|
| 1575 | + echo '$' . (string) $line_item->total(); |
|
| 1576 | + if ($breakdown) { |
|
| 1577 | + echo " ( {$breakdown} )"; |
|
| 1578 | + } |
|
| 1579 | + if ($line_item->is_taxable()) { |
|
| 1580 | + echo ' * taxable'; |
|
| 1581 | + } |
|
| 1582 | + if ($line_item->children()) { |
|
| 1583 | + foreach ($line_item->children() as $child) { |
|
| 1584 | + self::visualize($child, $indentation + 1); |
|
| 1585 | + } |
|
| 1586 | + } |
|
| 1587 | + } |
|
| 1588 | + |
|
| 1589 | + |
|
| 1590 | + /** |
|
| 1591 | + * Calculates the registration's final price, taking into account that they |
|
| 1592 | + * need to not only help pay for their OWN ticket, but also any transaction-wide surcharges and taxes, |
|
| 1593 | + * and receive a portion of any transaction-wide discounts. |
|
| 1594 | + * eg1, if I buy a $1 ticket and brent buys a $9 ticket, and we receive a $5 discount |
|
| 1595 | + * then I'll get 1/10 of that $5 discount, which is $0.50, and brent will get |
|
| 1596 | + * 9/10ths of that $5 discount, which is $4.50. So my final price should be $0.50 |
|
| 1597 | + * and brent's final price should be $5.50. |
|
| 1598 | + * In order to do this, we basically need to traverse the line item tree calculating |
|
| 1599 | + * the running totals (just as if we were recalculating the total), but when we identify |
|
| 1600 | + * regular line items, we need to keep track of their share of the grand total. |
|
| 1601 | + * Also, we need to keep track of the TAXABLE total for each ticket purchase, so |
|
| 1602 | + * we can know how to apply taxes to it. (Note: "taxable total" does not equal the "pretax total" |
|
| 1603 | + * when there are non-taxable items; otherwise they would be the same) |
|
| 1604 | + * |
|
| 1605 | + * @param EE_Line_Item $line_item |
|
| 1606 | + * @param array $billable_ticket_quantities array of EE_Ticket IDs and their corresponding quantity that |
|
| 1607 | + * can be included in price calculations at this moment |
|
| 1608 | + * @return array keys are line items for tickets IDs and values are their share of the running total, |
|
| 1609 | + * plus the key 'total', and 'taxable' which also has keys of all |
|
| 1610 | + * the ticket IDs. |
|
| 1611 | + * Eg array( |
|
| 1612 | + * 12 => 4.3 |
|
| 1613 | + * 23 => 8.0 |
|
| 1614 | + * 'total' => 16.6, |
|
| 1615 | + * 'taxable' => array( |
|
| 1616 | + * 12 => 10, |
|
| 1617 | + * 23 => 4 |
|
| 1618 | + * ). |
|
| 1619 | + * So to find which registrations have which final price, we need |
|
| 1620 | + * to find which line item is theirs, which can be done with |
|
| 1621 | + * `EEM_Line_Item::instance()->get_line_item_for_registration( |
|
| 1622 | + * $registration );` |
|
| 1623 | + * @throws EE_Error |
|
| 1624 | + * @throws InvalidArgumentException |
|
| 1625 | + * @throws InvalidDataTypeException |
|
| 1626 | + * @throws InvalidInterfaceException |
|
| 1627 | + * @throws ReflectionException |
|
| 1628 | + */ |
|
| 1629 | + public static function calculate_reg_final_prices_per_line_item( |
|
| 1630 | + EE_Line_Item $line_item, |
|
| 1631 | + $billable_ticket_quantities = array() |
|
| 1632 | + ) { |
|
| 1633 | + $running_totals = [ |
|
| 1634 | + 'total' => 0, |
|
| 1635 | + 'taxable' => ['total' => 0] |
|
| 1636 | + ]; |
|
| 1637 | + foreach ($line_item->children() as $child_line_item) { |
|
| 1638 | + switch ($child_line_item->type()) { |
|
| 1639 | + case EEM_Line_Item::type_sub_total: |
|
| 1640 | + $running_totals_from_subtotal = EEH_Line_Item::calculate_reg_final_prices_per_line_item( |
|
| 1641 | + $child_line_item, |
|
| 1642 | + $billable_ticket_quantities |
|
| 1643 | + ); |
|
| 1644 | + // combine arrays but preserve numeric keys |
|
| 1645 | + $running_totals = array_replace_recursive($running_totals_from_subtotal, $running_totals); |
|
| 1646 | + $running_totals['total'] += $running_totals_from_subtotal['total']; |
|
| 1647 | + $running_totals['taxable']['total'] += $running_totals_from_subtotal['taxable']['total']; |
|
| 1648 | + break; |
|
| 1649 | + |
|
| 1650 | + case EEM_Line_Item::type_tax_sub_total: |
|
| 1651 | + // find how much the taxes percentage is |
|
| 1652 | + if ($child_line_item->percent() !== 0) { |
|
| 1653 | + $tax_percent_decimal = $child_line_item->percent() / 100; |
|
| 1654 | + } else { |
|
| 1655 | + $tax_percent_decimal = EE_Taxes::get_total_taxes_percentage() / 100; |
|
| 1656 | + } |
|
| 1657 | + // and apply to all the taxable totals, and add to the pretax totals |
|
| 1658 | + foreach ($running_totals as $line_item_id => $this_running_total) { |
|
| 1659 | + // "total" and "taxable" array key is an exception |
|
| 1660 | + if ($line_item_id === 'taxable') { |
|
| 1661 | + continue; |
|
| 1662 | + } |
|
| 1663 | + $taxable_total = $running_totals['taxable'][ $line_item_id ]; |
|
| 1664 | + $running_totals[ $line_item_id ] += ($taxable_total * $tax_percent_decimal); |
|
| 1665 | + } |
|
| 1666 | + break; |
|
| 1667 | + |
|
| 1668 | + case EEM_Line_Item::type_line_item: |
|
| 1669 | + // ticket line items or ???? |
|
| 1670 | + if ($child_line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_TICKET) { |
|
| 1671 | + // kk it's a ticket |
|
| 1672 | + if (isset($running_totals[ $child_line_item->ID() ])) { |
|
| 1673 | + // huh? that shouldn't happen. |
|
| 1674 | + $running_totals['total'] += $child_line_item->total(); |
|
| 1675 | + } else { |
|
| 1676 | + // its not in our running totals yet. great. |
|
| 1677 | + if ($child_line_item->is_taxable()) { |
|
| 1678 | + $taxable_amount = $child_line_item->unit_price(); |
|
| 1679 | + } else { |
|
| 1680 | + $taxable_amount = 0; |
|
| 1681 | + } |
|
| 1682 | + // are we only calculating totals for some tickets? |
|
| 1683 | + if (isset($billable_ticket_quantities[ $child_line_item->OBJ_ID() ])) { |
|
| 1684 | + $quantity = $billable_ticket_quantities[ $child_line_item->OBJ_ID() ]; |
|
| 1685 | + $running_totals[ $child_line_item->ID() ] = $quantity |
|
| 1686 | + ? $child_line_item->unit_price() |
|
| 1687 | + : 0; |
|
| 1688 | + $running_totals['taxable'][ $child_line_item->ID() ] = $quantity |
|
| 1689 | + ? $taxable_amount |
|
| 1690 | + : 0; |
|
| 1691 | + } else { |
|
| 1692 | + $quantity = $child_line_item->quantity(); |
|
| 1693 | + $running_totals[ $child_line_item->ID() ] = $child_line_item->unit_price(); |
|
| 1694 | + $running_totals['taxable'][ $child_line_item->ID() ] = $taxable_amount; |
|
| 1695 | + } |
|
| 1696 | + $running_totals['taxable']['total'] += $taxable_amount * $quantity; |
|
| 1697 | + $running_totals['total'] += $child_line_item->unit_price() * $quantity; |
|
| 1698 | + } |
|
| 1699 | + } else { |
|
| 1700 | + // it's some other type of item added to the cart |
|
| 1701 | + // it should affect the running totals |
|
| 1702 | + // basically we want to convert it into a PERCENT modifier. Because |
|
| 1703 | + // more clearly affect all registration's final price equally |
|
| 1704 | + $line_items_percent_of_running_total = $running_totals['total'] > 0 |
|
| 1705 | + ? ($child_line_item->total() / $running_totals['total']) + 1 |
|
| 1706 | + : 1; |
|
| 1707 | + foreach ($running_totals as $line_item_id => $this_running_total) { |
|
| 1708 | + // the "taxable" array key is an exception |
|
| 1709 | + if ($line_item_id === 'taxable') { |
|
| 1710 | + continue; |
|
| 1711 | + } |
|
| 1712 | + // update the running totals |
|
| 1713 | + // yes this actually even works for the running grand total! |
|
| 1714 | + $running_totals[ $line_item_id ] = |
|
| 1715 | + $line_items_percent_of_running_total * $this_running_total; |
|
| 1716 | + |
|
| 1717 | + if ($child_line_item->is_taxable()) { |
|
| 1718 | + $running_totals['taxable'][ $line_item_id ] = |
|
| 1719 | + $line_items_percent_of_running_total * $running_totals['taxable'][ $line_item_id ]; |
|
| 1720 | + } |
|
| 1721 | + } |
|
| 1722 | + } |
|
| 1723 | + break; |
|
| 1724 | + } |
|
| 1725 | + } |
|
| 1726 | + return $running_totals; |
|
| 1727 | + } |
|
| 1728 | + |
|
| 1729 | + |
|
| 1730 | + /** |
|
| 1731 | + * @param EE_Line_Item $total_line_item |
|
| 1732 | + * @param EE_Line_Item $ticket_line_item |
|
| 1733 | + * @return float | null |
|
| 1734 | + * @throws EE_Error |
|
| 1735 | + * @throws InvalidArgumentException |
|
| 1736 | + * @throws InvalidDataTypeException |
|
| 1737 | + * @throws InvalidInterfaceException |
|
| 1738 | + * @throws OutOfRangeException |
|
| 1739 | + * @throws ReflectionException |
|
| 1740 | + */ |
|
| 1741 | + public static function calculate_final_price_for_ticket_line_item( |
|
| 1742 | + EE_Line_Item $total_line_item, |
|
| 1743 | + EE_Line_Item $ticket_line_item |
|
| 1744 | + ) { |
|
| 1745 | + static $final_prices_per_ticket_line_item = array(); |
|
| 1746 | + if (empty($final_prices_per_ticket_line_item) || empty($final_prices_per_ticket_line_item[ $total_line_item->ID() ])) { |
|
| 1747 | + $final_prices_per_ticket_line_item[ $total_line_item->ID() ] = EEH_Line_Item::calculate_reg_final_prices_per_line_item( |
|
| 1748 | + $total_line_item |
|
| 1749 | + ); |
|
| 1750 | + } |
|
| 1751 | + // ok now find this new registration's final price |
|
| 1752 | + if (isset($final_prices_per_ticket_line_item[ $total_line_item->ID() ][ $ticket_line_item->ID() ])) { |
|
| 1753 | + return $final_prices_per_ticket_line_item[ $total_line_item->ID() ][ $ticket_line_item->ID() ]; |
|
| 1754 | + } |
|
| 1755 | + $message = sprintf( |
|
| 1756 | + esc_html__( |
|
| 1757 | + 'The final price for the ticket line item (ID:%1$d) could not be calculated.', |
|
| 1758 | + 'event_espresso' |
|
| 1759 | + ), |
|
| 1760 | + $ticket_line_item->ID() |
|
| 1761 | + ); |
|
| 1762 | + if (WP_DEBUG) { |
|
| 1763 | + $message .= '<br>' . print_r($final_prices_per_ticket_line_item, true); |
|
| 1764 | + throw new OutOfRangeException($message); |
|
| 1765 | + } |
|
| 1766 | + EE_Log::instance()->log(__CLASS__, __FUNCTION__, $message); |
|
| 1767 | + return null; |
|
| 1768 | + } |
|
| 1769 | + |
|
| 1770 | + |
|
| 1771 | + /** |
|
| 1772 | + * Creates a duplicate of the line item tree, except only includes billable items |
|
| 1773 | + * and the portion of line items attributed to billable things |
|
| 1774 | + * |
|
| 1775 | + * @param EE_Line_Item $line_item |
|
| 1776 | + * @param EE_Registration[] $registrations |
|
| 1777 | + * @return EE_Line_Item |
|
| 1778 | + * @throws EE_Error |
|
| 1779 | + * @throws InvalidArgumentException |
|
| 1780 | + * @throws InvalidDataTypeException |
|
| 1781 | + * @throws InvalidInterfaceException |
|
| 1782 | + * @throws ReflectionException |
|
| 1783 | + */ |
|
| 1784 | + public static function billable_line_item_tree(EE_Line_Item $line_item, $registrations) |
|
| 1785 | + { |
|
| 1786 | + $copy_li = EEH_Line_Item::billable_line_item($line_item, $registrations); |
|
| 1787 | + foreach ($line_item->children() as $child_li) { |
|
| 1788 | + $copy_li->add_child_line_item( |
|
| 1789 | + EEH_Line_Item::billable_line_item_tree($child_li, $registrations) |
|
| 1790 | + ); |
|
| 1791 | + } |
|
| 1792 | + // if this is the grand total line item, make sure the totals all add up |
|
| 1793 | + // (we could have duplicated this logic AS we copied the line items, but |
|
| 1794 | + // it seems DRYer this way) |
|
| 1795 | + if ($copy_li->type() === EEM_Line_Item::type_total) { |
|
| 1796 | + $copy_li->recalculate_total_including_taxes(); |
|
| 1797 | + } |
|
| 1798 | + return $copy_li; |
|
| 1799 | + } |
|
| 1800 | + |
|
| 1801 | + |
|
| 1802 | + /** |
|
| 1803 | + * Creates a new, unsaved line item from $line_item that factors in the |
|
| 1804 | + * number of billable registrations on $registrations. |
|
| 1805 | + * |
|
| 1806 | + * @param EE_Line_Item $line_item |
|
| 1807 | + * @param EE_Registration[] $registrations |
|
| 1808 | + * @return EE_Line_Item |
|
| 1809 | + * @throws EE_Error |
|
| 1810 | + * @throws InvalidArgumentException |
|
| 1811 | + * @throws InvalidDataTypeException |
|
| 1812 | + * @throws InvalidInterfaceException |
|
| 1813 | + * @throws ReflectionException |
|
| 1814 | + */ |
|
| 1815 | + public static function billable_line_item(EE_Line_Item $line_item, $registrations) |
|
| 1816 | + { |
|
| 1817 | + $new_li_fields = $line_item->model_field_array(); |
|
| 1818 | + if ($line_item->type() === EEM_Line_Item::type_line_item && |
|
| 1819 | + $line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_TICKET |
|
| 1820 | + ) { |
|
| 1821 | + $count = 0; |
|
| 1822 | + foreach ($registrations as $registration) { |
|
| 1823 | + if ($line_item->OBJ_ID() === $registration->ticket_ID() && |
|
| 1824 | + in_array( |
|
| 1825 | + $registration->status_ID(), |
|
| 1826 | + EEM_Registration::reg_statuses_that_allow_payment(), |
|
| 1827 | + true |
|
| 1828 | + ) |
|
| 1829 | + ) { |
|
| 1830 | + $count++; |
|
| 1831 | + } |
|
| 1832 | + } |
|
| 1833 | + $new_li_fields['LIN_quantity'] = $count; |
|
| 1834 | + } |
|
| 1835 | + // don't set the total. We'll leave that up to the code that calculates it |
|
| 1836 | + unset($new_li_fields['LIN_ID'], $new_li_fields['LIN_parent'], $new_li_fields['LIN_total']); |
|
| 1837 | + return EE_Line_Item::new_instance($new_li_fields); |
|
| 1838 | + } |
|
| 1839 | + |
|
| 1840 | + |
|
| 1841 | + /** |
|
| 1842 | + * Returns a modified line item tree where all the subtotals which have a total of 0 |
|
| 1843 | + * are removed, and line items with a quantity of 0 |
|
| 1844 | + * |
|
| 1845 | + * @param EE_Line_Item $line_item |null |
|
| 1846 | + * @return EE_Line_Item|null |
|
| 1847 | + * @throws EE_Error |
|
| 1848 | + * @throws InvalidArgumentException |
|
| 1849 | + * @throws InvalidDataTypeException |
|
| 1850 | + * @throws InvalidInterfaceException |
|
| 1851 | + * @throws ReflectionException |
|
| 1852 | + */ |
|
| 1853 | + public static function non_empty_line_items(EE_Line_Item $line_item) |
|
| 1854 | + { |
|
| 1855 | + $copied_li = EEH_Line_Item::non_empty_line_item($line_item); |
|
| 1856 | + if ($copied_li === null) { |
|
| 1857 | + return null; |
|
| 1858 | + } |
|
| 1859 | + // if this is an event subtotal, we want to only include it if it |
|
| 1860 | + // has a non-zero total and at least one ticket line item child |
|
| 1861 | + $ticket_children = 0; |
|
| 1862 | + foreach ($line_item->children() as $child_li) { |
|
| 1863 | + $child_li_copy = EEH_Line_Item::non_empty_line_items($child_li); |
|
| 1864 | + if ($child_li_copy !== null) { |
|
| 1865 | + $copied_li->add_child_line_item($child_li_copy); |
|
| 1866 | + if ($child_li_copy->type() === EEM_Line_Item::type_line_item && |
|
| 1867 | + $child_li_copy->OBJ_type() === EEM_Line_Item::OBJ_TYPE_TICKET |
|
| 1868 | + ) { |
|
| 1869 | + $ticket_children++; |
|
| 1870 | + } |
|
| 1871 | + } |
|
| 1872 | + } |
|
| 1873 | + // if this is an event subtotal with NO ticket children |
|
| 1874 | + // we basically want to ignore it |
|
| 1875 | + if ($ticket_children === 0 |
|
| 1876 | + && $line_item->type() === EEM_Line_Item::type_sub_total |
|
| 1877 | + && $line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_EVENT |
|
| 1878 | + && $line_item->total() === 0 |
|
| 1879 | + ) { |
|
| 1880 | + return null; |
|
| 1881 | + } |
|
| 1882 | + return $copied_li; |
|
| 1883 | + } |
|
| 1884 | + |
|
| 1885 | + |
|
| 1886 | + /** |
|
| 1887 | + * Creates a new, unsaved line item, but if it's a ticket line item |
|
| 1888 | + * with a total of 0, or a subtotal of 0, returns null instead |
|
| 1889 | + * |
|
| 1890 | + * @param EE_Line_Item $line_item |
|
| 1891 | + * @return EE_Line_Item |
|
| 1892 | + * @throws EE_Error |
|
| 1893 | + * @throws InvalidArgumentException |
|
| 1894 | + * @throws InvalidDataTypeException |
|
| 1895 | + * @throws InvalidInterfaceException |
|
| 1896 | + * @throws ReflectionException |
|
| 1897 | + */ |
|
| 1898 | + public static function non_empty_line_item(EE_Line_Item $line_item) |
|
| 1899 | + { |
|
| 1900 | + if ($line_item->type() === EEM_Line_Item::type_line_item |
|
| 1901 | + && $line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_TICKET |
|
| 1902 | + && $line_item->quantity() === 0 |
|
| 1903 | + ) { |
|
| 1904 | + return null; |
|
| 1905 | + } |
|
| 1906 | + $new_li_fields = $line_item->model_field_array(); |
|
| 1907 | + // don't set the total. We'll leave that up to the code that calculates it |
|
| 1908 | + unset($new_li_fields['LIN_ID'], $new_li_fields['LIN_parent']); |
|
| 1909 | + return EE_Line_Item::new_instance($new_li_fields); |
|
| 1910 | + } |
|
| 1911 | + |
|
| 1912 | + |
|
| 1913 | + /** |
|
| 1914 | + * Cycles through all of the ticket line items for the supplied total line item |
|
| 1915 | + * and ensures that the line item's "is_taxable" field matches that of its corresponding ticket |
|
| 1916 | + * |
|
| 1917 | + * @param EE_Line_Item $total_line_item |
|
| 1918 | + * @since 4.9.79.p |
|
| 1919 | + * @throws EE_Error |
|
| 1920 | + * @throws InvalidArgumentException |
|
| 1921 | + * @throws InvalidDataTypeException |
|
| 1922 | + * @throws InvalidInterfaceException |
|
| 1923 | + * @throws ReflectionException |
|
| 1924 | + */ |
|
| 1925 | + public static function resetIsTaxableForTickets(EE_Line_Item $total_line_item) |
|
| 1926 | + { |
|
| 1927 | + $ticket_line_items = self::get_ticket_line_items($total_line_item); |
|
| 1928 | + foreach ($ticket_line_items as $ticket_line_item) { |
|
| 1929 | + if ($ticket_line_item instanceof EE_Line_Item |
|
| 1930 | + && $ticket_line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_TICKET |
|
| 1931 | + ) { |
|
| 1932 | + $ticket = $ticket_line_item->ticket(); |
|
| 1933 | + if ($ticket instanceof EE_Ticket && $ticket->taxable() !== $ticket_line_item->is_taxable()) { |
|
| 1934 | + $ticket_line_item->set_is_taxable($ticket->taxable()); |
|
| 1935 | + $ticket_line_item->save(); |
|
| 1936 | + } |
|
| 1937 | + } |
|
| 1938 | + } |
|
| 1939 | + } |
|
| 1940 | + |
|
| 1941 | + |
|
| 1942 | + |
|
| 1943 | + /**************************************** @DEPRECATED METHODS *************************************** */ |
|
| 1944 | + /** |
|
| 1945 | + * @deprecated |
|
| 1946 | + * @param EE_Line_Item $total_line_item |
|
| 1947 | + * @return EE_Line_Item |
|
| 1948 | + * @throws EE_Error |
|
| 1949 | + * @throws InvalidArgumentException |
|
| 1950 | + * @throws InvalidDataTypeException |
|
| 1951 | + * @throws InvalidInterfaceException |
|
| 1952 | + * @throws ReflectionException |
|
| 1953 | + */ |
|
| 1954 | + public static function get_items_subtotal(EE_Line_Item $total_line_item) |
|
| 1955 | + { |
|
| 1956 | + EE_Error::doing_it_wrong( |
|
| 1957 | + 'EEH_Line_Item::get_items_subtotal()', |
|
| 1958 | + sprintf( |
|
| 1959 | + esc_html__('Method replaced with %1$s', 'event_espresso'), |
|
| 1960 | + 'EEH_Line_Item::get_pre_tax_subtotal()' |
|
| 1961 | + ), |
|
| 1962 | + '4.6.0' |
|
| 1963 | + ); |
|
| 1964 | + return self::get_pre_tax_subtotal($total_line_item); |
|
| 1965 | + } |
|
| 1966 | + |
|
| 1967 | + |
|
| 1968 | + /** |
|
| 1969 | + * @deprecated |
|
| 1970 | + * @param EE_Transaction $transaction |
|
| 1971 | + * @return EE_Line_Item |
|
| 1972 | + * @throws EE_Error |
|
| 1973 | + * @throws InvalidArgumentException |
|
| 1974 | + * @throws InvalidDataTypeException |
|
| 1975 | + * @throws InvalidInterfaceException |
|
| 1976 | + * @throws ReflectionException |
|
| 1977 | + */ |
|
| 1978 | + public static function create_default_total_line_item($transaction = null) |
|
| 1979 | + { |
|
| 1980 | + EE_Error::doing_it_wrong( |
|
| 1981 | + 'EEH_Line_Item::create_default_total_line_item()', |
|
| 1982 | + sprintf( |
|
| 1983 | + esc_html__('Method replaced with %1$s', 'event_espresso'), |
|
| 1984 | + 'EEH_Line_Item::create_total_line_item()' |
|
| 1985 | + ), |
|
| 1986 | + '4.6.0' |
|
| 1987 | + ); |
|
| 1988 | + return self::create_total_line_item($transaction); |
|
| 1989 | + } |
|
| 1990 | + |
|
| 1991 | + |
|
| 1992 | + /** |
|
| 1993 | + * @deprecated |
|
| 1994 | + * @param EE_Line_Item $total_line_item |
|
| 1995 | + * @param EE_Transaction $transaction |
|
| 1996 | + * @return EE_Line_Item |
|
| 1997 | + * @throws EE_Error |
|
| 1998 | + * @throws InvalidArgumentException |
|
| 1999 | + * @throws InvalidDataTypeException |
|
| 2000 | + * @throws InvalidInterfaceException |
|
| 2001 | + * @throws ReflectionException |
|
| 2002 | + */ |
|
| 2003 | + public static function create_default_tickets_subtotal(EE_Line_Item $total_line_item, $transaction = null) |
|
| 2004 | + { |
|
| 2005 | + EE_Error::doing_it_wrong( |
|
| 2006 | + 'EEH_Line_Item::create_default_tickets_subtotal()', |
|
| 2007 | + sprintf( |
|
| 2008 | + esc_html__('Method replaced with %1$s', 'event_espresso'), |
|
| 2009 | + 'EEH_Line_Item::create_pre_tax_subtotal()' |
|
| 2010 | + ), |
|
| 2011 | + '4.6.0' |
|
| 2012 | + ); |
|
| 2013 | + return self::create_pre_tax_subtotal($total_line_item, $transaction); |
|
| 2014 | + } |
|
| 2015 | + |
|
| 2016 | + |
|
| 2017 | + /** |
|
| 2018 | + * @deprecated |
|
| 2019 | + * @param EE_Line_Item $total_line_item |
|
| 2020 | + * @param EE_Transaction $transaction |
|
| 2021 | + * @return EE_Line_Item |
|
| 2022 | + * @throws EE_Error |
|
| 2023 | + * @throws InvalidArgumentException |
|
| 2024 | + * @throws InvalidDataTypeException |
|
| 2025 | + * @throws InvalidInterfaceException |
|
| 2026 | + * @throws ReflectionException |
|
| 2027 | + */ |
|
| 2028 | + public static function create_default_taxes_subtotal(EE_Line_Item $total_line_item, $transaction = null) |
|
| 2029 | + { |
|
| 2030 | + EE_Error::doing_it_wrong( |
|
| 2031 | + 'EEH_Line_Item::create_default_taxes_subtotal()', |
|
| 2032 | + sprintf( |
|
| 2033 | + esc_html__('Method replaced with %1$s', 'event_espresso'), |
|
| 2034 | + 'EEH_Line_Item::create_taxes_subtotal()' |
|
| 2035 | + ), |
|
| 2036 | + '4.6.0' |
|
| 2037 | + ); |
|
| 2038 | + return self::create_taxes_subtotal($total_line_item, $transaction); |
|
| 2039 | + } |
|
| 2040 | + |
|
| 2041 | + |
|
| 2042 | + /** |
|
| 2043 | + * @deprecated |
|
| 2044 | + * @param EE_Line_Item $total_line_item |
|
| 2045 | + * @param EE_Transaction $transaction |
|
| 2046 | + * @return EE_Line_Item |
|
| 2047 | + * @throws EE_Error |
|
| 2048 | + * @throws InvalidArgumentException |
|
| 2049 | + * @throws InvalidDataTypeException |
|
| 2050 | + * @throws InvalidInterfaceException |
|
| 2051 | + * @throws ReflectionException |
|
| 2052 | + */ |
|
| 2053 | + public static function create_default_event_subtotal(EE_Line_Item $total_line_item, $transaction = null) |
|
| 2054 | + { |
|
| 2055 | + EE_Error::doing_it_wrong( |
|
| 2056 | + 'EEH_Line_Item::create_default_event_subtotal()', |
|
| 2057 | + sprintf( |
|
| 2058 | + esc_html__('Method replaced with %1$s', 'event_espresso'), |
|
| 2059 | + 'EEH_Line_Item::create_event_subtotal()' |
|
| 2060 | + ), |
|
| 2061 | + '4.6.0' |
|
| 2062 | + ); |
|
| 2063 | + return self::create_event_subtotal($total_line_item, $transaction); |
|
| 2064 | + } |
|
| 2065 | 2065 | } |
@@ -138,7 +138,7 @@ discard block |
||
| 138 | 138 | */ |
| 139 | 139 | public static function add_ticket_purchase(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1) |
| 140 | 140 | { |
| 141 | - if (! $total_line_item instanceof EE_Line_Item || ! $total_line_item->is_total()) { |
|
| 141 | + if ( ! $total_line_item instanceof EE_Line_Item || ! $total_line_item->is_total()) { |
|
| 142 | 142 | throw new EE_Error( |
| 143 | 143 | sprintf( |
| 144 | 144 | esc_html__( |
@@ -153,7 +153,7 @@ discard block |
||
| 153 | 153 | // either increment the qty for an existing ticket |
| 154 | 154 | $line_item = self::increment_ticket_qty_if_already_in_cart($total_line_item, $ticket, $qty); |
| 155 | 155 | // or add a new one |
| 156 | - if (! $line_item instanceof EE_Line_Item) { |
|
| 156 | + if ( ! $line_item instanceof EE_Line_Item) { |
|
| 157 | 157 | $line_item = self::create_ticket_line_item($total_line_item, $ticket, $qty); |
| 158 | 158 | } |
| 159 | 159 | $total_line_item->recalculate_total_including_taxes(); |
@@ -214,7 +214,7 @@ discard block |
||
| 214 | 214 | */ |
| 215 | 215 | public static function increment_quantity(EE_Line_Item $line_item, $qty = 1) |
| 216 | 216 | { |
| 217 | - if (! $line_item->is_percent()) { |
|
| 217 | + if ( ! $line_item->is_percent()) { |
|
| 218 | 218 | $qty += $line_item->quantity(); |
| 219 | 219 | $line_item->set_quantity($qty); |
| 220 | 220 | $line_item->set_total($line_item->unit_price() * $qty); |
@@ -243,7 +243,7 @@ discard block |
||
| 243 | 243 | */ |
| 244 | 244 | public static function decrement_quantity(EE_Line_Item $line_item, $qty = 1) |
| 245 | 245 | { |
| 246 | - if (! $line_item->is_percent()) { |
|
| 246 | + if ( ! $line_item->is_percent()) { |
|
| 247 | 247 | $qty = $line_item->quantity() - $qty; |
| 248 | 248 | $qty = max($qty, 0); |
| 249 | 249 | $line_item->set_quantity($qty); |
@@ -272,7 +272,7 @@ discard block |
||
| 272 | 272 | */ |
| 273 | 273 | public static function update_quantity(EE_Line_Item $line_item, $new_quantity) |
| 274 | 274 | { |
| 275 | - if (! $line_item->is_percent()) { |
|
| 275 | + if ( ! $line_item->is_percent()) { |
|
| 276 | 276 | $line_item->set_quantity($new_quantity); |
| 277 | 277 | $line_item->set_total($line_item->unit_price() * $new_quantity); |
| 278 | 278 | $line_item->save(); |
@@ -312,7 +312,7 @@ discard block |
||
| 312 | 312 | // add $ticket to cart |
| 313 | 313 | $line_item = EE_Line_Item::new_instance(array( |
| 314 | 314 | 'LIN_name' => $ticket->name(), |
| 315 | - 'LIN_desc' => $ticket->description() !== '' ? $ticket->description() . ' ' . $event : $event, |
|
| 315 | + 'LIN_desc' => $ticket->description() !== '' ? $ticket->description().' '.$event : $event, |
|
| 316 | 316 | 'LIN_unit_price' => $ticket->price(), |
| 317 | 317 | 'LIN_quantity' => $qty, |
| 318 | 318 | 'LIN_is_taxable' => $ticket->taxable(), |
@@ -462,7 +462,7 @@ discard block |
||
| 462 | 462 | 'event_espresso' |
| 463 | 463 | ), |
| 464 | 464 | $ticket_line_item->name(), |
| 465 | - current_time(get_option('date_format') . ' ' . get_option('time_format')) |
|
| 465 | + current_time(get_option('date_format').' '.get_option('time_format')) |
|
| 466 | 466 | ), |
| 467 | 467 | 'LIN_unit_price' => 0, // $ticket_line_item->unit_price() |
| 468 | 468 | 'LIN_quantity' => $qty, |
@@ -524,7 +524,7 @@ discard block |
||
| 524 | 524 | ); |
| 525 | 525 | $cancellation_line_item = reset($cancellation_line_item); |
| 526 | 526 | // verify that this ticket was indeed previously cancelled |
| 527 | - if (! $cancellation_line_item instanceof EE_Line_Item) { |
|
| 527 | + if ( ! $cancellation_line_item instanceof EE_Line_Item) { |
|
| 528 | 528 | return false; |
| 529 | 529 | } |
| 530 | 530 | if ($cancellation_line_item->quantity() > $qty) { |
@@ -729,7 +729,7 @@ discard block |
||
| 729 | 729 | 'LIN_code' => 'taxes', |
| 730 | 730 | 'LIN_name' => esc_html__('Taxes', 'event_espresso'), |
| 731 | 731 | 'LIN_type' => EEM_Line_Item::type_tax_sub_total, |
| 732 | - 'LIN_order' => 1000,// this should always come last |
|
| 732 | + 'LIN_order' => 1000, // this should always come last |
|
| 733 | 733 | )); |
| 734 | 734 | $tax_line_item = apply_filters( |
| 735 | 735 | 'FHEE__EEH_Line_Item__create_taxes_subtotal__tax_line_item', |
@@ -785,7 +785,7 @@ discard block |
||
| 785 | 785 | */ |
| 786 | 786 | public static function get_event_code($event) |
| 787 | 787 | { |
| 788 | - return 'event-' . ($event instanceof EE_Event ? $event->ID() : '0'); |
|
| 788 | + return 'event-'.($event instanceof EE_Event ? $event->ID() : '0'); |
|
| 789 | 789 | } |
| 790 | 790 | |
| 791 | 791 | |
@@ -834,7 +834,7 @@ discard block |
||
| 834 | 834 | public static function get_event_line_item_for_ticket(EE_Line_Item $grand_total, EE_Ticket $ticket) |
| 835 | 835 | { |
| 836 | 836 | $first_datetime = $ticket->first_datetime(); |
| 837 | - if (! $first_datetime instanceof EE_Datetime) { |
|
| 837 | + if ( ! $first_datetime instanceof EE_Datetime) { |
|
| 838 | 838 | throw new EE_Error( |
| 839 | 839 | sprintf( |
| 840 | 840 | __('The supplied ticket (ID %d) has no datetimes', 'event_espresso'), |
@@ -843,7 +843,7 @@ discard block |
||
| 843 | 843 | ); |
| 844 | 844 | } |
| 845 | 845 | $event = $first_datetime->event(); |
| 846 | - if (! $event instanceof EE_Event) { |
|
| 846 | + if ( ! $event instanceof EE_Event) { |
|
| 847 | 847 | throw new EE_Error( |
| 848 | 848 | sprintf( |
| 849 | 849 | esc_html__( |
@@ -855,7 +855,7 @@ discard block |
||
| 855 | 855 | ); |
| 856 | 856 | } |
| 857 | 857 | $events_sub_total = EEH_Line_Item::get_event_line_item($grand_total, $event); |
| 858 | - if (! $events_sub_total instanceof EE_Line_Item) { |
|
| 858 | + if ( ! $events_sub_total instanceof EE_Line_Item) { |
|
| 859 | 859 | throw new EE_Error( |
| 860 | 860 | sprintf( |
| 861 | 861 | esc_html__( |
@@ -891,7 +891,7 @@ discard block |
||
| 891 | 891 | $found = false; |
| 892 | 892 | foreach (EEH_Line_Item::get_event_subtotals($grand_total) as $event_line_item) { |
| 893 | 893 | // default event subtotal, we should only ever find this the first time this method is called |
| 894 | - if (! $event_line_item->OBJ_ID()) { |
|
| 894 | + if ( ! $event_line_item->OBJ_ID()) { |
|
| 895 | 895 | // let's use this! but first... set the event details |
| 896 | 896 | EEH_Line_Item::set_event_subtotal_details($event_line_item, $event); |
| 897 | 897 | $found = true; |
@@ -903,7 +903,7 @@ discard block |
||
| 903 | 903 | break; |
| 904 | 904 | } |
| 905 | 905 | } |
| 906 | - if (! $found) { |
|
| 906 | + if ( ! $found) { |
|
| 907 | 907 | // there is no event sub-total yet, so add it |
| 908 | 908 | $pre_tax_subtotal = EEH_Line_Item::get_pre_tax_subtotal($grand_total); |
| 909 | 909 | // create a new "event" subtotal below that |
@@ -1020,7 +1020,7 @@ discard block |
||
| 1020 | 1020 | public static function ensure_taxes_applied($total_line_item) |
| 1021 | 1021 | { |
| 1022 | 1022 | $taxes_subtotal = self::get_taxes_subtotal($total_line_item); |
| 1023 | - if (! $taxes_subtotal->children()) { |
|
| 1023 | + if ( ! $taxes_subtotal->children()) { |
|
| 1024 | 1024 | self::apply_taxes($total_line_item); |
| 1025 | 1025 | } |
| 1026 | 1026 | return $taxes_subtotal->total(); |
@@ -1087,7 +1087,7 @@ discard block |
||
| 1087 | 1087 | do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
| 1088 | 1088 | |
| 1089 | 1089 | // check if only a single line_item_id was passed |
| 1090 | - if (! empty($line_item_codes) && ! is_array($line_item_codes)) { |
|
| 1090 | + if ( ! empty($line_item_codes) && ! is_array($line_item_codes)) { |
|
| 1091 | 1091 | // place single line_item_id in an array to appear as multiple line_item_ids |
| 1092 | 1092 | $line_item_codes = array($line_item_codes); |
| 1093 | 1093 | } |
@@ -1194,7 +1194,7 @@ discard block |
||
| 1194 | 1194 | if ($code_substring_for_whitelist !== null) { |
| 1195 | 1195 | $whitelisted = strpos($line_item->code(), $code_substring_for_whitelist) !== false; |
| 1196 | 1196 | } |
| 1197 | - if (! $whitelisted && $line_item->is_line_item()) { |
|
| 1197 | + if ( ! $whitelisted && $line_item->is_line_item()) { |
|
| 1198 | 1198 | $line_item->set_is_taxable($taxable); |
| 1199 | 1199 | } |
| 1200 | 1200 | foreach ($line_item->children() as $child_line_item) { |
@@ -1556,7 +1556,7 @@ discard block |
||
| 1556 | 1556 | public static function visualize(EE_Line_Item $line_item, $indentation = 0) |
| 1557 | 1557 | { |
| 1558 | 1558 | echo defined('EE_TESTS_DIR') ? "\n" : '<br />'; |
| 1559 | - if (! $indentation) { |
|
| 1559 | + if ( ! $indentation) { |
|
| 1560 | 1560 | echo defined('EE_TESTS_DIR') ? "\n" : '<br />'; |
| 1561 | 1561 | } |
| 1562 | 1562 | for ($i = 0; $i < $indentation; $i++) { |
@@ -1567,12 +1567,12 @@ discard block |
||
| 1567 | 1567 | if ($line_item->is_percent()) { |
| 1568 | 1568 | $breakdown = "{$line_item->percent()}%"; |
| 1569 | 1569 | } else { |
| 1570 | - $breakdown = '$' . "{$line_item->unit_price()} x {$line_item->quantity()}"; |
|
| 1570 | + $breakdown = '$'."{$line_item->unit_price()} x {$line_item->quantity()}"; |
|
| 1571 | 1571 | } |
| 1572 | 1572 | } |
| 1573 | 1573 | echo $line_item->name(); |
| 1574 | 1574 | echo " [ ID:{$line_item->ID()} | qty:{$line_item->quantity()} ] {$line_item->type()} : "; |
| 1575 | - echo '$' . (string) $line_item->total(); |
|
| 1575 | + echo '$'.(string) $line_item->total(); |
|
| 1576 | 1576 | if ($breakdown) { |
| 1577 | 1577 | echo " ( {$breakdown} )"; |
| 1578 | 1578 | } |
@@ -1660,8 +1660,8 @@ discard block |
||
| 1660 | 1660 | if ($line_item_id === 'taxable') { |
| 1661 | 1661 | continue; |
| 1662 | 1662 | } |
| 1663 | - $taxable_total = $running_totals['taxable'][ $line_item_id ]; |
|
| 1664 | - $running_totals[ $line_item_id ] += ($taxable_total * $tax_percent_decimal); |
|
| 1663 | + $taxable_total = $running_totals['taxable'][$line_item_id]; |
|
| 1664 | + $running_totals[$line_item_id] += ($taxable_total * $tax_percent_decimal); |
|
| 1665 | 1665 | } |
| 1666 | 1666 | break; |
| 1667 | 1667 | |
@@ -1669,7 +1669,7 @@ discard block |
||
| 1669 | 1669 | // ticket line items or ???? |
| 1670 | 1670 | if ($child_line_item->OBJ_type() === EEM_Line_Item::OBJ_TYPE_TICKET) { |
| 1671 | 1671 | // kk it's a ticket |
| 1672 | - if (isset($running_totals[ $child_line_item->ID() ])) { |
|
| 1672 | + if (isset($running_totals[$child_line_item->ID()])) { |
|
| 1673 | 1673 | // huh? that shouldn't happen. |
| 1674 | 1674 | $running_totals['total'] += $child_line_item->total(); |
| 1675 | 1675 | } else { |
@@ -1680,18 +1680,18 @@ discard block |
||
| 1680 | 1680 | $taxable_amount = 0; |
| 1681 | 1681 | } |
| 1682 | 1682 | // are we only calculating totals for some tickets? |
| 1683 | - if (isset($billable_ticket_quantities[ $child_line_item->OBJ_ID() ])) { |
|
| 1684 | - $quantity = $billable_ticket_quantities[ $child_line_item->OBJ_ID() ]; |
|
| 1685 | - $running_totals[ $child_line_item->ID() ] = $quantity |
|
| 1683 | + if (isset($billable_ticket_quantities[$child_line_item->OBJ_ID()])) { |
|
| 1684 | + $quantity = $billable_ticket_quantities[$child_line_item->OBJ_ID()]; |
|
| 1685 | + $running_totals[$child_line_item->ID()] = $quantity |
|
| 1686 | 1686 | ? $child_line_item->unit_price() |
| 1687 | 1687 | : 0; |
| 1688 | - $running_totals['taxable'][ $child_line_item->ID() ] = $quantity |
|
| 1688 | + $running_totals['taxable'][$child_line_item->ID()] = $quantity |
|
| 1689 | 1689 | ? $taxable_amount |
| 1690 | 1690 | : 0; |
| 1691 | 1691 | } else { |
| 1692 | 1692 | $quantity = $child_line_item->quantity(); |
| 1693 | - $running_totals[ $child_line_item->ID() ] = $child_line_item->unit_price(); |
|
| 1694 | - $running_totals['taxable'][ $child_line_item->ID() ] = $taxable_amount; |
|
| 1693 | + $running_totals[$child_line_item->ID()] = $child_line_item->unit_price(); |
|
| 1694 | + $running_totals['taxable'][$child_line_item->ID()] = $taxable_amount; |
|
| 1695 | 1695 | } |
| 1696 | 1696 | $running_totals['taxable']['total'] += $taxable_amount * $quantity; |
| 1697 | 1697 | $running_totals['total'] += $child_line_item->unit_price() * $quantity; |
@@ -1711,12 +1711,12 @@ discard block |
||
| 1711 | 1711 | } |
| 1712 | 1712 | // update the running totals |
| 1713 | 1713 | // yes this actually even works for the running grand total! |
| 1714 | - $running_totals[ $line_item_id ] = |
|
| 1714 | + $running_totals[$line_item_id] = |
|
| 1715 | 1715 | $line_items_percent_of_running_total * $this_running_total; |
| 1716 | 1716 | |
| 1717 | 1717 | if ($child_line_item->is_taxable()) { |
| 1718 | - $running_totals['taxable'][ $line_item_id ] = |
|
| 1719 | - $line_items_percent_of_running_total * $running_totals['taxable'][ $line_item_id ]; |
|
| 1718 | + $running_totals['taxable'][$line_item_id] = |
|
| 1719 | + $line_items_percent_of_running_total * $running_totals['taxable'][$line_item_id]; |
|
| 1720 | 1720 | } |
| 1721 | 1721 | } |
| 1722 | 1722 | } |
@@ -1743,14 +1743,14 @@ discard block |
||
| 1743 | 1743 | EE_Line_Item $ticket_line_item |
| 1744 | 1744 | ) { |
| 1745 | 1745 | static $final_prices_per_ticket_line_item = array(); |
| 1746 | - if (empty($final_prices_per_ticket_line_item) || empty($final_prices_per_ticket_line_item[ $total_line_item->ID() ])) { |
|
| 1747 | - $final_prices_per_ticket_line_item[ $total_line_item->ID() ] = EEH_Line_Item::calculate_reg_final_prices_per_line_item( |
|
| 1746 | + if (empty($final_prices_per_ticket_line_item) || empty($final_prices_per_ticket_line_item[$total_line_item->ID()])) { |
|
| 1747 | + $final_prices_per_ticket_line_item[$total_line_item->ID()] = EEH_Line_Item::calculate_reg_final_prices_per_line_item( |
|
| 1748 | 1748 | $total_line_item |
| 1749 | 1749 | ); |
| 1750 | 1750 | } |
| 1751 | 1751 | // ok now find this new registration's final price |
| 1752 | - if (isset($final_prices_per_ticket_line_item[ $total_line_item->ID() ][ $ticket_line_item->ID() ])) { |
|
| 1753 | - return $final_prices_per_ticket_line_item[ $total_line_item->ID() ][ $ticket_line_item->ID() ]; |
|
| 1752 | + if (isset($final_prices_per_ticket_line_item[$total_line_item->ID()][$ticket_line_item->ID()])) { |
|
| 1753 | + return $final_prices_per_ticket_line_item[$total_line_item->ID()][$ticket_line_item->ID()]; |
|
| 1754 | 1754 | } |
| 1755 | 1755 | $message = sprintf( |
| 1756 | 1756 | esc_html__( |
@@ -1760,7 +1760,7 @@ discard block |
||
| 1760 | 1760 | $ticket_line_item->ID() |
| 1761 | 1761 | ); |
| 1762 | 1762 | if (WP_DEBUG) { |
| 1763 | - $message .= '<br>' . print_r($final_prices_per_ticket_line_item, true); |
|
| 1763 | + $message .= '<br>'.print_r($final_prices_per_ticket_line_item, true); |
|
| 1764 | 1764 | throw new OutOfRangeException($message); |
| 1765 | 1765 | } |
| 1766 | 1766 | EE_Log::instance()->log(__CLASS__, __FUNCTION__, $message); |
@@ -17,165 +17,165 @@ |
||
| 17 | 17 | class ClassInterfaceCache |
| 18 | 18 | { |
| 19 | 19 | |
| 20 | - /** |
|
| 21 | - * array of interfaces indexed by FQCNs where values are arrays of interface FQNs |
|
| 22 | - * |
|
| 23 | - * @var string[][] $interfaces |
|
| 24 | - */ |
|
| 25 | - private $interfaces = array(); |
|
| 26 | - |
|
| 27 | - /** |
|
| 28 | - * @type string[][] $aliases |
|
| 29 | - */ |
|
| 30 | - protected $aliases = array(); |
|
| 31 | - |
|
| 32 | - |
|
| 33 | - /** |
|
| 34 | - * @param string $fqn |
|
| 35 | - * @return string |
|
| 36 | - */ |
|
| 37 | - public function getFqn($fqn) |
|
| 38 | - { |
|
| 39 | - $fqn = $fqn instanceof FullyQualifiedName ? $fqn->string() : $fqn; |
|
| 40 | - return ltrim($fqn, '\\'); |
|
| 41 | - } |
|
| 42 | - |
|
| 43 | - |
|
| 44 | - /** |
|
| 45 | - * @param string $fqn |
|
| 46 | - * @return array |
|
| 47 | - */ |
|
| 48 | - public function getInterfaces($fqn) |
|
| 49 | - { |
|
| 50 | - $fqn = $this->getFqn($fqn); |
|
| 51 | - // have we already seen this FQCN ? |
|
| 52 | - if (! array_key_exists($fqn, $this->interfaces)) { |
|
| 53 | - $this->interfaces[ $fqn ] = array(); |
|
| 54 | - if (class_exists($fqn)) { |
|
| 55 | - $this->interfaces[ $fqn ] = class_implements($fqn, false); |
|
| 56 | - $this->interfaces[ $fqn ] = $this->interfaces[ $fqn ] !== false |
|
| 57 | - ? $this->interfaces[ $fqn ] |
|
| 58 | - : array(); |
|
| 59 | - } |
|
| 60 | - } |
|
| 61 | - return $this->interfaces[ $fqn ]; |
|
| 62 | - } |
|
| 63 | - |
|
| 64 | - |
|
| 65 | - /** |
|
| 66 | - * @param string $fqn |
|
| 67 | - * @param string $interface |
|
| 68 | - * @return bool |
|
| 69 | - */ |
|
| 70 | - public function hasInterface($fqn, $interface) |
|
| 71 | - { |
|
| 72 | - $fqn = $this->getFqn($fqn); |
|
| 73 | - $interfaces = $this->getInterfaces($fqn); |
|
| 74 | - return in_array($interface, $interfaces, true); |
|
| 75 | - } |
|
| 76 | - |
|
| 77 | - |
|
| 78 | - /** |
|
| 79 | - * adds an alias for a classname |
|
| 80 | - * |
|
| 81 | - * @param string $fqn the class name that should be used (concrete class to replace interface) |
|
| 82 | - * @param string $alias the class name that would be type hinted for (abstract parent or interface) |
|
| 83 | - * @param string $for_class the class that has the dependency (is type hinting for the interface) |
|
| 84 | - * @throws InvalidAliasException |
|
| 85 | - */ |
|
| 86 | - public function addAlias($fqn, $alias, $for_class = '') |
|
| 87 | - { |
|
| 88 | - $fqn = $this->getFqn($fqn); |
|
| 89 | - $alias = $this->getFqn($alias); |
|
| 90 | - if (strpos($alias, '\\') !== false && ! is_subclass_of($fqn, $alias)) { |
|
| 91 | - throw new InvalidAliasException($fqn, $alias); |
|
| 92 | - } |
|
| 93 | - // are we adding an alias for a specific class? |
|
| 94 | - if ($for_class !== '') { |
|
| 95 | - // make sure it's set up as an array |
|
| 96 | - if (! isset($this->aliases[ $for_class ])) { |
|
| 97 | - $this->aliases[ $for_class ] = array(); |
|
| 98 | - } |
|
| 99 | - $this->aliases[ $for_class ][ $alias ] = $fqn; |
|
| 100 | - return; |
|
| 101 | - } |
|
| 102 | - $this->aliases[ $alias ] = $fqn; |
|
| 103 | - } |
|
| 104 | - |
|
| 105 | - |
|
| 106 | - /** |
|
| 107 | - * returns TRUE if the provided FQN is an alias |
|
| 108 | - * |
|
| 109 | - * @param string $fqn |
|
| 110 | - * @param string $for_class |
|
| 111 | - * @return bool |
|
| 112 | - */ |
|
| 113 | - public function isAlias($fqn = '', $for_class = '') |
|
| 114 | - { |
|
| 115 | - $fqn = $this->getFqn($fqn); |
|
| 116 | - if ($this->isAliasForClass($fqn, $for_class)) { |
|
| 117 | - return true; |
|
| 118 | - } |
|
| 119 | - if ($this->isDirectAlias($fqn)) { |
|
| 120 | - return true; |
|
| 121 | - } |
|
| 122 | - return false; |
|
| 123 | - } |
|
| 124 | - |
|
| 125 | - |
|
| 126 | - /** |
|
| 127 | - * returns TRUE if the provided FQN is an alias |
|
| 128 | - * |
|
| 129 | - * @param string $fqn |
|
| 130 | - * @return bool |
|
| 131 | - */ |
|
| 132 | - protected function isDirectAlias($fqn = '') |
|
| 133 | - { |
|
| 134 | - return isset($this->aliases[ (string) $fqn ]) && ! is_array($this->aliases[ (string) $fqn ]); |
|
| 135 | - } |
|
| 136 | - |
|
| 137 | - |
|
| 138 | - /** |
|
| 139 | - * returns TRUE if the provided FQN is an alias for the specified class |
|
| 140 | - * |
|
| 141 | - * @param string $fqn |
|
| 142 | - * @param string $for_class |
|
| 143 | - * @return bool |
|
| 144 | - */ |
|
| 145 | - protected function isAliasForClass($fqn = '', $for_class = '') |
|
| 146 | - { |
|
| 147 | - return ( |
|
| 148 | - $for_class !== '' |
|
| 149 | - && isset($this->aliases[ (string) $for_class ][ (string) $fqn ]) |
|
| 150 | - ); |
|
| 151 | - } |
|
| 152 | - |
|
| 153 | - |
|
| 154 | - /** |
|
| 155 | - * returns FQN for provided alias if one exists, otherwise returns the original FQN |
|
| 156 | - * functions recursively, so that multiple aliases can be used to drill down to a FQN |
|
| 157 | - * for example: |
|
| 158 | - * if the following two entries were added to the aliases array: |
|
| 159 | - * array( |
|
| 160 | - * 'interface_alias' => 'some\namespace\interface' |
|
| 161 | - * 'some\namespace\interface' => 'some\namespace\classname' |
|
| 162 | - * ) |
|
| 163 | - * then one could use Loader::getNew( 'interface_alias' ) |
|
| 164 | - * to load an instance of 'some\namespace\classname' |
|
| 165 | - * |
|
| 166 | - * @param string $alias |
|
| 167 | - * @param string $for_class |
|
| 168 | - * @return string |
|
| 169 | - */ |
|
| 170 | - public function getFqnForAlias($alias = '', $for_class = '') |
|
| 171 | - { |
|
| 172 | - $alias = $this->getFqn($alias); |
|
| 173 | - if ($this->isAliasForClass($alias, $for_class)) { |
|
| 174 | - return $this->getFqnForAlias($this->aliases[ (string) $for_class ][ (string) $alias ], $for_class); |
|
| 175 | - } |
|
| 176 | - if ($this->isDirectAlias($alias)) { |
|
| 177 | - return $this->getFqnForAlias($this->aliases[ (string) $alias ], ''); |
|
| 178 | - } |
|
| 179 | - return $alias; |
|
| 180 | - } |
|
| 20 | + /** |
|
| 21 | + * array of interfaces indexed by FQCNs where values are arrays of interface FQNs |
|
| 22 | + * |
|
| 23 | + * @var string[][] $interfaces |
|
| 24 | + */ |
|
| 25 | + private $interfaces = array(); |
|
| 26 | + |
|
| 27 | + /** |
|
| 28 | + * @type string[][] $aliases |
|
| 29 | + */ |
|
| 30 | + protected $aliases = array(); |
|
| 31 | + |
|
| 32 | + |
|
| 33 | + /** |
|
| 34 | + * @param string $fqn |
|
| 35 | + * @return string |
|
| 36 | + */ |
|
| 37 | + public function getFqn($fqn) |
|
| 38 | + { |
|
| 39 | + $fqn = $fqn instanceof FullyQualifiedName ? $fqn->string() : $fqn; |
|
| 40 | + return ltrim($fqn, '\\'); |
|
| 41 | + } |
|
| 42 | + |
|
| 43 | + |
|
| 44 | + /** |
|
| 45 | + * @param string $fqn |
|
| 46 | + * @return array |
|
| 47 | + */ |
|
| 48 | + public function getInterfaces($fqn) |
|
| 49 | + { |
|
| 50 | + $fqn = $this->getFqn($fqn); |
|
| 51 | + // have we already seen this FQCN ? |
|
| 52 | + if (! array_key_exists($fqn, $this->interfaces)) { |
|
| 53 | + $this->interfaces[ $fqn ] = array(); |
|
| 54 | + if (class_exists($fqn)) { |
|
| 55 | + $this->interfaces[ $fqn ] = class_implements($fqn, false); |
|
| 56 | + $this->interfaces[ $fqn ] = $this->interfaces[ $fqn ] !== false |
|
| 57 | + ? $this->interfaces[ $fqn ] |
|
| 58 | + : array(); |
|
| 59 | + } |
|
| 60 | + } |
|
| 61 | + return $this->interfaces[ $fqn ]; |
|
| 62 | + } |
|
| 63 | + |
|
| 64 | + |
|
| 65 | + /** |
|
| 66 | + * @param string $fqn |
|
| 67 | + * @param string $interface |
|
| 68 | + * @return bool |
|
| 69 | + */ |
|
| 70 | + public function hasInterface($fqn, $interface) |
|
| 71 | + { |
|
| 72 | + $fqn = $this->getFqn($fqn); |
|
| 73 | + $interfaces = $this->getInterfaces($fqn); |
|
| 74 | + return in_array($interface, $interfaces, true); |
|
| 75 | + } |
|
| 76 | + |
|
| 77 | + |
|
| 78 | + /** |
|
| 79 | + * adds an alias for a classname |
|
| 80 | + * |
|
| 81 | + * @param string $fqn the class name that should be used (concrete class to replace interface) |
|
| 82 | + * @param string $alias the class name that would be type hinted for (abstract parent or interface) |
|
| 83 | + * @param string $for_class the class that has the dependency (is type hinting for the interface) |
|
| 84 | + * @throws InvalidAliasException |
|
| 85 | + */ |
|
| 86 | + public function addAlias($fqn, $alias, $for_class = '') |
|
| 87 | + { |
|
| 88 | + $fqn = $this->getFqn($fqn); |
|
| 89 | + $alias = $this->getFqn($alias); |
|
| 90 | + if (strpos($alias, '\\') !== false && ! is_subclass_of($fqn, $alias)) { |
|
| 91 | + throw new InvalidAliasException($fqn, $alias); |
|
| 92 | + } |
|
| 93 | + // are we adding an alias for a specific class? |
|
| 94 | + if ($for_class !== '') { |
|
| 95 | + // make sure it's set up as an array |
|
| 96 | + if (! isset($this->aliases[ $for_class ])) { |
|
| 97 | + $this->aliases[ $for_class ] = array(); |
|
| 98 | + } |
|
| 99 | + $this->aliases[ $for_class ][ $alias ] = $fqn; |
|
| 100 | + return; |
|
| 101 | + } |
|
| 102 | + $this->aliases[ $alias ] = $fqn; |
|
| 103 | + } |
|
| 104 | + |
|
| 105 | + |
|
| 106 | + /** |
|
| 107 | + * returns TRUE if the provided FQN is an alias |
|
| 108 | + * |
|
| 109 | + * @param string $fqn |
|
| 110 | + * @param string $for_class |
|
| 111 | + * @return bool |
|
| 112 | + */ |
|
| 113 | + public function isAlias($fqn = '', $for_class = '') |
|
| 114 | + { |
|
| 115 | + $fqn = $this->getFqn($fqn); |
|
| 116 | + if ($this->isAliasForClass($fqn, $for_class)) { |
|
| 117 | + return true; |
|
| 118 | + } |
|
| 119 | + if ($this->isDirectAlias($fqn)) { |
|
| 120 | + return true; |
|
| 121 | + } |
|
| 122 | + return false; |
|
| 123 | + } |
|
| 124 | + |
|
| 125 | + |
|
| 126 | + /** |
|
| 127 | + * returns TRUE if the provided FQN is an alias |
|
| 128 | + * |
|
| 129 | + * @param string $fqn |
|
| 130 | + * @return bool |
|
| 131 | + */ |
|
| 132 | + protected function isDirectAlias($fqn = '') |
|
| 133 | + { |
|
| 134 | + return isset($this->aliases[ (string) $fqn ]) && ! is_array($this->aliases[ (string) $fqn ]); |
|
| 135 | + } |
|
| 136 | + |
|
| 137 | + |
|
| 138 | + /** |
|
| 139 | + * returns TRUE if the provided FQN is an alias for the specified class |
|
| 140 | + * |
|
| 141 | + * @param string $fqn |
|
| 142 | + * @param string $for_class |
|
| 143 | + * @return bool |
|
| 144 | + */ |
|
| 145 | + protected function isAliasForClass($fqn = '', $for_class = '') |
|
| 146 | + { |
|
| 147 | + return ( |
|
| 148 | + $for_class !== '' |
|
| 149 | + && isset($this->aliases[ (string) $for_class ][ (string) $fqn ]) |
|
| 150 | + ); |
|
| 151 | + } |
|
| 152 | + |
|
| 153 | + |
|
| 154 | + /** |
|
| 155 | + * returns FQN for provided alias if one exists, otherwise returns the original FQN |
|
| 156 | + * functions recursively, so that multiple aliases can be used to drill down to a FQN |
|
| 157 | + * for example: |
|
| 158 | + * if the following two entries were added to the aliases array: |
|
| 159 | + * array( |
|
| 160 | + * 'interface_alias' => 'some\namespace\interface' |
|
| 161 | + * 'some\namespace\interface' => 'some\namespace\classname' |
|
| 162 | + * ) |
|
| 163 | + * then one could use Loader::getNew( 'interface_alias' ) |
|
| 164 | + * to load an instance of 'some\namespace\classname' |
|
| 165 | + * |
|
| 166 | + * @param string $alias |
|
| 167 | + * @param string $for_class |
|
| 168 | + * @return string |
|
| 169 | + */ |
|
| 170 | + public function getFqnForAlias($alias = '', $for_class = '') |
|
| 171 | + { |
|
| 172 | + $alias = $this->getFqn($alias); |
|
| 173 | + if ($this->isAliasForClass($alias, $for_class)) { |
|
| 174 | + return $this->getFqnForAlias($this->aliases[ (string) $for_class ][ (string) $alias ], $for_class); |
|
| 175 | + } |
|
| 176 | + if ($this->isDirectAlias($alias)) { |
|
| 177 | + return $this->getFqnForAlias($this->aliases[ (string) $alias ], ''); |
|
| 178 | + } |
|
| 179 | + return $alias; |
|
| 180 | + } |
|
| 181 | 181 | } |
@@ -20,1093 +20,1093 @@ |
||
| 20 | 20 | class EE_Dependency_Map |
| 21 | 21 | { |
| 22 | 22 | |
| 23 | - /** |
|
| 24 | - * This means that the requested class dependency is not present in the dependency map |
|
| 25 | - */ |
|
| 26 | - const not_registered = 0; |
|
| 27 | - |
|
| 28 | - /** |
|
| 29 | - * This instructs class loaders to ALWAYS return a newly instantiated object for the requested class. |
|
| 30 | - */ |
|
| 31 | - const load_new_object = 1; |
|
| 32 | - |
|
| 33 | - /** |
|
| 34 | - * This instructs class loaders to return a previously instantiated and cached object for the requested class. |
|
| 35 | - * IF a previously instantiated object does not exist, a new one will be created and added to the cache. |
|
| 36 | - */ |
|
| 37 | - const load_from_cache = 2; |
|
| 38 | - |
|
| 39 | - /** |
|
| 40 | - * When registering a dependency, |
|
| 41 | - * this indicates to keep any existing dependencies that already exist, |
|
| 42 | - * and simply discard any new dependencies declared in the incoming data |
|
| 43 | - */ |
|
| 44 | - const KEEP_EXISTING_DEPENDENCIES = 0; |
|
| 45 | - |
|
| 46 | - /** |
|
| 47 | - * When registering a dependency, |
|
| 48 | - * this indicates to overwrite any existing dependencies that already exist using the incoming data |
|
| 49 | - */ |
|
| 50 | - const OVERWRITE_DEPENDENCIES = 1; |
|
| 51 | - |
|
| 52 | - |
|
| 53 | - /** |
|
| 54 | - * @type EE_Dependency_Map $_instance |
|
| 55 | - */ |
|
| 56 | - protected static $_instance; |
|
| 57 | - |
|
| 58 | - /** |
|
| 59 | - * @var ClassInterfaceCache $class_cache |
|
| 60 | - */ |
|
| 61 | - private $class_cache; |
|
| 62 | - |
|
| 63 | - /** |
|
| 64 | - * @type RequestInterface $request |
|
| 65 | - */ |
|
| 66 | - protected $request; |
|
| 67 | - |
|
| 68 | - /** |
|
| 69 | - * @type LegacyRequestInterface $legacy_request |
|
| 70 | - */ |
|
| 71 | - protected $legacy_request; |
|
| 72 | - |
|
| 73 | - /** |
|
| 74 | - * @type ResponseInterface $response |
|
| 75 | - */ |
|
| 76 | - protected $response; |
|
| 77 | - |
|
| 78 | - /** |
|
| 79 | - * @type LoaderInterface $loader |
|
| 80 | - */ |
|
| 81 | - protected $loader; |
|
| 82 | - |
|
| 83 | - /** |
|
| 84 | - * @type array $_dependency_map |
|
| 85 | - */ |
|
| 86 | - protected $_dependency_map = array(); |
|
| 87 | - |
|
| 88 | - /** |
|
| 89 | - * @type array $_class_loaders |
|
| 90 | - */ |
|
| 91 | - protected $_class_loaders = array(); |
|
| 92 | - |
|
| 93 | - |
|
| 94 | - /** |
|
| 95 | - * EE_Dependency_Map constructor. |
|
| 96 | - * |
|
| 97 | - * @param ClassInterfaceCache $class_cache |
|
| 98 | - */ |
|
| 99 | - protected function __construct(ClassInterfaceCache $class_cache) |
|
| 100 | - { |
|
| 101 | - $this->class_cache = $class_cache; |
|
| 102 | - do_action('EE_Dependency_Map____construct', $this); |
|
| 103 | - } |
|
| 104 | - |
|
| 105 | - |
|
| 106 | - /** |
|
| 107 | - * @return void |
|
| 108 | - */ |
|
| 109 | - public function initialize() |
|
| 110 | - { |
|
| 111 | - $this->_register_core_dependencies(); |
|
| 112 | - $this->_register_core_class_loaders(); |
|
| 113 | - $this->_register_core_aliases(); |
|
| 114 | - } |
|
| 115 | - |
|
| 116 | - |
|
| 117 | - /** |
|
| 118 | - * @singleton method used to instantiate class object |
|
| 119 | - * @param ClassInterfaceCache|null $class_cache |
|
| 120 | - * @return EE_Dependency_Map |
|
| 121 | - */ |
|
| 122 | - public static function instance(ClassInterfaceCache $class_cache = null) |
|
| 123 | - { |
|
| 124 | - // check if class object is instantiated, and instantiated properly |
|
| 125 | - if (! self::$_instance instanceof EE_Dependency_Map |
|
| 126 | - && $class_cache instanceof ClassInterfaceCache |
|
| 127 | - ) { |
|
| 128 | - self::$_instance = new EE_Dependency_Map($class_cache); |
|
| 129 | - } |
|
| 130 | - return self::$_instance; |
|
| 131 | - } |
|
| 132 | - |
|
| 133 | - |
|
| 134 | - /** |
|
| 135 | - * @param RequestInterface $request |
|
| 136 | - */ |
|
| 137 | - public function setRequest(RequestInterface $request) |
|
| 138 | - { |
|
| 139 | - $this->request = $request; |
|
| 140 | - } |
|
| 141 | - |
|
| 142 | - |
|
| 143 | - /** |
|
| 144 | - * @param LegacyRequestInterface $legacy_request |
|
| 145 | - */ |
|
| 146 | - public function setLegacyRequest(LegacyRequestInterface $legacy_request) |
|
| 147 | - { |
|
| 148 | - $this->legacy_request = $legacy_request; |
|
| 149 | - } |
|
| 150 | - |
|
| 151 | - |
|
| 152 | - /** |
|
| 153 | - * @param ResponseInterface $response |
|
| 154 | - */ |
|
| 155 | - public function setResponse(ResponseInterface $response) |
|
| 156 | - { |
|
| 157 | - $this->response = $response; |
|
| 158 | - } |
|
| 159 | - |
|
| 160 | - |
|
| 161 | - /** |
|
| 162 | - * @param LoaderInterface $loader |
|
| 163 | - */ |
|
| 164 | - public function setLoader(LoaderInterface $loader) |
|
| 165 | - { |
|
| 166 | - $this->loader = $loader; |
|
| 167 | - } |
|
| 168 | - |
|
| 169 | - |
|
| 170 | - /** |
|
| 171 | - * @param string $class |
|
| 172 | - * @param array $dependencies |
|
| 173 | - * @param int $overwrite |
|
| 174 | - * @return bool |
|
| 175 | - */ |
|
| 176 | - public static function register_dependencies( |
|
| 177 | - $class, |
|
| 178 | - array $dependencies, |
|
| 179 | - $overwrite = EE_Dependency_Map::KEEP_EXISTING_DEPENDENCIES |
|
| 180 | - ) { |
|
| 181 | - return self::$_instance->registerDependencies($class, $dependencies, $overwrite); |
|
| 182 | - } |
|
| 183 | - |
|
| 184 | - |
|
| 185 | - /** |
|
| 186 | - * Assigns an array of class names and corresponding load sources (new or cached) |
|
| 187 | - * to the class specified by the first parameter. |
|
| 188 | - * IMPORTANT !!! |
|
| 189 | - * The order of elements in the incoming $dependencies array MUST match |
|
| 190 | - * the order of the constructor parameters for the class in question. |
|
| 191 | - * This is especially important when overriding any existing dependencies that are registered. |
|
| 192 | - * the third parameter controls whether any duplicate dependencies are overwritten or not. |
|
| 193 | - * |
|
| 194 | - * @param string $class |
|
| 195 | - * @param array $dependencies |
|
| 196 | - * @param int $overwrite |
|
| 197 | - * @return bool |
|
| 198 | - */ |
|
| 199 | - public function registerDependencies( |
|
| 200 | - $class, |
|
| 201 | - array $dependencies, |
|
| 202 | - $overwrite = EE_Dependency_Map::KEEP_EXISTING_DEPENDENCIES |
|
| 203 | - ) { |
|
| 204 | - $class = trim($class, '\\'); |
|
| 205 | - $registered = false; |
|
| 206 | - if (empty(self::$_instance->_dependency_map[ $class ])) { |
|
| 207 | - self::$_instance->_dependency_map[ $class ] = array(); |
|
| 208 | - } |
|
| 209 | - // we need to make sure that any aliases used when registering a dependency |
|
| 210 | - // get resolved to the correct class name |
|
| 211 | - foreach ($dependencies as $dependency => $load_source) { |
|
| 212 | - $alias = self::$_instance->getFqnForAlias($dependency); |
|
| 213 | - if ($overwrite === EE_Dependency_Map::OVERWRITE_DEPENDENCIES |
|
| 214 | - || ! isset(self::$_instance->_dependency_map[ $class ][ $alias ]) |
|
| 215 | - ) { |
|
| 216 | - unset($dependencies[ $dependency ]); |
|
| 217 | - $dependencies[ $alias ] = $load_source; |
|
| 218 | - $registered = true; |
|
| 219 | - } |
|
| 220 | - } |
|
| 221 | - // now add our two lists of dependencies together. |
|
| 222 | - // using Union (+=) favours the arrays in precedence from left to right, |
|
| 223 | - // so $dependencies is NOT overwritten because it is listed first |
|
| 224 | - // ie: with A = B + C, entries in B take precedence over duplicate entries in C |
|
| 225 | - // Union is way faster than array_merge() but should be used with caution... |
|
| 226 | - // especially with numerically indexed arrays |
|
| 227 | - $dependencies += self::$_instance->_dependency_map[ $class ]; |
|
| 228 | - // now we need to ensure that the resulting dependencies |
|
| 229 | - // array only has the entries that are required for the class |
|
| 230 | - // so first count how many dependencies were originally registered for the class |
|
| 231 | - $dependency_count = count(self::$_instance->_dependency_map[ $class ]); |
|
| 232 | - // if that count is non-zero (meaning dependencies were already registered) |
|
| 233 | - self::$_instance->_dependency_map[ $class ] = $dependency_count |
|
| 234 | - // then truncate the final array to match that count |
|
| 235 | - ? array_slice($dependencies, 0, $dependency_count) |
|
| 236 | - // otherwise just take the incoming array because nothing previously existed |
|
| 237 | - : $dependencies; |
|
| 238 | - return $registered; |
|
| 239 | - } |
|
| 240 | - |
|
| 241 | - |
|
| 242 | - /** |
|
| 243 | - * @param string $class_name |
|
| 244 | - * @param string $loader |
|
| 245 | - * @return bool |
|
| 246 | - * @throws DomainException |
|
| 247 | - */ |
|
| 248 | - public static function register_class_loader($class_name, $loader = 'load_core') |
|
| 249 | - { |
|
| 250 | - if (! $loader instanceof Closure && strpos($class_name, '\\') !== false) { |
|
| 251 | - throw new DomainException( |
|
| 252 | - esc_html__('Don\'t use class loaders for FQCNs.', 'event_espresso') |
|
| 253 | - ); |
|
| 254 | - } |
|
| 255 | - // check that loader is callable or method starts with "load_" and exists in EE_Registry |
|
| 256 | - if (! is_callable($loader) |
|
| 257 | - && ( |
|
| 258 | - strpos($loader, 'load_') !== 0 |
|
| 259 | - || ! method_exists('EE_Registry', $loader) |
|
| 260 | - ) |
|
| 261 | - ) { |
|
| 262 | - throw new DomainException( |
|
| 263 | - sprintf( |
|
| 264 | - esc_html__( |
|
| 265 | - '"%1$s" is not a valid loader method on EE_Registry.', |
|
| 266 | - 'event_espresso' |
|
| 267 | - ), |
|
| 268 | - $loader |
|
| 269 | - ) |
|
| 270 | - ); |
|
| 271 | - } |
|
| 272 | - $class_name = self::$_instance->getFqnForAlias($class_name); |
|
| 273 | - if (! isset(self::$_instance->_class_loaders[ $class_name ])) { |
|
| 274 | - self::$_instance->_class_loaders[ $class_name ] = $loader; |
|
| 275 | - return true; |
|
| 276 | - } |
|
| 277 | - return false; |
|
| 278 | - } |
|
| 279 | - |
|
| 280 | - |
|
| 281 | - /** |
|
| 282 | - * @return array |
|
| 283 | - */ |
|
| 284 | - public function dependency_map() |
|
| 285 | - { |
|
| 286 | - return $this->_dependency_map; |
|
| 287 | - } |
|
| 288 | - |
|
| 289 | - |
|
| 290 | - /** |
|
| 291 | - * returns TRUE if dependency map contains a listing for the provided class name |
|
| 292 | - * |
|
| 293 | - * @param string $class_name |
|
| 294 | - * @return boolean |
|
| 295 | - */ |
|
| 296 | - public function has($class_name = '') |
|
| 297 | - { |
|
| 298 | - // all legacy models have the same dependencies |
|
| 299 | - if (strpos($class_name, 'EEM_') === 0) { |
|
| 300 | - $class_name = 'LEGACY_MODELS'; |
|
| 301 | - } |
|
| 302 | - return isset($this->_dependency_map[ $class_name ]) ? true : false; |
|
| 303 | - } |
|
| 304 | - |
|
| 305 | - |
|
| 306 | - /** |
|
| 307 | - * returns TRUE if dependency map contains a listing for the provided class name AND dependency |
|
| 308 | - * |
|
| 309 | - * @param string $class_name |
|
| 310 | - * @param string $dependency |
|
| 311 | - * @return bool |
|
| 312 | - */ |
|
| 313 | - public function has_dependency_for_class($class_name = '', $dependency = '') |
|
| 314 | - { |
|
| 315 | - // all legacy models have the same dependencies |
|
| 316 | - if (strpos($class_name, 'EEM_') === 0) { |
|
| 317 | - $class_name = 'LEGACY_MODELS'; |
|
| 318 | - } |
|
| 319 | - $dependency = $this->getFqnForAlias($dependency, $class_name); |
|
| 320 | - return isset($this->_dependency_map[ $class_name ][ $dependency ]) |
|
| 321 | - ? true |
|
| 322 | - : false; |
|
| 323 | - } |
|
| 324 | - |
|
| 325 | - |
|
| 326 | - /** |
|
| 327 | - * returns loading strategy for whether a previously cached dependency should be loaded or a new instance returned |
|
| 328 | - * |
|
| 329 | - * @param string $class_name |
|
| 330 | - * @param string $dependency |
|
| 331 | - * @return int |
|
| 332 | - */ |
|
| 333 | - public function loading_strategy_for_class_dependency($class_name = '', $dependency = '') |
|
| 334 | - { |
|
| 335 | - // all legacy models have the same dependencies |
|
| 336 | - if (strpos($class_name, 'EEM_') === 0) { |
|
| 337 | - $class_name = 'LEGACY_MODELS'; |
|
| 338 | - } |
|
| 339 | - $dependency = $this->getFqnForAlias($dependency); |
|
| 340 | - return $this->has_dependency_for_class($class_name, $dependency) |
|
| 341 | - ? $this->_dependency_map[ $class_name ][ $dependency ] |
|
| 342 | - : EE_Dependency_Map::not_registered; |
|
| 343 | - } |
|
| 344 | - |
|
| 345 | - |
|
| 346 | - /** |
|
| 347 | - * @param string $class_name |
|
| 348 | - * @return string | Closure |
|
| 349 | - */ |
|
| 350 | - public function class_loader($class_name) |
|
| 351 | - { |
|
| 352 | - // all legacy models use load_model() |
|
| 353 | - if (strpos($class_name, 'EEM_') === 0) { |
|
| 354 | - return 'load_model'; |
|
| 355 | - } |
|
| 356 | - // EE_CPT_*_Strategy classes like EE_CPT_Event_Strategy, EE_CPT_Venue_Strategy, etc |
|
| 357 | - // perform strpos() first to avoid loading regex every time we load a class |
|
| 358 | - if (strpos($class_name, 'EE_CPT_') === 0 |
|
| 359 | - && preg_match('/^EE_CPT_([a-zA-Z]+)_Strategy$/', $class_name) |
|
| 360 | - ) { |
|
| 361 | - return 'load_core'; |
|
| 362 | - } |
|
| 363 | - $class_name = $this->getFqnForAlias($class_name); |
|
| 364 | - return isset($this->_class_loaders[ $class_name ]) ? $this->_class_loaders[ $class_name ] : ''; |
|
| 365 | - } |
|
| 366 | - |
|
| 367 | - |
|
| 368 | - /** |
|
| 369 | - * @return array |
|
| 370 | - */ |
|
| 371 | - public function class_loaders() |
|
| 372 | - { |
|
| 373 | - return $this->_class_loaders; |
|
| 374 | - } |
|
| 375 | - |
|
| 376 | - |
|
| 377 | - /** |
|
| 378 | - * adds an alias for a classname |
|
| 379 | - * |
|
| 380 | - * @param string $fqcn the class name that should be used (concrete class to replace interface) |
|
| 381 | - * @param string $alias the class name that would be type hinted for (abstract parent or interface) |
|
| 382 | - * @param string $for_class the class that has the dependency (is type hinting for the interface) |
|
| 383 | - */ |
|
| 384 | - public function add_alias($fqcn, $alias, $for_class = '') |
|
| 385 | - { |
|
| 386 | - $this->class_cache->addAlias($fqcn, $alias, $for_class); |
|
| 387 | - } |
|
| 388 | - |
|
| 389 | - |
|
| 390 | - /** |
|
| 391 | - * Returns TRUE if the provided fully qualified name IS an alias |
|
| 392 | - * WHY? |
|
| 393 | - * Because if a class is type hinting for a concretion, |
|
| 394 | - * then why would we need to find another class to supply it? |
|
| 395 | - * ie: if a class asks for `Fully/Qualified/Namespace/SpecificClassName`, |
|
| 396 | - * then give it an instance of `Fully/Qualified/Namespace/SpecificClassName`. |
|
| 397 | - * Don't go looking for some substitute. |
|
| 398 | - * Whereas if a class is type hinting for an interface... |
|
| 399 | - * then we need to find an actual class to use. |
|
| 400 | - * So the interface IS the alias for some other FQN, |
|
| 401 | - * and we need to find out if `Fully/Qualified/Namespace/SomeInterface` |
|
| 402 | - * represents some other class. |
|
| 403 | - * |
|
| 404 | - * @param string $fqn |
|
| 405 | - * @param string $for_class |
|
| 406 | - * @return bool |
|
| 407 | - */ |
|
| 408 | - public function isAlias($fqn = '', $for_class = '') |
|
| 409 | - { |
|
| 410 | - return $this->class_cache->isAlias($fqn, $for_class); |
|
| 411 | - } |
|
| 412 | - |
|
| 413 | - |
|
| 414 | - /** |
|
| 415 | - * Returns a FQN for provided alias if one exists, otherwise returns the original $alias |
|
| 416 | - * functions recursively, so that multiple aliases can be used to drill down to a FQN |
|
| 417 | - * for example: |
|
| 418 | - * if the following two entries were added to the _aliases array: |
|
| 419 | - * array( |
|
| 420 | - * 'interface_alias' => 'some\namespace\interface' |
|
| 421 | - * 'some\namespace\interface' => 'some\namespace\classname' |
|
| 422 | - * ) |
|
| 423 | - * then one could use EE_Registry::instance()->create( 'interface_alias' ) |
|
| 424 | - * to load an instance of 'some\namespace\classname' |
|
| 425 | - * |
|
| 426 | - * @param string $alias |
|
| 427 | - * @param string $for_class |
|
| 428 | - * @return string |
|
| 429 | - */ |
|
| 430 | - public function getFqnForAlias($alias = '', $for_class = '') |
|
| 431 | - { |
|
| 432 | - return (string) $this->class_cache->getFqnForAlias($alias, $for_class); |
|
| 433 | - } |
|
| 434 | - |
|
| 435 | - |
|
| 436 | - /** |
|
| 437 | - * Registers the core dependencies and whether a previously instantiated object should be loaded from the cache, |
|
| 438 | - * if one exists, or whether a new object should be generated every time the requested class is loaded. |
|
| 439 | - * This is done by using the following class constants: |
|
| 440 | - * EE_Dependency_Map::load_from_cache - loads previously instantiated object |
|
| 441 | - * EE_Dependency_Map::load_new_object - generates a new object every time |
|
| 442 | - */ |
|
| 443 | - protected function _register_core_dependencies() |
|
| 444 | - { |
|
| 445 | - $this->_dependency_map = array( |
|
| 446 | - 'EE_Request_Handler' => array( |
|
| 447 | - 'EE_Request' => EE_Dependency_Map::load_from_cache, |
|
| 448 | - ), |
|
| 449 | - 'EE_System' => array( |
|
| 450 | - 'EE_Registry' => EE_Dependency_Map::load_from_cache, |
|
| 451 | - 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 452 | - 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 453 | - 'EE_Maintenance_Mode' => EE_Dependency_Map::load_from_cache, |
|
| 454 | - ), |
|
| 455 | - 'EE_Session' => array( |
|
| 456 | - 'EventEspresso\core\services\cache\TransientCacheStorage' => EE_Dependency_Map::load_from_cache, |
|
| 457 | - 'EventEspresso\core\domain\values\session\SessionLifespan' => EE_Dependency_Map::load_from_cache, |
|
| 458 | - 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 459 | - 'EventEspresso\core\services\session\SessionStartHandler' => EE_Dependency_Map::load_from_cache, |
|
| 460 | - 'EE_Encryption' => EE_Dependency_Map::load_from_cache, |
|
| 461 | - ), |
|
| 462 | - 'EE_Cart' => array( |
|
| 463 | - 'EE_Session' => EE_Dependency_Map::load_from_cache, |
|
| 464 | - ), |
|
| 465 | - 'EE_Front_Controller' => array( |
|
| 466 | - 'EE_Registry' => EE_Dependency_Map::load_from_cache, |
|
| 467 | - 'EE_Request_Handler' => EE_Dependency_Map::load_from_cache, |
|
| 468 | - 'EE_Module_Request_Router' => EE_Dependency_Map::load_from_cache, |
|
| 469 | - ), |
|
| 470 | - 'EE_Messenger_Collection_Loader' => array( |
|
| 471 | - 'EE_Messenger_Collection' => EE_Dependency_Map::load_new_object, |
|
| 472 | - ), |
|
| 473 | - 'EE_Message_Type_Collection_Loader' => array( |
|
| 474 | - 'EE_Message_Type_Collection' => EE_Dependency_Map::load_new_object, |
|
| 475 | - ), |
|
| 476 | - 'EE_Message_Resource_Manager' => array( |
|
| 477 | - 'EE_Messenger_Collection_Loader' => EE_Dependency_Map::load_new_object, |
|
| 478 | - 'EE_Message_Type_Collection_Loader' => EE_Dependency_Map::load_new_object, |
|
| 479 | - 'EEM_Message_Template_Group' => EE_Dependency_Map::load_from_cache, |
|
| 480 | - ), |
|
| 481 | - 'EE_Message_Factory' => array( |
|
| 482 | - 'EE_Message_Resource_Manager' => EE_Dependency_Map::load_from_cache, |
|
| 483 | - ), |
|
| 484 | - 'EE_messages' => array( |
|
| 485 | - 'EE_Message_Resource_Manager' => EE_Dependency_Map::load_from_cache, |
|
| 486 | - ), |
|
| 487 | - 'EE_Messages_Generator' => array( |
|
| 488 | - 'EE_Messages_Queue' => EE_Dependency_Map::load_new_object, |
|
| 489 | - 'EE_Messages_Data_Handler_Collection' => EE_Dependency_Map::load_new_object, |
|
| 490 | - 'EE_Message_Template_Group_Collection' => EE_Dependency_Map::load_new_object, |
|
| 491 | - 'EEH_Parse_Shortcodes' => EE_Dependency_Map::load_from_cache, |
|
| 492 | - ), |
|
| 493 | - 'EE_Messages_Processor' => array( |
|
| 494 | - 'EE_Message_Resource_Manager' => EE_Dependency_Map::load_from_cache, |
|
| 495 | - ), |
|
| 496 | - 'EE_Messages_Queue' => array( |
|
| 497 | - 'EE_Message_Repository' => EE_Dependency_Map::load_new_object, |
|
| 498 | - ), |
|
| 499 | - 'EE_Messages_Template_Defaults' => array( |
|
| 500 | - 'EEM_Message_Template_Group' => EE_Dependency_Map::load_from_cache, |
|
| 501 | - 'EEM_Message_Template' => EE_Dependency_Map::load_from_cache, |
|
| 502 | - ), |
|
| 503 | - 'EE_Message_To_Generate_From_Request' => array( |
|
| 504 | - 'EE_Message_Resource_Manager' => EE_Dependency_Map::load_from_cache, |
|
| 505 | - 'EE_Request_Handler' => EE_Dependency_Map::load_from_cache, |
|
| 506 | - ), |
|
| 507 | - 'EventEspresso\core\services\commands\CommandBus' => array( |
|
| 508 | - 'EventEspresso\core\services\commands\CommandHandlerManager' => EE_Dependency_Map::load_from_cache, |
|
| 509 | - ), |
|
| 510 | - 'EventEspresso\services\commands\CommandHandler' => array( |
|
| 511 | - 'EE_Registry' => EE_Dependency_Map::load_from_cache, |
|
| 512 | - 'CommandBusInterface' => EE_Dependency_Map::load_from_cache, |
|
| 513 | - ), |
|
| 514 | - 'EventEspresso\core\services\commands\CommandHandlerManager' => array( |
|
| 515 | - 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 516 | - ), |
|
| 517 | - 'EventEspresso\core\services\commands\CompositeCommandHandler' => array( |
|
| 518 | - 'EventEspresso\core\services\commands\CommandBus' => EE_Dependency_Map::load_from_cache, |
|
| 519 | - 'EventEspresso\core\services\commands\CommandFactory' => EE_Dependency_Map::load_from_cache, |
|
| 520 | - ), |
|
| 521 | - 'EventEspresso\core\services\commands\CommandFactory' => array( |
|
| 522 | - 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 523 | - ), |
|
| 524 | - 'EventEspresso\core\services\commands\middleware\CapChecker' => array( |
|
| 525 | - 'EventEspresso\core\domain\services\capabilities\CapabilitiesChecker' => EE_Dependency_Map::load_from_cache, |
|
| 526 | - ), |
|
| 527 | - 'EventEspresso\core\domain\services\capabilities\CapabilitiesChecker' => array( |
|
| 528 | - 'EE_Capabilities' => EE_Dependency_Map::load_from_cache, |
|
| 529 | - ), |
|
| 530 | - 'EventEspresso\core\domain\services\capabilities\RegistrationsCapChecker' => array( |
|
| 531 | - 'EE_Capabilities' => EE_Dependency_Map::load_from_cache, |
|
| 532 | - ), |
|
| 533 | - 'EventEspresso\core\services\commands\registration\CreateRegistrationCommandHandler' => array( |
|
| 534 | - 'EventEspresso\core\domain\services\registration\CreateRegistrationService' => EE_Dependency_Map::load_from_cache, |
|
| 535 | - ), |
|
| 536 | - 'EventEspresso\core\services\commands\registration\CopyRegistrationDetailsCommandHandler' => array( |
|
| 537 | - 'EventEspresso\core\domain\services\registration\CopyRegistrationService' => EE_Dependency_Map::load_from_cache, |
|
| 538 | - ), |
|
| 539 | - 'EventEspresso\core\services\commands\registration\CopyRegistrationPaymentsCommandHandler' => array( |
|
| 540 | - 'EventEspresso\core\domain\services\registration\CopyRegistrationService' => EE_Dependency_Map::load_from_cache, |
|
| 541 | - ), |
|
| 542 | - 'EventEspresso\core\services\commands\registration\CancelRegistrationAndTicketLineItemCommandHandler' => array( |
|
| 543 | - 'EventEspresso\core\domain\services\registration\CancelTicketLineItemService' => EE_Dependency_Map::load_from_cache, |
|
| 544 | - ), |
|
| 545 | - 'EventEspresso\core\services\commands\registration\UpdateRegistrationAndTransactionAfterChangeCommandHandler' => array( |
|
| 546 | - 'EventEspresso\core\domain\services\registration\UpdateRegistrationService' => EE_Dependency_Map::load_from_cache, |
|
| 547 | - ), |
|
| 548 | - 'EventEspresso\core\services\commands\ticket\CreateTicketLineItemCommandHandler' => array( |
|
| 549 | - 'EventEspresso\core\domain\services\ticket\CreateTicketLineItemService' => EE_Dependency_Map::load_from_cache, |
|
| 550 | - ), |
|
| 551 | - 'EventEspresso\core\services\commands\ticket\CancelTicketLineItemCommandHandler' => array( |
|
| 552 | - 'EventEspresso\core\domain\services\ticket\CancelTicketLineItemService' => EE_Dependency_Map::load_from_cache, |
|
| 553 | - ), |
|
| 554 | - 'EventEspresso\core\domain\services\registration\CancelRegistrationService' => array( |
|
| 555 | - 'EventEspresso\core\domain\services\ticket\CancelTicketLineItemService' => EE_Dependency_Map::load_from_cache, |
|
| 556 | - ), |
|
| 557 | - 'EventEspresso\core\services\commands\attendee\CreateAttendeeCommandHandler' => array( |
|
| 558 | - 'EEM_Attendee' => EE_Dependency_Map::load_from_cache, |
|
| 559 | - ), |
|
| 560 | - 'EventEspresso\core\services\database\TableManager' => array( |
|
| 561 | - 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 562 | - ), |
|
| 563 | - 'EE_Data_Migration_Class_Base' => array( |
|
| 564 | - 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 565 | - 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 566 | - ), |
|
| 567 | - 'EE_DMS_Core_4_1_0' => array( |
|
| 568 | - 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 569 | - 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 570 | - ), |
|
| 571 | - 'EE_DMS_Core_4_2_0' => array( |
|
| 572 | - 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 573 | - 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 574 | - ), |
|
| 575 | - 'EE_DMS_Core_4_3_0' => array( |
|
| 576 | - 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 577 | - 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 578 | - ), |
|
| 579 | - 'EE_DMS_Core_4_4_0' => array( |
|
| 580 | - 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 581 | - 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 582 | - ), |
|
| 583 | - 'EE_DMS_Core_4_5_0' => array( |
|
| 584 | - 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 585 | - 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 586 | - ), |
|
| 587 | - 'EE_DMS_Core_4_6_0' => array( |
|
| 588 | - 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 589 | - 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 590 | - ), |
|
| 591 | - 'EE_DMS_Core_4_7_0' => array( |
|
| 592 | - 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 593 | - 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 594 | - ), |
|
| 595 | - 'EE_DMS_Core_4_8_0' => array( |
|
| 596 | - 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 597 | - 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 598 | - ), |
|
| 599 | - 'EE_DMS_Core_4_9_0' => array( |
|
| 600 | - 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 601 | - 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 602 | - ), |
|
| 603 | - 'EventEspresso\core\services\assets\I18nRegistry' => array( |
|
| 604 | - array(), |
|
| 605 | - 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 606 | - ), |
|
| 607 | - 'EventEspresso\core\services\assets\Registry' => array( |
|
| 608 | - 'EventEspresso\core\services\assets\AssetCollection' => EE_Dependency_Map::load_from_cache, |
|
| 609 | - 'EventEspresso\core\services\assets\I18nRegistry' => EE_Dependency_Map::load_from_cache, |
|
| 610 | - ), |
|
| 611 | - 'EventEspresso\core\domain\entities\shortcodes\EspressoCancelled' => array( |
|
| 612 | - 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 613 | - ), |
|
| 614 | - 'EventEspresso\core\domain\entities\shortcodes\EspressoCheckout' => array( |
|
| 615 | - 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 616 | - ), |
|
| 617 | - 'EventEspresso\core\domain\entities\shortcodes\EspressoEventAttendees' => array( |
|
| 618 | - 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 619 | - ), |
|
| 620 | - 'EventEspresso\core\domain\entities\shortcodes\EspressoEvents' => array( |
|
| 621 | - 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 622 | - ), |
|
| 623 | - 'EventEspresso\core\domain\entities\shortcodes\EspressoThankYou' => array( |
|
| 624 | - 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 625 | - ), |
|
| 626 | - 'EventEspresso\core\domain\entities\shortcodes\EspressoTicketSelector' => array( |
|
| 627 | - 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 628 | - ), |
|
| 629 | - 'EventEspresso\core\domain\entities\shortcodes\EspressoTxnPage' => array( |
|
| 630 | - 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 631 | - ), |
|
| 632 | - 'EventEspresso\core\services\cache\BasicCacheManager' => array( |
|
| 633 | - 'EventEspresso\core\services\cache\TransientCacheStorage' => EE_Dependency_Map::load_from_cache, |
|
| 634 | - ), |
|
| 635 | - 'EventEspresso\core\services\cache\PostRelatedCacheManager' => array( |
|
| 636 | - 'EventEspresso\core\services\cache\TransientCacheStorage' => EE_Dependency_Map::load_from_cache, |
|
| 637 | - ), |
|
| 638 | - 'EventEspresso\core\domain\services\validation\email\EmailValidationService' => array( |
|
| 639 | - 'EE_Registration_Config' => EE_Dependency_Map::load_from_cache, |
|
| 640 | - 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 641 | - ), |
|
| 642 | - 'EventEspresso\core\domain\values\EmailAddress' => array( |
|
| 643 | - null, |
|
| 644 | - 'EventEspresso\core\domain\services\validation\email\EmailValidationService' => EE_Dependency_Map::load_from_cache, |
|
| 645 | - ), |
|
| 646 | - 'EventEspresso\core\services\orm\ModelFieldFactory' => array( |
|
| 647 | - 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 648 | - ), |
|
| 649 | - 'LEGACY_MODELS' => array( |
|
| 650 | - null, |
|
| 651 | - 'EventEspresso\core\services\database\ModelFieldFactory' => EE_Dependency_Map::load_from_cache, |
|
| 652 | - ), |
|
| 653 | - 'EE_Module_Request_Router' => array( |
|
| 654 | - 'EE_Request' => EE_Dependency_Map::load_from_cache, |
|
| 655 | - ), |
|
| 656 | - 'EE_Registration_Processor' => array( |
|
| 657 | - 'EE_Request' => EE_Dependency_Map::load_from_cache, |
|
| 658 | - ), |
|
| 659 | - 'EventEspresso\core\services\notifications\PersistentAdminNoticeManager' => array( |
|
| 660 | - null, |
|
| 661 | - 'EventEspresso\core\domain\services\capabilities\CapabilitiesChecker' => EE_Dependency_Map::load_from_cache, |
|
| 662 | - 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 663 | - ), |
|
| 664 | - 'EventEspresso\core\services\licensing\LicenseService' => array( |
|
| 665 | - 'EventEspresso\core\domain\services\pue\Stats' => EE_Dependency_Map::load_from_cache, |
|
| 666 | - 'EventEspresso\core\domain\services\pue\Config' => EE_Dependency_Map::load_from_cache, |
|
| 667 | - ), |
|
| 668 | - 'EE_Admin_Transactions_List_Table' => array( |
|
| 669 | - null, |
|
| 670 | - 'EventEspresso\core\domain\values\session\SessionLifespan' => EE_Dependency_Map::load_from_cache, |
|
| 671 | - ), |
|
| 672 | - 'EventEspresso\core\domain\services\pue\Stats' => array( |
|
| 673 | - 'EventEspresso\core\domain\services\pue\Config' => EE_Dependency_Map::load_from_cache, |
|
| 674 | - 'EE_Maintenance_Mode' => EE_Dependency_Map::load_from_cache, |
|
| 675 | - 'EventEspresso\core\domain\services\pue\StatsGatherer' => EE_Dependency_Map::load_from_cache, |
|
| 676 | - ), |
|
| 677 | - 'EventEspresso\core\domain\services\pue\Config' => array( |
|
| 678 | - 'EE_Network_Config' => EE_Dependency_Map::load_from_cache, |
|
| 679 | - 'EE_Config' => EE_Dependency_Map::load_from_cache, |
|
| 680 | - ), |
|
| 681 | - 'EventEspresso\core\domain\services\pue\StatsGatherer' => array( |
|
| 682 | - 'EEM_Payment_Method' => EE_Dependency_Map::load_from_cache, |
|
| 683 | - 'EEM_Event' => EE_Dependency_Map::load_from_cache, |
|
| 684 | - 'EEM_Datetime' => EE_Dependency_Map::load_from_cache, |
|
| 685 | - 'EEM_Ticket' => EE_Dependency_Map::load_from_cache, |
|
| 686 | - 'EEM_Registration' => EE_Dependency_Map::load_from_cache, |
|
| 687 | - 'EEM_Transaction' => EE_Dependency_Map::load_from_cache, |
|
| 688 | - 'EE_Config' => EE_Dependency_Map::load_from_cache, |
|
| 689 | - ), |
|
| 690 | - 'EventEspresso\core\domain\services\admin\ExitModal' => array( |
|
| 691 | - 'EventEspresso\core\services\assets\Registry' => EE_Dependency_Map::load_from_cache, |
|
| 692 | - ), |
|
| 693 | - 'EventEspresso\core\domain\services\admin\PluginUpsells' => array( |
|
| 694 | - 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 695 | - ), |
|
| 696 | - 'EventEspresso\caffeinated\modules\recaptcha_invisible\InvisibleRecaptcha' => array( |
|
| 697 | - 'EE_Registration_Config' => EE_Dependency_Map::load_from_cache, |
|
| 698 | - 'EE_Session' => EE_Dependency_Map::load_from_cache, |
|
| 699 | - ), |
|
| 700 | - 'EventEspresso\caffeinated\modules\recaptcha_invisible\RecaptchaAdminSettings' => array( |
|
| 701 | - 'EE_Registration_Config' => EE_Dependency_Map::load_from_cache, |
|
| 702 | - ), |
|
| 703 | - 'EventEspresso\modules\ticket_selector\ProcessTicketSelector' => array( |
|
| 704 | - 'EE_Core_Config' => EE_Dependency_Map::load_from_cache, |
|
| 705 | - 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 706 | - 'EE_Session' => EE_Dependency_Map::load_from_cache, |
|
| 707 | - 'EEM_Ticket' => EE_Dependency_Map::load_from_cache, |
|
| 708 | - 'EventEspresso\modules\ticket_selector\TicketDatetimeAvailabilityTracker' => EE_Dependency_Map::load_from_cache, |
|
| 709 | - ), |
|
| 710 | - 'EventEspresso\modules\ticket_selector\TicketDatetimeAvailabilityTracker' => array( |
|
| 711 | - 'EEM_Datetime' => EE_Dependency_Map::load_from_cache, |
|
| 712 | - ), |
|
| 713 | - 'EventEspresso\core\domain\entities\custom_post_types\CustomPostTypeDefinitions' => array( |
|
| 714 | - 'EE_Core_Config' => EE_Dependency_Map::load_from_cache, |
|
| 715 | - 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 716 | - ), |
|
| 717 | - 'EventEspresso\core\domain\services\custom_post_types\RegisterCustomPostTypes' => array( |
|
| 718 | - 'EventEspresso\core\domain\entities\custom_post_types\CustomPostTypeDefinitions' => EE_Dependency_Map::load_from_cache, |
|
| 719 | - ), |
|
| 720 | - 'EventEspresso\core\domain\services\custom_post_types\RegisterCustomTaxonomies' => array( |
|
| 721 | - 'EventEspresso\core\domain\entities\custom_post_types\CustomTaxonomyDefinitions' => EE_Dependency_Map::load_from_cache, |
|
| 722 | - ), |
|
| 723 | - 'EE_CPT_Strategy' => array( |
|
| 724 | - 'EventEspresso\core\domain\entities\custom_post_types\CustomPostTypeDefinitions' => EE_Dependency_Map::load_from_cache, |
|
| 725 | - 'EventEspresso\core\domain\entities\custom_post_types\CustomTaxonomyDefinitions' => EE_Dependency_Map::load_from_cache, |
|
| 726 | - ), |
|
| 727 | - 'EventEspresso\core\services\loaders\ObjectIdentifier' => array( |
|
| 728 | - 'EventEspresso\core\services\loaders\ClassInterfaceCache' => EE_Dependency_Map::load_from_cache, |
|
| 729 | - ), |
|
| 730 | - 'EventEspresso\core\domain\services\assets\CoreAssetManager' => array( |
|
| 731 | - 'EventEspresso\core\services\assets\AssetCollection' => EE_Dependency_Map::load_from_cache, |
|
| 732 | - 'EE_Currency_Config' => EE_Dependency_Map::load_from_cache, |
|
| 733 | - 'EE_Template_Config' => EE_Dependency_Map::load_from_cache, |
|
| 734 | - 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 735 | - 'EventEspresso\core\services\assets\Registry' => EE_Dependency_Map::load_from_cache, |
|
| 736 | - ), |
|
| 737 | - 'EventEspresso\core\domain\services\admin\privacy\policy\PrivacyPolicy' => array( |
|
| 738 | - 'EEM_Payment_Method' => EE_Dependency_Map::load_from_cache, |
|
| 739 | - 'EventEspresso\core\domain\values\session\SessionLifespan' => EE_Dependency_Map::load_from_cache |
|
| 740 | - ), |
|
| 741 | - 'EventEspresso\core\domain\services\admin\privacy\export\ExportAttendee' => array( |
|
| 742 | - 'EEM_Attendee' => EE_Dependency_Map::load_from_cache, |
|
| 743 | - ), |
|
| 744 | - 'EventEspresso\core\domain\services\admin\privacy\export\ExportAttendeeBillingData' => array( |
|
| 745 | - 'EEM_Attendee' => EE_Dependency_Map::load_from_cache, |
|
| 746 | - 'EEM_Payment_Method' => EE_Dependency_Map::load_from_cache |
|
| 747 | - ), |
|
| 748 | - 'EventEspresso\core\domain\services\admin\privacy\export\ExportCheckins' => array( |
|
| 749 | - 'EEM_Checkin' => EE_Dependency_Map::load_from_cache, |
|
| 750 | - ), |
|
| 751 | - 'EventEspresso\core\domain\services\admin\privacy\export\ExportRegistration' => array( |
|
| 752 | - 'EEM_Registration' => EE_Dependency_Map::load_from_cache, |
|
| 753 | - ), |
|
| 754 | - 'EventEspresso\core\domain\services\admin\privacy\export\ExportTransaction' => array( |
|
| 755 | - 'EEM_Transaction' => EE_Dependency_Map::load_from_cache, |
|
| 756 | - ), |
|
| 757 | - 'EventEspresso\core\domain\services\admin\privacy\erasure\EraseAttendeeData' => array( |
|
| 758 | - 'EEM_Attendee' => EE_Dependency_Map::load_from_cache, |
|
| 759 | - ), |
|
| 760 | - 'EventEspresso\core\domain\services\admin\privacy\erasure\EraseAnswers' => array( |
|
| 761 | - 'EEM_Answer' => EE_Dependency_Map::load_from_cache, |
|
| 762 | - 'EEM_Question' => EE_Dependency_Map::load_from_cache, |
|
| 763 | - ), |
|
| 764 | - 'EventEspresso\core\CPTs\CptQueryModifier' => array( |
|
| 765 | - null, |
|
| 766 | - null, |
|
| 767 | - null, |
|
| 768 | - 'EE_Request_Handler' => EE_Dependency_Map::load_from_cache, |
|
| 769 | - 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 770 | - 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 771 | - ), |
|
| 772 | - 'EventEspresso\core\domain\services\admin\privacy\forms\PrivacySettingsFormHandler' => array( |
|
| 773 | - 'EE_Registry' => EE_Dependency_Map::load_from_cache, |
|
| 774 | - 'EE_Config' => EE_Dependency_Map::load_from_cache |
|
| 775 | - ), |
|
| 776 | - 'EventEspresso\core\services\editor\BlockRegistrationManager' => array( |
|
| 777 | - 'EventEspresso\core\services\assets\BlockAssetManagerCollection' => EE_Dependency_Map::load_from_cache, |
|
| 778 | - 'EventEspresso\core\domain\entities\editor\BlockCollection' => EE_Dependency_Map::load_from_cache, |
|
| 779 | - 'EventEspresso\core\services\route_match\RouteMatchSpecificationManager' => EE_Dependency_Map::load_from_cache, |
|
| 780 | - 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 781 | - ), |
|
| 782 | - 'EventEspresso\core\domain\entities\editor\CoreBlocksAssetManager' => array( |
|
| 783 | - 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 784 | - 'EventEspresso\core\services\assets\AssetCollection' => EE_Dependency_Map::load_from_cache, |
|
| 785 | - 'EventEspresso\core\services\assets\Registry' => EE_Dependency_Map::load_from_cache, |
|
| 786 | - ), |
|
| 787 | - 'EventEspresso\core\domain\services\blocks\EventAttendeesBlockRenderer' => array( |
|
| 788 | - 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 789 | - 'EEM_Attendee' => EE_Dependency_Map::load_from_cache, |
|
| 790 | - ), |
|
| 791 | - 'EventEspresso\core\domain\entities\editor\blocks\EventAttendees' => array( |
|
| 792 | - 'EventEspresso\core\domain\entities\editor\CoreBlocksAssetManager' => self::load_from_cache, |
|
| 793 | - 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 794 | - 'EventEspresso\core\domain\services\blocks\EventAttendeesBlockRenderer' => self::load_from_cache, |
|
| 795 | - ), |
|
| 796 | - 'EventEspresso\core\services\route_match\RouteMatchSpecificationDependencyResolver' => array( |
|
| 797 | - 'EventEspresso\core\services\container\Mirror' => EE_Dependency_Map::load_from_cache, |
|
| 798 | - 'EventEspresso\core\services\loaders\ClassInterfaceCache' => EE_Dependency_Map::load_from_cache, |
|
| 799 | - 'EE_Dependency_Map' => EE_Dependency_Map::load_from_cache, |
|
| 800 | - ), |
|
| 801 | - 'EventEspresso\core\services\route_match\RouteMatchSpecificationFactory' => array( |
|
| 802 | - 'EventEspresso\core\services\route_match\RouteMatchSpecificationDependencyResolver' => EE_Dependency_Map::load_from_cache, |
|
| 803 | - 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 804 | - ), |
|
| 805 | - 'EventEspresso\core\services\route_match\RouteMatchSpecificationManager' => array( |
|
| 806 | - 'EventEspresso\core\services\route_match\RouteMatchSpecificationCollection' => EE_Dependency_Map::load_from_cache, |
|
| 807 | - 'EventEspresso\core\services\route_match\RouteMatchSpecificationFactory' => EE_Dependency_Map::load_from_cache, |
|
| 808 | - ), |
|
| 809 | - 'EventEspresso\core\libraries\rest_api\CalculatedModelFields' => array( |
|
| 810 | - 'EventEspresso\core\libraries\rest_api\calculations\CalculatedModelFieldsFactory' => EE_Dependency_Map::load_from_cache |
|
| 811 | - ), |
|
| 812 | - 'EventEspresso\core\libraries\rest_api\calculations\CalculatedModelFieldsFactory' => array( |
|
| 813 | - 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 814 | - ), |
|
| 815 | - 'EventEspresso\core\libraries\rest_api\controllers\model\Read' => array( |
|
| 816 | - 'EventEspresso\core\libraries\rest_api\CalculatedModelFields' => EE_Dependency_Map::load_from_cache |
|
| 817 | - ), |
|
| 818 | - 'EventEspresso\core\libraries\rest_api\calculations\Datetime' => array( |
|
| 819 | - 'EEM_Datetime' => EE_Dependency_Map::load_from_cache, |
|
| 820 | - 'EEM_Registration' => EE_Dependency_Map::load_from_cache |
|
| 821 | - ), |
|
| 822 | - 'EventEspresso\core\libraries\rest_api\calculations\Event' => array( |
|
| 823 | - 'EEM_Event' => EE_Dependency_Map::load_from_cache, |
|
| 824 | - 'EEM_Registration' => EE_Dependency_Map::load_from_cache |
|
| 825 | - ), |
|
| 826 | - 'EventEspresso\core\libraries\rest_api\calculations\Registration' => array( |
|
| 827 | - 'EEM_Registration' => EE_Dependency_Map::load_from_cache |
|
| 828 | - ), |
|
| 829 | - 'EventEspresso\core\services\session\SessionStartHandler' => array( |
|
| 830 | - 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 831 | - ), |
|
| 832 | - 'EE_URL_Validation_Strategy' => array( |
|
| 833 | - null, |
|
| 834 | - null, |
|
| 835 | - 'EventEspresso\core\services\validators\URLValidator' => EE_Dependency_Map::load_from_cache |
|
| 836 | - ), |
|
| 837 | - 'EventEspresso\admin_pages\general_settings\OrganizationSettings' => array( |
|
| 838 | - 'EE_Registry' => EE_Dependency_Map::load_from_cache, |
|
| 839 | - 'EE_Organization_Config' => EE_Dependency_Map::load_from_cache, |
|
| 840 | - 'EE_Core_Config' => EE_Dependency_Map::load_from_cache, |
|
| 841 | - 'EE_Network_Core_Config' => EE_Dependency_Map::load_from_cache, |
|
| 842 | - 'EventEspresso\core\services\address\CountrySubRegionDao' => EE_Dependency_Map::load_from_cache, |
|
| 843 | - ), |
|
| 844 | - 'EventEspresso\core\services\address\CountrySubRegionDao' => array( |
|
| 845 | - 'EEM_State' => EE_Dependency_Map::load_from_cache, |
|
| 846 | - 'EventEspresso\core\services\validators\JsonValidator' => EE_Dependency_Map::load_from_cache |
|
| 847 | - ), |
|
| 848 | - 'EventEspresso\core\domain\services\admin\ajax\WordpressHeartbeat' => array( |
|
| 849 | - 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 850 | - 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 851 | - ), |
|
| 852 | - 'EventEspresso\core\domain\services\admin\ajax\EventEditorHeartbeat' => array( |
|
| 853 | - 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 854 | - 'EE_Environment_Config' => EE_Dependency_Map::load_from_cache, |
|
| 855 | - ), |
|
| 856 | - 'EventEspresso\core\services\request\files\FilesDataHandler' => array( |
|
| 857 | - 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 858 | - ), |
|
| 859 | - 'EventEspressoBatchRequest\BatchRequestProcessor' => [ |
|
| 860 | - 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 861 | - ] |
|
| 862 | - ); |
|
| 863 | - } |
|
| 864 | - |
|
| 865 | - |
|
| 866 | - /** |
|
| 867 | - * Registers how core classes are loaded. |
|
| 868 | - * This can either be done by simply providing the name of one of the EE_Registry loader methods such as: |
|
| 869 | - * 'EE_Request_Handler' => 'load_core' |
|
| 870 | - * 'EE_Messages_Queue' => 'load_lib' |
|
| 871 | - * 'EEH_Debug_Tools' => 'load_helper' |
|
| 872 | - * or, if greater control is required, by providing a custom closure. For example: |
|
| 873 | - * 'Some_Class' => function () { |
|
| 874 | - * return new Some_Class(); |
|
| 875 | - * }, |
|
| 876 | - * This is required for instantiating dependencies |
|
| 877 | - * where an interface has been type hinted in a class constructor. For example: |
|
| 878 | - * 'Required_Interface' => function () { |
|
| 879 | - * return new A_Class_That_Implements_Required_Interface(); |
|
| 880 | - * }, |
|
| 881 | - */ |
|
| 882 | - protected function _register_core_class_loaders() |
|
| 883 | - { |
|
| 884 | - $this->_class_loaders = array( |
|
| 885 | - // load_core |
|
| 886 | - 'EE_Dependency_Map' => function () { |
|
| 887 | - return $this; |
|
| 888 | - }, |
|
| 889 | - 'EE_Capabilities' => 'load_core', |
|
| 890 | - 'EE_Encryption' => 'load_core', |
|
| 891 | - 'EE_Front_Controller' => 'load_core', |
|
| 892 | - 'EE_Module_Request_Router' => 'load_core', |
|
| 893 | - 'EE_Registry' => 'load_core', |
|
| 894 | - 'EE_Request' => function () { |
|
| 895 | - return $this->legacy_request; |
|
| 896 | - }, |
|
| 897 | - 'EventEspresso\core\services\request\Request' => function () { |
|
| 898 | - return $this->request; |
|
| 899 | - }, |
|
| 900 | - 'EventEspresso\core\services\request\Response' => function () { |
|
| 901 | - return $this->response; |
|
| 902 | - }, |
|
| 903 | - 'EE_Base' => 'load_core', |
|
| 904 | - 'EE_Request_Handler' => 'load_core', |
|
| 905 | - 'EE_Session' => 'load_core', |
|
| 906 | - 'EE_Cron_Tasks' => 'load_core', |
|
| 907 | - 'EE_System' => 'load_core', |
|
| 908 | - 'EE_Maintenance_Mode' => 'load_core', |
|
| 909 | - 'EE_Register_CPTs' => 'load_core', |
|
| 910 | - 'EE_Admin' => 'load_core', |
|
| 911 | - 'EE_CPT_Strategy' => 'load_core', |
|
| 912 | - // load_class |
|
| 913 | - 'EE_Registration_Processor' => 'load_class', |
|
| 914 | - // load_lib |
|
| 915 | - 'EE_Message_Resource_Manager' => 'load_lib', |
|
| 916 | - 'EE_Message_Type_Collection' => 'load_lib', |
|
| 917 | - 'EE_Message_Type_Collection_Loader' => 'load_lib', |
|
| 918 | - 'EE_Messenger_Collection' => 'load_lib', |
|
| 919 | - 'EE_Messenger_Collection_Loader' => 'load_lib', |
|
| 920 | - 'EE_Messages_Processor' => 'load_lib', |
|
| 921 | - 'EE_Message_Repository' => 'load_lib', |
|
| 922 | - 'EE_Messages_Queue' => 'load_lib', |
|
| 923 | - 'EE_Messages_Data_Handler_Collection' => 'load_lib', |
|
| 924 | - 'EE_Message_Template_Group_Collection' => 'load_lib', |
|
| 925 | - 'EE_Payment_Method_Manager' => 'load_lib', |
|
| 926 | - 'EE_Messages_Generator' => function () { |
|
| 927 | - return EE_Registry::instance()->load_lib( |
|
| 928 | - 'Messages_Generator', |
|
| 929 | - array(), |
|
| 930 | - false, |
|
| 931 | - false |
|
| 932 | - ); |
|
| 933 | - }, |
|
| 934 | - 'EE_Messages_Template_Defaults' => function ($arguments = array()) { |
|
| 935 | - return EE_Registry::instance()->load_lib( |
|
| 936 | - 'Messages_Template_Defaults', |
|
| 937 | - $arguments, |
|
| 938 | - false, |
|
| 939 | - false |
|
| 940 | - ); |
|
| 941 | - }, |
|
| 942 | - // load_helper |
|
| 943 | - 'EEH_Parse_Shortcodes' => function () { |
|
| 944 | - if (EE_Registry::instance()->load_helper('Parse_Shortcodes')) { |
|
| 945 | - return new EEH_Parse_Shortcodes(); |
|
| 946 | - } |
|
| 947 | - return null; |
|
| 948 | - }, |
|
| 949 | - 'EE_Template_Config' => function () { |
|
| 950 | - return EE_Config::instance()->template_settings; |
|
| 951 | - }, |
|
| 952 | - 'EE_Currency_Config' => function () { |
|
| 953 | - return EE_Config::instance()->currency; |
|
| 954 | - }, |
|
| 955 | - 'EE_Registration_Config' => function () { |
|
| 956 | - return EE_Config::instance()->registration; |
|
| 957 | - }, |
|
| 958 | - 'EE_Core_Config' => function () { |
|
| 959 | - return EE_Config::instance()->core; |
|
| 960 | - }, |
|
| 961 | - 'EventEspresso\core\services\loaders\Loader' => function () { |
|
| 962 | - return LoaderFactory::getLoader(); |
|
| 963 | - }, |
|
| 964 | - 'EE_Network_Config' => function () { |
|
| 965 | - return EE_Network_Config::instance(); |
|
| 966 | - }, |
|
| 967 | - 'EE_Config' => function () { |
|
| 968 | - return EE_Config::instance(); |
|
| 969 | - }, |
|
| 970 | - 'EventEspresso\core\domain\Domain' => function () { |
|
| 971 | - return DomainFactory::getEventEspressoCoreDomain(); |
|
| 972 | - }, |
|
| 973 | - 'EE_Admin_Config' => function () { |
|
| 974 | - return EE_Config::instance()->admin; |
|
| 975 | - }, |
|
| 976 | - 'EE_Organization_Config' => function () { |
|
| 977 | - return EE_Config::instance()->organization; |
|
| 978 | - }, |
|
| 979 | - 'EE_Network_Core_Config' => function () { |
|
| 980 | - return EE_Network_Config::instance()->core; |
|
| 981 | - }, |
|
| 982 | - 'EE_Environment_Config' => function () { |
|
| 983 | - return EE_Config::instance()->environment; |
|
| 984 | - }, |
|
| 985 | - ); |
|
| 986 | - } |
|
| 987 | - |
|
| 988 | - |
|
| 989 | - /** |
|
| 990 | - * can be used for supplying alternate names for classes, |
|
| 991 | - * or for connecting interface names to instantiable classes |
|
| 992 | - */ |
|
| 993 | - protected function _register_core_aliases() |
|
| 994 | - { |
|
| 995 | - $aliases = array( |
|
| 996 | - 'CommandBusInterface' => 'EventEspresso\core\services\commands\CommandBusInterface', |
|
| 997 | - 'EventEspresso\core\services\commands\CommandBusInterface' => 'EventEspresso\core\services\commands\CommandBus', |
|
| 998 | - 'CommandHandlerManagerInterface' => 'EventEspresso\core\services\commands\CommandHandlerManagerInterface', |
|
| 999 | - 'EventEspresso\core\services\commands\CommandHandlerManagerInterface' => 'EventEspresso\core\services\commands\CommandHandlerManager', |
|
| 1000 | - 'CapChecker' => 'EventEspresso\core\services\commands\middleware\CapChecker', |
|
| 1001 | - 'AddActionHook' => 'EventEspresso\core\services\commands\middleware\AddActionHook', |
|
| 1002 | - 'CapabilitiesChecker' => 'EventEspresso\core\domain\services\capabilities\CapabilitiesChecker', |
|
| 1003 | - 'CapabilitiesCheckerInterface' => 'EventEspresso\core\domain\services\capabilities\CapabilitiesCheckerInterface', |
|
| 1004 | - 'EventEspresso\core\domain\services\capabilities\CapabilitiesCheckerInterface' => 'EventEspresso\core\domain\services\capabilities\CapabilitiesChecker', |
|
| 1005 | - 'CreateRegistrationService' => 'EventEspresso\core\domain\services\registration\CreateRegistrationService', |
|
| 1006 | - 'CreateRegistrationCommandHandler' => 'EventEspresso\core\services\commands\registration\CreateRegistrationCommand', |
|
| 1007 | - 'CopyRegistrationDetailsCommandHandler' => 'EventEspresso\core\services\commands\registration\CopyRegistrationDetailsCommand', |
|
| 1008 | - 'CopyRegistrationPaymentsCommandHandler' => 'EventEspresso\core\services\commands\registration\CopyRegistrationPaymentsCommand', |
|
| 1009 | - 'CancelRegistrationAndTicketLineItemCommandHandler' => 'EventEspresso\core\services\commands\registration\CancelRegistrationAndTicketLineItemCommandHandler', |
|
| 1010 | - 'UpdateRegistrationAndTransactionAfterChangeCommandHandler' => 'EventEspresso\core\services\commands\registration\UpdateRegistrationAndTransactionAfterChangeCommandHandler', |
|
| 1011 | - 'CreateTicketLineItemCommandHandler' => 'EventEspresso\core\services\commands\ticket\CreateTicketLineItemCommand', |
|
| 1012 | - 'CreateTransactionCommandHandler' => 'EventEspresso\core\services\commands\transaction\CreateTransactionCommandHandler', |
|
| 1013 | - 'CreateAttendeeCommandHandler' => 'EventEspresso\core\services\commands\attendee\CreateAttendeeCommandHandler', |
|
| 1014 | - 'TableManager' => 'EventEspresso\core\services\database\TableManager', |
|
| 1015 | - 'TableAnalysis' => 'EventEspresso\core\services\database\TableAnalysis', |
|
| 1016 | - 'EspressoShortcode' => 'EventEspresso\core\services\shortcodes\EspressoShortcode', |
|
| 1017 | - 'ShortcodeInterface' => 'EventEspresso\core\services\shortcodes\ShortcodeInterface', |
|
| 1018 | - 'EventEspresso\core\services\shortcodes\ShortcodeInterface' => 'EventEspresso\core\services\shortcodes\EspressoShortcode', |
|
| 1019 | - 'EventEspresso\core\services\cache\CacheStorageInterface' => 'EventEspresso\core\services\cache\TransientCacheStorage', |
|
| 1020 | - 'LoaderInterface' => 'EventEspresso\core\services\loaders\LoaderInterface', |
|
| 1021 | - 'EventEspresso\core\services\loaders\LoaderInterface' => 'EventEspresso\core\services\loaders\Loader', |
|
| 1022 | - 'CommandFactoryInterface' => 'EventEspresso\core\services\commands\CommandFactoryInterface', |
|
| 1023 | - 'EventEspresso\core\services\commands\CommandFactoryInterface' => 'EventEspresso\core\services\commands\CommandFactory', |
|
| 1024 | - 'EmailValidatorInterface' => 'EventEspresso\core\domain\services\validation\email\EmailValidatorInterface', |
|
| 1025 | - 'EventEspresso\core\domain\services\validation\email\EmailValidatorInterface' => 'EventEspresso\core\domain\services\validation\email\EmailValidationService', |
|
| 1026 | - 'NoticeConverterInterface' => 'EventEspresso\core\services\notices\NoticeConverterInterface', |
|
| 1027 | - 'EventEspresso\core\services\notices\NoticeConverterInterface' => 'EventEspresso\core\services\notices\ConvertNoticesToEeErrors', |
|
| 1028 | - 'NoticesContainerInterface' => 'EventEspresso\core\services\notices\NoticesContainerInterface', |
|
| 1029 | - 'EventEspresso\core\services\notices\NoticesContainerInterface' => 'EventEspresso\core\services\notices\NoticesContainer', |
|
| 1030 | - 'EventEspresso\core\services\request\RequestInterface' => 'EventEspresso\core\services\request\Request', |
|
| 1031 | - 'EventEspresso\core\services\request\ResponseInterface' => 'EventEspresso\core\services\request\Response', |
|
| 1032 | - 'EventEspresso\core\domain\DomainInterface' => 'EventEspresso\core\domain\Domain', |
|
| 1033 | - 'Registration_Processor' => 'EE_Registration_Processor', |
|
| 1034 | - ); |
|
| 1035 | - foreach ($aliases as $alias => $fqn) { |
|
| 1036 | - if (is_array($fqn)) { |
|
| 1037 | - foreach ($fqn as $class => $for_class) { |
|
| 1038 | - $this->class_cache->addAlias($class, $alias, $for_class); |
|
| 1039 | - } |
|
| 1040 | - continue; |
|
| 1041 | - } |
|
| 1042 | - $this->class_cache->addAlias($fqn, $alias); |
|
| 1043 | - } |
|
| 1044 | - if (! (defined('DOING_AJAX') && DOING_AJAX) && is_admin()) { |
|
| 1045 | - $this->class_cache->addAlias( |
|
| 1046 | - 'EventEspresso\core\services\notices\ConvertNoticesToAdminNotices', |
|
| 1047 | - 'EventEspresso\core\services\notices\NoticeConverterInterface' |
|
| 1048 | - ); |
|
| 1049 | - } |
|
| 1050 | - } |
|
| 1051 | - |
|
| 1052 | - |
|
| 1053 | - /** |
|
| 1054 | - * This is used to reset the internal map and class_loaders to their original default state at the beginning of the |
|
| 1055 | - * request Primarily used by unit tests. |
|
| 1056 | - */ |
|
| 1057 | - public function reset() |
|
| 1058 | - { |
|
| 1059 | - $this->_register_core_class_loaders(); |
|
| 1060 | - $this->_register_core_dependencies(); |
|
| 1061 | - } |
|
| 1062 | - |
|
| 1063 | - |
|
| 1064 | - /** |
|
| 1065 | - * PLZ NOTE: a better name for this method would be is_alias() |
|
| 1066 | - * because it returns TRUE if the provided fully qualified name IS an alias |
|
| 1067 | - * WHY? |
|
| 1068 | - * Because if a class is type hinting for a concretion, |
|
| 1069 | - * then why would we need to find another class to supply it? |
|
| 1070 | - * ie: if a class asks for `Fully/Qualified/Namespace/SpecificClassName`, |
|
| 1071 | - * then give it an instance of `Fully/Qualified/Namespace/SpecificClassName`. |
|
| 1072 | - * Don't go looking for some substitute. |
|
| 1073 | - * Whereas if a class is type hinting for an interface... |
|
| 1074 | - * then we need to find an actual class to use. |
|
| 1075 | - * So the interface IS the alias for some other FQN, |
|
| 1076 | - * and we need to find out if `Fully/Qualified/Namespace/SomeInterface` |
|
| 1077 | - * represents some other class. |
|
| 1078 | - * |
|
| 1079 | - * @deprecated 4.9.62.p |
|
| 1080 | - * @param string $fqn |
|
| 1081 | - * @param string $for_class |
|
| 1082 | - * @return bool |
|
| 1083 | - */ |
|
| 1084 | - public function has_alias($fqn = '', $for_class = '') |
|
| 1085 | - { |
|
| 1086 | - return $this->isAlias($fqn, $for_class); |
|
| 1087 | - } |
|
| 1088 | - |
|
| 1089 | - |
|
| 1090 | - /** |
|
| 1091 | - * PLZ NOTE: a better name for this method would be get_fqn_for_alias() |
|
| 1092 | - * because it returns a FQN for provided alias if one exists, otherwise returns the original $alias |
|
| 1093 | - * functions recursively, so that multiple aliases can be used to drill down to a FQN |
|
| 1094 | - * for example: |
|
| 1095 | - * if the following two entries were added to the _aliases array: |
|
| 1096 | - * array( |
|
| 1097 | - * 'interface_alias' => 'some\namespace\interface' |
|
| 1098 | - * 'some\namespace\interface' => 'some\namespace\classname' |
|
| 1099 | - * ) |
|
| 1100 | - * then one could use EE_Registry::instance()->create( 'interface_alias' ) |
|
| 1101 | - * to load an instance of 'some\namespace\classname' |
|
| 1102 | - * |
|
| 1103 | - * @deprecated 4.9.62.p |
|
| 1104 | - * @param string $alias |
|
| 1105 | - * @param string $for_class |
|
| 1106 | - * @return string |
|
| 1107 | - */ |
|
| 1108 | - public function get_alias($alias = '', $for_class = '') |
|
| 1109 | - { |
|
| 1110 | - return $this->getFqnForAlias($alias, $for_class); |
|
| 1111 | - } |
|
| 23 | + /** |
|
| 24 | + * This means that the requested class dependency is not present in the dependency map |
|
| 25 | + */ |
|
| 26 | + const not_registered = 0; |
|
| 27 | + |
|
| 28 | + /** |
|
| 29 | + * This instructs class loaders to ALWAYS return a newly instantiated object for the requested class. |
|
| 30 | + */ |
|
| 31 | + const load_new_object = 1; |
|
| 32 | + |
|
| 33 | + /** |
|
| 34 | + * This instructs class loaders to return a previously instantiated and cached object for the requested class. |
|
| 35 | + * IF a previously instantiated object does not exist, a new one will be created and added to the cache. |
|
| 36 | + */ |
|
| 37 | + const load_from_cache = 2; |
|
| 38 | + |
|
| 39 | + /** |
|
| 40 | + * When registering a dependency, |
|
| 41 | + * this indicates to keep any existing dependencies that already exist, |
|
| 42 | + * and simply discard any new dependencies declared in the incoming data |
|
| 43 | + */ |
|
| 44 | + const KEEP_EXISTING_DEPENDENCIES = 0; |
|
| 45 | + |
|
| 46 | + /** |
|
| 47 | + * When registering a dependency, |
|
| 48 | + * this indicates to overwrite any existing dependencies that already exist using the incoming data |
|
| 49 | + */ |
|
| 50 | + const OVERWRITE_DEPENDENCIES = 1; |
|
| 51 | + |
|
| 52 | + |
|
| 53 | + /** |
|
| 54 | + * @type EE_Dependency_Map $_instance |
|
| 55 | + */ |
|
| 56 | + protected static $_instance; |
|
| 57 | + |
|
| 58 | + /** |
|
| 59 | + * @var ClassInterfaceCache $class_cache |
|
| 60 | + */ |
|
| 61 | + private $class_cache; |
|
| 62 | + |
|
| 63 | + /** |
|
| 64 | + * @type RequestInterface $request |
|
| 65 | + */ |
|
| 66 | + protected $request; |
|
| 67 | + |
|
| 68 | + /** |
|
| 69 | + * @type LegacyRequestInterface $legacy_request |
|
| 70 | + */ |
|
| 71 | + protected $legacy_request; |
|
| 72 | + |
|
| 73 | + /** |
|
| 74 | + * @type ResponseInterface $response |
|
| 75 | + */ |
|
| 76 | + protected $response; |
|
| 77 | + |
|
| 78 | + /** |
|
| 79 | + * @type LoaderInterface $loader |
|
| 80 | + */ |
|
| 81 | + protected $loader; |
|
| 82 | + |
|
| 83 | + /** |
|
| 84 | + * @type array $_dependency_map |
|
| 85 | + */ |
|
| 86 | + protected $_dependency_map = array(); |
|
| 87 | + |
|
| 88 | + /** |
|
| 89 | + * @type array $_class_loaders |
|
| 90 | + */ |
|
| 91 | + protected $_class_loaders = array(); |
|
| 92 | + |
|
| 93 | + |
|
| 94 | + /** |
|
| 95 | + * EE_Dependency_Map constructor. |
|
| 96 | + * |
|
| 97 | + * @param ClassInterfaceCache $class_cache |
|
| 98 | + */ |
|
| 99 | + protected function __construct(ClassInterfaceCache $class_cache) |
|
| 100 | + { |
|
| 101 | + $this->class_cache = $class_cache; |
|
| 102 | + do_action('EE_Dependency_Map____construct', $this); |
|
| 103 | + } |
|
| 104 | + |
|
| 105 | + |
|
| 106 | + /** |
|
| 107 | + * @return void |
|
| 108 | + */ |
|
| 109 | + public function initialize() |
|
| 110 | + { |
|
| 111 | + $this->_register_core_dependencies(); |
|
| 112 | + $this->_register_core_class_loaders(); |
|
| 113 | + $this->_register_core_aliases(); |
|
| 114 | + } |
|
| 115 | + |
|
| 116 | + |
|
| 117 | + /** |
|
| 118 | + * @singleton method used to instantiate class object |
|
| 119 | + * @param ClassInterfaceCache|null $class_cache |
|
| 120 | + * @return EE_Dependency_Map |
|
| 121 | + */ |
|
| 122 | + public static function instance(ClassInterfaceCache $class_cache = null) |
|
| 123 | + { |
|
| 124 | + // check if class object is instantiated, and instantiated properly |
|
| 125 | + if (! self::$_instance instanceof EE_Dependency_Map |
|
| 126 | + && $class_cache instanceof ClassInterfaceCache |
|
| 127 | + ) { |
|
| 128 | + self::$_instance = new EE_Dependency_Map($class_cache); |
|
| 129 | + } |
|
| 130 | + return self::$_instance; |
|
| 131 | + } |
|
| 132 | + |
|
| 133 | + |
|
| 134 | + /** |
|
| 135 | + * @param RequestInterface $request |
|
| 136 | + */ |
|
| 137 | + public function setRequest(RequestInterface $request) |
|
| 138 | + { |
|
| 139 | + $this->request = $request; |
|
| 140 | + } |
|
| 141 | + |
|
| 142 | + |
|
| 143 | + /** |
|
| 144 | + * @param LegacyRequestInterface $legacy_request |
|
| 145 | + */ |
|
| 146 | + public function setLegacyRequest(LegacyRequestInterface $legacy_request) |
|
| 147 | + { |
|
| 148 | + $this->legacy_request = $legacy_request; |
|
| 149 | + } |
|
| 150 | + |
|
| 151 | + |
|
| 152 | + /** |
|
| 153 | + * @param ResponseInterface $response |
|
| 154 | + */ |
|
| 155 | + public function setResponse(ResponseInterface $response) |
|
| 156 | + { |
|
| 157 | + $this->response = $response; |
|
| 158 | + } |
|
| 159 | + |
|
| 160 | + |
|
| 161 | + /** |
|
| 162 | + * @param LoaderInterface $loader |
|
| 163 | + */ |
|
| 164 | + public function setLoader(LoaderInterface $loader) |
|
| 165 | + { |
|
| 166 | + $this->loader = $loader; |
|
| 167 | + } |
|
| 168 | + |
|
| 169 | + |
|
| 170 | + /** |
|
| 171 | + * @param string $class |
|
| 172 | + * @param array $dependencies |
|
| 173 | + * @param int $overwrite |
|
| 174 | + * @return bool |
|
| 175 | + */ |
|
| 176 | + public static function register_dependencies( |
|
| 177 | + $class, |
|
| 178 | + array $dependencies, |
|
| 179 | + $overwrite = EE_Dependency_Map::KEEP_EXISTING_DEPENDENCIES |
|
| 180 | + ) { |
|
| 181 | + return self::$_instance->registerDependencies($class, $dependencies, $overwrite); |
|
| 182 | + } |
|
| 183 | + |
|
| 184 | + |
|
| 185 | + /** |
|
| 186 | + * Assigns an array of class names and corresponding load sources (new or cached) |
|
| 187 | + * to the class specified by the first parameter. |
|
| 188 | + * IMPORTANT !!! |
|
| 189 | + * The order of elements in the incoming $dependencies array MUST match |
|
| 190 | + * the order of the constructor parameters for the class in question. |
|
| 191 | + * This is especially important when overriding any existing dependencies that are registered. |
|
| 192 | + * the third parameter controls whether any duplicate dependencies are overwritten or not. |
|
| 193 | + * |
|
| 194 | + * @param string $class |
|
| 195 | + * @param array $dependencies |
|
| 196 | + * @param int $overwrite |
|
| 197 | + * @return bool |
|
| 198 | + */ |
|
| 199 | + public function registerDependencies( |
|
| 200 | + $class, |
|
| 201 | + array $dependencies, |
|
| 202 | + $overwrite = EE_Dependency_Map::KEEP_EXISTING_DEPENDENCIES |
|
| 203 | + ) { |
|
| 204 | + $class = trim($class, '\\'); |
|
| 205 | + $registered = false; |
|
| 206 | + if (empty(self::$_instance->_dependency_map[ $class ])) { |
|
| 207 | + self::$_instance->_dependency_map[ $class ] = array(); |
|
| 208 | + } |
|
| 209 | + // we need to make sure that any aliases used when registering a dependency |
|
| 210 | + // get resolved to the correct class name |
|
| 211 | + foreach ($dependencies as $dependency => $load_source) { |
|
| 212 | + $alias = self::$_instance->getFqnForAlias($dependency); |
|
| 213 | + if ($overwrite === EE_Dependency_Map::OVERWRITE_DEPENDENCIES |
|
| 214 | + || ! isset(self::$_instance->_dependency_map[ $class ][ $alias ]) |
|
| 215 | + ) { |
|
| 216 | + unset($dependencies[ $dependency ]); |
|
| 217 | + $dependencies[ $alias ] = $load_source; |
|
| 218 | + $registered = true; |
|
| 219 | + } |
|
| 220 | + } |
|
| 221 | + // now add our two lists of dependencies together. |
|
| 222 | + // using Union (+=) favours the arrays in precedence from left to right, |
|
| 223 | + // so $dependencies is NOT overwritten because it is listed first |
|
| 224 | + // ie: with A = B + C, entries in B take precedence over duplicate entries in C |
|
| 225 | + // Union is way faster than array_merge() but should be used with caution... |
|
| 226 | + // especially with numerically indexed arrays |
|
| 227 | + $dependencies += self::$_instance->_dependency_map[ $class ]; |
|
| 228 | + // now we need to ensure that the resulting dependencies |
|
| 229 | + // array only has the entries that are required for the class |
|
| 230 | + // so first count how many dependencies were originally registered for the class |
|
| 231 | + $dependency_count = count(self::$_instance->_dependency_map[ $class ]); |
|
| 232 | + // if that count is non-zero (meaning dependencies were already registered) |
|
| 233 | + self::$_instance->_dependency_map[ $class ] = $dependency_count |
|
| 234 | + // then truncate the final array to match that count |
|
| 235 | + ? array_slice($dependencies, 0, $dependency_count) |
|
| 236 | + // otherwise just take the incoming array because nothing previously existed |
|
| 237 | + : $dependencies; |
|
| 238 | + return $registered; |
|
| 239 | + } |
|
| 240 | + |
|
| 241 | + |
|
| 242 | + /** |
|
| 243 | + * @param string $class_name |
|
| 244 | + * @param string $loader |
|
| 245 | + * @return bool |
|
| 246 | + * @throws DomainException |
|
| 247 | + */ |
|
| 248 | + public static function register_class_loader($class_name, $loader = 'load_core') |
|
| 249 | + { |
|
| 250 | + if (! $loader instanceof Closure && strpos($class_name, '\\') !== false) { |
|
| 251 | + throw new DomainException( |
|
| 252 | + esc_html__('Don\'t use class loaders for FQCNs.', 'event_espresso') |
|
| 253 | + ); |
|
| 254 | + } |
|
| 255 | + // check that loader is callable or method starts with "load_" and exists in EE_Registry |
|
| 256 | + if (! is_callable($loader) |
|
| 257 | + && ( |
|
| 258 | + strpos($loader, 'load_') !== 0 |
|
| 259 | + || ! method_exists('EE_Registry', $loader) |
|
| 260 | + ) |
|
| 261 | + ) { |
|
| 262 | + throw new DomainException( |
|
| 263 | + sprintf( |
|
| 264 | + esc_html__( |
|
| 265 | + '"%1$s" is not a valid loader method on EE_Registry.', |
|
| 266 | + 'event_espresso' |
|
| 267 | + ), |
|
| 268 | + $loader |
|
| 269 | + ) |
|
| 270 | + ); |
|
| 271 | + } |
|
| 272 | + $class_name = self::$_instance->getFqnForAlias($class_name); |
|
| 273 | + if (! isset(self::$_instance->_class_loaders[ $class_name ])) { |
|
| 274 | + self::$_instance->_class_loaders[ $class_name ] = $loader; |
|
| 275 | + return true; |
|
| 276 | + } |
|
| 277 | + return false; |
|
| 278 | + } |
|
| 279 | + |
|
| 280 | + |
|
| 281 | + /** |
|
| 282 | + * @return array |
|
| 283 | + */ |
|
| 284 | + public function dependency_map() |
|
| 285 | + { |
|
| 286 | + return $this->_dependency_map; |
|
| 287 | + } |
|
| 288 | + |
|
| 289 | + |
|
| 290 | + /** |
|
| 291 | + * returns TRUE if dependency map contains a listing for the provided class name |
|
| 292 | + * |
|
| 293 | + * @param string $class_name |
|
| 294 | + * @return boolean |
|
| 295 | + */ |
|
| 296 | + public function has($class_name = '') |
|
| 297 | + { |
|
| 298 | + // all legacy models have the same dependencies |
|
| 299 | + if (strpos($class_name, 'EEM_') === 0) { |
|
| 300 | + $class_name = 'LEGACY_MODELS'; |
|
| 301 | + } |
|
| 302 | + return isset($this->_dependency_map[ $class_name ]) ? true : false; |
|
| 303 | + } |
|
| 304 | + |
|
| 305 | + |
|
| 306 | + /** |
|
| 307 | + * returns TRUE if dependency map contains a listing for the provided class name AND dependency |
|
| 308 | + * |
|
| 309 | + * @param string $class_name |
|
| 310 | + * @param string $dependency |
|
| 311 | + * @return bool |
|
| 312 | + */ |
|
| 313 | + public function has_dependency_for_class($class_name = '', $dependency = '') |
|
| 314 | + { |
|
| 315 | + // all legacy models have the same dependencies |
|
| 316 | + if (strpos($class_name, 'EEM_') === 0) { |
|
| 317 | + $class_name = 'LEGACY_MODELS'; |
|
| 318 | + } |
|
| 319 | + $dependency = $this->getFqnForAlias($dependency, $class_name); |
|
| 320 | + return isset($this->_dependency_map[ $class_name ][ $dependency ]) |
|
| 321 | + ? true |
|
| 322 | + : false; |
|
| 323 | + } |
|
| 324 | + |
|
| 325 | + |
|
| 326 | + /** |
|
| 327 | + * returns loading strategy for whether a previously cached dependency should be loaded or a new instance returned |
|
| 328 | + * |
|
| 329 | + * @param string $class_name |
|
| 330 | + * @param string $dependency |
|
| 331 | + * @return int |
|
| 332 | + */ |
|
| 333 | + public function loading_strategy_for_class_dependency($class_name = '', $dependency = '') |
|
| 334 | + { |
|
| 335 | + // all legacy models have the same dependencies |
|
| 336 | + if (strpos($class_name, 'EEM_') === 0) { |
|
| 337 | + $class_name = 'LEGACY_MODELS'; |
|
| 338 | + } |
|
| 339 | + $dependency = $this->getFqnForAlias($dependency); |
|
| 340 | + return $this->has_dependency_for_class($class_name, $dependency) |
|
| 341 | + ? $this->_dependency_map[ $class_name ][ $dependency ] |
|
| 342 | + : EE_Dependency_Map::not_registered; |
|
| 343 | + } |
|
| 344 | + |
|
| 345 | + |
|
| 346 | + /** |
|
| 347 | + * @param string $class_name |
|
| 348 | + * @return string | Closure |
|
| 349 | + */ |
|
| 350 | + public function class_loader($class_name) |
|
| 351 | + { |
|
| 352 | + // all legacy models use load_model() |
|
| 353 | + if (strpos($class_name, 'EEM_') === 0) { |
|
| 354 | + return 'load_model'; |
|
| 355 | + } |
|
| 356 | + // EE_CPT_*_Strategy classes like EE_CPT_Event_Strategy, EE_CPT_Venue_Strategy, etc |
|
| 357 | + // perform strpos() first to avoid loading regex every time we load a class |
|
| 358 | + if (strpos($class_name, 'EE_CPT_') === 0 |
|
| 359 | + && preg_match('/^EE_CPT_([a-zA-Z]+)_Strategy$/', $class_name) |
|
| 360 | + ) { |
|
| 361 | + return 'load_core'; |
|
| 362 | + } |
|
| 363 | + $class_name = $this->getFqnForAlias($class_name); |
|
| 364 | + return isset($this->_class_loaders[ $class_name ]) ? $this->_class_loaders[ $class_name ] : ''; |
|
| 365 | + } |
|
| 366 | + |
|
| 367 | + |
|
| 368 | + /** |
|
| 369 | + * @return array |
|
| 370 | + */ |
|
| 371 | + public function class_loaders() |
|
| 372 | + { |
|
| 373 | + return $this->_class_loaders; |
|
| 374 | + } |
|
| 375 | + |
|
| 376 | + |
|
| 377 | + /** |
|
| 378 | + * adds an alias for a classname |
|
| 379 | + * |
|
| 380 | + * @param string $fqcn the class name that should be used (concrete class to replace interface) |
|
| 381 | + * @param string $alias the class name that would be type hinted for (abstract parent or interface) |
|
| 382 | + * @param string $for_class the class that has the dependency (is type hinting for the interface) |
|
| 383 | + */ |
|
| 384 | + public function add_alias($fqcn, $alias, $for_class = '') |
|
| 385 | + { |
|
| 386 | + $this->class_cache->addAlias($fqcn, $alias, $for_class); |
|
| 387 | + } |
|
| 388 | + |
|
| 389 | + |
|
| 390 | + /** |
|
| 391 | + * Returns TRUE if the provided fully qualified name IS an alias |
|
| 392 | + * WHY? |
|
| 393 | + * Because if a class is type hinting for a concretion, |
|
| 394 | + * then why would we need to find another class to supply it? |
|
| 395 | + * ie: if a class asks for `Fully/Qualified/Namespace/SpecificClassName`, |
|
| 396 | + * then give it an instance of `Fully/Qualified/Namespace/SpecificClassName`. |
|
| 397 | + * Don't go looking for some substitute. |
|
| 398 | + * Whereas if a class is type hinting for an interface... |
|
| 399 | + * then we need to find an actual class to use. |
|
| 400 | + * So the interface IS the alias for some other FQN, |
|
| 401 | + * and we need to find out if `Fully/Qualified/Namespace/SomeInterface` |
|
| 402 | + * represents some other class. |
|
| 403 | + * |
|
| 404 | + * @param string $fqn |
|
| 405 | + * @param string $for_class |
|
| 406 | + * @return bool |
|
| 407 | + */ |
|
| 408 | + public function isAlias($fqn = '', $for_class = '') |
|
| 409 | + { |
|
| 410 | + return $this->class_cache->isAlias($fqn, $for_class); |
|
| 411 | + } |
|
| 412 | + |
|
| 413 | + |
|
| 414 | + /** |
|
| 415 | + * Returns a FQN for provided alias if one exists, otherwise returns the original $alias |
|
| 416 | + * functions recursively, so that multiple aliases can be used to drill down to a FQN |
|
| 417 | + * for example: |
|
| 418 | + * if the following two entries were added to the _aliases array: |
|
| 419 | + * array( |
|
| 420 | + * 'interface_alias' => 'some\namespace\interface' |
|
| 421 | + * 'some\namespace\interface' => 'some\namespace\classname' |
|
| 422 | + * ) |
|
| 423 | + * then one could use EE_Registry::instance()->create( 'interface_alias' ) |
|
| 424 | + * to load an instance of 'some\namespace\classname' |
|
| 425 | + * |
|
| 426 | + * @param string $alias |
|
| 427 | + * @param string $for_class |
|
| 428 | + * @return string |
|
| 429 | + */ |
|
| 430 | + public function getFqnForAlias($alias = '', $for_class = '') |
|
| 431 | + { |
|
| 432 | + return (string) $this->class_cache->getFqnForAlias($alias, $for_class); |
|
| 433 | + } |
|
| 434 | + |
|
| 435 | + |
|
| 436 | + /** |
|
| 437 | + * Registers the core dependencies and whether a previously instantiated object should be loaded from the cache, |
|
| 438 | + * if one exists, or whether a new object should be generated every time the requested class is loaded. |
|
| 439 | + * This is done by using the following class constants: |
|
| 440 | + * EE_Dependency_Map::load_from_cache - loads previously instantiated object |
|
| 441 | + * EE_Dependency_Map::load_new_object - generates a new object every time |
|
| 442 | + */ |
|
| 443 | + protected function _register_core_dependencies() |
|
| 444 | + { |
|
| 445 | + $this->_dependency_map = array( |
|
| 446 | + 'EE_Request_Handler' => array( |
|
| 447 | + 'EE_Request' => EE_Dependency_Map::load_from_cache, |
|
| 448 | + ), |
|
| 449 | + 'EE_System' => array( |
|
| 450 | + 'EE_Registry' => EE_Dependency_Map::load_from_cache, |
|
| 451 | + 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 452 | + 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 453 | + 'EE_Maintenance_Mode' => EE_Dependency_Map::load_from_cache, |
|
| 454 | + ), |
|
| 455 | + 'EE_Session' => array( |
|
| 456 | + 'EventEspresso\core\services\cache\TransientCacheStorage' => EE_Dependency_Map::load_from_cache, |
|
| 457 | + 'EventEspresso\core\domain\values\session\SessionLifespan' => EE_Dependency_Map::load_from_cache, |
|
| 458 | + 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 459 | + 'EventEspresso\core\services\session\SessionStartHandler' => EE_Dependency_Map::load_from_cache, |
|
| 460 | + 'EE_Encryption' => EE_Dependency_Map::load_from_cache, |
|
| 461 | + ), |
|
| 462 | + 'EE_Cart' => array( |
|
| 463 | + 'EE_Session' => EE_Dependency_Map::load_from_cache, |
|
| 464 | + ), |
|
| 465 | + 'EE_Front_Controller' => array( |
|
| 466 | + 'EE_Registry' => EE_Dependency_Map::load_from_cache, |
|
| 467 | + 'EE_Request_Handler' => EE_Dependency_Map::load_from_cache, |
|
| 468 | + 'EE_Module_Request_Router' => EE_Dependency_Map::load_from_cache, |
|
| 469 | + ), |
|
| 470 | + 'EE_Messenger_Collection_Loader' => array( |
|
| 471 | + 'EE_Messenger_Collection' => EE_Dependency_Map::load_new_object, |
|
| 472 | + ), |
|
| 473 | + 'EE_Message_Type_Collection_Loader' => array( |
|
| 474 | + 'EE_Message_Type_Collection' => EE_Dependency_Map::load_new_object, |
|
| 475 | + ), |
|
| 476 | + 'EE_Message_Resource_Manager' => array( |
|
| 477 | + 'EE_Messenger_Collection_Loader' => EE_Dependency_Map::load_new_object, |
|
| 478 | + 'EE_Message_Type_Collection_Loader' => EE_Dependency_Map::load_new_object, |
|
| 479 | + 'EEM_Message_Template_Group' => EE_Dependency_Map::load_from_cache, |
|
| 480 | + ), |
|
| 481 | + 'EE_Message_Factory' => array( |
|
| 482 | + 'EE_Message_Resource_Manager' => EE_Dependency_Map::load_from_cache, |
|
| 483 | + ), |
|
| 484 | + 'EE_messages' => array( |
|
| 485 | + 'EE_Message_Resource_Manager' => EE_Dependency_Map::load_from_cache, |
|
| 486 | + ), |
|
| 487 | + 'EE_Messages_Generator' => array( |
|
| 488 | + 'EE_Messages_Queue' => EE_Dependency_Map::load_new_object, |
|
| 489 | + 'EE_Messages_Data_Handler_Collection' => EE_Dependency_Map::load_new_object, |
|
| 490 | + 'EE_Message_Template_Group_Collection' => EE_Dependency_Map::load_new_object, |
|
| 491 | + 'EEH_Parse_Shortcodes' => EE_Dependency_Map::load_from_cache, |
|
| 492 | + ), |
|
| 493 | + 'EE_Messages_Processor' => array( |
|
| 494 | + 'EE_Message_Resource_Manager' => EE_Dependency_Map::load_from_cache, |
|
| 495 | + ), |
|
| 496 | + 'EE_Messages_Queue' => array( |
|
| 497 | + 'EE_Message_Repository' => EE_Dependency_Map::load_new_object, |
|
| 498 | + ), |
|
| 499 | + 'EE_Messages_Template_Defaults' => array( |
|
| 500 | + 'EEM_Message_Template_Group' => EE_Dependency_Map::load_from_cache, |
|
| 501 | + 'EEM_Message_Template' => EE_Dependency_Map::load_from_cache, |
|
| 502 | + ), |
|
| 503 | + 'EE_Message_To_Generate_From_Request' => array( |
|
| 504 | + 'EE_Message_Resource_Manager' => EE_Dependency_Map::load_from_cache, |
|
| 505 | + 'EE_Request_Handler' => EE_Dependency_Map::load_from_cache, |
|
| 506 | + ), |
|
| 507 | + 'EventEspresso\core\services\commands\CommandBus' => array( |
|
| 508 | + 'EventEspresso\core\services\commands\CommandHandlerManager' => EE_Dependency_Map::load_from_cache, |
|
| 509 | + ), |
|
| 510 | + 'EventEspresso\services\commands\CommandHandler' => array( |
|
| 511 | + 'EE_Registry' => EE_Dependency_Map::load_from_cache, |
|
| 512 | + 'CommandBusInterface' => EE_Dependency_Map::load_from_cache, |
|
| 513 | + ), |
|
| 514 | + 'EventEspresso\core\services\commands\CommandHandlerManager' => array( |
|
| 515 | + 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 516 | + ), |
|
| 517 | + 'EventEspresso\core\services\commands\CompositeCommandHandler' => array( |
|
| 518 | + 'EventEspresso\core\services\commands\CommandBus' => EE_Dependency_Map::load_from_cache, |
|
| 519 | + 'EventEspresso\core\services\commands\CommandFactory' => EE_Dependency_Map::load_from_cache, |
|
| 520 | + ), |
|
| 521 | + 'EventEspresso\core\services\commands\CommandFactory' => array( |
|
| 522 | + 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 523 | + ), |
|
| 524 | + 'EventEspresso\core\services\commands\middleware\CapChecker' => array( |
|
| 525 | + 'EventEspresso\core\domain\services\capabilities\CapabilitiesChecker' => EE_Dependency_Map::load_from_cache, |
|
| 526 | + ), |
|
| 527 | + 'EventEspresso\core\domain\services\capabilities\CapabilitiesChecker' => array( |
|
| 528 | + 'EE_Capabilities' => EE_Dependency_Map::load_from_cache, |
|
| 529 | + ), |
|
| 530 | + 'EventEspresso\core\domain\services\capabilities\RegistrationsCapChecker' => array( |
|
| 531 | + 'EE_Capabilities' => EE_Dependency_Map::load_from_cache, |
|
| 532 | + ), |
|
| 533 | + 'EventEspresso\core\services\commands\registration\CreateRegistrationCommandHandler' => array( |
|
| 534 | + 'EventEspresso\core\domain\services\registration\CreateRegistrationService' => EE_Dependency_Map::load_from_cache, |
|
| 535 | + ), |
|
| 536 | + 'EventEspresso\core\services\commands\registration\CopyRegistrationDetailsCommandHandler' => array( |
|
| 537 | + 'EventEspresso\core\domain\services\registration\CopyRegistrationService' => EE_Dependency_Map::load_from_cache, |
|
| 538 | + ), |
|
| 539 | + 'EventEspresso\core\services\commands\registration\CopyRegistrationPaymentsCommandHandler' => array( |
|
| 540 | + 'EventEspresso\core\domain\services\registration\CopyRegistrationService' => EE_Dependency_Map::load_from_cache, |
|
| 541 | + ), |
|
| 542 | + 'EventEspresso\core\services\commands\registration\CancelRegistrationAndTicketLineItemCommandHandler' => array( |
|
| 543 | + 'EventEspresso\core\domain\services\registration\CancelTicketLineItemService' => EE_Dependency_Map::load_from_cache, |
|
| 544 | + ), |
|
| 545 | + 'EventEspresso\core\services\commands\registration\UpdateRegistrationAndTransactionAfterChangeCommandHandler' => array( |
|
| 546 | + 'EventEspresso\core\domain\services\registration\UpdateRegistrationService' => EE_Dependency_Map::load_from_cache, |
|
| 547 | + ), |
|
| 548 | + 'EventEspresso\core\services\commands\ticket\CreateTicketLineItemCommandHandler' => array( |
|
| 549 | + 'EventEspresso\core\domain\services\ticket\CreateTicketLineItemService' => EE_Dependency_Map::load_from_cache, |
|
| 550 | + ), |
|
| 551 | + 'EventEspresso\core\services\commands\ticket\CancelTicketLineItemCommandHandler' => array( |
|
| 552 | + 'EventEspresso\core\domain\services\ticket\CancelTicketLineItemService' => EE_Dependency_Map::load_from_cache, |
|
| 553 | + ), |
|
| 554 | + 'EventEspresso\core\domain\services\registration\CancelRegistrationService' => array( |
|
| 555 | + 'EventEspresso\core\domain\services\ticket\CancelTicketLineItemService' => EE_Dependency_Map::load_from_cache, |
|
| 556 | + ), |
|
| 557 | + 'EventEspresso\core\services\commands\attendee\CreateAttendeeCommandHandler' => array( |
|
| 558 | + 'EEM_Attendee' => EE_Dependency_Map::load_from_cache, |
|
| 559 | + ), |
|
| 560 | + 'EventEspresso\core\services\database\TableManager' => array( |
|
| 561 | + 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 562 | + ), |
|
| 563 | + 'EE_Data_Migration_Class_Base' => array( |
|
| 564 | + 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 565 | + 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 566 | + ), |
|
| 567 | + 'EE_DMS_Core_4_1_0' => array( |
|
| 568 | + 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 569 | + 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 570 | + ), |
|
| 571 | + 'EE_DMS_Core_4_2_0' => array( |
|
| 572 | + 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 573 | + 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 574 | + ), |
|
| 575 | + 'EE_DMS_Core_4_3_0' => array( |
|
| 576 | + 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 577 | + 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 578 | + ), |
|
| 579 | + 'EE_DMS_Core_4_4_0' => array( |
|
| 580 | + 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 581 | + 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 582 | + ), |
|
| 583 | + 'EE_DMS_Core_4_5_0' => array( |
|
| 584 | + 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 585 | + 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 586 | + ), |
|
| 587 | + 'EE_DMS_Core_4_6_0' => array( |
|
| 588 | + 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 589 | + 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 590 | + ), |
|
| 591 | + 'EE_DMS_Core_4_7_0' => array( |
|
| 592 | + 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 593 | + 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 594 | + ), |
|
| 595 | + 'EE_DMS_Core_4_8_0' => array( |
|
| 596 | + 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 597 | + 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 598 | + ), |
|
| 599 | + 'EE_DMS_Core_4_9_0' => array( |
|
| 600 | + 'EventEspresso\core\services\database\TableAnalysis' => EE_Dependency_Map::load_from_cache, |
|
| 601 | + 'EventEspresso\core\services\database\TableManager' => EE_Dependency_Map::load_from_cache, |
|
| 602 | + ), |
|
| 603 | + 'EventEspresso\core\services\assets\I18nRegistry' => array( |
|
| 604 | + array(), |
|
| 605 | + 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 606 | + ), |
|
| 607 | + 'EventEspresso\core\services\assets\Registry' => array( |
|
| 608 | + 'EventEspresso\core\services\assets\AssetCollection' => EE_Dependency_Map::load_from_cache, |
|
| 609 | + 'EventEspresso\core\services\assets\I18nRegistry' => EE_Dependency_Map::load_from_cache, |
|
| 610 | + ), |
|
| 611 | + 'EventEspresso\core\domain\entities\shortcodes\EspressoCancelled' => array( |
|
| 612 | + 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 613 | + ), |
|
| 614 | + 'EventEspresso\core\domain\entities\shortcodes\EspressoCheckout' => array( |
|
| 615 | + 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 616 | + ), |
|
| 617 | + 'EventEspresso\core\domain\entities\shortcodes\EspressoEventAttendees' => array( |
|
| 618 | + 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 619 | + ), |
|
| 620 | + 'EventEspresso\core\domain\entities\shortcodes\EspressoEvents' => array( |
|
| 621 | + 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 622 | + ), |
|
| 623 | + 'EventEspresso\core\domain\entities\shortcodes\EspressoThankYou' => array( |
|
| 624 | + 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 625 | + ), |
|
| 626 | + 'EventEspresso\core\domain\entities\shortcodes\EspressoTicketSelector' => array( |
|
| 627 | + 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 628 | + ), |
|
| 629 | + 'EventEspresso\core\domain\entities\shortcodes\EspressoTxnPage' => array( |
|
| 630 | + 'EventEspresso\core\services\cache\PostRelatedCacheManager' => EE_Dependency_Map::load_from_cache, |
|
| 631 | + ), |
|
| 632 | + 'EventEspresso\core\services\cache\BasicCacheManager' => array( |
|
| 633 | + 'EventEspresso\core\services\cache\TransientCacheStorage' => EE_Dependency_Map::load_from_cache, |
|
| 634 | + ), |
|
| 635 | + 'EventEspresso\core\services\cache\PostRelatedCacheManager' => array( |
|
| 636 | + 'EventEspresso\core\services\cache\TransientCacheStorage' => EE_Dependency_Map::load_from_cache, |
|
| 637 | + ), |
|
| 638 | + 'EventEspresso\core\domain\services\validation\email\EmailValidationService' => array( |
|
| 639 | + 'EE_Registration_Config' => EE_Dependency_Map::load_from_cache, |
|
| 640 | + 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 641 | + ), |
|
| 642 | + 'EventEspresso\core\domain\values\EmailAddress' => array( |
|
| 643 | + null, |
|
| 644 | + 'EventEspresso\core\domain\services\validation\email\EmailValidationService' => EE_Dependency_Map::load_from_cache, |
|
| 645 | + ), |
|
| 646 | + 'EventEspresso\core\services\orm\ModelFieldFactory' => array( |
|
| 647 | + 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 648 | + ), |
|
| 649 | + 'LEGACY_MODELS' => array( |
|
| 650 | + null, |
|
| 651 | + 'EventEspresso\core\services\database\ModelFieldFactory' => EE_Dependency_Map::load_from_cache, |
|
| 652 | + ), |
|
| 653 | + 'EE_Module_Request_Router' => array( |
|
| 654 | + 'EE_Request' => EE_Dependency_Map::load_from_cache, |
|
| 655 | + ), |
|
| 656 | + 'EE_Registration_Processor' => array( |
|
| 657 | + 'EE_Request' => EE_Dependency_Map::load_from_cache, |
|
| 658 | + ), |
|
| 659 | + 'EventEspresso\core\services\notifications\PersistentAdminNoticeManager' => array( |
|
| 660 | + null, |
|
| 661 | + 'EventEspresso\core\domain\services\capabilities\CapabilitiesChecker' => EE_Dependency_Map::load_from_cache, |
|
| 662 | + 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 663 | + ), |
|
| 664 | + 'EventEspresso\core\services\licensing\LicenseService' => array( |
|
| 665 | + 'EventEspresso\core\domain\services\pue\Stats' => EE_Dependency_Map::load_from_cache, |
|
| 666 | + 'EventEspresso\core\domain\services\pue\Config' => EE_Dependency_Map::load_from_cache, |
|
| 667 | + ), |
|
| 668 | + 'EE_Admin_Transactions_List_Table' => array( |
|
| 669 | + null, |
|
| 670 | + 'EventEspresso\core\domain\values\session\SessionLifespan' => EE_Dependency_Map::load_from_cache, |
|
| 671 | + ), |
|
| 672 | + 'EventEspresso\core\domain\services\pue\Stats' => array( |
|
| 673 | + 'EventEspresso\core\domain\services\pue\Config' => EE_Dependency_Map::load_from_cache, |
|
| 674 | + 'EE_Maintenance_Mode' => EE_Dependency_Map::load_from_cache, |
|
| 675 | + 'EventEspresso\core\domain\services\pue\StatsGatherer' => EE_Dependency_Map::load_from_cache, |
|
| 676 | + ), |
|
| 677 | + 'EventEspresso\core\domain\services\pue\Config' => array( |
|
| 678 | + 'EE_Network_Config' => EE_Dependency_Map::load_from_cache, |
|
| 679 | + 'EE_Config' => EE_Dependency_Map::load_from_cache, |
|
| 680 | + ), |
|
| 681 | + 'EventEspresso\core\domain\services\pue\StatsGatherer' => array( |
|
| 682 | + 'EEM_Payment_Method' => EE_Dependency_Map::load_from_cache, |
|
| 683 | + 'EEM_Event' => EE_Dependency_Map::load_from_cache, |
|
| 684 | + 'EEM_Datetime' => EE_Dependency_Map::load_from_cache, |
|
| 685 | + 'EEM_Ticket' => EE_Dependency_Map::load_from_cache, |
|
| 686 | + 'EEM_Registration' => EE_Dependency_Map::load_from_cache, |
|
| 687 | + 'EEM_Transaction' => EE_Dependency_Map::load_from_cache, |
|
| 688 | + 'EE_Config' => EE_Dependency_Map::load_from_cache, |
|
| 689 | + ), |
|
| 690 | + 'EventEspresso\core\domain\services\admin\ExitModal' => array( |
|
| 691 | + 'EventEspresso\core\services\assets\Registry' => EE_Dependency_Map::load_from_cache, |
|
| 692 | + ), |
|
| 693 | + 'EventEspresso\core\domain\services\admin\PluginUpsells' => array( |
|
| 694 | + 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 695 | + ), |
|
| 696 | + 'EventEspresso\caffeinated\modules\recaptcha_invisible\InvisibleRecaptcha' => array( |
|
| 697 | + 'EE_Registration_Config' => EE_Dependency_Map::load_from_cache, |
|
| 698 | + 'EE_Session' => EE_Dependency_Map::load_from_cache, |
|
| 699 | + ), |
|
| 700 | + 'EventEspresso\caffeinated\modules\recaptcha_invisible\RecaptchaAdminSettings' => array( |
|
| 701 | + 'EE_Registration_Config' => EE_Dependency_Map::load_from_cache, |
|
| 702 | + ), |
|
| 703 | + 'EventEspresso\modules\ticket_selector\ProcessTicketSelector' => array( |
|
| 704 | + 'EE_Core_Config' => EE_Dependency_Map::load_from_cache, |
|
| 705 | + 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 706 | + 'EE_Session' => EE_Dependency_Map::load_from_cache, |
|
| 707 | + 'EEM_Ticket' => EE_Dependency_Map::load_from_cache, |
|
| 708 | + 'EventEspresso\modules\ticket_selector\TicketDatetimeAvailabilityTracker' => EE_Dependency_Map::load_from_cache, |
|
| 709 | + ), |
|
| 710 | + 'EventEspresso\modules\ticket_selector\TicketDatetimeAvailabilityTracker' => array( |
|
| 711 | + 'EEM_Datetime' => EE_Dependency_Map::load_from_cache, |
|
| 712 | + ), |
|
| 713 | + 'EventEspresso\core\domain\entities\custom_post_types\CustomPostTypeDefinitions' => array( |
|
| 714 | + 'EE_Core_Config' => EE_Dependency_Map::load_from_cache, |
|
| 715 | + 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 716 | + ), |
|
| 717 | + 'EventEspresso\core\domain\services\custom_post_types\RegisterCustomPostTypes' => array( |
|
| 718 | + 'EventEspresso\core\domain\entities\custom_post_types\CustomPostTypeDefinitions' => EE_Dependency_Map::load_from_cache, |
|
| 719 | + ), |
|
| 720 | + 'EventEspresso\core\domain\services\custom_post_types\RegisterCustomTaxonomies' => array( |
|
| 721 | + 'EventEspresso\core\domain\entities\custom_post_types\CustomTaxonomyDefinitions' => EE_Dependency_Map::load_from_cache, |
|
| 722 | + ), |
|
| 723 | + 'EE_CPT_Strategy' => array( |
|
| 724 | + 'EventEspresso\core\domain\entities\custom_post_types\CustomPostTypeDefinitions' => EE_Dependency_Map::load_from_cache, |
|
| 725 | + 'EventEspresso\core\domain\entities\custom_post_types\CustomTaxonomyDefinitions' => EE_Dependency_Map::load_from_cache, |
|
| 726 | + ), |
|
| 727 | + 'EventEspresso\core\services\loaders\ObjectIdentifier' => array( |
|
| 728 | + 'EventEspresso\core\services\loaders\ClassInterfaceCache' => EE_Dependency_Map::load_from_cache, |
|
| 729 | + ), |
|
| 730 | + 'EventEspresso\core\domain\services\assets\CoreAssetManager' => array( |
|
| 731 | + 'EventEspresso\core\services\assets\AssetCollection' => EE_Dependency_Map::load_from_cache, |
|
| 732 | + 'EE_Currency_Config' => EE_Dependency_Map::load_from_cache, |
|
| 733 | + 'EE_Template_Config' => EE_Dependency_Map::load_from_cache, |
|
| 734 | + 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 735 | + 'EventEspresso\core\services\assets\Registry' => EE_Dependency_Map::load_from_cache, |
|
| 736 | + ), |
|
| 737 | + 'EventEspresso\core\domain\services\admin\privacy\policy\PrivacyPolicy' => array( |
|
| 738 | + 'EEM_Payment_Method' => EE_Dependency_Map::load_from_cache, |
|
| 739 | + 'EventEspresso\core\domain\values\session\SessionLifespan' => EE_Dependency_Map::load_from_cache |
|
| 740 | + ), |
|
| 741 | + 'EventEspresso\core\domain\services\admin\privacy\export\ExportAttendee' => array( |
|
| 742 | + 'EEM_Attendee' => EE_Dependency_Map::load_from_cache, |
|
| 743 | + ), |
|
| 744 | + 'EventEspresso\core\domain\services\admin\privacy\export\ExportAttendeeBillingData' => array( |
|
| 745 | + 'EEM_Attendee' => EE_Dependency_Map::load_from_cache, |
|
| 746 | + 'EEM_Payment_Method' => EE_Dependency_Map::load_from_cache |
|
| 747 | + ), |
|
| 748 | + 'EventEspresso\core\domain\services\admin\privacy\export\ExportCheckins' => array( |
|
| 749 | + 'EEM_Checkin' => EE_Dependency_Map::load_from_cache, |
|
| 750 | + ), |
|
| 751 | + 'EventEspresso\core\domain\services\admin\privacy\export\ExportRegistration' => array( |
|
| 752 | + 'EEM_Registration' => EE_Dependency_Map::load_from_cache, |
|
| 753 | + ), |
|
| 754 | + 'EventEspresso\core\domain\services\admin\privacy\export\ExportTransaction' => array( |
|
| 755 | + 'EEM_Transaction' => EE_Dependency_Map::load_from_cache, |
|
| 756 | + ), |
|
| 757 | + 'EventEspresso\core\domain\services\admin\privacy\erasure\EraseAttendeeData' => array( |
|
| 758 | + 'EEM_Attendee' => EE_Dependency_Map::load_from_cache, |
|
| 759 | + ), |
|
| 760 | + 'EventEspresso\core\domain\services\admin\privacy\erasure\EraseAnswers' => array( |
|
| 761 | + 'EEM_Answer' => EE_Dependency_Map::load_from_cache, |
|
| 762 | + 'EEM_Question' => EE_Dependency_Map::load_from_cache, |
|
| 763 | + ), |
|
| 764 | + 'EventEspresso\core\CPTs\CptQueryModifier' => array( |
|
| 765 | + null, |
|
| 766 | + null, |
|
| 767 | + null, |
|
| 768 | + 'EE_Request_Handler' => EE_Dependency_Map::load_from_cache, |
|
| 769 | + 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 770 | + 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 771 | + ), |
|
| 772 | + 'EventEspresso\core\domain\services\admin\privacy\forms\PrivacySettingsFormHandler' => array( |
|
| 773 | + 'EE_Registry' => EE_Dependency_Map::load_from_cache, |
|
| 774 | + 'EE_Config' => EE_Dependency_Map::load_from_cache |
|
| 775 | + ), |
|
| 776 | + 'EventEspresso\core\services\editor\BlockRegistrationManager' => array( |
|
| 777 | + 'EventEspresso\core\services\assets\BlockAssetManagerCollection' => EE_Dependency_Map::load_from_cache, |
|
| 778 | + 'EventEspresso\core\domain\entities\editor\BlockCollection' => EE_Dependency_Map::load_from_cache, |
|
| 779 | + 'EventEspresso\core\services\route_match\RouteMatchSpecificationManager' => EE_Dependency_Map::load_from_cache, |
|
| 780 | + 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 781 | + ), |
|
| 782 | + 'EventEspresso\core\domain\entities\editor\CoreBlocksAssetManager' => array( |
|
| 783 | + 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 784 | + 'EventEspresso\core\services\assets\AssetCollection' => EE_Dependency_Map::load_from_cache, |
|
| 785 | + 'EventEspresso\core\services\assets\Registry' => EE_Dependency_Map::load_from_cache, |
|
| 786 | + ), |
|
| 787 | + 'EventEspresso\core\domain\services\blocks\EventAttendeesBlockRenderer' => array( |
|
| 788 | + 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 789 | + 'EEM_Attendee' => EE_Dependency_Map::load_from_cache, |
|
| 790 | + ), |
|
| 791 | + 'EventEspresso\core\domain\entities\editor\blocks\EventAttendees' => array( |
|
| 792 | + 'EventEspresso\core\domain\entities\editor\CoreBlocksAssetManager' => self::load_from_cache, |
|
| 793 | + 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 794 | + 'EventEspresso\core\domain\services\blocks\EventAttendeesBlockRenderer' => self::load_from_cache, |
|
| 795 | + ), |
|
| 796 | + 'EventEspresso\core\services\route_match\RouteMatchSpecificationDependencyResolver' => array( |
|
| 797 | + 'EventEspresso\core\services\container\Mirror' => EE_Dependency_Map::load_from_cache, |
|
| 798 | + 'EventEspresso\core\services\loaders\ClassInterfaceCache' => EE_Dependency_Map::load_from_cache, |
|
| 799 | + 'EE_Dependency_Map' => EE_Dependency_Map::load_from_cache, |
|
| 800 | + ), |
|
| 801 | + 'EventEspresso\core\services\route_match\RouteMatchSpecificationFactory' => array( |
|
| 802 | + 'EventEspresso\core\services\route_match\RouteMatchSpecificationDependencyResolver' => EE_Dependency_Map::load_from_cache, |
|
| 803 | + 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 804 | + ), |
|
| 805 | + 'EventEspresso\core\services\route_match\RouteMatchSpecificationManager' => array( |
|
| 806 | + 'EventEspresso\core\services\route_match\RouteMatchSpecificationCollection' => EE_Dependency_Map::load_from_cache, |
|
| 807 | + 'EventEspresso\core\services\route_match\RouteMatchSpecificationFactory' => EE_Dependency_Map::load_from_cache, |
|
| 808 | + ), |
|
| 809 | + 'EventEspresso\core\libraries\rest_api\CalculatedModelFields' => array( |
|
| 810 | + 'EventEspresso\core\libraries\rest_api\calculations\CalculatedModelFieldsFactory' => EE_Dependency_Map::load_from_cache |
|
| 811 | + ), |
|
| 812 | + 'EventEspresso\core\libraries\rest_api\calculations\CalculatedModelFieldsFactory' => array( |
|
| 813 | + 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 814 | + ), |
|
| 815 | + 'EventEspresso\core\libraries\rest_api\controllers\model\Read' => array( |
|
| 816 | + 'EventEspresso\core\libraries\rest_api\CalculatedModelFields' => EE_Dependency_Map::load_from_cache |
|
| 817 | + ), |
|
| 818 | + 'EventEspresso\core\libraries\rest_api\calculations\Datetime' => array( |
|
| 819 | + 'EEM_Datetime' => EE_Dependency_Map::load_from_cache, |
|
| 820 | + 'EEM_Registration' => EE_Dependency_Map::load_from_cache |
|
| 821 | + ), |
|
| 822 | + 'EventEspresso\core\libraries\rest_api\calculations\Event' => array( |
|
| 823 | + 'EEM_Event' => EE_Dependency_Map::load_from_cache, |
|
| 824 | + 'EEM_Registration' => EE_Dependency_Map::load_from_cache |
|
| 825 | + ), |
|
| 826 | + 'EventEspresso\core\libraries\rest_api\calculations\Registration' => array( |
|
| 827 | + 'EEM_Registration' => EE_Dependency_Map::load_from_cache |
|
| 828 | + ), |
|
| 829 | + 'EventEspresso\core\services\session\SessionStartHandler' => array( |
|
| 830 | + 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 831 | + ), |
|
| 832 | + 'EE_URL_Validation_Strategy' => array( |
|
| 833 | + null, |
|
| 834 | + null, |
|
| 835 | + 'EventEspresso\core\services\validators\URLValidator' => EE_Dependency_Map::load_from_cache |
|
| 836 | + ), |
|
| 837 | + 'EventEspresso\admin_pages\general_settings\OrganizationSettings' => array( |
|
| 838 | + 'EE_Registry' => EE_Dependency_Map::load_from_cache, |
|
| 839 | + 'EE_Organization_Config' => EE_Dependency_Map::load_from_cache, |
|
| 840 | + 'EE_Core_Config' => EE_Dependency_Map::load_from_cache, |
|
| 841 | + 'EE_Network_Core_Config' => EE_Dependency_Map::load_from_cache, |
|
| 842 | + 'EventEspresso\core\services\address\CountrySubRegionDao' => EE_Dependency_Map::load_from_cache, |
|
| 843 | + ), |
|
| 844 | + 'EventEspresso\core\services\address\CountrySubRegionDao' => array( |
|
| 845 | + 'EEM_State' => EE_Dependency_Map::load_from_cache, |
|
| 846 | + 'EventEspresso\core\services\validators\JsonValidator' => EE_Dependency_Map::load_from_cache |
|
| 847 | + ), |
|
| 848 | + 'EventEspresso\core\domain\services\admin\ajax\WordpressHeartbeat' => array( |
|
| 849 | + 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 850 | + 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 851 | + ), |
|
| 852 | + 'EventEspresso\core\domain\services\admin\ajax\EventEditorHeartbeat' => array( |
|
| 853 | + 'EventEspresso\core\domain\Domain' => EE_Dependency_Map::load_from_cache, |
|
| 854 | + 'EE_Environment_Config' => EE_Dependency_Map::load_from_cache, |
|
| 855 | + ), |
|
| 856 | + 'EventEspresso\core\services\request\files\FilesDataHandler' => array( |
|
| 857 | + 'EventEspresso\core\services\request\Request' => EE_Dependency_Map::load_from_cache, |
|
| 858 | + ), |
|
| 859 | + 'EventEspressoBatchRequest\BatchRequestProcessor' => [ |
|
| 860 | + 'EventEspresso\core\services\loaders\Loader' => EE_Dependency_Map::load_from_cache, |
|
| 861 | + ] |
|
| 862 | + ); |
|
| 863 | + } |
|
| 864 | + |
|
| 865 | + |
|
| 866 | + /** |
|
| 867 | + * Registers how core classes are loaded. |
|
| 868 | + * This can either be done by simply providing the name of one of the EE_Registry loader methods such as: |
|
| 869 | + * 'EE_Request_Handler' => 'load_core' |
|
| 870 | + * 'EE_Messages_Queue' => 'load_lib' |
|
| 871 | + * 'EEH_Debug_Tools' => 'load_helper' |
|
| 872 | + * or, if greater control is required, by providing a custom closure. For example: |
|
| 873 | + * 'Some_Class' => function () { |
|
| 874 | + * return new Some_Class(); |
|
| 875 | + * }, |
|
| 876 | + * This is required for instantiating dependencies |
|
| 877 | + * where an interface has been type hinted in a class constructor. For example: |
|
| 878 | + * 'Required_Interface' => function () { |
|
| 879 | + * return new A_Class_That_Implements_Required_Interface(); |
|
| 880 | + * }, |
|
| 881 | + */ |
|
| 882 | + protected function _register_core_class_loaders() |
|
| 883 | + { |
|
| 884 | + $this->_class_loaders = array( |
|
| 885 | + // load_core |
|
| 886 | + 'EE_Dependency_Map' => function () { |
|
| 887 | + return $this; |
|
| 888 | + }, |
|
| 889 | + 'EE_Capabilities' => 'load_core', |
|
| 890 | + 'EE_Encryption' => 'load_core', |
|
| 891 | + 'EE_Front_Controller' => 'load_core', |
|
| 892 | + 'EE_Module_Request_Router' => 'load_core', |
|
| 893 | + 'EE_Registry' => 'load_core', |
|
| 894 | + 'EE_Request' => function () { |
|
| 895 | + return $this->legacy_request; |
|
| 896 | + }, |
|
| 897 | + 'EventEspresso\core\services\request\Request' => function () { |
|
| 898 | + return $this->request; |
|
| 899 | + }, |
|
| 900 | + 'EventEspresso\core\services\request\Response' => function () { |
|
| 901 | + return $this->response; |
|
| 902 | + }, |
|
| 903 | + 'EE_Base' => 'load_core', |
|
| 904 | + 'EE_Request_Handler' => 'load_core', |
|
| 905 | + 'EE_Session' => 'load_core', |
|
| 906 | + 'EE_Cron_Tasks' => 'load_core', |
|
| 907 | + 'EE_System' => 'load_core', |
|
| 908 | + 'EE_Maintenance_Mode' => 'load_core', |
|
| 909 | + 'EE_Register_CPTs' => 'load_core', |
|
| 910 | + 'EE_Admin' => 'load_core', |
|
| 911 | + 'EE_CPT_Strategy' => 'load_core', |
|
| 912 | + // load_class |
|
| 913 | + 'EE_Registration_Processor' => 'load_class', |
|
| 914 | + // load_lib |
|
| 915 | + 'EE_Message_Resource_Manager' => 'load_lib', |
|
| 916 | + 'EE_Message_Type_Collection' => 'load_lib', |
|
| 917 | + 'EE_Message_Type_Collection_Loader' => 'load_lib', |
|
| 918 | + 'EE_Messenger_Collection' => 'load_lib', |
|
| 919 | + 'EE_Messenger_Collection_Loader' => 'load_lib', |
|
| 920 | + 'EE_Messages_Processor' => 'load_lib', |
|
| 921 | + 'EE_Message_Repository' => 'load_lib', |
|
| 922 | + 'EE_Messages_Queue' => 'load_lib', |
|
| 923 | + 'EE_Messages_Data_Handler_Collection' => 'load_lib', |
|
| 924 | + 'EE_Message_Template_Group_Collection' => 'load_lib', |
|
| 925 | + 'EE_Payment_Method_Manager' => 'load_lib', |
|
| 926 | + 'EE_Messages_Generator' => function () { |
|
| 927 | + return EE_Registry::instance()->load_lib( |
|
| 928 | + 'Messages_Generator', |
|
| 929 | + array(), |
|
| 930 | + false, |
|
| 931 | + false |
|
| 932 | + ); |
|
| 933 | + }, |
|
| 934 | + 'EE_Messages_Template_Defaults' => function ($arguments = array()) { |
|
| 935 | + return EE_Registry::instance()->load_lib( |
|
| 936 | + 'Messages_Template_Defaults', |
|
| 937 | + $arguments, |
|
| 938 | + false, |
|
| 939 | + false |
|
| 940 | + ); |
|
| 941 | + }, |
|
| 942 | + // load_helper |
|
| 943 | + 'EEH_Parse_Shortcodes' => function () { |
|
| 944 | + if (EE_Registry::instance()->load_helper('Parse_Shortcodes')) { |
|
| 945 | + return new EEH_Parse_Shortcodes(); |
|
| 946 | + } |
|
| 947 | + return null; |
|
| 948 | + }, |
|
| 949 | + 'EE_Template_Config' => function () { |
|
| 950 | + return EE_Config::instance()->template_settings; |
|
| 951 | + }, |
|
| 952 | + 'EE_Currency_Config' => function () { |
|
| 953 | + return EE_Config::instance()->currency; |
|
| 954 | + }, |
|
| 955 | + 'EE_Registration_Config' => function () { |
|
| 956 | + return EE_Config::instance()->registration; |
|
| 957 | + }, |
|
| 958 | + 'EE_Core_Config' => function () { |
|
| 959 | + return EE_Config::instance()->core; |
|
| 960 | + }, |
|
| 961 | + 'EventEspresso\core\services\loaders\Loader' => function () { |
|
| 962 | + return LoaderFactory::getLoader(); |
|
| 963 | + }, |
|
| 964 | + 'EE_Network_Config' => function () { |
|
| 965 | + return EE_Network_Config::instance(); |
|
| 966 | + }, |
|
| 967 | + 'EE_Config' => function () { |
|
| 968 | + return EE_Config::instance(); |
|
| 969 | + }, |
|
| 970 | + 'EventEspresso\core\domain\Domain' => function () { |
|
| 971 | + return DomainFactory::getEventEspressoCoreDomain(); |
|
| 972 | + }, |
|
| 973 | + 'EE_Admin_Config' => function () { |
|
| 974 | + return EE_Config::instance()->admin; |
|
| 975 | + }, |
|
| 976 | + 'EE_Organization_Config' => function () { |
|
| 977 | + return EE_Config::instance()->organization; |
|
| 978 | + }, |
|
| 979 | + 'EE_Network_Core_Config' => function () { |
|
| 980 | + return EE_Network_Config::instance()->core; |
|
| 981 | + }, |
|
| 982 | + 'EE_Environment_Config' => function () { |
|
| 983 | + return EE_Config::instance()->environment; |
|
| 984 | + }, |
|
| 985 | + ); |
|
| 986 | + } |
|
| 987 | + |
|
| 988 | + |
|
| 989 | + /** |
|
| 990 | + * can be used for supplying alternate names for classes, |
|
| 991 | + * or for connecting interface names to instantiable classes |
|
| 992 | + */ |
|
| 993 | + protected function _register_core_aliases() |
|
| 994 | + { |
|
| 995 | + $aliases = array( |
|
| 996 | + 'CommandBusInterface' => 'EventEspresso\core\services\commands\CommandBusInterface', |
|
| 997 | + 'EventEspresso\core\services\commands\CommandBusInterface' => 'EventEspresso\core\services\commands\CommandBus', |
|
| 998 | + 'CommandHandlerManagerInterface' => 'EventEspresso\core\services\commands\CommandHandlerManagerInterface', |
|
| 999 | + 'EventEspresso\core\services\commands\CommandHandlerManagerInterface' => 'EventEspresso\core\services\commands\CommandHandlerManager', |
|
| 1000 | + 'CapChecker' => 'EventEspresso\core\services\commands\middleware\CapChecker', |
|
| 1001 | + 'AddActionHook' => 'EventEspresso\core\services\commands\middleware\AddActionHook', |
|
| 1002 | + 'CapabilitiesChecker' => 'EventEspresso\core\domain\services\capabilities\CapabilitiesChecker', |
|
| 1003 | + 'CapabilitiesCheckerInterface' => 'EventEspresso\core\domain\services\capabilities\CapabilitiesCheckerInterface', |
|
| 1004 | + 'EventEspresso\core\domain\services\capabilities\CapabilitiesCheckerInterface' => 'EventEspresso\core\domain\services\capabilities\CapabilitiesChecker', |
|
| 1005 | + 'CreateRegistrationService' => 'EventEspresso\core\domain\services\registration\CreateRegistrationService', |
|
| 1006 | + 'CreateRegistrationCommandHandler' => 'EventEspresso\core\services\commands\registration\CreateRegistrationCommand', |
|
| 1007 | + 'CopyRegistrationDetailsCommandHandler' => 'EventEspresso\core\services\commands\registration\CopyRegistrationDetailsCommand', |
|
| 1008 | + 'CopyRegistrationPaymentsCommandHandler' => 'EventEspresso\core\services\commands\registration\CopyRegistrationPaymentsCommand', |
|
| 1009 | + 'CancelRegistrationAndTicketLineItemCommandHandler' => 'EventEspresso\core\services\commands\registration\CancelRegistrationAndTicketLineItemCommandHandler', |
|
| 1010 | + 'UpdateRegistrationAndTransactionAfterChangeCommandHandler' => 'EventEspresso\core\services\commands\registration\UpdateRegistrationAndTransactionAfterChangeCommandHandler', |
|
| 1011 | + 'CreateTicketLineItemCommandHandler' => 'EventEspresso\core\services\commands\ticket\CreateTicketLineItemCommand', |
|
| 1012 | + 'CreateTransactionCommandHandler' => 'EventEspresso\core\services\commands\transaction\CreateTransactionCommandHandler', |
|
| 1013 | + 'CreateAttendeeCommandHandler' => 'EventEspresso\core\services\commands\attendee\CreateAttendeeCommandHandler', |
|
| 1014 | + 'TableManager' => 'EventEspresso\core\services\database\TableManager', |
|
| 1015 | + 'TableAnalysis' => 'EventEspresso\core\services\database\TableAnalysis', |
|
| 1016 | + 'EspressoShortcode' => 'EventEspresso\core\services\shortcodes\EspressoShortcode', |
|
| 1017 | + 'ShortcodeInterface' => 'EventEspresso\core\services\shortcodes\ShortcodeInterface', |
|
| 1018 | + 'EventEspresso\core\services\shortcodes\ShortcodeInterface' => 'EventEspresso\core\services\shortcodes\EspressoShortcode', |
|
| 1019 | + 'EventEspresso\core\services\cache\CacheStorageInterface' => 'EventEspresso\core\services\cache\TransientCacheStorage', |
|
| 1020 | + 'LoaderInterface' => 'EventEspresso\core\services\loaders\LoaderInterface', |
|
| 1021 | + 'EventEspresso\core\services\loaders\LoaderInterface' => 'EventEspresso\core\services\loaders\Loader', |
|
| 1022 | + 'CommandFactoryInterface' => 'EventEspresso\core\services\commands\CommandFactoryInterface', |
|
| 1023 | + 'EventEspresso\core\services\commands\CommandFactoryInterface' => 'EventEspresso\core\services\commands\CommandFactory', |
|
| 1024 | + 'EmailValidatorInterface' => 'EventEspresso\core\domain\services\validation\email\EmailValidatorInterface', |
|
| 1025 | + 'EventEspresso\core\domain\services\validation\email\EmailValidatorInterface' => 'EventEspresso\core\domain\services\validation\email\EmailValidationService', |
|
| 1026 | + 'NoticeConverterInterface' => 'EventEspresso\core\services\notices\NoticeConverterInterface', |
|
| 1027 | + 'EventEspresso\core\services\notices\NoticeConverterInterface' => 'EventEspresso\core\services\notices\ConvertNoticesToEeErrors', |
|
| 1028 | + 'NoticesContainerInterface' => 'EventEspresso\core\services\notices\NoticesContainerInterface', |
|
| 1029 | + 'EventEspresso\core\services\notices\NoticesContainerInterface' => 'EventEspresso\core\services\notices\NoticesContainer', |
|
| 1030 | + 'EventEspresso\core\services\request\RequestInterface' => 'EventEspresso\core\services\request\Request', |
|
| 1031 | + 'EventEspresso\core\services\request\ResponseInterface' => 'EventEspresso\core\services\request\Response', |
|
| 1032 | + 'EventEspresso\core\domain\DomainInterface' => 'EventEspresso\core\domain\Domain', |
|
| 1033 | + 'Registration_Processor' => 'EE_Registration_Processor', |
|
| 1034 | + ); |
|
| 1035 | + foreach ($aliases as $alias => $fqn) { |
|
| 1036 | + if (is_array($fqn)) { |
|
| 1037 | + foreach ($fqn as $class => $for_class) { |
|
| 1038 | + $this->class_cache->addAlias($class, $alias, $for_class); |
|
| 1039 | + } |
|
| 1040 | + continue; |
|
| 1041 | + } |
|
| 1042 | + $this->class_cache->addAlias($fqn, $alias); |
|
| 1043 | + } |
|
| 1044 | + if (! (defined('DOING_AJAX') && DOING_AJAX) && is_admin()) { |
|
| 1045 | + $this->class_cache->addAlias( |
|
| 1046 | + 'EventEspresso\core\services\notices\ConvertNoticesToAdminNotices', |
|
| 1047 | + 'EventEspresso\core\services\notices\NoticeConverterInterface' |
|
| 1048 | + ); |
|
| 1049 | + } |
|
| 1050 | + } |
|
| 1051 | + |
|
| 1052 | + |
|
| 1053 | + /** |
|
| 1054 | + * This is used to reset the internal map and class_loaders to their original default state at the beginning of the |
|
| 1055 | + * request Primarily used by unit tests. |
|
| 1056 | + */ |
|
| 1057 | + public function reset() |
|
| 1058 | + { |
|
| 1059 | + $this->_register_core_class_loaders(); |
|
| 1060 | + $this->_register_core_dependencies(); |
|
| 1061 | + } |
|
| 1062 | + |
|
| 1063 | + |
|
| 1064 | + /** |
|
| 1065 | + * PLZ NOTE: a better name for this method would be is_alias() |
|
| 1066 | + * because it returns TRUE if the provided fully qualified name IS an alias |
|
| 1067 | + * WHY? |
|
| 1068 | + * Because if a class is type hinting for a concretion, |
|
| 1069 | + * then why would we need to find another class to supply it? |
|
| 1070 | + * ie: if a class asks for `Fully/Qualified/Namespace/SpecificClassName`, |
|
| 1071 | + * then give it an instance of `Fully/Qualified/Namespace/SpecificClassName`. |
|
| 1072 | + * Don't go looking for some substitute. |
|
| 1073 | + * Whereas if a class is type hinting for an interface... |
|
| 1074 | + * then we need to find an actual class to use. |
|
| 1075 | + * So the interface IS the alias for some other FQN, |
|
| 1076 | + * and we need to find out if `Fully/Qualified/Namespace/SomeInterface` |
|
| 1077 | + * represents some other class. |
|
| 1078 | + * |
|
| 1079 | + * @deprecated 4.9.62.p |
|
| 1080 | + * @param string $fqn |
|
| 1081 | + * @param string $for_class |
|
| 1082 | + * @return bool |
|
| 1083 | + */ |
|
| 1084 | + public function has_alias($fqn = '', $for_class = '') |
|
| 1085 | + { |
|
| 1086 | + return $this->isAlias($fqn, $for_class); |
|
| 1087 | + } |
|
| 1088 | + |
|
| 1089 | + |
|
| 1090 | + /** |
|
| 1091 | + * PLZ NOTE: a better name for this method would be get_fqn_for_alias() |
|
| 1092 | + * because it returns a FQN for provided alias if one exists, otherwise returns the original $alias |
|
| 1093 | + * functions recursively, so that multiple aliases can be used to drill down to a FQN |
|
| 1094 | + * for example: |
|
| 1095 | + * if the following two entries were added to the _aliases array: |
|
| 1096 | + * array( |
|
| 1097 | + * 'interface_alias' => 'some\namespace\interface' |
|
| 1098 | + * 'some\namespace\interface' => 'some\namespace\classname' |
|
| 1099 | + * ) |
|
| 1100 | + * then one could use EE_Registry::instance()->create( 'interface_alias' ) |
|
| 1101 | + * to load an instance of 'some\namespace\classname' |
|
| 1102 | + * |
|
| 1103 | + * @deprecated 4.9.62.p |
|
| 1104 | + * @param string $alias |
|
| 1105 | + * @param string $for_class |
|
| 1106 | + * @return string |
|
| 1107 | + */ |
|
| 1108 | + public function get_alias($alias = '', $for_class = '') |
|
| 1109 | + { |
|
| 1110 | + return $this->getFqnForAlias($alias, $for_class); |
|
| 1111 | + } |
|
| 1112 | 1112 | } |
@@ -15,296 +15,296 @@ |
||
| 15 | 15 | { |
| 16 | 16 | |
| 17 | 17 | |
| 18 | - /** |
|
| 19 | - * instance of the EE_System object |
|
| 20 | - * |
|
| 21 | - * @var $_instance |
|
| 22 | - * @access private |
|
| 23 | - */ |
|
| 24 | - private static $_instance = null; |
|
| 25 | - |
|
| 26 | - /** |
|
| 27 | - * $_autoloaders |
|
| 28 | - * @var array $_autoloaders |
|
| 29 | - * @access private |
|
| 30 | - */ |
|
| 31 | - private static $_autoloaders; |
|
| 32 | - |
|
| 33 | - /** |
|
| 34 | - * set to "paths" to display autoloader class => path mappings |
|
| 35 | - * set to "times" to display autoloader loading times |
|
| 36 | - * set to "all" to display both |
|
| 37 | - * |
|
| 38 | - * @var string $debug |
|
| 39 | - * @access private |
|
| 40 | - */ |
|
| 41 | - public static $debug = false; |
|
| 42 | - |
|
| 43 | - |
|
| 44 | - /** |
|
| 45 | - * class constructor |
|
| 46 | - * |
|
| 47 | - * @access private |
|
| 48 | - * @return \EEH_Autoloader |
|
| 49 | - * @throws Exception |
|
| 50 | - */ |
|
| 51 | - private function __construct() |
|
| 52 | - { |
|
| 53 | - if (self::$_autoloaders === null) { |
|
| 54 | - self::$_autoloaders = array(); |
|
| 55 | - $this->_register_custom_autoloaders(); |
|
| 56 | - spl_autoload_register(array( $this, 'espresso_autoloader' )); |
|
| 57 | - } |
|
| 58 | - } |
|
| 59 | - |
|
| 60 | - |
|
| 61 | - |
|
| 62 | - /** |
|
| 63 | - * @access public |
|
| 64 | - * @return EEH_Autoloader |
|
| 65 | - */ |
|
| 66 | - public static function instance() |
|
| 67 | - { |
|
| 68 | - // check if class object is instantiated |
|
| 69 | - if (! self::$_instance instanceof EEH_Autoloader) { |
|
| 70 | - self::$_instance = new self(); |
|
| 71 | - } |
|
| 72 | - return self::$_instance; |
|
| 73 | - } |
|
| 74 | - |
|
| 75 | - |
|
| 76 | - |
|
| 77 | - /** |
|
| 78 | - * espresso_autoloader |
|
| 79 | - * |
|
| 80 | - * @access public |
|
| 81 | - * @param $class_name |
|
| 82 | - * @internal param $className |
|
| 83 | - * @internal param string $class_name - simple class name ie: session |
|
| 84 | - * @return void |
|
| 85 | - */ |
|
| 86 | - public static function espresso_autoloader($class_name) |
|
| 87 | - { |
|
| 88 | - if (isset(self::$_autoloaders[ $class_name ])) { |
|
| 89 | - require_once(self::$_autoloaders[ $class_name ]); |
|
| 90 | - } |
|
| 91 | - } |
|
| 92 | - |
|
| 93 | - |
|
| 94 | - |
|
| 95 | - /** |
|
| 96 | - * register_autoloader |
|
| 97 | - * |
|
| 98 | - * @access public |
|
| 99 | - * @param array | string $class_paths - array of key => value pairings between class names and paths |
|
| 100 | - * @param bool $read_check true if we need to check whether the file is readable or not. |
|
| 101 | - * @param bool $debug **deprecated** |
|
| 102 | - * @throws \EE_Error |
|
| 103 | - */ |
|
| 104 | - public static function register_autoloader($class_paths, $read_check = true, $debug = false) |
|
| 105 | - { |
|
| 106 | - $class_paths = is_array($class_paths) ? $class_paths : array( $class_paths ); |
|
| 107 | - foreach ($class_paths as $class => $path) { |
|
| 108 | - // skip all files that are not PHP |
|
| 109 | - if (substr($path, strlen($path) - 3) !== 'php') { |
|
| 110 | - continue; |
|
| 111 | - } |
|
| 112 | - // don't give up! you gotta... |
|
| 113 | - // get some class |
|
| 114 | - if (empty($class)) { |
|
| 115 | - throw new EE_Error(sprintf(__('No Class name was specified while registering an autoloader for the following path: %s.', 'event_espresso'), $path)); |
|
| 116 | - } |
|
| 117 | - // one day you will find the path young grasshopper |
|
| 118 | - if (empty($path)) { |
|
| 119 | - throw new EE_Error(sprintf(__('No path was specified while registering an autoloader for the %s class.', 'event_espresso'), $class)); |
|
| 120 | - } |
|
| 121 | - // is file readable ? |
|
| 122 | - if ($read_check && ! is_readable($path)) { |
|
| 123 | - throw new EE_Error(sprintf(__('The file for the %s class could not be found or is not readable due to file permissions. Please ensure the following path is correct: %s', 'event_espresso'), $class, $path)); |
|
| 124 | - } |
|
| 125 | - if (! isset(self::$_autoloaders[ $class ])) { |
|
| 126 | - self::$_autoloaders[ $class ] = str_replace(array( '/', '\\' ), DS, $path); |
|
| 127 | - if (EE_DEBUG && ( EEH_Autoloader::$debug === 'paths' || EEH_Autoloader::$debug === 'all' || $debug )) { |
|
| 128 | - EEH_Debug_Tools::printr(self::$_autoloaders[ $class ], $class, __FILE__, __LINE__); |
|
| 129 | - } |
|
| 130 | - } |
|
| 131 | - } |
|
| 132 | - } |
|
| 133 | - |
|
| 134 | - |
|
| 135 | - |
|
| 136 | - |
|
| 137 | - /** |
|
| 138 | - * get_autoloaders |
|
| 139 | - * |
|
| 140 | - * @access public |
|
| 141 | - * @return array |
|
| 142 | - */ |
|
| 143 | - public static function get_autoloaders() |
|
| 144 | - { |
|
| 145 | - return self::$_autoloaders; |
|
| 146 | - } |
|
| 147 | - |
|
| 148 | - |
|
| 149 | - /** |
|
| 150 | - * register core, model and class 'autoloaders' |
|
| 151 | - * |
|
| 152 | - * @access private |
|
| 153 | - * @return void |
|
| 154 | - * @throws EE_Error |
|
| 155 | - */ |
|
| 156 | - private function _register_custom_autoloaders() |
|
| 157 | - { |
|
| 158 | - EEH_Autoloader::$debug = ''; |
|
| 159 | - \EEH_Autoloader::register_helpers_autoloaders(); |
|
| 160 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CORE . 'interfaces'); |
|
| 161 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CORE); |
|
| 162 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_INTERFACES, true); |
|
| 163 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_MODELS, true); |
|
| 164 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CLASSES); |
|
| 165 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_FORM_SECTIONS, true); |
|
| 166 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'messages'); |
|
| 167 | - if (EEH_Autoloader::$debug === 'times' || EEH_Autoloader::$debug === 'all') { |
|
| 168 | - EEH_Debug_Tools::instance()->show_times(); |
|
| 169 | - } |
|
| 170 | - } |
|
| 171 | - |
|
| 172 | - |
|
| 173 | - |
|
| 174 | - /** |
|
| 175 | - * register core, model and class 'autoloaders' |
|
| 176 | - * |
|
| 177 | - * @access public |
|
| 178 | - */ |
|
| 179 | - public static function register_helpers_autoloaders() |
|
| 180 | - { |
|
| 181 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_HELPERS); |
|
| 182 | - } |
|
| 183 | - |
|
| 184 | - |
|
| 185 | - |
|
| 186 | - |
|
| 187 | - /** |
|
| 188 | - * register core, model and class 'autoloaders' |
|
| 189 | - * |
|
| 190 | - * @access public |
|
| 191 | - * @return void |
|
| 192 | - */ |
|
| 193 | - public static function register_form_sections_autoloaders() |
|
| 194 | - { |
|
| 195 | - // EEH_Autoloader::register_autoloaders_for_each_file_in_folder( EE_FORM_SECTIONS, true ); |
|
| 196 | - } |
|
| 197 | - |
|
| 198 | - |
|
| 199 | - /** |
|
| 200 | - * register core, model and class 'autoloaders' |
|
| 201 | - * |
|
| 202 | - * @access public |
|
| 203 | - * @return void |
|
| 204 | - * @throws EE_Error |
|
| 205 | - */ |
|
| 206 | - public static function register_line_item_display_autoloaders() |
|
| 207 | - { |
|
| 208 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'line_item_display', true); |
|
| 209 | - } |
|
| 210 | - |
|
| 211 | - |
|
| 212 | - /** |
|
| 213 | - * register core, model and class 'autoloaders' |
|
| 214 | - * |
|
| 215 | - * @access public |
|
| 216 | - * @return void |
|
| 217 | - * @throws EE_Error |
|
| 218 | - */ |
|
| 219 | - public static function register_line_item_filter_autoloaders() |
|
| 220 | - { |
|
| 221 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'line_item_filters', true); |
|
| 222 | - } |
|
| 223 | - |
|
| 224 | - |
|
| 225 | - /** |
|
| 226 | - * register template part 'autoloaders' |
|
| 227 | - * |
|
| 228 | - * @access public |
|
| 229 | - * @return void |
|
| 230 | - * @throws EE_Error |
|
| 231 | - */ |
|
| 232 | - public static function register_template_part_autoloaders() |
|
| 233 | - { |
|
| 234 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'template_parts', true); |
|
| 235 | - } |
|
| 236 | - |
|
| 237 | - |
|
| 238 | - /** |
|
| 239 | - * @return void |
|
| 240 | - * @throws EE_Error |
|
| 241 | - */ |
|
| 242 | - public static function register_business_classes() |
|
| 243 | - { |
|
| 244 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CORE . 'business'); |
|
| 245 | - } |
|
| 246 | - |
|
| 247 | - |
|
| 248 | - |
|
| 249 | - /** |
|
| 250 | - * Assumes all the files in this folder have the normal naming scheme (namely that their classname |
|
| 251 | - * is the file's name, plus ".whatever.php".) and adds each of them to the autoloader list. |
|
| 252 | - * If that's not the case, you'll need to improve this function or just use EEH_File::get_classname_from_filepath_with_standard_filename() directly. |
|
| 253 | - * Yes this has to scan the directory for files, but it only does it once -- not on EACH |
|
| 254 | - * time the autoloader is used |
|
| 255 | - * |
|
| 256 | - * @param string $folder name, with or without trailing /, doesn't matter |
|
| 257 | - * @param bool $recursive |
|
| 258 | - * @param bool $debug **deprecated** |
|
| 259 | - * @throws \EE_Error |
|
| 260 | - */ |
|
| 261 | - public static function register_autoloaders_for_each_file_in_folder($folder, $recursive = false, $debug = false) |
|
| 262 | - { |
|
| 263 | - if (EEH_Autoloader::$debug === 'times' || EEH_Autoloader::$debug === 'all' || $debug) { |
|
| 264 | - EEH_Debug_Tools::instance()->start_timer(basename($folder)); |
|
| 265 | - } |
|
| 266 | - // make sure last char is a / |
|
| 267 | - $folder .= $folder[ strlen($folder)-1 ] !== DS ? DS : ''; |
|
| 268 | - $class_to_filepath_map = array(); |
|
| 269 | - $exclude = array( 'index' ); |
|
| 270 | - // get all the files in that folder that end in php |
|
| 271 | - $filepaths = glob($folder.'*'); |
|
| 272 | - |
|
| 273 | - if (empty($filepaths)) { |
|
| 274 | - return; |
|
| 275 | - } |
|
| 276 | - |
|
| 277 | - foreach ($filepaths as $filepath) { |
|
| 278 | - if (substr($filepath, -4, 4) === '.php') { |
|
| 279 | - $class_name = EEH_File::get_classname_from_filepath_with_standard_filename($filepath); |
|
| 280 | - if (! in_array($class_name, $exclude)) { |
|
| 281 | - $class_to_filepath_map [ $class_name ] = $filepath; |
|
| 282 | - } |
|
| 283 | - } elseif ($recursive) { |
|
| 284 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder($filepath, $recursive, $debug); |
|
| 285 | - } |
|
| 286 | - } |
|
| 287 | - // we remove the necessity to do a is_readable() check via the $read_check flag because glob by nature will not return non_readable files/directories. |
|
| 288 | - self::register_autoloader($class_to_filepath_map, false, $debug); |
|
| 289 | - if (EEH_Autoloader::$debug === 'times' || EEH_Autoloader::$debug === 'all') { |
|
| 290 | - EEH_Debug_Tools::instance()->stop_timer(basename($folder)); |
|
| 291 | - } |
|
| 292 | - } |
|
| 293 | - |
|
| 294 | - |
|
| 295 | - |
|
| 296 | - /** |
|
| 297 | - * add_alias |
|
| 298 | - * register additional autoloader based on variation of the classname for an existing autoloader |
|
| 299 | - * |
|
| 300 | - * @access public |
|
| 301 | - * @param string $class_name - simple class name ie: EE_Session |
|
| 302 | - * @param string $alias - variation on class name ie: EE_session, session, etc |
|
| 303 | - */ |
|
| 304 | - public static function add_alias($class_name, $alias) |
|
| 305 | - { |
|
| 306 | - if (isset(self::$_autoloaders[ $class_name ])) { |
|
| 307 | - self::$_autoloaders[ $alias ] = self::$_autoloaders[ $class_name ]; |
|
| 308 | - } |
|
| 309 | - } |
|
| 18 | + /** |
|
| 19 | + * instance of the EE_System object |
|
| 20 | + * |
|
| 21 | + * @var $_instance |
|
| 22 | + * @access private |
|
| 23 | + */ |
|
| 24 | + private static $_instance = null; |
|
| 25 | + |
|
| 26 | + /** |
|
| 27 | + * $_autoloaders |
|
| 28 | + * @var array $_autoloaders |
|
| 29 | + * @access private |
|
| 30 | + */ |
|
| 31 | + private static $_autoloaders; |
|
| 32 | + |
|
| 33 | + /** |
|
| 34 | + * set to "paths" to display autoloader class => path mappings |
|
| 35 | + * set to "times" to display autoloader loading times |
|
| 36 | + * set to "all" to display both |
|
| 37 | + * |
|
| 38 | + * @var string $debug |
|
| 39 | + * @access private |
|
| 40 | + */ |
|
| 41 | + public static $debug = false; |
|
| 42 | + |
|
| 43 | + |
|
| 44 | + /** |
|
| 45 | + * class constructor |
|
| 46 | + * |
|
| 47 | + * @access private |
|
| 48 | + * @return \EEH_Autoloader |
|
| 49 | + * @throws Exception |
|
| 50 | + */ |
|
| 51 | + private function __construct() |
|
| 52 | + { |
|
| 53 | + if (self::$_autoloaders === null) { |
|
| 54 | + self::$_autoloaders = array(); |
|
| 55 | + $this->_register_custom_autoloaders(); |
|
| 56 | + spl_autoload_register(array( $this, 'espresso_autoloader' )); |
|
| 57 | + } |
|
| 58 | + } |
|
| 59 | + |
|
| 60 | + |
|
| 61 | + |
|
| 62 | + /** |
|
| 63 | + * @access public |
|
| 64 | + * @return EEH_Autoloader |
|
| 65 | + */ |
|
| 66 | + public static function instance() |
|
| 67 | + { |
|
| 68 | + // check if class object is instantiated |
|
| 69 | + if (! self::$_instance instanceof EEH_Autoloader) { |
|
| 70 | + self::$_instance = new self(); |
|
| 71 | + } |
|
| 72 | + return self::$_instance; |
|
| 73 | + } |
|
| 74 | + |
|
| 75 | + |
|
| 76 | + |
|
| 77 | + /** |
|
| 78 | + * espresso_autoloader |
|
| 79 | + * |
|
| 80 | + * @access public |
|
| 81 | + * @param $class_name |
|
| 82 | + * @internal param $className |
|
| 83 | + * @internal param string $class_name - simple class name ie: session |
|
| 84 | + * @return void |
|
| 85 | + */ |
|
| 86 | + public static function espresso_autoloader($class_name) |
|
| 87 | + { |
|
| 88 | + if (isset(self::$_autoloaders[ $class_name ])) { |
|
| 89 | + require_once(self::$_autoloaders[ $class_name ]); |
|
| 90 | + } |
|
| 91 | + } |
|
| 92 | + |
|
| 93 | + |
|
| 94 | + |
|
| 95 | + /** |
|
| 96 | + * register_autoloader |
|
| 97 | + * |
|
| 98 | + * @access public |
|
| 99 | + * @param array | string $class_paths - array of key => value pairings between class names and paths |
|
| 100 | + * @param bool $read_check true if we need to check whether the file is readable or not. |
|
| 101 | + * @param bool $debug **deprecated** |
|
| 102 | + * @throws \EE_Error |
|
| 103 | + */ |
|
| 104 | + public static function register_autoloader($class_paths, $read_check = true, $debug = false) |
|
| 105 | + { |
|
| 106 | + $class_paths = is_array($class_paths) ? $class_paths : array( $class_paths ); |
|
| 107 | + foreach ($class_paths as $class => $path) { |
|
| 108 | + // skip all files that are not PHP |
|
| 109 | + if (substr($path, strlen($path) - 3) !== 'php') { |
|
| 110 | + continue; |
|
| 111 | + } |
|
| 112 | + // don't give up! you gotta... |
|
| 113 | + // get some class |
|
| 114 | + if (empty($class)) { |
|
| 115 | + throw new EE_Error(sprintf(__('No Class name was specified while registering an autoloader for the following path: %s.', 'event_espresso'), $path)); |
|
| 116 | + } |
|
| 117 | + // one day you will find the path young grasshopper |
|
| 118 | + if (empty($path)) { |
|
| 119 | + throw new EE_Error(sprintf(__('No path was specified while registering an autoloader for the %s class.', 'event_espresso'), $class)); |
|
| 120 | + } |
|
| 121 | + // is file readable ? |
|
| 122 | + if ($read_check && ! is_readable($path)) { |
|
| 123 | + throw new EE_Error(sprintf(__('The file for the %s class could not be found or is not readable due to file permissions. Please ensure the following path is correct: %s', 'event_espresso'), $class, $path)); |
|
| 124 | + } |
|
| 125 | + if (! isset(self::$_autoloaders[ $class ])) { |
|
| 126 | + self::$_autoloaders[ $class ] = str_replace(array( '/', '\\' ), DS, $path); |
|
| 127 | + if (EE_DEBUG && ( EEH_Autoloader::$debug === 'paths' || EEH_Autoloader::$debug === 'all' || $debug )) { |
|
| 128 | + EEH_Debug_Tools::printr(self::$_autoloaders[ $class ], $class, __FILE__, __LINE__); |
|
| 129 | + } |
|
| 130 | + } |
|
| 131 | + } |
|
| 132 | + } |
|
| 133 | + |
|
| 134 | + |
|
| 135 | + |
|
| 136 | + |
|
| 137 | + /** |
|
| 138 | + * get_autoloaders |
|
| 139 | + * |
|
| 140 | + * @access public |
|
| 141 | + * @return array |
|
| 142 | + */ |
|
| 143 | + public static function get_autoloaders() |
|
| 144 | + { |
|
| 145 | + return self::$_autoloaders; |
|
| 146 | + } |
|
| 147 | + |
|
| 148 | + |
|
| 149 | + /** |
|
| 150 | + * register core, model and class 'autoloaders' |
|
| 151 | + * |
|
| 152 | + * @access private |
|
| 153 | + * @return void |
|
| 154 | + * @throws EE_Error |
|
| 155 | + */ |
|
| 156 | + private function _register_custom_autoloaders() |
|
| 157 | + { |
|
| 158 | + EEH_Autoloader::$debug = ''; |
|
| 159 | + \EEH_Autoloader::register_helpers_autoloaders(); |
|
| 160 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CORE . 'interfaces'); |
|
| 161 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CORE); |
|
| 162 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_INTERFACES, true); |
|
| 163 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_MODELS, true); |
|
| 164 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CLASSES); |
|
| 165 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_FORM_SECTIONS, true); |
|
| 166 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'messages'); |
|
| 167 | + if (EEH_Autoloader::$debug === 'times' || EEH_Autoloader::$debug === 'all') { |
|
| 168 | + EEH_Debug_Tools::instance()->show_times(); |
|
| 169 | + } |
|
| 170 | + } |
|
| 171 | + |
|
| 172 | + |
|
| 173 | + |
|
| 174 | + /** |
|
| 175 | + * register core, model and class 'autoloaders' |
|
| 176 | + * |
|
| 177 | + * @access public |
|
| 178 | + */ |
|
| 179 | + public static function register_helpers_autoloaders() |
|
| 180 | + { |
|
| 181 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_HELPERS); |
|
| 182 | + } |
|
| 183 | + |
|
| 184 | + |
|
| 185 | + |
|
| 186 | + |
|
| 187 | + /** |
|
| 188 | + * register core, model and class 'autoloaders' |
|
| 189 | + * |
|
| 190 | + * @access public |
|
| 191 | + * @return void |
|
| 192 | + */ |
|
| 193 | + public static function register_form_sections_autoloaders() |
|
| 194 | + { |
|
| 195 | + // EEH_Autoloader::register_autoloaders_for_each_file_in_folder( EE_FORM_SECTIONS, true ); |
|
| 196 | + } |
|
| 197 | + |
|
| 198 | + |
|
| 199 | + /** |
|
| 200 | + * register core, model and class 'autoloaders' |
|
| 201 | + * |
|
| 202 | + * @access public |
|
| 203 | + * @return void |
|
| 204 | + * @throws EE_Error |
|
| 205 | + */ |
|
| 206 | + public static function register_line_item_display_autoloaders() |
|
| 207 | + { |
|
| 208 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'line_item_display', true); |
|
| 209 | + } |
|
| 210 | + |
|
| 211 | + |
|
| 212 | + /** |
|
| 213 | + * register core, model and class 'autoloaders' |
|
| 214 | + * |
|
| 215 | + * @access public |
|
| 216 | + * @return void |
|
| 217 | + * @throws EE_Error |
|
| 218 | + */ |
|
| 219 | + public static function register_line_item_filter_autoloaders() |
|
| 220 | + { |
|
| 221 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'line_item_filters', true); |
|
| 222 | + } |
|
| 223 | + |
|
| 224 | + |
|
| 225 | + /** |
|
| 226 | + * register template part 'autoloaders' |
|
| 227 | + * |
|
| 228 | + * @access public |
|
| 229 | + * @return void |
|
| 230 | + * @throws EE_Error |
|
| 231 | + */ |
|
| 232 | + public static function register_template_part_autoloaders() |
|
| 233 | + { |
|
| 234 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'template_parts', true); |
|
| 235 | + } |
|
| 236 | + |
|
| 237 | + |
|
| 238 | + /** |
|
| 239 | + * @return void |
|
| 240 | + * @throws EE_Error |
|
| 241 | + */ |
|
| 242 | + public static function register_business_classes() |
|
| 243 | + { |
|
| 244 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CORE . 'business'); |
|
| 245 | + } |
|
| 246 | + |
|
| 247 | + |
|
| 248 | + |
|
| 249 | + /** |
|
| 250 | + * Assumes all the files in this folder have the normal naming scheme (namely that their classname |
|
| 251 | + * is the file's name, plus ".whatever.php".) and adds each of them to the autoloader list. |
|
| 252 | + * If that's not the case, you'll need to improve this function or just use EEH_File::get_classname_from_filepath_with_standard_filename() directly. |
|
| 253 | + * Yes this has to scan the directory for files, but it only does it once -- not on EACH |
|
| 254 | + * time the autoloader is used |
|
| 255 | + * |
|
| 256 | + * @param string $folder name, with or without trailing /, doesn't matter |
|
| 257 | + * @param bool $recursive |
|
| 258 | + * @param bool $debug **deprecated** |
|
| 259 | + * @throws \EE_Error |
|
| 260 | + */ |
|
| 261 | + public static function register_autoloaders_for_each_file_in_folder($folder, $recursive = false, $debug = false) |
|
| 262 | + { |
|
| 263 | + if (EEH_Autoloader::$debug === 'times' || EEH_Autoloader::$debug === 'all' || $debug) { |
|
| 264 | + EEH_Debug_Tools::instance()->start_timer(basename($folder)); |
|
| 265 | + } |
|
| 266 | + // make sure last char is a / |
|
| 267 | + $folder .= $folder[ strlen($folder)-1 ] !== DS ? DS : ''; |
|
| 268 | + $class_to_filepath_map = array(); |
|
| 269 | + $exclude = array( 'index' ); |
|
| 270 | + // get all the files in that folder that end in php |
|
| 271 | + $filepaths = glob($folder.'*'); |
|
| 272 | + |
|
| 273 | + if (empty($filepaths)) { |
|
| 274 | + return; |
|
| 275 | + } |
|
| 276 | + |
|
| 277 | + foreach ($filepaths as $filepath) { |
|
| 278 | + if (substr($filepath, -4, 4) === '.php') { |
|
| 279 | + $class_name = EEH_File::get_classname_from_filepath_with_standard_filename($filepath); |
|
| 280 | + if (! in_array($class_name, $exclude)) { |
|
| 281 | + $class_to_filepath_map [ $class_name ] = $filepath; |
|
| 282 | + } |
|
| 283 | + } elseif ($recursive) { |
|
| 284 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder($filepath, $recursive, $debug); |
|
| 285 | + } |
|
| 286 | + } |
|
| 287 | + // we remove the necessity to do a is_readable() check via the $read_check flag because glob by nature will not return non_readable files/directories. |
|
| 288 | + self::register_autoloader($class_to_filepath_map, false, $debug); |
|
| 289 | + if (EEH_Autoloader::$debug === 'times' || EEH_Autoloader::$debug === 'all') { |
|
| 290 | + EEH_Debug_Tools::instance()->stop_timer(basename($folder)); |
|
| 291 | + } |
|
| 292 | + } |
|
| 293 | + |
|
| 294 | + |
|
| 295 | + |
|
| 296 | + /** |
|
| 297 | + * add_alias |
|
| 298 | + * register additional autoloader based on variation of the classname for an existing autoloader |
|
| 299 | + * |
|
| 300 | + * @access public |
|
| 301 | + * @param string $class_name - simple class name ie: EE_Session |
|
| 302 | + * @param string $alias - variation on class name ie: EE_session, session, etc |
|
| 303 | + */ |
|
| 304 | + public static function add_alias($class_name, $alias) |
|
| 305 | + { |
|
| 306 | + if (isset(self::$_autoloaders[ $class_name ])) { |
|
| 307 | + self::$_autoloaders[ $alias ] = self::$_autoloaders[ $class_name ]; |
|
| 308 | + } |
|
| 309 | + } |
|
| 310 | 310 | } |
@@ -53,7 +53,7 @@ discard block |
||
| 53 | 53 | if (self::$_autoloaders === null) { |
| 54 | 54 | self::$_autoloaders = array(); |
| 55 | 55 | $this->_register_custom_autoloaders(); |
| 56 | - spl_autoload_register(array( $this, 'espresso_autoloader' )); |
|
| 56 | + spl_autoload_register(array($this, 'espresso_autoloader')); |
|
| 57 | 57 | } |
| 58 | 58 | } |
| 59 | 59 | |
@@ -66,7 +66,7 @@ discard block |
||
| 66 | 66 | public static function instance() |
| 67 | 67 | { |
| 68 | 68 | // check if class object is instantiated |
| 69 | - if (! self::$_instance instanceof EEH_Autoloader) { |
|
| 69 | + if ( ! self::$_instance instanceof EEH_Autoloader) { |
|
| 70 | 70 | self::$_instance = new self(); |
| 71 | 71 | } |
| 72 | 72 | return self::$_instance; |
@@ -85,8 +85,8 @@ discard block |
||
| 85 | 85 | */ |
| 86 | 86 | public static function espresso_autoloader($class_name) |
| 87 | 87 | { |
| 88 | - if (isset(self::$_autoloaders[ $class_name ])) { |
|
| 89 | - require_once(self::$_autoloaders[ $class_name ]); |
|
| 88 | + if (isset(self::$_autoloaders[$class_name])) { |
|
| 89 | + require_once(self::$_autoloaders[$class_name]); |
|
| 90 | 90 | } |
| 91 | 91 | } |
| 92 | 92 | |
@@ -103,7 +103,7 @@ discard block |
||
| 103 | 103 | */ |
| 104 | 104 | public static function register_autoloader($class_paths, $read_check = true, $debug = false) |
| 105 | 105 | { |
| 106 | - $class_paths = is_array($class_paths) ? $class_paths : array( $class_paths ); |
|
| 106 | + $class_paths = is_array($class_paths) ? $class_paths : array($class_paths); |
|
| 107 | 107 | foreach ($class_paths as $class => $path) { |
| 108 | 108 | // skip all files that are not PHP |
| 109 | 109 | if (substr($path, strlen($path) - 3) !== 'php') { |
@@ -122,10 +122,10 @@ discard block |
||
| 122 | 122 | if ($read_check && ! is_readable($path)) { |
| 123 | 123 | throw new EE_Error(sprintf(__('The file for the %s class could not be found or is not readable due to file permissions. Please ensure the following path is correct: %s', 'event_espresso'), $class, $path)); |
| 124 | 124 | } |
| 125 | - if (! isset(self::$_autoloaders[ $class ])) { |
|
| 126 | - self::$_autoloaders[ $class ] = str_replace(array( '/', '\\' ), DS, $path); |
|
| 127 | - if (EE_DEBUG && ( EEH_Autoloader::$debug === 'paths' || EEH_Autoloader::$debug === 'all' || $debug )) { |
|
| 128 | - EEH_Debug_Tools::printr(self::$_autoloaders[ $class ], $class, __FILE__, __LINE__); |
|
| 125 | + if ( ! isset(self::$_autoloaders[$class])) { |
|
| 126 | + self::$_autoloaders[$class] = str_replace(array('/', '\\'), DS, $path); |
|
| 127 | + if (EE_DEBUG && (EEH_Autoloader::$debug === 'paths' || EEH_Autoloader::$debug === 'all' || $debug)) { |
|
| 128 | + EEH_Debug_Tools::printr(self::$_autoloaders[$class], $class, __FILE__, __LINE__); |
|
| 129 | 129 | } |
| 130 | 130 | } |
| 131 | 131 | } |
@@ -157,13 +157,13 @@ discard block |
||
| 157 | 157 | { |
| 158 | 158 | EEH_Autoloader::$debug = ''; |
| 159 | 159 | \EEH_Autoloader::register_helpers_autoloaders(); |
| 160 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CORE . 'interfaces'); |
|
| 160 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CORE.'interfaces'); |
|
| 161 | 161 | EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CORE); |
| 162 | 162 | EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_INTERFACES, true); |
| 163 | 163 | EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_MODELS, true); |
| 164 | 164 | EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CLASSES); |
| 165 | 165 | EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_FORM_SECTIONS, true); |
| 166 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'messages'); |
|
| 166 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES.'messages'); |
|
| 167 | 167 | if (EEH_Autoloader::$debug === 'times' || EEH_Autoloader::$debug === 'all') { |
| 168 | 168 | EEH_Debug_Tools::instance()->show_times(); |
| 169 | 169 | } |
@@ -205,7 +205,7 @@ discard block |
||
| 205 | 205 | */ |
| 206 | 206 | public static function register_line_item_display_autoloaders() |
| 207 | 207 | { |
| 208 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'line_item_display', true); |
|
| 208 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES.'line_item_display', true); |
|
| 209 | 209 | } |
| 210 | 210 | |
| 211 | 211 | |
@@ -218,7 +218,7 @@ discard block |
||
| 218 | 218 | */ |
| 219 | 219 | public static function register_line_item_filter_autoloaders() |
| 220 | 220 | { |
| 221 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'line_item_filters', true); |
|
| 221 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES.'line_item_filters', true); |
|
| 222 | 222 | } |
| 223 | 223 | |
| 224 | 224 | |
@@ -231,7 +231,7 @@ discard block |
||
| 231 | 231 | */ |
| 232 | 232 | public static function register_template_part_autoloaders() |
| 233 | 233 | { |
| 234 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES . 'template_parts', true); |
|
| 234 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_LIBRARIES.'template_parts', true); |
|
| 235 | 235 | } |
| 236 | 236 | |
| 237 | 237 | |
@@ -241,7 +241,7 @@ discard block |
||
| 241 | 241 | */ |
| 242 | 242 | public static function register_business_classes() |
| 243 | 243 | { |
| 244 | - EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CORE . 'business'); |
|
| 244 | + EEH_Autoloader::register_autoloaders_for_each_file_in_folder(EE_CORE.'business'); |
|
| 245 | 245 | } |
| 246 | 246 | |
| 247 | 247 | |
@@ -264,9 +264,9 @@ discard block |
||
| 264 | 264 | EEH_Debug_Tools::instance()->start_timer(basename($folder)); |
| 265 | 265 | } |
| 266 | 266 | // make sure last char is a / |
| 267 | - $folder .= $folder[ strlen($folder)-1 ] !== DS ? DS : ''; |
|
| 267 | + $folder .= $folder[strlen($folder) - 1] !== DS ? DS : ''; |
|
| 268 | 268 | $class_to_filepath_map = array(); |
| 269 | - $exclude = array( 'index' ); |
|
| 269 | + $exclude = array('index'); |
|
| 270 | 270 | // get all the files in that folder that end in php |
| 271 | 271 | $filepaths = glob($folder.'*'); |
| 272 | 272 | |
@@ -277,8 +277,8 @@ discard block |
||
| 277 | 277 | foreach ($filepaths as $filepath) { |
| 278 | 278 | if (substr($filepath, -4, 4) === '.php') { |
| 279 | 279 | $class_name = EEH_File::get_classname_from_filepath_with_standard_filename($filepath); |
| 280 | - if (! in_array($class_name, $exclude)) { |
|
| 281 | - $class_to_filepath_map [ $class_name ] = $filepath; |
|
| 280 | + if ( ! in_array($class_name, $exclude)) { |
|
| 281 | + $class_to_filepath_map [$class_name] = $filepath; |
|
| 282 | 282 | } |
| 283 | 283 | } elseif ($recursive) { |
| 284 | 284 | EEH_Autoloader::register_autoloaders_for_each_file_in_folder($filepath, $recursive, $debug); |
@@ -303,8 +303,8 @@ discard block |
||
| 303 | 303 | */ |
| 304 | 304 | public static function add_alias($class_name, $alias) |
| 305 | 305 | { |
| 306 | - if (isset(self::$_autoloaders[ $class_name ])) { |
|
| 307 | - self::$_autoloaders[ $alias ] = self::$_autoloaders[ $class_name ]; |
|
| 306 | + if (isset(self::$_autoloaders[$class_name])) { |
|
| 307 | + self::$_autoloaders[$alias] = self::$_autoloaders[$class_name]; |
|
| 308 | 308 | } |
| 309 | 309 | } |
| 310 | 310 | } |
@@ -12,288 +12,288 @@ |
||
| 12 | 12 | abstract class EE_Form_Section_Layout_Base |
| 13 | 13 | { |
| 14 | 14 | |
| 15 | - /** |
|
| 16 | - * Form form section to lay out |
|
| 17 | - * |
|
| 18 | - * @var EE_Form_Section_Proper |
|
| 19 | - */ |
|
| 20 | - protected $_form_section; |
|
| 21 | - |
|
| 22 | - |
|
| 23 | - |
|
| 24 | - /** |
|
| 25 | - * __construct |
|
| 26 | - */ |
|
| 27 | - public function __construct() |
|
| 28 | - { |
|
| 29 | - } |
|
| 30 | - |
|
| 31 | - |
|
| 32 | - |
|
| 33 | - /** |
|
| 34 | - * The form section on which this strategy is to perform |
|
| 35 | - * |
|
| 36 | - * @param EE_Form_Section_Proper $form |
|
| 37 | - */ |
|
| 38 | - public function _construct_finalize(EE_Form_Section_Proper $form) |
|
| 39 | - { |
|
| 40 | - $this->_form_section = $form; |
|
| 41 | - } |
|
| 42 | - |
|
| 43 | - |
|
| 44 | - |
|
| 45 | - /** |
|
| 46 | - * @return EE_Form_Section_Proper |
|
| 47 | - */ |
|
| 48 | - public function form_section() |
|
| 49 | - { |
|
| 50 | - return $this->_form_section; |
|
| 51 | - } |
|
| 52 | - |
|
| 53 | - |
|
| 54 | - |
|
| 55 | - /** |
|
| 56 | - * Also has teh side effect of enqueuing any needed JS and CSS for |
|
| 57 | - * this form. |
|
| 58 | - * Creates all the HTML necessary for displaying this form, its inputs, and |
|
| 59 | - * proper subsections. |
|
| 60 | - * Returns the HTML |
|
| 61 | - * |
|
| 62 | - * @return string HTML for displaying |
|
| 63 | - * @throws EE_Error |
|
| 64 | - */ |
|
| 65 | - public function layout_form() |
|
| 66 | - { |
|
| 67 | - $html = ''; |
|
| 68 | - // layout_form_begin |
|
| 69 | - $html .= apply_filters( |
|
| 70 | - 'FHEE__EE_Form_Section_Layout_Base__layout_form__start__for_' . $this->_form_section->name(), |
|
| 71 | - $this->layout_form_begin(), |
|
| 72 | - $this->_form_section |
|
| 73 | - ); |
|
| 74 | - // layout_form_loop |
|
| 75 | - $html .= apply_filters( |
|
| 76 | - 'FHEE__EE_Form_Section_Layout_Base__layout_form__loop__for_' . $this->_form_section->name(), |
|
| 77 | - $this->layout_form_loop(), |
|
| 78 | - $this->_form_section |
|
| 79 | - ); |
|
| 80 | - // layout_form_end |
|
| 81 | - $html .= apply_filters( |
|
| 82 | - 'FHEE__EE_Form_Section_Layout_Base__layout_form__end__for_' . $this->_form_section->name(), |
|
| 83 | - $this->layout_form_end(), |
|
| 84 | - $this->_form_section |
|
| 85 | - ); |
|
| 86 | - $html = $this->add_form_section_hooks_and_filters($html); |
|
| 87 | - return $html; |
|
| 88 | - } |
|
| 89 | - |
|
| 90 | - |
|
| 91 | - |
|
| 92 | - /** |
|
| 93 | - * @return string |
|
| 94 | - * @throws EE_Error |
|
| 95 | - */ |
|
| 96 | - public function layout_form_loop() |
|
| 97 | - { |
|
| 98 | - $html = ''; |
|
| 99 | - foreach ($this->_form_section->subsections() as $name => $subsection) { |
|
| 100 | - if ($subsection instanceof EE_Form_Input_Base) { |
|
| 101 | - $html .= apply_filters( |
|
| 102 | - 'FHEE__EE_Form_Section_Layout_Base__layout_form__loop_for_input_' |
|
| 103 | - . $name . '__in_' . $this->_form_section->name(), |
|
| 104 | - $this->layout_input($subsection), |
|
| 105 | - $this->_form_section, |
|
| 106 | - $subsection |
|
| 107 | - ); |
|
| 108 | - } elseif ($subsection instanceof EE_Form_Section_Base) { |
|
| 109 | - $html .= apply_filters( |
|
| 110 | - 'FHEE__EE_Form_Section_Layout_Base__layout_form__loop_for_non_input_' |
|
| 111 | - . $name . '__in_' . $this->_form_section->name(), |
|
| 112 | - $this->layout_subsection($subsection), |
|
| 113 | - $this->_form_section, |
|
| 114 | - $subsection |
|
| 115 | - ); |
|
| 116 | - } |
|
| 117 | - } |
|
| 118 | - return $html; |
|
| 119 | - } |
|
| 120 | - |
|
| 121 | - |
|
| 122 | - |
|
| 123 | - /** |
|
| 124 | - * Should be used to start teh form section (Eg a table tag, or a div tag, etc.) |
|
| 125 | - * |
|
| 126 | - * @return string |
|
| 127 | - */ |
|
| 128 | - abstract public function layout_form_begin(); |
|
| 129 | - |
|
| 130 | - |
|
| 131 | - |
|
| 132 | - /** |
|
| 133 | - * Should be used to end the form section (eg a /table tag, or a /div tag, etc) |
|
| 134 | - * |
|
| 135 | - * @return string |
|
| 136 | - */ |
|
| 137 | - abstract public function layout_form_end(); |
|
| 138 | - |
|
| 139 | - |
|
| 140 | - |
|
| 141 | - /** |
|
| 142 | - * Should be used internally by layout_form() to layout each input (eg, if this layout |
|
| 143 | - * is putting each input in a row of its own, this should probably be called by a |
|
| 144 | - * foreach loop in layout_form() (WITHOUT adding any content directly within layout_form()'s foreach loop. |
|
| 145 | - * Eg, this method should add the tr and td tags). This method is exposed in case you want to completely |
|
| 146 | - * customize the form's layout, but would like to make use of it for laying out |
|
| 147 | - * 'easy-to-layout' inputs |
|
| 148 | - * |
|
| 149 | - * @param EE_Form_Input_Base $input |
|
| 150 | - * @return string html |
|
| 151 | - */ |
|
| 152 | - abstract public function layout_input($input); |
|
| 153 | - |
|
| 154 | - |
|
| 155 | - |
|
| 156 | - /** |
|
| 157 | - * Similar to layout_input(), should be used internally by layout_form() within a |
|
| 158 | - * loop to layout each proper subsection. Unlike layout_input(), however, it is assumed |
|
| 159 | - * that the proper subsection will layout its container, label, etc on its own. |
|
| 160 | - * |
|
| 161 | - * @param EE_Form_Section_Base $subsection |
|
| 162 | - * @return string html |
|
| 163 | - */ |
|
| 164 | - abstract public function layout_subsection($subsection); |
|
| 165 | - |
|
| 166 | - |
|
| 167 | - |
|
| 168 | - /** |
|
| 169 | - * Gets the HTML for the label tag and its contents for the input |
|
| 170 | - * |
|
| 171 | - * @param EE_Form_Input_Base $input |
|
| 172 | - * @return string |
|
| 173 | - */ |
|
| 174 | - public function display_label($input) |
|
| 175 | - { |
|
| 176 | - if ($input->get_display_strategy() instanceof EE_Hidden_Display_Strategy) { |
|
| 177 | - return ''; |
|
| 178 | - } |
|
| 179 | - $class = $input->required() |
|
| 180 | - ? 'ee-required-label ' . $input->html_label_class() |
|
| 181 | - : $input->html_label_class(); |
|
| 182 | - $label_text = $input->required() |
|
| 183 | - ? $input->html_label_text() . '<span class="ee-asterisk">*</span>' |
|
| 184 | - : $input->html_label_text(); |
|
| 185 | - return '<label id="' |
|
| 186 | - . $input->html_label_id() |
|
| 187 | - . '" class="' |
|
| 188 | - . $class |
|
| 189 | - . '" style="' |
|
| 190 | - . $input->html_label_style() |
|
| 191 | - . '" for="' . $input->html_id() |
|
| 192 | - . '">' |
|
| 193 | - . $label_text |
|
| 194 | - . '</label>'; |
|
| 195 | - } |
|
| 196 | - |
|
| 197 | - |
|
| 198 | - |
|
| 199 | - /** |
|
| 200 | - * Gets the HTML for all the form's form-wide errors (ie, errors which |
|
| 201 | - * are not for specific inputs. E.g., if two inputs somehow disagree, |
|
| 202 | - * those errors would probably be on the form section, not one of its inputs) |
|
| 203 | - * @return string |
|
| 204 | - */ |
|
| 205 | - public function display_form_wide_errors() |
|
| 206 | - { |
|
| 207 | - $html = ''; |
|
| 208 | - if ($this->_form_section->get_validation_errors()) { |
|
| 209 | - $html .= "<div class='ee-form-wide-errors'>"; |
|
| 210 | - // get all the errors on THIS form section (errors which aren't |
|
| 211 | - // for specific inputs, but instead for the entire form section) |
|
| 212 | - foreach ($this->_form_section->get_validation_errors() as $error) { |
|
| 213 | - $html .= $error->getMessage() . '<br>'; |
|
| 214 | - } |
|
| 215 | - $html .= '</div>'; |
|
| 216 | - } |
|
| 217 | - return apply_filters( |
|
| 218 | - 'FHEE__EE_Form_Section_Layout_Base__display_form_wide_errors', |
|
| 219 | - $html, |
|
| 220 | - $this |
|
| 221 | - ); |
|
| 222 | - } |
|
| 223 | - |
|
| 224 | - |
|
| 225 | - |
|
| 226 | - /** |
|
| 227 | - * returns the HTML for the server-side validation errors for the specified input |
|
| 228 | - * Note that if JS is enabled, it should remove these and instead |
|
| 229 | - * populate the form's errors in the jquery validate fashion |
|
| 230 | - * using the localized data provided to the JS |
|
| 231 | - * |
|
| 232 | - * @param EE_Form_Input_Base $input |
|
| 233 | - * @return string |
|
| 234 | - */ |
|
| 235 | - public function display_errors($input) |
|
| 236 | - { |
|
| 237 | - if ($input->get_validation_errors()) { |
|
| 238 | - return "<label id='" |
|
| 239 | - . $input->html_id() |
|
| 240 | - . "-error' class='error' for='{$input->html_name()}'>" |
|
| 241 | - . $input->get_validation_error_string() |
|
| 242 | - . '</label>'; |
|
| 243 | - } |
|
| 244 | - return ''; |
|
| 245 | - } |
|
| 246 | - |
|
| 247 | - |
|
| 248 | - |
|
| 249 | - /** |
|
| 250 | - * Displays the help span for the specified input |
|
| 251 | - * |
|
| 252 | - * @param EE_Form_Input_Base $input |
|
| 253 | - * @return string |
|
| 254 | - */ |
|
| 255 | - public function display_help_text($input) |
|
| 256 | - { |
|
| 257 | - $help_text = $input->html_help_text(); |
|
| 258 | - if ($help_text !== '' && $help_text !== null) { |
|
| 259 | - $tag = is_admin() ? 'p' : 'span'; |
|
| 260 | - return '<' |
|
| 261 | - . $tag |
|
| 262 | - . ' id="' |
|
| 263 | - . $input->html_id() |
|
| 264 | - . '-help" class="' |
|
| 265 | - . $input->html_help_class() |
|
| 266 | - . '" style="' |
|
| 267 | - . $input->html_help_style() |
|
| 268 | - . '">' |
|
| 269 | - . $help_text |
|
| 270 | - . '</' |
|
| 271 | - . $tag |
|
| 272 | - . '>'; |
|
| 273 | - } |
|
| 274 | - return ''; |
|
| 275 | - } |
|
| 276 | - |
|
| 277 | - |
|
| 278 | - |
|
| 279 | - /** |
|
| 280 | - * Does an action and hook onto the end of teh form |
|
| 281 | - * |
|
| 282 | - * @param string $html |
|
| 283 | - * @return string |
|
| 284 | - */ |
|
| 285 | - public function add_form_section_hooks_and_filters($html) |
|
| 286 | - { |
|
| 287 | - // replace dashes and spaces with underscores |
|
| 288 | - $hook_name = str_replace(array('-', ' '), '_', $this->_form_section->html_id()); |
|
| 289 | - do_action('AHEE__Form_Section_Layout__' . $hook_name, $this->_form_section); |
|
| 290 | - $html = (string) apply_filters( |
|
| 291 | - 'AFEE__Form_Section_Layout__' . $hook_name . '__html', |
|
| 292 | - $html, |
|
| 293 | - $this->_form_section |
|
| 294 | - ); |
|
| 295 | - $html .= EEH_HTML::nl() . '<!-- AHEE__Form_Section_Layout__' . $hook_name . '__html -->'; |
|
| 296 | - $html .= EEH_HTML::nl() . '<!-- AFEE__Form_Section_Layout__' . $hook_name . ' -->'; |
|
| 297 | - return $html; |
|
| 298 | - } |
|
| 15 | + /** |
|
| 16 | + * Form form section to lay out |
|
| 17 | + * |
|
| 18 | + * @var EE_Form_Section_Proper |
|
| 19 | + */ |
|
| 20 | + protected $_form_section; |
|
| 21 | + |
|
| 22 | + |
|
| 23 | + |
|
| 24 | + /** |
|
| 25 | + * __construct |
|
| 26 | + */ |
|
| 27 | + public function __construct() |
|
| 28 | + { |
|
| 29 | + } |
|
| 30 | + |
|
| 31 | + |
|
| 32 | + |
|
| 33 | + /** |
|
| 34 | + * The form section on which this strategy is to perform |
|
| 35 | + * |
|
| 36 | + * @param EE_Form_Section_Proper $form |
|
| 37 | + */ |
|
| 38 | + public function _construct_finalize(EE_Form_Section_Proper $form) |
|
| 39 | + { |
|
| 40 | + $this->_form_section = $form; |
|
| 41 | + } |
|
| 42 | + |
|
| 43 | + |
|
| 44 | + |
|
| 45 | + /** |
|
| 46 | + * @return EE_Form_Section_Proper |
|
| 47 | + */ |
|
| 48 | + public function form_section() |
|
| 49 | + { |
|
| 50 | + return $this->_form_section; |
|
| 51 | + } |
|
| 52 | + |
|
| 53 | + |
|
| 54 | + |
|
| 55 | + /** |
|
| 56 | + * Also has teh side effect of enqueuing any needed JS and CSS for |
|
| 57 | + * this form. |
|
| 58 | + * Creates all the HTML necessary for displaying this form, its inputs, and |
|
| 59 | + * proper subsections. |
|
| 60 | + * Returns the HTML |
|
| 61 | + * |
|
| 62 | + * @return string HTML for displaying |
|
| 63 | + * @throws EE_Error |
|
| 64 | + */ |
|
| 65 | + public function layout_form() |
|
| 66 | + { |
|
| 67 | + $html = ''; |
|
| 68 | + // layout_form_begin |
|
| 69 | + $html .= apply_filters( |
|
| 70 | + 'FHEE__EE_Form_Section_Layout_Base__layout_form__start__for_' . $this->_form_section->name(), |
|
| 71 | + $this->layout_form_begin(), |
|
| 72 | + $this->_form_section |
|
| 73 | + ); |
|
| 74 | + // layout_form_loop |
|
| 75 | + $html .= apply_filters( |
|
| 76 | + 'FHEE__EE_Form_Section_Layout_Base__layout_form__loop__for_' . $this->_form_section->name(), |
|
| 77 | + $this->layout_form_loop(), |
|
| 78 | + $this->_form_section |
|
| 79 | + ); |
|
| 80 | + // layout_form_end |
|
| 81 | + $html .= apply_filters( |
|
| 82 | + 'FHEE__EE_Form_Section_Layout_Base__layout_form__end__for_' . $this->_form_section->name(), |
|
| 83 | + $this->layout_form_end(), |
|
| 84 | + $this->_form_section |
|
| 85 | + ); |
|
| 86 | + $html = $this->add_form_section_hooks_and_filters($html); |
|
| 87 | + return $html; |
|
| 88 | + } |
|
| 89 | + |
|
| 90 | + |
|
| 91 | + |
|
| 92 | + /** |
|
| 93 | + * @return string |
|
| 94 | + * @throws EE_Error |
|
| 95 | + */ |
|
| 96 | + public function layout_form_loop() |
|
| 97 | + { |
|
| 98 | + $html = ''; |
|
| 99 | + foreach ($this->_form_section->subsections() as $name => $subsection) { |
|
| 100 | + if ($subsection instanceof EE_Form_Input_Base) { |
|
| 101 | + $html .= apply_filters( |
|
| 102 | + 'FHEE__EE_Form_Section_Layout_Base__layout_form__loop_for_input_' |
|
| 103 | + . $name . '__in_' . $this->_form_section->name(), |
|
| 104 | + $this->layout_input($subsection), |
|
| 105 | + $this->_form_section, |
|
| 106 | + $subsection |
|
| 107 | + ); |
|
| 108 | + } elseif ($subsection instanceof EE_Form_Section_Base) { |
|
| 109 | + $html .= apply_filters( |
|
| 110 | + 'FHEE__EE_Form_Section_Layout_Base__layout_form__loop_for_non_input_' |
|
| 111 | + . $name . '__in_' . $this->_form_section->name(), |
|
| 112 | + $this->layout_subsection($subsection), |
|
| 113 | + $this->_form_section, |
|
| 114 | + $subsection |
|
| 115 | + ); |
|
| 116 | + } |
|
| 117 | + } |
|
| 118 | + return $html; |
|
| 119 | + } |
|
| 120 | + |
|
| 121 | + |
|
| 122 | + |
|
| 123 | + /** |
|
| 124 | + * Should be used to start teh form section (Eg a table tag, or a div tag, etc.) |
|
| 125 | + * |
|
| 126 | + * @return string |
|
| 127 | + */ |
|
| 128 | + abstract public function layout_form_begin(); |
|
| 129 | + |
|
| 130 | + |
|
| 131 | + |
|
| 132 | + /** |
|
| 133 | + * Should be used to end the form section (eg a /table tag, or a /div tag, etc) |
|
| 134 | + * |
|
| 135 | + * @return string |
|
| 136 | + */ |
|
| 137 | + abstract public function layout_form_end(); |
|
| 138 | + |
|
| 139 | + |
|
| 140 | + |
|
| 141 | + /** |
|
| 142 | + * Should be used internally by layout_form() to layout each input (eg, if this layout |
|
| 143 | + * is putting each input in a row of its own, this should probably be called by a |
|
| 144 | + * foreach loop in layout_form() (WITHOUT adding any content directly within layout_form()'s foreach loop. |
|
| 145 | + * Eg, this method should add the tr and td tags). This method is exposed in case you want to completely |
|
| 146 | + * customize the form's layout, but would like to make use of it for laying out |
|
| 147 | + * 'easy-to-layout' inputs |
|
| 148 | + * |
|
| 149 | + * @param EE_Form_Input_Base $input |
|
| 150 | + * @return string html |
|
| 151 | + */ |
|
| 152 | + abstract public function layout_input($input); |
|
| 153 | + |
|
| 154 | + |
|
| 155 | + |
|
| 156 | + /** |
|
| 157 | + * Similar to layout_input(), should be used internally by layout_form() within a |
|
| 158 | + * loop to layout each proper subsection. Unlike layout_input(), however, it is assumed |
|
| 159 | + * that the proper subsection will layout its container, label, etc on its own. |
|
| 160 | + * |
|
| 161 | + * @param EE_Form_Section_Base $subsection |
|
| 162 | + * @return string html |
|
| 163 | + */ |
|
| 164 | + abstract public function layout_subsection($subsection); |
|
| 165 | + |
|
| 166 | + |
|
| 167 | + |
|
| 168 | + /** |
|
| 169 | + * Gets the HTML for the label tag and its contents for the input |
|
| 170 | + * |
|
| 171 | + * @param EE_Form_Input_Base $input |
|
| 172 | + * @return string |
|
| 173 | + */ |
|
| 174 | + public function display_label($input) |
|
| 175 | + { |
|
| 176 | + if ($input->get_display_strategy() instanceof EE_Hidden_Display_Strategy) { |
|
| 177 | + return ''; |
|
| 178 | + } |
|
| 179 | + $class = $input->required() |
|
| 180 | + ? 'ee-required-label ' . $input->html_label_class() |
|
| 181 | + : $input->html_label_class(); |
|
| 182 | + $label_text = $input->required() |
|
| 183 | + ? $input->html_label_text() . '<span class="ee-asterisk">*</span>' |
|
| 184 | + : $input->html_label_text(); |
|
| 185 | + return '<label id="' |
|
| 186 | + . $input->html_label_id() |
|
| 187 | + . '" class="' |
|
| 188 | + . $class |
|
| 189 | + . '" style="' |
|
| 190 | + . $input->html_label_style() |
|
| 191 | + . '" for="' . $input->html_id() |
|
| 192 | + . '">' |
|
| 193 | + . $label_text |
|
| 194 | + . '</label>'; |
|
| 195 | + } |
|
| 196 | + |
|
| 197 | + |
|
| 198 | + |
|
| 199 | + /** |
|
| 200 | + * Gets the HTML for all the form's form-wide errors (ie, errors which |
|
| 201 | + * are not for specific inputs. E.g., if two inputs somehow disagree, |
|
| 202 | + * those errors would probably be on the form section, not one of its inputs) |
|
| 203 | + * @return string |
|
| 204 | + */ |
|
| 205 | + public function display_form_wide_errors() |
|
| 206 | + { |
|
| 207 | + $html = ''; |
|
| 208 | + if ($this->_form_section->get_validation_errors()) { |
|
| 209 | + $html .= "<div class='ee-form-wide-errors'>"; |
|
| 210 | + // get all the errors on THIS form section (errors which aren't |
|
| 211 | + // for specific inputs, but instead for the entire form section) |
|
| 212 | + foreach ($this->_form_section->get_validation_errors() as $error) { |
|
| 213 | + $html .= $error->getMessage() . '<br>'; |
|
| 214 | + } |
|
| 215 | + $html .= '</div>'; |
|
| 216 | + } |
|
| 217 | + return apply_filters( |
|
| 218 | + 'FHEE__EE_Form_Section_Layout_Base__display_form_wide_errors', |
|
| 219 | + $html, |
|
| 220 | + $this |
|
| 221 | + ); |
|
| 222 | + } |
|
| 223 | + |
|
| 224 | + |
|
| 225 | + |
|
| 226 | + /** |
|
| 227 | + * returns the HTML for the server-side validation errors for the specified input |
|
| 228 | + * Note that if JS is enabled, it should remove these and instead |
|
| 229 | + * populate the form's errors in the jquery validate fashion |
|
| 230 | + * using the localized data provided to the JS |
|
| 231 | + * |
|
| 232 | + * @param EE_Form_Input_Base $input |
|
| 233 | + * @return string |
|
| 234 | + */ |
|
| 235 | + public function display_errors($input) |
|
| 236 | + { |
|
| 237 | + if ($input->get_validation_errors()) { |
|
| 238 | + return "<label id='" |
|
| 239 | + . $input->html_id() |
|
| 240 | + . "-error' class='error' for='{$input->html_name()}'>" |
|
| 241 | + . $input->get_validation_error_string() |
|
| 242 | + . '</label>'; |
|
| 243 | + } |
|
| 244 | + return ''; |
|
| 245 | + } |
|
| 246 | + |
|
| 247 | + |
|
| 248 | + |
|
| 249 | + /** |
|
| 250 | + * Displays the help span for the specified input |
|
| 251 | + * |
|
| 252 | + * @param EE_Form_Input_Base $input |
|
| 253 | + * @return string |
|
| 254 | + */ |
|
| 255 | + public function display_help_text($input) |
|
| 256 | + { |
|
| 257 | + $help_text = $input->html_help_text(); |
|
| 258 | + if ($help_text !== '' && $help_text !== null) { |
|
| 259 | + $tag = is_admin() ? 'p' : 'span'; |
|
| 260 | + return '<' |
|
| 261 | + . $tag |
|
| 262 | + . ' id="' |
|
| 263 | + . $input->html_id() |
|
| 264 | + . '-help" class="' |
|
| 265 | + . $input->html_help_class() |
|
| 266 | + . '" style="' |
|
| 267 | + . $input->html_help_style() |
|
| 268 | + . '">' |
|
| 269 | + . $help_text |
|
| 270 | + . '</' |
|
| 271 | + . $tag |
|
| 272 | + . '>'; |
|
| 273 | + } |
|
| 274 | + return ''; |
|
| 275 | + } |
|
| 276 | + |
|
| 277 | + |
|
| 278 | + |
|
| 279 | + /** |
|
| 280 | + * Does an action and hook onto the end of teh form |
|
| 281 | + * |
|
| 282 | + * @param string $html |
|
| 283 | + * @return string |
|
| 284 | + */ |
|
| 285 | + public function add_form_section_hooks_and_filters($html) |
|
| 286 | + { |
|
| 287 | + // replace dashes and spaces with underscores |
|
| 288 | + $hook_name = str_replace(array('-', ' '), '_', $this->_form_section->html_id()); |
|
| 289 | + do_action('AHEE__Form_Section_Layout__' . $hook_name, $this->_form_section); |
|
| 290 | + $html = (string) apply_filters( |
|
| 291 | + 'AFEE__Form_Section_Layout__' . $hook_name . '__html', |
|
| 292 | + $html, |
|
| 293 | + $this->_form_section |
|
| 294 | + ); |
|
| 295 | + $html .= EEH_HTML::nl() . '<!-- AHEE__Form_Section_Layout__' . $hook_name . '__html -->'; |
|
| 296 | + $html .= EEH_HTML::nl() . '<!-- AFEE__Form_Section_Layout__' . $hook_name . ' -->'; |
|
| 297 | + return $html; |
|
| 298 | + } |
|
| 299 | 299 | } |
@@ -67,19 +67,19 @@ discard block |
||
| 67 | 67 | $html = ''; |
| 68 | 68 | // layout_form_begin |
| 69 | 69 | $html .= apply_filters( |
| 70 | - 'FHEE__EE_Form_Section_Layout_Base__layout_form__start__for_' . $this->_form_section->name(), |
|
| 70 | + 'FHEE__EE_Form_Section_Layout_Base__layout_form__start__for_'.$this->_form_section->name(), |
|
| 71 | 71 | $this->layout_form_begin(), |
| 72 | 72 | $this->_form_section |
| 73 | 73 | ); |
| 74 | 74 | // layout_form_loop |
| 75 | 75 | $html .= apply_filters( |
| 76 | - 'FHEE__EE_Form_Section_Layout_Base__layout_form__loop__for_' . $this->_form_section->name(), |
|
| 76 | + 'FHEE__EE_Form_Section_Layout_Base__layout_form__loop__for_'.$this->_form_section->name(), |
|
| 77 | 77 | $this->layout_form_loop(), |
| 78 | 78 | $this->_form_section |
| 79 | 79 | ); |
| 80 | 80 | // layout_form_end |
| 81 | 81 | $html .= apply_filters( |
| 82 | - 'FHEE__EE_Form_Section_Layout_Base__layout_form__end__for_' . $this->_form_section->name(), |
|
| 82 | + 'FHEE__EE_Form_Section_Layout_Base__layout_form__end__for_'.$this->_form_section->name(), |
|
| 83 | 83 | $this->layout_form_end(), |
| 84 | 84 | $this->_form_section |
| 85 | 85 | ); |
@@ -100,7 +100,7 @@ discard block |
||
| 100 | 100 | if ($subsection instanceof EE_Form_Input_Base) { |
| 101 | 101 | $html .= apply_filters( |
| 102 | 102 | 'FHEE__EE_Form_Section_Layout_Base__layout_form__loop_for_input_' |
| 103 | - . $name . '__in_' . $this->_form_section->name(), |
|
| 103 | + . $name.'__in_'.$this->_form_section->name(), |
|
| 104 | 104 | $this->layout_input($subsection), |
| 105 | 105 | $this->_form_section, |
| 106 | 106 | $subsection |
@@ -108,7 +108,7 @@ discard block |
||
| 108 | 108 | } elseif ($subsection instanceof EE_Form_Section_Base) { |
| 109 | 109 | $html .= apply_filters( |
| 110 | 110 | 'FHEE__EE_Form_Section_Layout_Base__layout_form__loop_for_non_input_' |
| 111 | - . $name . '__in_' . $this->_form_section->name(), |
|
| 111 | + . $name.'__in_'.$this->_form_section->name(), |
|
| 112 | 112 | $this->layout_subsection($subsection), |
| 113 | 113 | $this->_form_section, |
| 114 | 114 | $subsection |
@@ -177,10 +177,10 @@ discard block |
||
| 177 | 177 | return ''; |
| 178 | 178 | } |
| 179 | 179 | $class = $input->required() |
| 180 | - ? 'ee-required-label ' . $input->html_label_class() |
|
| 180 | + ? 'ee-required-label '.$input->html_label_class() |
|
| 181 | 181 | : $input->html_label_class(); |
| 182 | 182 | $label_text = $input->required() |
| 183 | - ? $input->html_label_text() . '<span class="ee-asterisk">*</span>' |
|
| 183 | + ? $input->html_label_text().'<span class="ee-asterisk">*</span>' |
|
| 184 | 184 | : $input->html_label_text(); |
| 185 | 185 | return '<label id="' |
| 186 | 186 | . $input->html_label_id() |
@@ -188,7 +188,7 @@ discard block |
||
| 188 | 188 | . $class |
| 189 | 189 | . '" style="' |
| 190 | 190 | . $input->html_label_style() |
| 191 | - . '" for="' . $input->html_id() |
|
| 191 | + . '" for="'.$input->html_id() |
|
| 192 | 192 | . '">' |
| 193 | 193 | . $label_text |
| 194 | 194 | . '</label>'; |
@@ -210,7 +210,7 @@ discard block |
||
| 210 | 210 | // get all the errors on THIS form section (errors which aren't |
| 211 | 211 | // for specific inputs, but instead for the entire form section) |
| 212 | 212 | foreach ($this->_form_section->get_validation_errors() as $error) { |
| 213 | - $html .= $error->getMessage() . '<br>'; |
|
| 213 | + $html .= $error->getMessage().'<br>'; |
|
| 214 | 214 | } |
| 215 | 215 | $html .= '</div>'; |
| 216 | 216 | } |
@@ -254,7 +254,7 @@ discard block |
||
| 254 | 254 | */ |
| 255 | 255 | public function display_help_text($input) |
| 256 | 256 | { |
| 257 | - $help_text = $input->html_help_text(); |
|
| 257 | + $help_text = $input->html_help_text(); |
|
| 258 | 258 | if ($help_text !== '' && $help_text !== null) { |
| 259 | 259 | $tag = is_admin() ? 'p' : 'span'; |
| 260 | 260 | return '<' |
@@ -286,14 +286,14 @@ discard block |
||
| 286 | 286 | { |
| 287 | 287 | // replace dashes and spaces with underscores |
| 288 | 288 | $hook_name = str_replace(array('-', ' '), '_', $this->_form_section->html_id()); |
| 289 | - do_action('AHEE__Form_Section_Layout__' . $hook_name, $this->_form_section); |
|
| 289 | + do_action('AHEE__Form_Section_Layout__'.$hook_name, $this->_form_section); |
|
| 290 | 290 | $html = (string) apply_filters( |
| 291 | - 'AFEE__Form_Section_Layout__' . $hook_name . '__html', |
|
| 291 | + 'AFEE__Form_Section_Layout__'.$hook_name.'__html', |
|
| 292 | 292 | $html, |
| 293 | 293 | $this->_form_section |
| 294 | 294 | ); |
| 295 | - $html .= EEH_HTML::nl() . '<!-- AHEE__Form_Section_Layout__' . $hook_name . '__html -->'; |
|
| 296 | - $html .= EEH_HTML::nl() . '<!-- AFEE__Form_Section_Layout__' . $hook_name . ' -->'; |
|
| 295 | + $html .= EEH_HTML::nl().'<!-- AHEE__Form_Section_Layout__'.$hook_name.'__html -->'; |
|
| 296 | + $html .= EEH_HTML::nl().'<!-- AFEE__Form_Section_Layout__'.$hook_name.' -->'; |
|
| 297 | 297 | return $html; |
| 298 | 298 | } |
| 299 | 299 | } |
@@ -45,1558 +45,1558 @@ |
||
| 45 | 45 | { |
| 46 | 46 | |
| 47 | 47 | |
| 48 | - /** |
|
| 49 | - * @var CalculatedModelFields |
|
| 50 | - */ |
|
| 51 | - protected $fields_calculator; |
|
| 52 | - |
|
| 53 | - |
|
| 54 | - /** |
|
| 55 | - * Read constructor. |
|
| 56 | - * @param CalculatedModelFields $fields_calculator |
|
| 57 | - */ |
|
| 58 | - public function __construct(CalculatedModelFields $fields_calculator) |
|
| 59 | - { |
|
| 60 | - parent::__construct(); |
|
| 61 | - $this->fields_calculator = $fields_calculator; |
|
| 62 | - } |
|
| 63 | - |
|
| 64 | - |
|
| 65 | - /** |
|
| 66 | - * Handles requests to get all (or a filtered subset) of entities for a particular model |
|
| 67 | - * |
|
| 68 | - * @param WP_REST_Request $request |
|
| 69 | - * @param string $version |
|
| 70 | - * @param string $model_name |
|
| 71 | - * @return WP_REST_Response|WP_Error |
|
| 72 | - * @throws InvalidArgumentException |
|
| 73 | - * @throws InvalidDataTypeException |
|
| 74 | - * @throws InvalidInterfaceException |
|
| 75 | - */ |
|
| 76 | - public static function handleRequestGetAll(WP_REST_Request $request, $version, $model_name) |
|
| 77 | - { |
|
| 78 | - $controller = LoaderFactory::getLoader()->getNew('EventEspresso\core\libraries\rest_api\controllers\model\Read'); |
|
| 79 | - try { |
|
| 80 | - $controller->setRequestedVersion($version); |
|
| 81 | - if (! $controller->getModelVersionInfo()->isModelNameInThisVersion($model_name)) { |
|
| 82 | - return $controller->sendResponse( |
|
| 83 | - new WP_Error( |
|
| 84 | - 'endpoint_parsing_error', |
|
| 85 | - sprintf( |
|
| 86 | - __( |
|
| 87 | - 'There is no model for endpoint %s. Please contact event espresso support', |
|
| 88 | - 'event_espresso' |
|
| 89 | - ), |
|
| 90 | - $model_name |
|
| 91 | - ) |
|
| 92 | - ) |
|
| 93 | - ); |
|
| 94 | - } |
|
| 95 | - return $controller->sendResponse( |
|
| 96 | - $controller->getEntitiesFromModel( |
|
| 97 | - $controller->getModelVersionInfo()->loadModel($model_name), |
|
| 98 | - $request |
|
| 99 | - ) |
|
| 100 | - ); |
|
| 101 | - } catch (Exception $e) { |
|
| 102 | - return $controller->sendResponse($e); |
|
| 103 | - } |
|
| 104 | - } |
|
| 105 | - |
|
| 106 | - |
|
| 107 | - /** |
|
| 108 | - * Prepares and returns schema for any OPTIONS request. |
|
| 109 | - * |
|
| 110 | - * @param string $version The API endpoint version being used. |
|
| 111 | - * @param string $model_name Something like `Event` or `Registration` |
|
| 112 | - * @return array |
|
| 113 | - * @throws InvalidArgumentException |
|
| 114 | - * @throws InvalidDataTypeException |
|
| 115 | - * @throws InvalidInterfaceException |
|
| 116 | - */ |
|
| 117 | - public static function handleSchemaRequest($version, $model_name) |
|
| 118 | - { |
|
| 119 | - $controller = LoaderFactory::getLoader()->getNew('EventEspresso\core\libraries\rest_api\controllers\model\Read'); |
|
| 120 | - try { |
|
| 121 | - $controller->setRequestedVersion($version); |
|
| 122 | - if (! $controller->getModelVersionInfo()->isModelNameInThisVersion($model_name)) { |
|
| 123 | - return array(); |
|
| 124 | - } |
|
| 125 | - // get the model for this version |
|
| 126 | - $model = $controller->getModelVersionInfo()->loadModel($model_name); |
|
| 127 | - $model_schema = new JsonModelSchema($model, LoaderFactory::getLoader()->getShared('EventEspresso\core\libraries\rest_api\CalculatedModelFields')); |
|
| 128 | - return $model_schema->getModelSchemaForRelations( |
|
| 129 | - $controller->getModelVersionInfo()->relationSettings($model), |
|
| 130 | - $controller->customizeSchemaForRestResponse( |
|
| 131 | - $model, |
|
| 132 | - $model_schema->getModelSchemaForFields( |
|
| 133 | - $controller->getModelVersionInfo()->fieldsOnModelInThisVersion($model), |
|
| 134 | - $model_schema->getInitialSchemaStructure() |
|
| 135 | - ) |
|
| 136 | - ) |
|
| 137 | - ); |
|
| 138 | - } catch (Exception $e) { |
|
| 139 | - return array(); |
|
| 140 | - } |
|
| 141 | - } |
|
| 142 | - |
|
| 143 | - |
|
| 144 | - /** |
|
| 145 | - * This loops through each field in the given schema for the model and does the following: |
|
| 146 | - * - add any extra fields that are REST API specific and related to existing fields. |
|
| 147 | - * - transform default values into the correct format for a REST API response. |
|
| 148 | - * |
|
| 149 | - * @param EEM_Base $model |
|
| 150 | - * @param array $schema |
|
| 151 | - * @return array The final schema. |
|
| 152 | - */ |
|
| 153 | - protected function customizeSchemaForRestResponse(EEM_Base $model, array $schema) |
|
| 154 | - { |
|
| 155 | - foreach ($this->getModelVersionInfo()->fieldsOnModelInThisVersion($model) as $field_name => $field) { |
|
| 156 | - $schema = $this->translateDefaultsForRestResponse( |
|
| 157 | - $field_name, |
|
| 158 | - $field, |
|
| 159 | - $this->maybeAddExtraFieldsToSchema($field_name, $field, $schema) |
|
| 160 | - ); |
|
| 161 | - } |
|
| 162 | - return $schema; |
|
| 163 | - } |
|
| 164 | - |
|
| 165 | - |
|
| 166 | - /** |
|
| 167 | - * This is used to ensure that the 'default' value set in the schema response is formatted correctly for the REST |
|
| 168 | - * response. |
|
| 169 | - * |
|
| 170 | - * @param $field_name |
|
| 171 | - * @param EE_Model_Field_Base $field |
|
| 172 | - * @param array $schema |
|
| 173 | - * @return array |
|
| 174 | - * @throws ObjectDetectedException if a default value has a PHP object, which should never do (and if we |
|
| 175 | - * did, let's know about it ASAP, so let the exception bubble up) |
|
| 176 | - */ |
|
| 177 | - protected function translateDefaultsForRestResponse($field_name, EE_Model_Field_Base $field, array $schema) |
|
| 178 | - { |
|
| 179 | - if (isset($schema['properties'][ $field_name ]['default'])) { |
|
| 180 | - if (is_array($schema['properties'][ $field_name ]['default'])) { |
|
| 181 | - foreach ($schema['properties'][ $field_name ]['default'] as $default_key => $default_value) { |
|
| 182 | - if ($default_key === 'raw') { |
|
| 183 | - $schema['properties'][ $field_name ]['default'][ $default_key ] = |
|
| 184 | - ModelDataTranslator::prepareFieldValueForJson( |
|
| 185 | - $field, |
|
| 186 | - $default_value, |
|
| 187 | - $this->getModelVersionInfo()->requestedVersion() |
|
| 188 | - ); |
|
| 189 | - } |
|
| 190 | - } |
|
| 191 | - } else { |
|
| 192 | - $schema['properties'][ $field_name ]['default'] = ModelDataTranslator::prepareFieldValueForJson( |
|
| 193 | - $field, |
|
| 194 | - $schema['properties'][ $field_name ]['default'], |
|
| 195 | - $this->getModelVersionInfo()->requestedVersion() |
|
| 196 | - ); |
|
| 197 | - } |
|
| 198 | - } |
|
| 199 | - return $schema; |
|
| 200 | - } |
|
| 201 | - |
|
| 202 | - |
|
| 203 | - /** |
|
| 204 | - * Adds additional fields to the schema |
|
| 205 | - * The REST API returns a GMT value field for each datetime field in the resource. Thus the description about this |
|
| 206 | - * needs to be added to the schema. |
|
| 207 | - * |
|
| 208 | - * @param $field_name |
|
| 209 | - * @param EE_Model_Field_Base $field |
|
| 210 | - * @param array $schema |
|
| 211 | - * @return array |
|
| 212 | - */ |
|
| 213 | - protected function maybeAddExtraFieldsToSchema($field_name, EE_Model_Field_Base $field, array $schema) |
|
| 214 | - { |
|
| 215 | - if ($field instanceof EE_Datetime_Field) { |
|
| 216 | - $schema['properties'][ $field_name . '_gmt' ] = $field->getSchema(); |
|
| 217 | - // modify the description |
|
| 218 | - $schema['properties'][ $field_name . '_gmt' ]['description'] = sprintf( |
|
| 219 | - esc_html__('%s - the value for this field is in GMT.', 'event_espresso'), |
|
| 220 | - wp_specialchars_decode($field->get_nicename(), ENT_QUOTES) |
|
| 221 | - ); |
|
| 222 | - } |
|
| 223 | - return $schema; |
|
| 224 | - } |
|
| 225 | - |
|
| 226 | - |
|
| 227 | - /** |
|
| 228 | - * Used to figure out the route from the request when a `WP_REST_Request` object is not available |
|
| 229 | - * |
|
| 230 | - * @return string |
|
| 231 | - */ |
|
| 232 | - protected function getRouteFromRequest() |
|
| 233 | - { |
|
| 234 | - if (isset($GLOBALS['wp']) |
|
| 235 | - && $GLOBALS['wp'] instanceof \WP |
|
| 236 | - && isset($GLOBALS['wp']->query_vars['rest_route']) |
|
| 237 | - ) { |
|
| 238 | - return $GLOBALS['wp']->query_vars['rest_route']; |
|
| 239 | - } else { |
|
| 240 | - return isset($_SERVER['PATH_INFO']) ? $_SERVER['PATH_INFO'] : '/'; |
|
| 241 | - } |
|
| 242 | - } |
|
| 243 | - |
|
| 244 | - |
|
| 245 | - /** |
|
| 246 | - * Gets a single entity related to the model indicated in the path and its id |
|
| 247 | - * |
|
| 248 | - * @param WP_REST_Request $request |
|
| 249 | - * @param string $version |
|
| 250 | - * @param string $model_name |
|
| 251 | - * @return WP_REST_Response|WP_Error |
|
| 252 | - * @throws InvalidDataTypeException |
|
| 253 | - * @throws InvalidInterfaceException |
|
| 254 | - * @throws InvalidArgumentException |
|
| 255 | - */ |
|
| 256 | - public static function handleRequestGetOne(WP_REST_Request $request, $version, $model_name) |
|
| 257 | - { |
|
| 258 | - $controller = LoaderFactory::getLoader()->getNew('EventEspresso\core\libraries\rest_api\controllers\model\Read'); |
|
| 259 | - try { |
|
| 260 | - $controller->setRequestedVersion($version); |
|
| 261 | - if (! $controller->getModelVersionInfo()->isModelNameInThisVersion($model_name)) { |
|
| 262 | - return $controller->sendResponse( |
|
| 263 | - new WP_Error( |
|
| 264 | - 'endpoint_parsing_error', |
|
| 265 | - sprintf( |
|
| 266 | - __( |
|
| 267 | - 'There is no model for endpoint %s. Please contact event espresso support', |
|
| 268 | - 'event_espresso' |
|
| 269 | - ), |
|
| 270 | - $model_name |
|
| 271 | - ) |
|
| 272 | - ) |
|
| 273 | - ); |
|
| 274 | - } |
|
| 275 | - return $controller->sendResponse( |
|
| 276 | - $controller->getEntityFromModel( |
|
| 277 | - $controller->getModelVersionInfo()->loadModel($model_name), |
|
| 278 | - $request |
|
| 279 | - ) |
|
| 280 | - ); |
|
| 281 | - } catch (Exception $e) { |
|
| 282 | - return $controller->sendResponse($e); |
|
| 283 | - } |
|
| 284 | - } |
|
| 285 | - |
|
| 286 | - |
|
| 287 | - /** |
|
| 288 | - * Gets all the related entities (or if its a belongs-to relation just the one) |
|
| 289 | - * to the item with the given id |
|
| 290 | - * |
|
| 291 | - * @param WP_REST_Request $request |
|
| 292 | - * @param string $version |
|
| 293 | - * @param string $model_name |
|
| 294 | - * @param string $related_model_name |
|
| 295 | - * @return WP_REST_Response|WP_Error |
|
| 296 | - * @throws InvalidDataTypeException |
|
| 297 | - * @throws InvalidInterfaceException |
|
| 298 | - * @throws InvalidArgumentException |
|
| 299 | - */ |
|
| 300 | - public static function handleRequestGetRelated( |
|
| 301 | - WP_REST_Request $request, |
|
| 302 | - $version, |
|
| 303 | - $model_name, |
|
| 304 | - $related_model_name |
|
| 305 | - ) { |
|
| 306 | - $controller = LoaderFactory::getLoader()->getNew('EventEspresso\core\libraries\rest_api\controllers\model\Read'); |
|
| 307 | - try { |
|
| 308 | - $controller->setRequestedVersion($version); |
|
| 309 | - $main_model = $controller->validateModel($model_name); |
|
| 310 | - $controller->validateModel($related_model_name); |
|
| 311 | - return $controller->sendResponse( |
|
| 312 | - $controller->getEntitiesFromRelation( |
|
| 313 | - $request->get_param('id'), |
|
| 314 | - $main_model->related_settings_for($related_model_name), |
|
| 315 | - $request |
|
| 316 | - ) |
|
| 317 | - ); |
|
| 318 | - } catch (Exception $e) { |
|
| 319 | - return $controller->sendResponse($e); |
|
| 320 | - } |
|
| 321 | - } |
|
| 322 | - |
|
| 323 | - |
|
| 324 | - /** |
|
| 325 | - * Gets a collection for the given model and filters |
|
| 326 | - * |
|
| 327 | - * @param EEM_Base $model |
|
| 328 | - * @param WP_REST_Request $request |
|
| 329 | - * @return array |
|
| 330 | - * @throws EE_Error |
|
| 331 | - * @throws InvalidArgumentException |
|
| 332 | - * @throws InvalidDataTypeException |
|
| 333 | - * @throws InvalidInterfaceException |
|
| 334 | - * @throws ReflectionException |
|
| 335 | - * @throws RestException |
|
| 336 | - */ |
|
| 337 | - public function getEntitiesFromModel($model, $request) |
|
| 338 | - { |
|
| 339 | - $query_params = $this->createModelQueryParams($model, $request->get_params()); |
|
| 340 | - if (! Capabilities::currentUserHasPartialAccessTo($model, $query_params['caps'])) { |
|
| 341 | - $model_name_plural = EEH_Inflector::pluralize_and_lower($model->get_this_model_name()); |
|
| 342 | - throw new RestException( |
|
| 343 | - sprintf('rest_%s_cannot_list', $model_name_plural), |
|
| 344 | - sprintf( |
|
| 345 | - __('Sorry, you are not allowed to list %1$s. Missing permissions: %2$s', 'event_espresso'), |
|
| 346 | - $model_name_plural, |
|
| 347 | - Capabilities::getMissingPermissionsString($model, $query_params['caps']) |
|
| 348 | - ), |
|
| 349 | - array('status' => 403) |
|
| 350 | - ); |
|
| 351 | - } |
|
| 352 | - if (! $request->get_header('no_rest_headers')) { |
|
| 353 | - $this->setHeadersFromQueryParams($model, $query_params); |
|
| 354 | - } |
|
| 355 | - /** @type array $results */ |
|
| 356 | - $results = $model->get_all_wpdb_results($query_params); |
|
| 357 | - $nice_results = array(); |
|
| 358 | - foreach ($results as $result) { |
|
| 359 | - $nice_results[] = $this->createEntityFromWpdbResult( |
|
| 360 | - $model, |
|
| 361 | - $result, |
|
| 362 | - $request |
|
| 363 | - ); |
|
| 364 | - } |
|
| 365 | - return $nice_results; |
|
| 366 | - } |
|
| 367 | - |
|
| 368 | - |
|
| 369 | - /** |
|
| 370 | - * Gets the collection for given relation object |
|
| 371 | - * The same as Read::get_entities_from_model(), except if the relation |
|
| 372 | - * is a HABTM relation, in which case it merges any non-foreign-key fields from |
|
| 373 | - * the join-model-object into the results |
|
| 374 | - * |
|
| 375 | - * @param array $primary_model_query_params query params for finding the item from which |
|
| 376 | - * relations will be based |
|
| 377 | - * @param \EE_Model_Relation_Base $relation |
|
| 378 | - * @param WP_REST_Request $request |
|
| 379 | - * @return array |
|
| 380 | - * @throws EE_Error |
|
| 381 | - * @throws InvalidArgumentException |
|
| 382 | - * @throws InvalidDataTypeException |
|
| 383 | - * @throws InvalidInterfaceException |
|
| 384 | - * @throws ReflectionException |
|
| 385 | - * @throws RestException |
|
| 386 | - * @throws \EventEspresso\core\exceptions\ModelConfigurationException |
|
| 387 | - */ |
|
| 388 | - protected function getEntitiesFromRelationUsingModelQueryParams($primary_model_query_params, $relation, $request) |
|
| 389 | - { |
|
| 390 | - $context = $this->validateContext($request->get_param('caps')); |
|
| 391 | - $model = $relation->get_this_model(); |
|
| 392 | - $related_model = $relation->get_other_model(); |
|
| 393 | - if (! isset($primary_model_query_params[0])) { |
|
| 394 | - $primary_model_query_params[0] = array(); |
|
| 395 | - } |
|
| 396 | - // check if they can access the 1st model object |
|
| 397 | - $primary_model_query_params = array( |
|
| 398 | - 0 => $primary_model_query_params[0], |
|
| 399 | - 'limit' => 1, |
|
| 400 | - ); |
|
| 401 | - if ($model instanceof EEM_Soft_Delete_Base) { |
|
| 402 | - $primary_model_query_params = $model->alter_query_params_so_deleted_and_undeleted_items_included( |
|
| 403 | - $primary_model_query_params |
|
| 404 | - ); |
|
| 405 | - } |
|
| 406 | - $restricted_query_params = $primary_model_query_params; |
|
| 407 | - $restricted_query_params['caps'] = $context; |
|
| 408 | - $restricted_query_params['limit'] = 1; |
|
| 409 | - $this->setDebugInfo('main model query params', $restricted_query_params); |
|
| 410 | - $this->setDebugInfo('missing caps', Capabilities::getMissingPermissionsString($related_model, $context)); |
|
| 411 | - $primary_model_rows = $model->get_all_wpdb_results($restricted_query_params); |
|
| 412 | - $primary_model_row = null; |
|
| 413 | - if (is_array($primary_model_rows)) { |
|
| 414 | - $primary_model_row = reset($primary_model_rows); |
|
| 415 | - } |
|
| 416 | - if (! ( |
|
| 417 | - Capabilities::currentUserHasPartialAccessTo($related_model, $context) |
|
| 418 | - && $primary_model_row |
|
| 419 | - ) |
|
| 420 | - ) { |
|
| 421 | - if ($relation instanceof EE_Belongs_To_Relation) { |
|
| 422 | - $related_model_name_maybe_plural = strtolower($related_model->get_this_model_name()); |
|
| 423 | - } else { |
|
| 424 | - $related_model_name_maybe_plural = EEH_Inflector::pluralize_and_lower( |
|
| 425 | - $related_model->get_this_model_name() |
|
| 426 | - ); |
|
| 427 | - } |
|
| 428 | - throw new RestException( |
|
| 429 | - sprintf('rest_%s_cannot_list', $related_model_name_maybe_plural), |
|
| 430 | - sprintf( |
|
| 431 | - __( |
|
| 432 | - 'Sorry, you are not allowed to list %1$s related to %2$s. Missing permissions: %3$s', |
|
| 433 | - 'event_espresso' |
|
| 434 | - ), |
|
| 435 | - $related_model_name_maybe_plural, |
|
| 436 | - $relation->get_this_model()->get_this_model_name(), |
|
| 437 | - implode( |
|
| 438 | - ',', |
|
| 439 | - array_keys( |
|
| 440 | - Capabilities::getMissingPermissions($related_model, $context) |
|
| 441 | - ) |
|
| 442 | - ) |
|
| 443 | - ), |
|
| 444 | - array('status' => 403) |
|
| 445 | - ); |
|
| 446 | - } |
|
| 447 | - |
|
| 448 | - $this->checkPassword( |
|
| 449 | - $model, |
|
| 450 | - $primary_model_row, |
|
| 451 | - $restricted_query_params, |
|
| 452 | - $request |
|
| 453 | - ); |
|
| 454 | - $query_params = $this->createModelQueryParams($relation->get_other_model(), $request->get_params()); |
|
| 455 | - foreach ($primary_model_query_params[0] as $where_condition_key => $where_condition_value) { |
|
| 456 | - $query_params[0][ $relation->get_this_model()->get_this_model_name() |
|
| 457 | - . '.' |
|
| 458 | - . $where_condition_key ] = $where_condition_value; |
|
| 459 | - } |
|
| 460 | - $query_params['default_where_conditions'] = 'none'; |
|
| 461 | - $query_params['caps'] = $context; |
|
| 462 | - if (! $request->get_header('no_rest_headers')) { |
|
| 463 | - $this->setHeadersFromQueryParams($relation->get_other_model(), $query_params); |
|
| 464 | - } |
|
| 465 | - /** @type array $results */ |
|
| 466 | - $results = $relation->get_other_model()->get_all_wpdb_results($query_params); |
|
| 467 | - $nice_results = array(); |
|
| 468 | - foreach ($results as $result) { |
|
| 469 | - $nice_result = $this->createEntityFromWpdbResult( |
|
| 470 | - $relation->get_other_model(), |
|
| 471 | - $result, |
|
| 472 | - $request |
|
| 473 | - ); |
|
| 474 | - if ($relation instanceof \EE_HABTM_Relation) { |
|
| 475 | - // put the unusual stuff (properties from the HABTM relation) first, and make sure |
|
| 476 | - // if there are conflicts we prefer the properties from the main model |
|
| 477 | - $join_model_result = $this->createEntityFromWpdbResult( |
|
| 478 | - $relation->get_join_model(), |
|
| 479 | - $result, |
|
| 480 | - $request |
|
| 481 | - ); |
|
| 482 | - $joined_result = array_merge($join_model_result, $nice_result); |
|
| 483 | - // but keep the meta stuff from the main model |
|
| 484 | - if (isset($nice_result['meta'])) { |
|
| 485 | - $joined_result['meta'] = $nice_result['meta']; |
|
| 486 | - } |
|
| 487 | - $nice_result = $joined_result; |
|
| 488 | - } |
|
| 489 | - $nice_results[] = $nice_result; |
|
| 490 | - } |
|
| 491 | - if ($relation instanceof EE_Belongs_To_Relation) { |
|
| 492 | - return array_shift($nice_results); |
|
| 493 | - } else { |
|
| 494 | - return $nice_results; |
|
| 495 | - } |
|
| 496 | - } |
|
| 497 | - |
|
| 498 | - |
|
| 499 | - /** |
|
| 500 | - * Gets the collection for given relation object |
|
| 501 | - * The same as Read::get_entities_from_model(), except if the relation |
|
| 502 | - * is a HABTM relation, in which case it merges any non-foreign-key fields from |
|
| 503 | - * the join-model-object into the results |
|
| 504 | - * |
|
| 505 | - * @param string $id the ID of the thing we are fetching related stuff from |
|
| 506 | - * @param \EE_Model_Relation_Base $relation |
|
| 507 | - * @param WP_REST_Request $request |
|
| 508 | - * @return array |
|
| 509 | - * @throws EE_Error |
|
| 510 | - */ |
|
| 511 | - public function getEntitiesFromRelation($id, $relation, $request) |
|
| 512 | - { |
|
| 513 | - if (! $relation->get_this_model()->has_primary_key_field()) { |
|
| 514 | - throw new EE_Error( |
|
| 515 | - sprintf( |
|
| 516 | - __( |
|
| 517 | - // @codingStandardsIgnoreStart |
|
| 518 | - 'Read::get_entities_from_relation should only be called from a model with a primary key, it was called from %1$s', |
|
| 519 | - // @codingStandardsIgnoreEnd |
|
| 520 | - 'event_espresso' |
|
| 521 | - ), |
|
| 522 | - $relation->get_this_model()->get_this_model_name() |
|
| 523 | - ) |
|
| 524 | - ); |
|
| 525 | - } |
|
| 526 | - // can we edit that main item? |
|
| 527 | - // if not, show nothing but an error |
|
| 528 | - // otherwise, please proceed |
|
| 529 | - return $this->getEntitiesFromRelationUsingModelQueryParams( |
|
| 530 | - array( |
|
| 531 | - array( |
|
| 532 | - $relation->get_this_model()->primary_key_name() => $id, |
|
| 533 | - ), |
|
| 534 | - ), |
|
| 535 | - $relation, |
|
| 536 | - $request |
|
| 537 | - ); |
|
| 538 | - } |
|
| 539 | - |
|
| 540 | - |
|
| 541 | - /** |
|
| 542 | - * Sets the headers that are based on the model and query params, |
|
| 543 | - * like the total records. This should only be called on the original request |
|
| 544 | - * from the client, not on subsequent internal |
|
| 545 | - * |
|
| 546 | - * @param EEM_Base $model |
|
| 547 | - * @param array $query_params |
|
| 548 | - * @return void |
|
| 549 | - */ |
|
| 550 | - protected function setHeadersFromQueryParams($model, $query_params) |
|
| 551 | - { |
|
| 552 | - $this->setDebugInfo('model query params', $query_params); |
|
| 553 | - $this->setDebugInfo( |
|
| 554 | - 'missing caps', |
|
| 555 | - Capabilities::getMissingPermissionsString($model, $query_params['caps']) |
|
| 556 | - ); |
|
| 557 | - // normally the limit to a 2-part array, where the 2nd item is the limit |
|
| 558 | - if (! isset($query_params['limit'])) { |
|
| 559 | - $query_params['limit'] = EED_Core_Rest_Api::get_default_query_limit(); |
|
| 560 | - } |
|
| 561 | - if (is_array($query_params['limit'])) { |
|
| 562 | - $limit_parts = $query_params['limit']; |
|
| 563 | - } else { |
|
| 564 | - $limit_parts = explode(',', $query_params['limit']); |
|
| 565 | - if (count($limit_parts) == 1) { |
|
| 566 | - $limit_parts = array(0, $limit_parts[0]); |
|
| 567 | - } |
|
| 568 | - } |
|
| 569 | - // remove the group by and having parts of the query, as those will |
|
| 570 | - // make the sql query return an array of values, instead of just a single value |
|
| 571 | - unset($query_params['group_by'], $query_params['having'], $query_params['limit']); |
|
| 572 | - $count = $model->count($query_params, null, true); |
|
| 573 | - $pages = $count / $limit_parts[1]; |
|
| 574 | - $this->setResponseHeader('Total', $count, false); |
|
| 575 | - $this->setResponseHeader('PageSize', $limit_parts[1], false); |
|
| 576 | - $this->setResponseHeader('TotalPages', ceil($pages), false); |
|
| 577 | - } |
|
| 578 | - |
|
| 579 | - |
|
| 580 | - /** |
|
| 581 | - * Changes database results into REST API entities |
|
| 582 | - * |
|
| 583 | - * @param EEM_Base $model |
|
| 584 | - * @param array $db_row like results from $wpdb->get_results() |
|
| 585 | - * @param WP_REST_Request $rest_request |
|
| 586 | - * @param string $deprecated no longer used |
|
| 587 | - * @return array ready for being converted into json for sending to client |
|
| 588 | - * @throws EE_Error |
|
| 589 | - * @throws RestException |
|
| 590 | - * @throws InvalidDataTypeException |
|
| 591 | - * @throws InvalidInterfaceException |
|
| 592 | - * @throws InvalidArgumentException |
|
| 593 | - * @throws ReflectionException |
|
| 594 | - */ |
|
| 595 | - public function createEntityFromWpdbResult($model, $db_row, $rest_request, $deprecated = null) |
|
| 596 | - { |
|
| 597 | - if (! $rest_request instanceof WP_REST_Request) { |
|
| 598 | - // ok so this was called in the old style, where the 3rd arg was |
|
| 599 | - // $include, and the 4th arg was $context |
|
| 600 | - // now setup the request just to avoid fatal errors, although we won't be able |
|
| 601 | - // to truly make use of it because it's kinda devoid of info |
|
| 602 | - $rest_request = new WP_REST_Request(); |
|
| 603 | - $rest_request->set_param('include', $rest_request); |
|
| 604 | - $rest_request->set_param('caps', $deprecated); |
|
| 605 | - } |
|
| 606 | - if ($rest_request->get_param('caps') == null) { |
|
| 607 | - $rest_request->set_param('caps', EEM_Base::caps_read); |
|
| 608 | - } |
|
| 609 | - $current_user_full_access_to_entity = $model->currentUserCan( |
|
| 610 | - EEM_Base::caps_read_admin, |
|
| 611 | - $model->deduce_fields_n_values_from_cols_n_values($db_row) |
|
| 612 | - ); |
|
| 613 | - $entity_array = $this->createBareEntityFromWpdbResults($model, $db_row); |
|
| 614 | - $entity_array = $this->addExtraFields($model, $db_row, $entity_array); |
|
| 615 | - $entity_array['_links'] = $this->getEntityLinks($model, $db_row, $entity_array); |
|
| 616 | - // when it's a regular read request for a model with a password and the password wasn't provided |
|
| 617 | - // remove the password protected fields |
|
| 618 | - $has_protected_fields = false; |
|
| 619 | - try { |
|
| 620 | - $this->checkPassword( |
|
| 621 | - $model, |
|
| 622 | - $db_row, |
|
| 623 | - $model->alter_query_params_to_restrict_by_ID( |
|
| 624 | - $model->get_index_primary_key_string( |
|
| 625 | - $model->deduce_fields_n_values_from_cols_n_values($db_row) |
|
| 626 | - ) |
|
| 627 | - ), |
|
| 628 | - $rest_request |
|
| 629 | - ); |
|
| 630 | - } catch (RestPasswordRequiredException $e) { |
|
| 631 | - if ($model->hasPassword()) { |
|
| 632 | - // just remove protected fields |
|
| 633 | - $has_protected_fields = true; |
|
| 634 | - $entity_array = Capabilities::filterOutPasswordProtectedFields( |
|
| 635 | - $entity_array, |
|
| 636 | - $model, |
|
| 637 | - $this->getModelVersionInfo() |
|
| 638 | - ); |
|
| 639 | - } else { |
|
| 640 | - // that's a problem. None of this should be accessible if no password was provided |
|
| 641 | - throw $e; |
|
| 642 | - } |
|
| 643 | - } |
|
| 644 | - |
|
| 645 | - $entity_array['_calculated_fields'] = $this->getEntityCalculations($model, $db_row, $rest_request, $has_protected_fields); |
|
| 646 | - $entity_array = apply_filters( |
|
| 647 | - 'FHEE__Read__create_entity_from_wpdb_results__entity_before_including_requested_models', |
|
| 648 | - $entity_array, |
|
| 649 | - $model, |
|
| 650 | - $rest_request->get_param('caps'), |
|
| 651 | - $rest_request, |
|
| 652 | - $this |
|
| 653 | - ); |
|
| 654 | - // add an empty protected property for now. If it's still around after we remove everything the request didn't |
|
| 655 | - // want, we'll populate it then. k? |
|
| 656 | - $entity_array['_protected'] = array(); |
|
| 657 | - // remove any properties the request didn't want. This way _protected won't bother mentioning them |
|
| 658 | - $entity_array = $this->includeOnlyRequestedProperties($model, $rest_request, $entity_array); |
|
| 659 | - $entity_array = $this->includeRequestedModels($model, $rest_request, $entity_array, $db_row, $has_protected_fields); |
|
| 660 | - // if they still wanted the _protected property, add it. |
|
| 661 | - if (isset($entity_array['_protected'])) { |
|
| 662 | - $entity_array = $this->addProtectedProperty($model, $entity_array, $has_protected_fields); |
|
| 663 | - } |
|
| 664 | - $entity_array = apply_filters( |
|
| 665 | - 'FHEE__Read__create_entity_from_wpdb_results__entity_before_inaccessible_field_removal', |
|
| 666 | - $entity_array, |
|
| 667 | - $model, |
|
| 668 | - $rest_request->get_param('caps'), |
|
| 669 | - $rest_request, |
|
| 670 | - $this |
|
| 671 | - ); |
|
| 672 | - if (! $current_user_full_access_to_entity) { |
|
| 673 | - $result_without_inaccessible_fields = Capabilities::filterOutInaccessibleEntityFields( |
|
| 674 | - $entity_array, |
|
| 675 | - $model, |
|
| 676 | - $rest_request->get_param('caps'), |
|
| 677 | - $this->getModelVersionInfo() |
|
| 678 | - ); |
|
| 679 | - } else { |
|
| 680 | - $result_without_inaccessible_fields = $entity_array; |
|
| 681 | - } |
|
| 682 | - $this->setDebugInfo( |
|
| 683 | - 'inaccessible fields', |
|
| 684 | - array_keys(array_diff_key((array) $entity_array, (array) $result_without_inaccessible_fields)) |
|
| 685 | - ); |
|
| 686 | - return apply_filters( |
|
| 687 | - 'FHEE__Read__create_entity_from_wpdb_results__entity_return', |
|
| 688 | - $result_without_inaccessible_fields, |
|
| 689 | - $model, |
|
| 690 | - $rest_request->get_param('caps') |
|
| 691 | - ); |
|
| 692 | - } |
|
| 693 | - |
|
| 694 | - /** |
|
| 695 | - * Returns an array describing which fields can be protected, and which actually were removed this request |
|
| 696 | - * @since 4.9.74.p |
|
| 697 | - * @param $model |
|
| 698 | - * @param $results_so_far |
|
| 699 | - * @param $protected |
|
| 700 | - * @return array results |
|
| 701 | - */ |
|
| 702 | - protected function addProtectedProperty(EEM_Base $model, $results_so_far, $protected) |
|
| 703 | - { |
|
| 704 | - if (! $model->hasPassword() || ! $protected) { |
|
| 705 | - return $results_so_far; |
|
| 706 | - } |
|
| 707 | - $password_field = $model->getPasswordField(); |
|
| 708 | - $all_protected = array_merge( |
|
| 709 | - array($password_field->get_name()), |
|
| 710 | - $password_field->protectedFields() |
|
| 711 | - ); |
|
| 712 | - $fields_included = array_keys($results_so_far); |
|
| 713 | - $fields_included = array_intersect( |
|
| 714 | - $all_protected, |
|
| 715 | - $fields_included |
|
| 716 | - ); |
|
| 717 | - foreach ($fields_included as $field_name) { |
|
| 718 | - $results_so_far['_protected'][] = $field_name ; |
|
| 719 | - } |
|
| 720 | - return $results_so_far; |
|
| 721 | - } |
|
| 722 | - |
|
| 723 | - /** |
|
| 724 | - * Creates a REST entity array (JSON object we're going to return in the response, but |
|
| 725 | - * for now still a PHP array, but soon enough we'll call json_encode on it, don't worry), |
|
| 726 | - * from $wpdb->get_row( $sql, ARRAY_A) |
|
| 727 | - * |
|
| 728 | - * @param EEM_Base $model |
|
| 729 | - * @param array $db_row |
|
| 730 | - * @return array entity mostly ready for converting to JSON and sending in the response |
|
| 731 | - */ |
|
| 732 | - protected function createBareEntityFromWpdbResults(EEM_Base $model, $db_row) |
|
| 733 | - { |
|
| 734 | - $result = $model->deduce_fields_n_values_from_cols_n_values($db_row); |
|
| 735 | - $result = array_intersect_key( |
|
| 736 | - $result, |
|
| 737 | - $this->getModelVersionInfo()->fieldsOnModelInThisVersion($model) |
|
| 738 | - ); |
|
| 739 | - // if this is a CPT, we need to set the global $post to it, |
|
| 740 | - // otherwise shortcodes etc won't work properly while rendering it |
|
| 741 | - if ($model instanceof \EEM_CPT_Base) { |
|
| 742 | - $do_chevy_shuffle = true; |
|
| 743 | - } else { |
|
| 744 | - $do_chevy_shuffle = false; |
|
| 745 | - } |
|
| 746 | - if ($do_chevy_shuffle) { |
|
| 747 | - global $post; |
|
| 748 | - $old_post = $post; |
|
| 749 | - $post = get_post($result[ $model->primary_key_name() ]); |
|
| 750 | - if (! $post instanceof \WP_Post) { |
|
| 751 | - // well that's weird, because $result is what we JUST fetched from the database |
|
| 752 | - throw new RestException( |
|
| 753 | - 'error_fetching_post_from_database_results', |
|
| 754 | - esc_html__( |
|
| 755 | - 'An item was retrieved from the database but it\'s not a WP_Post like it should be.', |
|
| 756 | - 'event_espresso' |
|
| 757 | - ) |
|
| 758 | - ); |
|
| 759 | - } |
|
| 760 | - $model_object_classname = 'EE_' . $model->get_this_model_name(); |
|
| 761 | - $post->{$model_object_classname} = \EE_Registry::instance()->load_class( |
|
| 762 | - $model_object_classname, |
|
| 763 | - $result, |
|
| 764 | - false, |
|
| 765 | - false |
|
| 766 | - ); |
|
| 767 | - } |
|
| 768 | - foreach ($result as $field_name => $field_value) { |
|
| 769 | - $field_obj = $model->field_settings_for($field_name); |
|
| 770 | - if ($this->isSubclassOfOne($field_obj, $this->getModelVersionInfo()->fieldsIgnored())) { |
|
| 771 | - unset($result[ $field_name ]); |
|
| 772 | - } elseif ($this->isSubclassOfOne( |
|
| 773 | - $field_obj, |
|
| 774 | - $this->getModelVersionInfo()->fieldsThatHaveRenderedFormat() |
|
| 775 | - ) |
|
| 776 | - ) { |
|
| 777 | - $result[ $field_name ] = array( |
|
| 778 | - 'raw' => $this->prepareFieldObjValueForJson($field_obj, $field_value), |
|
| 779 | - 'rendered' => $this->prepareFieldObjValueForJson($field_obj, $field_value, 'pretty'), |
|
| 780 | - ); |
|
| 781 | - } elseif ($this->isSubclassOfOne( |
|
| 782 | - $field_obj, |
|
| 783 | - $this->getModelVersionInfo()->fieldsThatHavePrettyFormat() |
|
| 784 | - ) |
|
| 785 | - ) { |
|
| 786 | - $result[ $field_name ] = array( |
|
| 787 | - 'raw' => $this->prepareFieldObjValueForJson($field_obj, $field_value), |
|
| 788 | - 'pretty' => $this->prepareFieldObjValueForJson($field_obj, $field_value, 'pretty'), |
|
| 789 | - ); |
|
| 790 | - } elseif ($field_obj instanceof \EE_Datetime_Field) { |
|
| 791 | - $field_value = $field_obj->prepare_for_set_from_db($field_value); |
|
| 792 | - // if the value is null, but we're not supposed to permit null, then set to the field's default |
|
| 793 | - if (is_null($field_value)) { |
|
| 794 | - $field_value = $field_obj->getDefaultDateTimeObj(); |
|
| 795 | - } |
|
| 796 | - if (is_null($field_value)) { |
|
| 797 | - $gmt_date = $local_date = ModelDataTranslator::prepareFieldValuesForJson( |
|
| 798 | - $field_obj, |
|
| 799 | - $field_value, |
|
| 800 | - $this->getModelVersionInfo()->requestedVersion() |
|
| 801 | - ); |
|
| 802 | - } else { |
|
| 803 | - $timezone = $field_value->getTimezone(); |
|
| 804 | - EEH_DTT_Helper::setTimezone($field_value, new DateTimeZone('UTC')); |
|
| 805 | - $gmt_date = ModelDataTranslator::prepareFieldValuesForJson( |
|
| 806 | - $field_obj, |
|
| 807 | - $field_value, |
|
| 808 | - $this->getModelVersionInfo()->requestedVersion() |
|
| 809 | - ); |
|
| 810 | - EEH_DTT_Helper::setTimezone($field_value, $timezone); |
|
| 811 | - $local_date = ModelDataTranslator::prepareFieldValuesForJson( |
|
| 812 | - $field_obj, |
|
| 813 | - $field_value, |
|
| 814 | - $this->getModelVersionInfo()->requestedVersion() |
|
| 815 | - ); |
|
| 816 | - } |
|
| 817 | - $result[ $field_name . '_gmt' ] = $gmt_date; |
|
| 818 | - $result[ $field_name ] = $local_date; |
|
| 819 | - } else { |
|
| 820 | - $result[ $field_name ] = $this->prepareFieldObjValueForJson($field_obj, $field_value); |
|
| 821 | - } |
|
| 822 | - } |
|
| 823 | - if ($do_chevy_shuffle) { |
|
| 824 | - $post = $old_post; |
|
| 825 | - } |
|
| 826 | - return $result; |
|
| 827 | - } |
|
| 828 | - |
|
| 829 | - |
|
| 830 | - /** |
|
| 831 | - * Takes a value all the way from the DB representation, to the model object's representation, to the |
|
| 832 | - * user-facing PHP representation, to the REST API representation. (Assumes you've already taken from the DB |
|
| 833 | - * representation using $field_obj->prepare_for_set_from_db()) |
|
| 834 | - * |
|
| 835 | - * @param EE_Model_Field_Base $field_obj |
|
| 836 | - * @param mixed $value as it's stored on a model object |
|
| 837 | - * @param string $format valid values are 'normal' (default), 'pretty', 'datetime_obj' |
|
| 838 | - * @return mixed |
|
| 839 | - * @throws ObjectDetectedException if $value contains a PHP object |
|
| 840 | - */ |
|
| 841 | - protected function prepareFieldObjValueForJson(EE_Model_Field_Base $field_obj, $value, $format = 'normal') |
|
| 842 | - { |
|
| 843 | - $value = $field_obj->prepare_for_set_from_db($value); |
|
| 844 | - switch ($format) { |
|
| 845 | - case 'pretty': |
|
| 846 | - $value = $field_obj->prepare_for_pretty_echoing($value); |
|
| 847 | - break; |
|
| 848 | - case 'normal': |
|
| 849 | - default: |
|
| 850 | - $value = $field_obj->prepare_for_get($value); |
|
| 851 | - break; |
|
| 852 | - } |
|
| 853 | - return ModelDataTranslator::prepareFieldValuesForJson( |
|
| 854 | - $field_obj, |
|
| 855 | - $value, |
|
| 856 | - $this->getModelVersionInfo()->requestedVersion() |
|
| 857 | - ); |
|
| 858 | - } |
|
| 859 | - |
|
| 860 | - |
|
| 861 | - /** |
|
| 862 | - * Adds a few extra fields to the entity response |
|
| 863 | - * |
|
| 864 | - * @param EEM_Base $model |
|
| 865 | - * @param array $db_row |
|
| 866 | - * @param array $entity_array |
|
| 867 | - * @return array modified entity |
|
| 868 | - */ |
|
| 869 | - protected function addExtraFields(EEM_Base $model, $db_row, $entity_array) |
|
| 870 | - { |
|
| 871 | - if ($model instanceof EEM_CPT_Base) { |
|
| 872 | - $entity_array['link'] = get_permalink($db_row[ $model->get_primary_key_field()->get_qualified_column() ]); |
|
| 873 | - } |
|
| 874 | - return $entity_array; |
|
| 875 | - } |
|
| 876 | - |
|
| 877 | - |
|
| 878 | - /** |
|
| 879 | - * Gets links we want to add to the response |
|
| 880 | - * |
|
| 881 | - * @global \WP_REST_Server $wp_rest_server |
|
| 882 | - * @param EEM_Base $model |
|
| 883 | - * @param array $db_row |
|
| 884 | - * @param array $entity_array |
|
| 885 | - * @return array the _links item in the entity |
|
| 886 | - */ |
|
| 887 | - protected function getEntityLinks($model, $db_row, $entity_array) |
|
| 888 | - { |
|
| 889 | - // add basic links |
|
| 890 | - $links = array(); |
|
| 891 | - if ($model->has_primary_key_field()) { |
|
| 892 | - $links['self'] = array( |
|
| 893 | - array( |
|
| 894 | - 'href' => $this->getVersionedLinkTo( |
|
| 895 | - EEH_Inflector::pluralize_and_lower($model->get_this_model_name()) |
|
| 896 | - . '/' |
|
| 897 | - . $entity_array[ $model->primary_key_name() ] |
|
| 898 | - ), |
|
| 899 | - ), |
|
| 900 | - ); |
|
| 901 | - } |
|
| 902 | - $links['collection'] = array( |
|
| 903 | - array( |
|
| 904 | - 'href' => $this->getVersionedLinkTo( |
|
| 905 | - EEH_Inflector::pluralize_and_lower($model->get_this_model_name()) |
|
| 906 | - ), |
|
| 907 | - ), |
|
| 908 | - ); |
|
| 909 | - // add links to related models |
|
| 910 | - if ($model->has_primary_key_field()) { |
|
| 911 | - foreach ($this->getModelVersionInfo()->relationSettings($model) as $relation_name => $relation_obj) { |
|
| 912 | - $related_model_part = Read::getRelatedEntityName($relation_name, $relation_obj); |
|
| 913 | - $links[ EED_Core_Rest_Api::ee_api_link_namespace . $related_model_part ] = array( |
|
| 914 | - array( |
|
| 915 | - 'href' => $this->getVersionedLinkTo( |
|
| 916 | - EEH_Inflector::pluralize_and_lower($model->get_this_model_name()) |
|
| 917 | - . '/' |
|
| 918 | - . $entity_array[ $model->primary_key_name() ] |
|
| 919 | - . '/' |
|
| 920 | - . $related_model_part |
|
| 921 | - ), |
|
| 922 | - 'single' => $relation_obj instanceof EE_Belongs_To_Relation ? true : false, |
|
| 923 | - ), |
|
| 924 | - ); |
|
| 925 | - } |
|
| 926 | - } |
|
| 927 | - return $links; |
|
| 928 | - } |
|
| 929 | - |
|
| 930 | - |
|
| 931 | - /** |
|
| 932 | - * Adds the included models indicated in the request to the entity provided |
|
| 933 | - * |
|
| 934 | - * @param EEM_Base $model |
|
| 935 | - * @param WP_REST_Request $rest_request |
|
| 936 | - * @param array $entity_array |
|
| 937 | - * @param array $db_row |
|
| 938 | - * @param boolean $included_items_protected if the original item is password protected, don't include any related models. |
|
| 939 | - * @return array the modified entity |
|
| 940 | - * @throws RestException |
|
| 941 | - */ |
|
| 942 | - protected function includeRequestedModels( |
|
| 943 | - EEM_Base $model, |
|
| 944 | - WP_REST_Request $rest_request, |
|
| 945 | - $entity_array, |
|
| 946 | - $db_row = array(), |
|
| 947 | - $included_items_protected = false |
|
| 948 | - ) { |
|
| 949 | - // if $db_row not included, hope the entity array has what we need |
|
| 950 | - if (! $db_row) { |
|
| 951 | - $db_row = $entity_array; |
|
| 952 | - } |
|
| 953 | - $relation_settings = $this->getModelVersionInfo()->relationSettings($model); |
|
| 954 | - foreach ($relation_settings as $relation_name => $relation_obj) { |
|
| 955 | - $related_fields_to_include = $this->explodeAndGetItemsPrefixedWith( |
|
| 956 | - $rest_request->get_param('include'), |
|
| 957 | - $relation_name |
|
| 958 | - ); |
|
| 959 | - $related_fields_to_calculate = $this->explodeAndGetItemsPrefixedWith( |
|
| 960 | - $rest_request->get_param('calculate'), |
|
| 961 | - $relation_name |
|
| 962 | - ); |
|
| 963 | - // did they specify they wanted to include a related model, or |
|
| 964 | - // specific fields from a related model? |
|
| 965 | - // or did they specify to calculate a field from a related model? |
|
| 966 | - if ($related_fields_to_include || $related_fields_to_calculate) { |
|
| 967 | - // if so, we should include at least some part of the related model |
|
| 968 | - $pretend_related_request = new WP_REST_Request(); |
|
| 969 | - $pretend_related_request->set_query_params( |
|
| 970 | - array( |
|
| 971 | - 'caps' => $rest_request->get_param('caps'), |
|
| 972 | - 'include' => $related_fields_to_include, |
|
| 973 | - 'calculate' => $related_fields_to_calculate, |
|
| 974 | - 'password' => $rest_request->get_param('password') |
|
| 975 | - ) |
|
| 976 | - ); |
|
| 977 | - $pretend_related_request->add_header('no_rest_headers', true); |
|
| 978 | - $primary_model_query_params = $model->alter_query_params_to_restrict_by_ID( |
|
| 979 | - $model->get_index_primary_key_string( |
|
| 980 | - $model->deduce_fields_n_values_from_cols_n_values($db_row) |
|
| 981 | - ) |
|
| 982 | - ); |
|
| 983 | - if (! $included_items_protected) { |
|
| 984 | - try { |
|
| 985 | - $related_results = $this->getEntitiesFromRelationUsingModelQueryParams( |
|
| 986 | - $primary_model_query_params, |
|
| 987 | - $relation_obj, |
|
| 988 | - $pretend_related_request |
|
| 989 | - ); |
|
| 990 | - } catch (RestException $e) { |
|
| 991 | - $related_results = null; |
|
| 992 | - } |
|
| 993 | - } else { |
|
| 994 | - // they're protected, hide them. |
|
| 995 | - $related_results = null; |
|
| 996 | - $entity_array['_protected'][] = Read::getRelatedEntityName($relation_name, $relation_obj); |
|
| 997 | - } |
|
| 998 | - if ($related_results instanceof WP_Error || $related_results === null) { |
|
| 999 | - $related_results = $relation_obj instanceof EE_Belongs_To_Relation ? null : array(); |
|
| 1000 | - } |
|
| 1001 | - $entity_array[ Read::getRelatedEntityName($relation_name, $relation_obj) ] = $related_results; |
|
| 1002 | - } |
|
| 1003 | - } |
|
| 1004 | - return $entity_array; |
|
| 1005 | - } |
|
| 1006 | - |
|
| 1007 | - /** |
|
| 1008 | - * If the user has requested only specific properties (including meta properties like _links or _protected) |
|
| 1009 | - * remove everything else. |
|
| 1010 | - * @since 4.9.74.p |
|
| 1011 | - * @param EEM_Base $model |
|
| 1012 | - * @param WP_REST_Request $rest_request |
|
| 1013 | - * @param $entity_array |
|
| 1014 | - * @return array |
|
| 1015 | - * @throws EE_Error |
|
| 1016 | - */ |
|
| 1017 | - protected function includeOnlyRequestedProperties( |
|
| 1018 | - EEM_Base $model, |
|
| 1019 | - WP_REST_Request $rest_request, |
|
| 1020 | - $entity_array |
|
| 1021 | - ) { |
|
| 1022 | - |
|
| 1023 | - $includes_for_this_model = $this->explodeAndGetItemsPrefixedWith($rest_request->get_param('include'), ''); |
|
| 1024 | - $includes_for_this_model = $this->removeModelNamesFromArray($includes_for_this_model); |
|
| 1025 | - // if they passed in * or didn't specify any includes, return everything |
|
| 1026 | - if (! in_array('*', $includes_for_this_model) |
|
| 1027 | - && ! empty($includes_for_this_model) |
|
| 1028 | - ) { |
|
| 1029 | - if ($model->has_primary_key_field()) { |
|
| 1030 | - // always include the primary key. ya just gotta know that at least |
|
| 1031 | - $includes_for_this_model[] = $model->primary_key_name(); |
|
| 1032 | - } |
|
| 1033 | - if ($this->explodeAndGetItemsPrefixedWith($rest_request->get_param('calculate'), '')) { |
|
| 1034 | - $includes_for_this_model[] = '_calculated_fields'; |
|
| 1035 | - } |
|
| 1036 | - $entity_array = array_intersect_key($entity_array, array_flip($includes_for_this_model)); |
|
| 1037 | - } |
|
| 1038 | - return $entity_array; |
|
| 1039 | - } |
|
| 1040 | - |
|
| 1041 | - |
|
| 1042 | - /** |
|
| 1043 | - * Returns a new array with all the names of models removed. Eg |
|
| 1044 | - * array( 'Event', 'Datetime.*', 'foobar' ) would become array( 'Datetime.*', 'foobar' ) |
|
| 1045 | - * |
|
| 1046 | - * @param array $arr |
|
| 1047 | - * @return array |
|
| 1048 | - */ |
|
| 1049 | - private function removeModelNamesFromArray($arr) |
|
| 1050 | - { |
|
| 1051 | - return array_diff($arr, array_keys(EE_Registry::instance()->non_abstract_db_models)); |
|
| 1052 | - } |
|
| 1053 | - |
|
| 1054 | - |
|
| 1055 | - /** |
|
| 1056 | - * Gets the calculated fields for the response |
|
| 1057 | - * |
|
| 1058 | - * @param EEM_Base $model |
|
| 1059 | - * @param array $wpdb_row |
|
| 1060 | - * @param WP_REST_Request $rest_request |
|
| 1061 | - * @param boolean $row_is_protected whether this row is password protected or not |
|
| 1062 | - * @return \stdClass the _calculations item in the entity |
|
| 1063 | - * @throws ObjectDetectedException if a default value has a PHP object, which should never do (and if we |
|
| 1064 | - * did, let's know about it ASAP, so let the exception bubble up) |
|
| 1065 | - */ |
|
| 1066 | - protected function getEntityCalculations($model, $wpdb_row, $rest_request, $row_is_protected = false) |
|
| 1067 | - { |
|
| 1068 | - $calculated_fields = $this->explodeAndGetItemsPrefixedWith( |
|
| 1069 | - $rest_request->get_param('calculate'), |
|
| 1070 | - '' |
|
| 1071 | - ); |
|
| 1072 | - // note: setting calculate=* doesn't do anything |
|
| 1073 | - $calculated_fields_to_return = new \stdClass(); |
|
| 1074 | - $protected_fields = array(); |
|
| 1075 | - foreach ($calculated_fields as $field_to_calculate) { |
|
| 1076 | - try { |
|
| 1077 | - // it's password protected, so they shouldn't be able to read this. Remove the value |
|
| 1078 | - $schema = $this->fields_calculator->getJsonSchemaForModel($model); |
|
| 1079 | - if ($row_is_protected |
|
| 1080 | - && isset($schema['properties'][ $field_to_calculate ]['protected']) |
|
| 1081 | - && $schema['properties'][ $field_to_calculate ]['protected']) { |
|
| 1082 | - $calculated_value = null; |
|
| 1083 | - $protected_fields[] = $field_to_calculate; |
|
| 1084 | - if ($schema['properties'][ $field_to_calculate ]['type']) { |
|
| 1085 | - switch ($schema['properties'][ $field_to_calculate ]['type']) { |
|
| 1086 | - case 'boolean': |
|
| 1087 | - $calculated_value = false; |
|
| 1088 | - break; |
|
| 1089 | - case 'integer': |
|
| 1090 | - $calculated_value = 0; |
|
| 1091 | - break; |
|
| 1092 | - case 'string': |
|
| 1093 | - $calculated_value = ''; |
|
| 1094 | - break; |
|
| 1095 | - case 'array': |
|
| 1096 | - $calculated_value = array(); |
|
| 1097 | - break; |
|
| 1098 | - case 'object': |
|
| 1099 | - $calculated_value = new stdClass(); |
|
| 1100 | - break; |
|
| 1101 | - } |
|
| 1102 | - } |
|
| 1103 | - } else { |
|
| 1104 | - $calculated_value = ModelDataTranslator::prepareFieldValueForJson( |
|
| 1105 | - null, |
|
| 1106 | - $this->fields_calculator->retrieveCalculatedFieldValue( |
|
| 1107 | - $model, |
|
| 1108 | - $field_to_calculate, |
|
| 1109 | - $wpdb_row, |
|
| 1110 | - $rest_request, |
|
| 1111 | - $this |
|
| 1112 | - ), |
|
| 1113 | - $this->getModelVersionInfo()->requestedVersion() |
|
| 1114 | - ); |
|
| 1115 | - } |
|
| 1116 | - $calculated_fields_to_return->{$field_to_calculate} = $calculated_value; |
|
| 1117 | - } catch (RestException $e) { |
|
| 1118 | - // if we don't have permission to read it, just leave it out. but let devs know about the problem |
|
| 1119 | - $this->setResponseHeader( |
|
| 1120 | - 'Notices-Field-Calculation-Errors[' |
|
| 1121 | - . $e->getStringCode() |
|
| 1122 | - . '][' |
|
| 1123 | - . $model->get_this_model_name() |
|
| 1124 | - . '][' |
|
| 1125 | - . $field_to_calculate |
|
| 1126 | - . ']', |
|
| 1127 | - $e->getMessage(), |
|
| 1128 | - true |
|
| 1129 | - ); |
|
| 1130 | - } |
|
| 1131 | - } |
|
| 1132 | - $calculated_fields_to_return->_protected = $protected_fields; |
|
| 1133 | - return $calculated_fields_to_return; |
|
| 1134 | - } |
|
| 1135 | - |
|
| 1136 | - |
|
| 1137 | - /** |
|
| 1138 | - * Gets the full URL to the resource, taking the requested version into account |
|
| 1139 | - * |
|
| 1140 | - * @param string $link_part_after_version_and_slash eg "events/10/datetimes" |
|
| 1141 | - * @return string url eg "http://mysite.com/wp-json/ee/v4.6/events/10/datetimes" |
|
| 1142 | - */ |
|
| 1143 | - public function getVersionedLinkTo($link_part_after_version_and_slash) |
|
| 1144 | - { |
|
| 1145 | - return rest_url( |
|
| 1146 | - EED_Core_Rest_Api::get_versioned_route_to( |
|
| 1147 | - $link_part_after_version_and_slash, |
|
| 1148 | - $this->getModelVersionInfo()->requestedVersion() |
|
| 1149 | - ) |
|
| 1150 | - ); |
|
| 1151 | - } |
|
| 1152 | - |
|
| 1153 | - |
|
| 1154 | - /** |
|
| 1155 | - * Gets the correct lowercase name for the relation in the API according |
|
| 1156 | - * to the relation's type |
|
| 1157 | - * |
|
| 1158 | - * @param string $relation_name |
|
| 1159 | - * @param \EE_Model_Relation_Base $relation_obj |
|
| 1160 | - * @return string |
|
| 1161 | - */ |
|
| 1162 | - public static function getRelatedEntityName($relation_name, $relation_obj) |
|
| 1163 | - { |
|
| 1164 | - if ($relation_obj instanceof EE_Belongs_To_Relation) { |
|
| 1165 | - return strtolower($relation_name); |
|
| 1166 | - } else { |
|
| 1167 | - return EEH_Inflector::pluralize_and_lower($relation_name); |
|
| 1168 | - } |
|
| 1169 | - } |
|
| 1170 | - |
|
| 1171 | - |
|
| 1172 | - /** |
|
| 1173 | - * Gets the one model object with the specified id for the specified model |
|
| 1174 | - * |
|
| 1175 | - * @param EEM_Base $model |
|
| 1176 | - * @param WP_REST_Request $request |
|
| 1177 | - * @return array |
|
| 1178 | - */ |
|
| 1179 | - public function getEntityFromModel($model, $request) |
|
| 1180 | - { |
|
| 1181 | - $context = $this->validateContext($request->get_param('caps')); |
|
| 1182 | - return $this->getOneOrReportPermissionError($model, $request, $context); |
|
| 1183 | - } |
|
| 1184 | - |
|
| 1185 | - |
|
| 1186 | - /** |
|
| 1187 | - * If a context is provided which isn't valid, maybe it was added in a future |
|
| 1188 | - * version so just treat it as a default read |
|
| 1189 | - * |
|
| 1190 | - * @param string $context |
|
| 1191 | - * @return string array key of EEM_Base::cap_contexts_to_cap_action_map() |
|
| 1192 | - */ |
|
| 1193 | - public function validateContext($context) |
|
| 1194 | - { |
|
| 1195 | - if (! $context) { |
|
| 1196 | - $context = EEM_Base::caps_read; |
|
| 1197 | - } |
|
| 1198 | - $valid_contexts = EEM_Base::valid_cap_contexts(); |
|
| 1199 | - if (in_array($context, $valid_contexts)) { |
|
| 1200 | - return $context; |
|
| 1201 | - } else { |
|
| 1202 | - return EEM_Base::caps_read; |
|
| 1203 | - } |
|
| 1204 | - } |
|
| 1205 | - |
|
| 1206 | - |
|
| 1207 | - /** |
|
| 1208 | - * Verifies the passed in value is an allowable default where conditions value. |
|
| 1209 | - * |
|
| 1210 | - * @param $default_query_params |
|
| 1211 | - * @return string |
|
| 1212 | - */ |
|
| 1213 | - public function validateDefaultQueryParams($default_query_params) |
|
| 1214 | - { |
|
| 1215 | - $valid_default_where_conditions_for_api_calls = array( |
|
| 1216 | - EEM_Base::default_where_conditions_all, |
|
| 1217 | - EEM_Base::default_where_conditions_minimum_all, |
|
| 1218 | - EEM_Base::default_where_conditions_minimum_others, |
|
| 1219 | - ); |
|
| 1220 | - if (! $default_query_params) { |
|
| 1221 | - $default_query_params = EEM_Base::default_where_conditions_all; |
|
| 1222 | - } |
|
| 1223 | - if (in_array( |
|
| 1224 | - $default_query_params, |
|
| 1225 | - $valid_default_where_conditions_for_api_calls, |
|
| 1226 | - true |
|
| 1227 | - )) { |
|
| 1228 | - return $default_query_params; |
|
| 1229 | - } else { |
|
| 1230 | - return EEM_Base::default_where_conditions_all; |
|
| 1231 | - } |
|
| 1232 | - } |
|
| 1233 | - |
|
| 1234 | - |
|
| 1235 | - /** |
|
| 1236 | - * Translates API filter get parameter into model query params @see https://github.com/eventespresso/event-espresso-core/tree/master/docs/G--Model-System/model-query-params.md#0-where-conditions. |
|
| 1237 | - * Note: right now the query parameter keys for fields (and related fields) |
|
| 1238 | - * can be left as-is, but it's quite possible this will change someday. |
|
| 1239 | - * Also, this method's contents might be candidate for moving to Model_Data_Translator |
|
| 1240 | - * |
|
| 1241 | - * @param EEM_Base $model |
|
| 1242 | - * @param array $query_parameters from $_GET parameter @see Read:handle_request_get_all |
|
| 1243 | - * @return array model query params (@see https://github.com/eventespresso/event-espresso-core/tree/master/docs/G--Model-System/model-query-params.md#0-where-conditions) |
|
| 1244 | - * or FALSE to indicate that absolutely no results should be returned |
|
| 1245 | - * @throws EE_Error |
|
| 1246 | - * @throws RestException |
|
| 1247 | - */ |
|
| 1248 | - public function createModelQueryParams($model, $query_params) |
|
| 1249 | - { |
|
| 1250 | - $model_query_params = array(); |
|
| 1251 | - if (isset($query_params['where'])) { |
|
| 1252 | - $model_query_params[0] = ModelDataTranslator::prepareConditionsQueryParamsForModels( |
|
| 1253 | - $query_params['where'], |
|
| 1254 | - $model, |
|
| 1255 | - $this->getModelVersionInfo()->requestedVersion() |
|
| 1256 | - ); |
|
| 1257 | - } |
|
| 1258 | - if (isset($query_params['order_by'])) { |
|
| 1259 | - $order_by = $query_params['order_by']; |
|
| 1260 | - } elseif (isset($query_params['orderby'])) { |
|
| 1261 | - $order_by = $query_params['orderby']; |
|
| 1262 | - } else { |
|
| 1263 | - $order_by = null; |
|
| 1264 | - } |
|
| 1265 | - if ($order_by !== null) { |
|
| 1266 | - if (is_array($order_by)) { |
|
| 1267 | - $order_by = ModelDataTranslator::prepareFieldNamesInArrayKeysFromJson($order_by); |
|
| 1268 | - } else { |
|
| 1269 | - // it's a single item |
|
| 1270 | - $order_by = ModelDataTranslator::prepareFieldNameFromJson($order_by); |
|
| 1271 | - } |
|
| 1272 | - $model_query_params['order_by'] = $order_by; |
|
| 1273 | - } |
|
| 1274 | - if (isset($query_params['group_by'])) { |
|
| 1275 | - $group_by = $query_params['group_by']; |
|
| 1276 | - } elseif (isset($query_params['groupby'])) { |
|
| 1277 | - $group_by = $query_params['groupby']; |
|
| 1278 | - } else { |
|
| 1279 | - $group_by = array_keys($model->get_combined_primary_key_fields()); |
|
| 1280 | - } |
|
| 1281 | - // make sure they're all real names |
|
| 1282 | - if (is_array($group_by)) { |
|
| 1283 | - $group_by = ModelDataTranslator::prepareFieldNamesFromJson($group_by); |
|
| 1284 | - } |
|
| 1285 | - if ($group_by !== null) { |
|
| 1286 | - $model_query_params['group_by'] = $group_by; |
|
| 1287 | - } |
|
| 1288 | - if (isset($query_params['having'])) { |
|
| 1289 | - $model_query_params['having'] = ModelDataTranslator::prepareConditionsQueryParamsForModels( |
|
| 1290 | - $query_params['having'], |
|
| 1291 | - $model, |
|
| 1292 | - $this->getModelVersionInfo()->requestedVersion() |
|
| 1293 | - ); |
|
| 1294 | - } |
|
| 1295 | - if (isset($query_params['order'])) { |
|
| 1296 | - $model_query_params['order'] = $query_params['order']; |
|
| 1297 | - } |
|
| 1298 | - if (isset($query_params['mine'])) { |
|
| 1299 | - $model_query_params = $model->alter_query_params_to_only_include_mine($model_query_params); |
|
| 1300 | - } |
|
| 1301 | - if (isset($query_params['limit'])) { |
|
| 1302 | - // limit should be either a string like '23' or '23,43', or an array with two items in it |
|
| 1303 | - if (! is_array($query_params['limit'])) { |
|
| 1304 | - $limit_array = explode(',', (string) $query_params['limit']); |
|
| 1305 | - } else { |
|
| 1306 | - $limit_array = $query_params['limit']; |
|
| 1307 | - } |
|
| 1308 | - $sanitized_limit = array(); |
|
| 1309 | - foreach ($limit_array as $key => $limit_part) { |
|
| 1310 | - if ($this->debug_mode && (! is_numeric($limit_part) || count($sanitized_limit) > 2)) { |
|
| 1311 | - throw new EE_Error( |
|
| 1312 | - sprintf( |
|
| 1313 | - __( |
|
| 1314 | - // @codingStandardsIgnoreStart |
|
| 1315 | - 'An invalid limit filter was provided. It was: %s. If the EE4 JSON REST API weren\'t in debug mode, this message would not appear.', |
|
| 1316 | - // @codingStandardsIgnoreEnd |
|
| 1317 | - 'event_espresso' |
|
| 1318 | - ), |
|
| 1319 | - wp_json_encode($query_params['limit']) |
|
| 1320 | - ) |
|
| 1321 | - ); |
|
| 1322 | - } |
|
| 1323 | - $sanitized_limit[] = (int) $limit_part; |
|
| 1324 | - } |
|
| 1325 | - $model_query_params['limit'] = implode(',', $sanitized_limit); |
|
| 1326 | - } else { |
|
| 1327 | - $model_query_params['limit'] = EED_Core_Rest_Api::get_default_query_limit(); |
|
| 1328 | - } |
|
| 1329 | - if (isset($query_params['caps'])) { |
|
| 1330 | - $model_query_params['caps'] = $this->validateContext($query_params['caps']); |
|
| 1331 | - } else { |
|
| 1332 | - $model_query_params['caps'] = EEM_Base::caps_read; |
|
| 1333 | - } |
|
| 1334 | - if (isset($query_params['default_where_conditions'])) { |
|
| 1335 | - $model_query_params['default_where_conditions'] = $this->validateDefaultQueryParams( |
|
| 1336 | - $query_params['default_where_conditions'] |
|
| 1337 | - ); |
|
| 1338 | - } |
|
| 1339 | - // if this is a model protected by a password on another model, exclude the password protected |
|
| 1340 | - // entities by default. But if they passed in a password, try to show them all. If the password is wrong, |
|
| 1341 | - // though, they'll get an error (see Read::createEntityFromWpdbResult() which calls Read::checkPassword) |
|
| 1342 | - if (! $model->hasPassword() |
|
| 1343 | - && $model->restrictedByRelatedModelPassword() |
|
| 1344 | - && $model_query_params['caps'] === EEM_Base::caps_read) { |
|
| 1345 | - if (empty($query_params['password'])) { |
|
| 1346 | - $model_query_params['exclude_protected'] = true; |
|
| 1347 | - } |
|
| 1348 | - } |
|
| 1349 | - |
|
| 1350 | - return apply_filters('FHEE__Read__create_model_query_params', $model_query_params, $query_params, $model); |
|
| 1351 | - } |
|
| 1352 | - |
|
| 1353 | - |
|
| 1354 | - /** |
|
| 1355 | - * Changes the REST-style query params for use in the models |
|
| 1356 | - * |
|
| 1357 | - * @deprecated |
|
| 1358 | - * @param EEM_Base $model |
|
| 1359 | - * @param array $query_params sub-array from @see EEM_Base::get_all() |
|
| 1360 | - * @return array |
|
| 1361 | - */ |
|
| 1362 | - public function prepareRestQueryParamsKeyForModels($model, $query_params) |
|
| 1363 | - { |
|
| 1364 | - $model_ready_query_params = array(); |
|
| 1365 | - foreach ($query_params as $key => $value) { |
|
| 1366 | - if (is_array($value)) { |
|
| 1367 | - $model_ready_query_params[ $key ] = $this->prepareRestQueryParamsKeyForModels($model, $value); |
|
| 1368 | - } else { |
|
| 1369 | - $model_ready_query_params[ $key ] = $value; |
|
| 1370 | - } |
|
| 1371 | - } |
|
| 1372 | - return $model_ready_query_params; |
|
| 1373 | - } |
|
| 1374 | - |
|
| 1375 | - |
|
| 1376 | - /** |
|
| 1377 | - * @deprecated instead use ModelDataTranslator::prepareFieldValuesFromJson() |
|
| 1378 | - * @param $model |
|
| 1379 | - * @param $query_params |
|
| 1380 | - * @return array |
|
| 1381 | - */ |
|
| 1382 | - public function prepareRestQueryParamsValuesForModels($model, $query_params) |
|
| 1383 | - { |
|
| 1384 | - $model_ready_query_params = array(); |
|
| 1385 | - foreach ($query_params as $key => $value) { |
|
| 1386 | - if (is_array($value)) { |
|
| 1387 | - $model_ready_query_params[ $key ] = $this->prepareRestQueryParamsValuesForModels($model, $value); |
|
| 1388 | - } else { |
|
| 1389 | - $model_ready_query_params[ $key ] = $value; |
|
| 1390 | - } |
|
| 1391 | - } |
|
| 1392 | - return $model_ready_query_params; |
|
| 1393 | - } |
|
| 1394 | - |
|
| 1395 | - |
|
| 1396 | - /** |
|
| 1397 | - * Explodes the string on commas, and only returns items with $prefix followed by a period. |
|
| 1398 | - * If no prefix is specified, returns items with no period. |
|
| 1399 | - * |
|
| 1400 | - * @param string|array $string_to_explode eg "jibba,jabba, blah, blah, blah" or array('jibba', 'jabba' ) |
|
| 1401 | - * @param string $prefix "Event" or "foobar" |
|
| 1402 | - * @return array $string_to_exploded exploded on COMMAS, and if a prefix was specified |
|
| 1403 | - * we only return strings starting with that and a period; if no prefix was |
|
| 1404 | - * specified we return all items containing NO periods |
|
| 1405 | - */ |
|
| 1406 | - public function explodeAndGetItemsPrefixedWith($string_to_explode, $prefix) |
|
| 1407 | - { |
|
| 1408 | - if (is_string($string_to_explode)) { |
|
| 1409 | - $exploded_contents = explode(',', $string_to_explode); |
|
| 1410 | - } elseif (is_array($string_to_explode)) { |
|
| 1411 | - $exploded_contents = $string_to_explode; |
|
| 1412 | - } else { |
|
| 1413 | - $exploded_contents = array(); |
|
| 1414 | - } |
|
| 1415 | - // if the string was empty, we want an empty array |
|
| 1416 | - $exploded_contents = array_filter($exploded_contents); |
|
| 1417 | - $contents_with_prefix = array(); |
|
| 1418 | - foreach ($exploded_contents as $item) { |
|
| 1419 | - $item = trim($item); |
|
| 1420 | - // if no prefix was provided, so we look for items with no "." in them |
|
| 1421 | - if (! $prefix) { |
|
| 1422 | - // does this item have a period? |
|
| 1423 | - if (strpos($item, '.') === false) { |
|
| 1424 | - // if not, then its what we're looking for |
|
| 1425 | - $contents_with_prefix[] = $item; |
|
| 1426 | - } |
|
| 1427 | - } elseif (strpos($item, $prefix . '.') === 0) { |
|
| 1428 | - // this item has the prefix and a period, grab it |
|
| 1429 | - $contents_with_prefix[] = substr( |
|
| 1430 | - $item, |
|
| 1431 | - strpos($item, $prefix . '.') + strlen($prefix . '.') |
|
| 1432 | - ); |
|
| 1433 | - } elseif ($item === $prefix) { |
|
| 1434 | - // this item is JUST the prefix |
|
| 1435 | - // so let's grab everything after, which is a blank string |
|
| 1436 | - $contents_with_prefix[] = ''; |
|
| 1437 | - } |
|
| 1438 | - } |
|
| 1439 | - return $contents_with_prefix; |
|
| 1440 | - } |
|
| 1441 | - |
|
| 1442 | - |
|
| 1443 | - /** |
|
| 1444 | - * @deprecated since 4.8.36.rc.001 You should instead use Read::explode_and_get_items_prefixed_with. |
|
| 1445 | - * Deprecated because its return values were really quite confusing- sometimes it returned |
|
| 1446 | - * an empty array (when the include string was blank or '*') or sometimes it returned |
|
| 1447 | - * array('*') (when you provided a model and a model of that kind was found). |
|
| 1448 | - * Parses the $include_string so we fetch all the field names relating to THIS model |
|
| 1449 | - * (ie have NO period in them), or for the provided model (ie start with the model |
|
| 1450 | - * name and then a period). |
|
| 1451 | - * @param string $include_string @see Read:handle_request_get_all |
|
| 1452 | - * @param string $model_name |
|
| 1453 | - * @return array of fields for this model. If $model_name is provided, then |
|
| 1454 | - * the fields for that model, with the model's name removed from each. |
|
| 1455 | - * If $include_string was blank or '*' returns an empty array |
|
| 1456 | - */ |
|
| 1457 | - public function extractIncludesForThisModel($include_string, $model_name = null) |
|
| 1458 | - { |
|
| 1459 | - if (is_array($include_string)) { |
|
| 1460 | - $include_string = implode(',', $include_string); |
|
| 1461 | - } |
|
| 1462 | - if ($include_string === '*' || $include_string === '') { |
|
| 1463 | - return array(); |
|
| 1464 | - } |
|
| 1465 | - $includes = explode(',', $include_string); |
|
| 1466 | - $extracted_fields_to_include = array(); |
|
| 1467 | - if ($model_name) { |
|
| 1468 | - foreach ($includes as $field_to_include) { |
|
| 1469 | - $field_to_include = trim($field_to_include); |
|
| 1470 | - if (strpos($field_to_include, $model_name . '.') === 0) { |
|
| 1471 | - // found the model name at the exact start |
|
| 1472 | - $field_sans_model_name = str_replace($model_name . '.', '', $field_to_include); |
|
| 1473 | - $extracted_fields_to_include[] = $field_sans_model_name; |
|
| 1474 | - } elseif ($field_to_include == $model_name) { |
|
| 1475 | - $extracted_fields_to_include[] = '*'; |
|
| 1476 | - } |
|
| 1477 | - } |
|
| 1478 | - } else { |
|
| 1479 | - // look for ones with no period |
|
| 1480 | - foreach ($includes as $field_to_include) { |
|
| 1481 | - $field_to_include = trim($field_to_include); |
|
| 1482 | - if (strpos($field_to_include, '.') === false |
|
| 1483 | - && ! $this->getModelVersionInfo()->isModelNameInThisVersion($field_to_include) |
|
| 1484 | - ) { |
|
| 1485 | - $extracted_fields_to_include[] = $field_to_include; |
|
| 1486 | - } |
|
| 1487 | - } |
|
| 1488 | - } |
|
| 1489 | - return $extracted_fields_to_include; |
|
| 1490 | - } |
|
| 1491 | - |
|
| 1492 | - |
|
| 1493 | - /** |
|
| 1494 | - * Gets the single item using the model according to the request in the context given, otherwise |
|
| 1495 | - * returns that it's inaccessible to the current user |
|
| 1496 | - * |
|
| 1497 | - * @param EEM_Base $model |
|
| 1498 | - * @param WP_REST_Request $request |
|
| 1499 | - * @param null $context |
|
| 1500 | - * @return array |
|
| 1501 | - * @throws EE_Error |
|
| 1502 | - */ |
|
| 1503 | - public function getOneOrReportPermissionError(EEM_Base $model, WP_REST_Request $request, $context = null) |
|
| 1504 | - { |
|
| 1505 | - $query_params = array(array($model->primary_key_name() => $request->get_param('id')), 'limit' => 1); |
|
| 1506 | - if ($model instanceof EEM_Soft_Delete_Base) { |
|
| 1507 | - $query_params = $model->alter_query_params_so_deleted_and_undeleted_items_included($query_params); |
|
| 1508 | - } |
|
| 1509 | - $restricted_query_params = $query_params; |
|
| 1510 | - $restricted_query_params['caps'] = $context; |
|
| 1511 | - $this->setDebugInfo('model query params', $restricted_query_params); |
|
| 1512 | - $model_rows = $model->get_all_wpdb_results($restricted_query_params); |
|
| 1513 | - if (! empty($model_rows)) { |
|
| 1514 | - return $this->createEntityFromWpdbResult( |
|
| 1515 | - $model, |
|
| 1516 | - reset($model_rows), |
|
| 1517 | - $request |
|
| 1518 | - ); |
|
| 1519 | - } else { |
|
| 1520 | - // ok let's test to see if we WOULD have found it, had we not had restrictions from missing capabilities |
|
| 1521 | - $lowercase_model_name = strtolower($model->get_this_model_name()); |
|
| 1522 | - if ($model->exists($query_params)) { |
|
| 1523 | - // you got shafted- it existed but we didn't want to tell you! |
|
| 1524 | - throw new RestException( |
|
| 1525 | - 'rest_user_cannot_' . $context, |
|
| 1526 | - sprintf( |
|
| 1527 | - __('Sorry, you cannot %1$s this %2$s. Missing permissions are: %3$s', 'event_espresso'), |
|
| 1528 | - $context, |
|
| 1529 | - $lowercase_model_name, |
|
| 1530 | - Capabilities::getMissingPermissionsString( |
|
| 1531 | - $model, |
|
| 1532 | - $context |
|
| 1533 | - ) |
|
| 1534 | - ), |
|
| 1535 | - array('status' => 403) |
|
| 1536 | - ); |
|
| 1537 | - } else { |
|
| 1538 | - // it's not you. It just doesn't exist |
|
| 1539 | - throw new RestException( |
|
| 1540 | - sprintf('rest_%s_invalid_id', $lowercase_model_name), |
|
| 1541 | - sprintf(__('Invalid %s ID.', 'event_espresso'), $lowercase_model_name), |
|
| 1542 | - array('status' => 404) |
|
| 1543 | - ); |
|
| 1544 | - } |
|
| 1545 | - } |
|
| 1546 | - } |
|
| 1547 | - |
|
| 1548 | - /** |
|
| 1549 | - * Checks that if this content requires a password to be read, that it's been provided and is correct. |
|
| 1550 | - * @since 4.9.74.p |
|
| 1551 | - * @param EEM_Base $model |
|
| 1552 | - * @param $model_row |
|
| 1553 | - * @param $query_params Adds 'default_where_conditions' => 'minimum' to ensure we don't confuse trashed with |
|
| 1554 | - * password protected. |
|
| 1555 | - * @param WP_REST_Request $request |
|
| 1556 | - * @throws EE_Error |
|
| 1557 | - * @throws InvalidArgumentException |
|
| 1558 | - * @throws InvalidDataTypeException |
|
| 1559 | - * @throws InvalidInterfaceException |
|
| 1560 | - * @throws RestPasswordRequiredException |
|
| 1561 | - * @throws RestPasswordIncorrectException |
|
| 1562 | - * @throws \EventEspresso\core\exceptions\ModelConfigurationException |
|
| 1563 | - * @throws ReflectionException |
|
| 1564 | - */ |
|
| 1565 | - protected function checkPassword(EEM_Base $model, $model_row, $query_params, WP_REST_Request $request) |
|
| 1566 | - { |
|
| 1567 | - $query_params['default_where_conditions'] = 'minimum'; |
|
| 1568 | - // stuff is only "protected" for front-end requests. Elsewhere, you either get full permission to access the object |
|
| 1569 | - // or you don't. |
|
| 1570 | - $request_caps = $request->get_param('caps'); |
|
| 1571 | - if (isset($request_caps) && $request_caps !== EEM_Base::caps_read) { |
|
| 1572 | - return; |
|
| 1573 | - } |
|
| 1574 | - // if this entity requires a password, they better give it and it better be right! |
|
| 1575 | - if ($model->hasPassword() |
|
| 1576 | - && $model_row[ $model->getPasswordField()->get_qualified_column() ] !== '') { |
|
| 1577 | - if (empty($request['password'])) { |
|
| 1578 | - throw new RestPasswordRequiredException(); |
|
| 1579 | - } elseif (!hash_equals( |
|
| 1580 | - $model_row[ $model->getPasswordField()->get_qualified_column() ], |
|
| 1581 | - $request['password'] |
|
| 1582 | - )) { |
|
| 1583 | - throw new RestPasswordIncorrectException(); |
|
| 1584 | - } |
|
| 1585 | - } // wait! maybe this content is password protected |
|
| 1586 | - elseif ($model->restrictedByRelatedModelPassword() |
|
| 1587 | - && $request->get_param('caps') === EEM_Base::caps_read) { |
|
| 1588 | - $password_supplied = $request->get_param('password'); |
|
| 1589 | - if (empty($password_supplied)) { |
|
| 1590 | - $query_params['exclude_protected'] = true; |
|
| 1591 | - if (!$model->exists($query_params)) { |
|
| 1592 | - throw new RestPasswordRequiredException(); |
|
| 1593 | - } |
|
| 1594 | - } else { |
|
| 1595 | - $query_params[0][ $model->modelChainAndPassword() ] = $password_supplied; |
|
| 1596 | - if (!$model->exists($query_params)) { |
|
| 1597 | - throw new RestPasswordIncorrectException(); |
|
| 1598 | - } |
|
| 1599 | - } |
|
| 1600 | - } |
|
| 1601 | - } |
|
| 48 | + /** |
|
| 49 | + * @var CalculatedModelFields |
|
| 50 | + */ |
|
| 51 | + protected $fields_calculator; |
|
| 52 | + |
|
| 53 | + |
|
| 54 | + /** |
|
| 55 | + * Read constructor. |
|
| 56 | + * @param CalculatedModelFields $fields_calculator |
|
| 57 | + */ |
|
| 58 | + public function __construct(CalculatedModelFields $fields_calculator) |
|
| 59 | + { |
|
| 60 | + parent::__construct(); |
|
| 61 | + $this->fields_calculator = $fields_calculator; |
|
| 62 | + } |
|
| 63 | + |
|
| 64 | + |
|
| 65 | + /** |
|
| 66 | + * Handles requests to get all (or a filtered subset) of entities for a particular model |
|
| 67 | + * |
|
| 68 | + * @param WP_REST_Request $request |
|
| 69 | + * @param string $version |
|
| 70 | + * @param string $model_name |
|
| 71 | + * @return WP_REST_Response|WP_Error |
|
| 72 | + * @throws InvalidArgumentException |
|
| 73 | + * @throws InvalidDataTypeException |
|
| 74 | + * @throws InvalidInterfaceException |
|
| 75 | + */ |
|
| 76 | + public static function handleRequestGetAll(WP_REST_Request $request, $version, $model_name) |
|
| 77 | + { |
|
| 78 | + $controller = LoaderFactory::getLoader()->getNew('EventEspresso\core\libraries\rest_api\controllers\model\Read'); |
|
| 79 | + try { |
|
| 80 | + $controller->setRequestedVersion($version); |
|
| 81 | + if (! $controller->getModelVersionInfo()->isModelNameInThisVersion($model_name)) { |
|
| 82 | + return $controller->sendResponse( |
|
| 83 | + new WP_Error( |
|
| 84 | + 'endpoint_parsing_error', |
|
| 85 | + sprintf( |
|
| 86 | + __( |
|
| 87 | + 'There is no model for endpoint %s. Please contact event espresso support', |
|
| 88 | + 'event_espresso' |
|
| 89 | + ), |
|
| 90 | + $model_name |
|
| 91 | + ) |
|
| 92 | + ) |
|
| 93 | + ); |
|
| 94 | + } |
|
| 95 | + return $controller->sendResponse( |
|
| 96 | + $controller->getEntitiesFromModel( |
|
| 97 | + $controller->getModelVersionInfo()->loadModel($model_name), |
|
| 98 | + $request |
|
| 99 | + ) |
|
| 100 | + ); |
|
| 101 | + } catch (Exception $e) { |
|
| 102 | + return $controller->sendResponse($e); |
|
| 103 | + } |
|
| 104 | + } |
|
| 105 | + |
|
| 106 | + |
|
| 107 | + /** |
|
| 108 | + * Prepares and returns schema for any OPTIONS request. |
|
| 109 | + * |
|
| 110 | + * @param string $version The API endpoint version being used. |
|
| 111 | + * @param string $model_name Something like `Event` or `Registration` |
|
| 112 | + * @return array |
|
| 113 | + * @throws InvalidArgumentException |
|
| 114 | + * @throws InvalidDataTypeException |
|
| 115 | + * @throws InvalidInterfaceException |
|
| 116 | + */ |
|
| 117 | + public static function handleSchemaRequest($version, $model_name) |
|
| 118 | + { |
|
| 119 | + $controller = LoaderFactory::getLoader()->getNew('EventEspresso\core\libraries\rest_api\controllers\model\Read'); |
|
| 120 | + try { |
|
| 121 | + $controller->setRequestedVersion($version); |
|
| 122 | + if (! $controller->getModelVersionInfo()->isModelNameInThisVersion($model_name)) { |
|
| 123 | + return array(); |
|
| 124 | + } |
|
| 125 | + // get the model for this version |
|
| 126 | + $model = $controller->getModelVersionInfo()->loadModel($model_name); |
|
| 127 | + $model_schema = new JsonModelSchema($model, LoaderFactory::getLoader()->getShared('EventEspresso\core\libraries\rest_api\CalculatedModelFields')); |
|
| 128 | + return $model_schema->getModelSchemaForRelations( |
|
| 129 | + $controller->getModelVersionInfo()->relationSettings($model), |
|
| 130 | + $controller->customizeSchemaForRestResponse( |
|
| 131 | + $model, |
|
| 132 | + $model_schema->getModelSchemaForFields( |
|
| 133 | + $controller->getModelVersionInfo()->fieldsOnModelInThisVersion($model), |
|
| 134 | + $model_schema->getInitialSchemaStructure() |
|
| 135 | + ) |
|
| 136 | + ) |
|
| 137 | + ); |
|
| 138 | + } catch (Exception $e) { |
|
| 139 | + return array(); |
|
| 140 | + } |
|
| 141 | + } |
|
| 142 | + |
|
| 143 | + |
|
| 144 | + /** |
|
| 145 | + * This loops through each field in the given schema for the model and does the following: |
|
| 146 | + * - add any extra fields that are REST API specific and related to existing fields. |
|
| 147 | + * - transform default values into the correct format for a REST API response. |
|
| 148 | + * |
|
| 149 | + * @param EEM_Base $model |
|
| 150 | + * @param array $schema |
|
| 151 | + * @return array The final schema. |
|
| 152 | + */ |
|
| 153 | + protected function customizeSchemaForRestResponse(EEM_Base $model, array $schema) |
|
| 154 | + { |
|
| 155 | + foreach ($this->getModelVersionInfo()->fieldsOnModelInThisVersion($model) as $field_name => $field) { |
|
| 156 | + $schema = $this->translateDefaultsForRestResponse( |
|
| 157 | + $field_name, |
|
| 158 | + $field, |
|
| 159 | + $this->maybeAddExtraFieldsToSchema($field_name, $field, $schema) |
|
| 160 | + ); |
|
| 161 | + } |
|
| 162 | + return $schema; |
|
| 163 | + } |
|
| 164 | + |
|
| 165 | + |
|
| 166 | + /** |
|
| 167 | + * This is used to ensure that the 'default' value set in the schema response is formatted correctly for the REST |
|
| 168 | + * response. |
|
| 169 | + * |
|
| 170 | + * @param $field_name |
|
| 171 | + * @param EE_Model_Field_Base $field |
|
| 172 | + * @param array $schema |
|
| 173 | + * @return array |
|
| 174 | + * @throws ObjectDetectedException if a default value has a PHP object, which should never do (and if we |
|
| 175 | + * did, let's know about it ASAP, so let the exception bubble up) |
|
| 176 | + */ |
|
| 177 | + protected function translateDefaultsForRestResponse($field_name, EE_Model_Field_Base $field, array $schema) |
|
| 178 | + { |
|
| 179 | + if (isset($schema['properties'][ $field_name ]['default'])) { |
|
| 180 | + if (is_array($schema['properties'][ $field_name ]['default'])) { |
|
| 181 | + foreach ($schema['properties'][ $field_name ]['default'] as $default_key => $default_value) { |
|
| 182 | + if ($default_key === 'raw') { |
|
| 183 | + $schema['properties'][ $field_name ]['default'][ $default_key ] = |
|
| 184 | + ModelDataTranslator::prepareFieldValueForJson( |
|
| 185 | + $field, |
|
| 186 | + $default_value, |
|
| 187 | + $this->getModelVersionInfo()->requestedVersion() |
|
| 188 | + ); |
|
| 189 | + } |
|
| 190 | + } |
|
| 191 | + } else { |
|
| 192 | + $schema['properties'][ $field_name ]['default'] = ModelDataTranslator::prepareFieldValueForJson( |
|
| 193 | + $field, |
|
| 194 | + $schema['properties'][ $field_name ]['default'], |
|
| 195 | + $this->getModelVersionInfo()->requestedVersion() |
|
| 196 | + ); |
|
| 197 | + } |
|
| 198 | + } |
|
| 199 | + return $schema; |
|
| 200 | + } |
|
| 201 | + |
|
| 202 | + |
|
| 203 | + /** |
|
| 204 | + * Adds additional fields to the schema |
|
| 205 | + * The REST API returns a GMT value field for each datetime field in the resource. Thus the description about this |
|
| 206 | + * needs to be added to the schema. |
|
| 207 | + * |
|
| 208 | + * @param $field_name |
|
| 209 | + * @param EE_Model_Field_Base $field |
|
| 210 | + * @param array $schema |
|
| 211 | + * @return array |
|
| 212 | + */ |
|
| 213 | + protected function maybeAddExtraFieldsToSchema($field_name, EE_Model_Field_Base $field, array $schema) |
|
| 214 | + { |
|
| 215 | + if ($field instanceof EE_Datetime_Field) { |
|
| 216 | + $schema['properties'][ $field_name . '_gmt' ] = $field->getSchema(); |
|
| 217 | + // modify the description |
|
| 218 | + $schema['properties'][ $field_name . '_gmt' ]['description'] = sprintf( |
|
| 219 | + esc_html__('%s - the value for this field is in GMT.', 'event_espresso'), |
|
| 220 | + wp_specialchars_decode($field->get_nicename(), ENT_QUOTES) |
|
| 221 | + ); |
|
| 222 | + } |
|
| 223 | + return $schema; |
|
| 224 | + } |
|
| 225 | + |
|
| 226 | + |
|
| 227 | + /** |
|
| 228 | + * Used to figure out the route from the request when a `WP_REST_Request` object is not available |
|
| 229 | + * |
|
| 230 | + * @return string |
|
| 231 | + */ |
|
| 232 | + protected function getRouteFromRequest() |
|
| 233 | + { |
|
| 234 | + if (isset($GLOBALS['wp']) |
|
| 235 | + && $GLOBALS['wp'] instanceof \WP |
|
| 236 | + && isset($GLOBALS['wp']->query_vars['rest_route']) |
|
| 237 | + ) { |
|
| 238 | + return $GLOBALS['wp']->query_vars['rest_route']; |
|
| 239 | + } else { |
|
| 240 | + return isset($_SERVER['PATH_INFO']) ? $_SERVER['PATH_INFO'] : '/'; |
|
| 241 | + } |
|
| 242 | + } |
|
| 243 | + |
|
| 244 | + |
|
| 245 | + /** |
|
| 246 | + * Gets a single entity related to the model indicated in the path and its id |
|
| 247 | + * |
|
| 248 | + * @param WP_REST_Request $request |
|
| 249 | + * @param string $version |
|
| 250 | + * @param string $model_name |
|
| 251 | + * @return WP_REST_Response|WP_Error |
|
| 252 | + * @throws InvalidDataTypeException |
|
| 253 | + * @throws InvalidInterfaceException |
|
| 254 | + * @throws InvalidArgumentException |
|
| 255 | + */ |
|
| 256 | + public static function handleRequestGetOne(WP_REST_Request $request, $version, $model_name) |
|
| 257 | + { |
|
| 258 | + $controller = LoaderFactory::getLoader()->getNew('EventEspresso\core\libraries\rest_api\controllers\model\Read'); |
|
| 259 | + try { |
|
| 260 | + $controller->setRequestedVersion($version); |
|
| 261 | + if (! $controller->getModelVersionInfo()->isModelNameInThisVersion($model_name)) { |
|
| 262 | + return $controller->sendResponse( |
|
| 263 | + new WP_Error( |
|
| 264 | + 'endpoint_parsing_error', |
|
| 265 | + sprintf( |
|
| 266 | + __( |
|
| 267 | + 'There is no model for endpoint %s. Please contact event espresso support', |
|
| 268 | + 'event_espresso' |
|
| 269 | + ), |
|
| 270 | + $model_name |
|
| 271 | + ) |
|
| 272 | + ) |
|
| 273 | + ); |
|
| 274 | + } |
|
| 275 | + return $controller->sendResponse( |
|
| 276 | + $controller->getEntityFromModel( |
|
| 277 | + $controller->getModelVersionInfo()->loadModel($model_name), |
|
| 278 | + $request |
|
| 279 | + ) |
|
| 280 | + ); |
|
| 281 | + } catch (Exception $e) { |
|
| 282 | + return $controller->sendResponse($e); |
|
| 283 | + } |
|
| 284 | + } |
|
| 285 | + |
|
| 286 | + |
|
| 287 | + /** |
|
| 288 | + * Gets all the related entities (or if its a belongs-to relation just the one) |
|
| 289 | + * to the item with the given id |
|
| 290 | + * |
|
| 291 | + * @param WP_REST_Request $request |
|
| 292 | + * @param string $version |
|
| 293 | + * @param string $model_name |
|
| 294 | + * @param string $related_model_name |
|
| 295 | + * @return WP_REST_Response|WP_Error |
|
| 296 | + * @throws InvalidDataTypeException |
|
| 297 | + * @throws InvalidInterfaceException |
|
| 298 | + * @throws InvalidArgumentException |
|
| 299 | + */ |
|
| 300 | + public static function handleRequestGetRelated( |
|
| 301 | + WP_REST_Request $request, |
|
| 302 | + $version, |
|
| 303 | + $model_name, |
|
| 304 | + $related_model_name |
|
| 305 | + ) { |
|
| 306 | + $controller = LoaderFactory::getLoader()->getNew('EventEspresso\core\libraries\rest_api\controllers\model\Read'); |
|
| 307 | + try { |
|
| 308 | + $controller->setRequestedVersion($version); |
|
| 309 | + $main_model = $controller->validateModel($model_name); |
|
| 310 | + $controller->validateModel($related_model_name); |
|
| 311 | + return $controller->sendResponse( |
|
| 312 | + $controller->getEntitiesFromRelation( |
|
| 313 | + $request->get_param('id'), |
|
| 314 | + $main_model->related_settings_for($related_model_name), |
|
| 315 | + $request |
|
| 316 | + ) |
|
| 317 | + ); |
|
| 318 | + } catch (Exception $e) { |
|
| 319 | + return $controller->sendResponse($e); |
|
| 320 | + } |
|
| 321 | + } |
|
| 322 | + |
|
| 323 | + |
|
| 324 | + /** |
|
| 325 | + * Gets a collection for the given model and filters |
|
| 326 | + * |
|
| 327 | + * @param EEM_Base $model |
|
| 328 | + * @param WP_REST_Request $request |
|
| 329 | + * @return array |
|
| 330 | + * @throws EE_Error |
|
| 331 | + * @throws InvalidArgumentException |
|
| 332 | + * @throws InvalidDataTypeException |
|
| 333 | + * @throws InvalidInterfaceException |
|
| 334 | + * @throws ReflectionException |
|
| 335 | + * @throws RestException |
|
| 336 | + */ |
|
| 337 | + public function getEntitiesFromModel($model, $request) |
|
| 338 | + { |
|
| 339 | + $query_params = $this->createModelQueryParams($model, $request->get_params()); |
|
| 340 | + if (! Capabilities::currentUserHasPartialAccessTo($model, $query_params['caps'])) { |
|
| 341 | + $model_name_plural = EEH_Inflector::pluralize_and_lower($model->get_this_model_name()); |
|
| 342 | + throw new RestException( |
|
| 343 | + sprintf('rest_%s_cannot_list', $model_name_plural), |
|
| 344 | + sprintf( |
|
| 345 | + __('Sorry, you are not allowed to list %1$s. Missing permissions: %2$s', 'event_espresso'), |
|
| 346 | + $model_name_plural, |
|
| 347 | + Capabilities::getMissingPermissionsString($model, $query_params['caps']) |
|
| 348 | + ), |
|
| 349 | + array('status' => 403) |
|
| 350 | + ); |
|
| 351 | + } |
|
| 352 | + if (! $request->get_header('no_rest_headers')) { |
|
| 353 | + $this->setHeadersFromQueryParams($model, $query_params); |
|
| 354 | + } |
|
| 355 | + /** @type array $results */ |
|
| 356 | + $results = $model->get_all_wpdb_results($query_params); |
|
| 357 | + $nice_results = array(); |
|
| 358 | + foreach ($results as $result) { |
|
| 359 | + $nice_results[] = $this->createEntityFromWpdbResult( |
|
| 360 | + $model, |
|
| 361 | + $result, |
|
| 362 | + $request |
|
| 363 | + ); |
|
| 364 | + } |
|
| 365 | + return $nice_results; |
|
| 366 | + } |
|
| 367 | + |
|
| 368 | + |
|
| 369 | + /** |
|
| 370 | + * Gets the collection for given relation object |
|
| 371 | + * The same as Read::get_entities_from_model(), except if the relation |
|
| 372 | + * is a HABTM relation, in which case it merges any non-foreign-key fields from |
|
| 373 | + * the join-model-object into the results |
|
| 374 | + * |
|
| 375 | + * @param array $primary_model_query_params query params for finding the item from which |
|
| 376 | + * relations will be based |
|
| 377 | + * @param \EE_Model_Relation_Base $relation |
|
| 378 | + * @param WP_REST_Request $request |
|
| 379 | + * @return array |
|
| 380 | + * @throws EE_Error |
|
| 381 | + * @throws InvalidArgumentException |
|
| 382 | + * @throws InvalidDataTypeException |
|
| 383 | + * @throws InvalidInterfaceException |
|
| 384 | + * @throws ReflectionException |
|
| 385 | + * @throws RestException |
|
| 386 | + * @throws \EventEspresso\core\exceptions\ModelConfigurationException |
|
| 387 | + */ |
|
| 388 | + protected function getEntitiesFromRelationUsingModelQueryParams($primary_model_query_params, $relation, $request) |
|
| 389 | + { |
|
| 390 | + $context = $this->validateContext($request->get_param('caps')); |
|
| 391 | + $model = $relation->get_this_model(); |
|
| 392 | + $related_model = $relation->get_other_model(); |
|
| 393 | + if (! isset($primary_model_query_params[0])) { |
|
| 394 | + $primary_model_query_params[0] = array(); |
|
| 395 | + } |
|
| 396 | + // check if they can access the 1st model object |
|
| 397 | + $primary_model_query_params = array( |
|
| 398 | + 0 => $primary_model_query_params[0], |
|
| 399 | + 'limit' => 1, |
|
| 400 | + ); |
|
| 401 | + if ($model instanceof EEM_Soft_Delete_Base) { |
|
| 402 | + $primary_model_query_params = $model->alter_query_params_so_deleted_and_undeleted_items_included( |
|
| 403 | + $primary_model_query_params |
|
| 404 | + ); |
|
| 405 | + } |
|
| 406 | + $restricted_query_params = $primary_model_query_params; |
|
| 407 | + $restricted_query_params['caps'] = $context; |
|
| 408 | + $restricted_query_params['limit'] = 1; |
|
| 409 | + $this->setDebugInfo('main model query params', $restricted_query_params); |
|
| 410 | + $this->setDebugInfo('missing caps', Capabilities::getMissingPermissionsString($related_model, $context)); |
|
| 411 | + $primary_model_rows = $model->get_all_wpdb_results($restricted_query_params); |
|
| 412 | + $primary_model_row = null; |
|
| 413 | + if (is_array($primary_model_rows)) { |
|
| 414 | + $primary_model_row = reset($primary_model_rows); |
|
| 415 | + } |
|
| 416 | + if (! ( |
|
| 417 | + Capabilities::currentUserHasPartialAccessTo($related_model, $context) |
|
| 418 | + && $primary_model_row |
|
| 419 | + ) |
|
| 420 | + ) { |
|
| 421 | + if ($relation instanceof EE_Belongs_To_Relation) { |
|
| 422 | + $related_model_name_maybe_plural = strtolower($related_model->get_this_model_name()); |
|
| 423 | + } else { |
|
| 424 | + $related_model_name_maybe_plural = EEH_Inflector::pluralize_and_lower( |
|
| 425 | + $related_model->get_this_model_name() |
|
| 426 | + ); |
|
| 427 | + } |
|
| 428 | + throw new RestException( |
|
| 429 | + sprintf('rest_%s_cannot_list', $related_model_name_maybe_plural), |
|
| 430 | + sprintf( |
|
| 431 | + __( |
|
| 432 | + 'Sorry, you are not allowed to list %1$s related to %2$s. Missing permissions: %3$s', |
|
| 433 | + 'event_espresso' |
|
| 434 | + ), |
|
| 435 | + $related_model_name_maybe_plural, |
|
| 436 | + $relation->get_this_model()->get_this_model_name(), |
|
| 437 | + implode( |
|
| 438 | + ',', |
|
| 439 | + array_keys( |
|
| 440 | + Capabilities::getMissingPermissions($related_model, $context) |
|
| 441 | + ) |
|
| 442 | + ) |
|
| 443 | + ), |
|
| 444 | + array('status' => 403) |
|
| 445 | + ); |
|
| 446 | + } |
|
| 447 | + |
|
| 448 | + $this->checkPassword( |
|
| 449 | + $model, |
|
| 450 | + $primary_model_row, |
|
| 451 | + $restricted_query_params, |
|
| 452 | + $request |
|
| 453 | + ); |
|
| 454 | + $query_params = $this->createModelQueryParams($relation->get_other_model(), $request->get_params()); |
|
| 455 | + foreach ($primary_model_query_params[0] as $where_condition_key => $where_condition_value) { |
|
| 456 | + $query_params[0][ $relation->get_this_model()->get_this_model_name() |
|
| 457 | + . '.' |
|
| 458 | + . $where_condition_key ] = $where_condition_value; |
|
| 459 | + } |
|
| 460 | + $query_params['default_where_conditions'] = 'none'; |
|
| 461 | + $query_params['caps'] = $context; |
|
| 462 | + if (! $request->get_header('no_rest_headers')) { |
|
| 463 | + $this->setHeadersFromQueryParams($relation->get_other_model(), $query_params); |
|
| 464 | + } |
|
| 465 | + /** @type array $results */ |
|
| 466 | + $results = $relation->get_other_model()->get_all_wpdb_results($query_params); |
|
| 467 | + $nice_results = array(); |
|
| 468 | + foreach ($results as $result) { |
|
| 469 | + $nice_result = $this->createEntityFromWpdbResult( |
|
| 470 | + $relation->get_other_model(), |
|
| 471 | + $result, |
|
| 472 | + $request |
|
| 473 | + ); |
|
| 474 | + if ($relation instanceof \EE_HABTM_Relation) { |
|
| 475 | + // put the unusual stuff (properties from the HABTM relation) first, and make sure |
|
| 476 | + // if there are conflicts we prefer the properties from the main model |
|
| 477 | + $join_model_result = $this->createEntityFromWpdbResult( |
|
| 478 | + $relation->get_join_model(), |
|
| 479 | + $result, |
|
| 480 | + $request |
|
| 481 | + ); |
|
| 482 | + $joined_result = array_merge($join_model_result, $nice_result); |
|
| 483 | + // but keep the meta stuff from the main model |
|
| 484 | + if (isset($nice_result['meta'])) { |
|
| 485 | + $joined_result['meta'] = $nice_result['meta']; |
|
| 486 | + } |
|
| 487 | + $nice_result = $joined_result; |
|
| 488 | + } |
|
| 489 | + $nice_results[] = $nice_result; |
|
| 490 | + } |
|
| 491 | + if ($relation instanceof EE_Belongs_To_Relation) { |
|
| 492 | + return array_shift($nice_results); |
|
| 493 | + } else { |
|
| 494 | + return $nice_results; |
|
| 495 | + } |
|
| 496 | + } |
|
| 497 | + |
|
| 498 | + |
|
| 499 | + /** |
|
| 500 | + * Gets the collection for given relation object |
|
| 501 | + * The same as Read::get_entities_from_model(), except if the relation |
|
| 502 | + * is a HABTM relation, in which case it merges any non-foreign-key fields from |
|
| 503 | + * the join-model-object into the results |
|
| 504 | + * |
|
| 505 | + * @param string $id the ID of the thing we are fetching related stuff from |
|
| 506 | + * @param \EE_Model_Relation_Base $relation |
|
| 507 | + * @param WP_REST_Request $request |
|
| 508 | + * @return array |
|
| 509 | + * @throws EE_Error |
|
| 510 | + */ |
|
| 511 | + public function getEntitiesFromRelation($id, $relation, $request) |
|
| 512 | + { |
|
| 513 | + if (! $relation->get_this_model()->has_primary_key_field()) { |
|
| 514 | + throw new EE_Error( |
|
| 515 | + sprintf( |
|
| 516 | + __( |
|
| 517 | + // @codingStandardsIgnoreStart |
|
| 518 | + 'Read::get_entities_from_relation should only be called from a model with a primary key, it was called from %1$s', |
|
| 519 | + // @codingStandardsIgnoreEnd |
|
| 520 | + 'event_espresso' |
|
| 521 | + ), |
|
| 522 | + $relation->get_this_model()->get_this_model_name() |
|
| 523 | + ) |
|
| 524 | + ); |
|
| 525 | + } |
|
| 526 | + // can we edit that main item? |
|
| 527 | + // if not, show nothing but an error |
|
| 528 | + // otherwise, please proceed |
|
| 529 | + return $this->getEntitiesFromRelationUsingModelQueryParams( |
|
| 530 | + array( |
|
| 531 | + array( |
|
| 532 | + $relation->get_this_model()->primary_key_name() => $id, |
|
| 533 | + ), |
|
| 534 | + ), |
|
| 535 | + $relation, |
|
| 536 | + $request |
|
| 537 | + ); |
|
| 538 | + } |
|
| 539 | + |
|
| 540 | + |
|
| 541 | + /** |
|
| 542 | + * Sets the headers that are based on the model and query params, |
|
| 543 | + * like the total records. This should only be called on the original request |
|
| 544 | + * from the client, not on subsequent internal |
|
| 545 | + * |
|
| 546 | + * @param EEM_Base $model |
|
| 547 | + * @param array $query_params |
|
| 548 | + * @return void |
|
| 549 | + */ |
|
| 550 | + protected function setHeadersFromQueryParams($model, $query_params) |
|
| 551 | + { |
|
| 552 | + $this->setDebugInfo('model query params', $query_params); |
|
| 553 | + $this->setDebugInfo( |
|
| 554 | + 'missing caps', |
|
| 555 | + Capabilities::getMissingPermissionsString($model, $query_params['caps']) |
|
| 556 | + ); |
|
| 557 | + // normally the limit to a 2-part array, where the 2nd item is the limit |
|
| 558 | + if (! isset($query_params['limit'])) { |
|
| 559 | + $query_params['limit'] = EED_Core_Rest_Api::get_default_query_limit(); |
|
| 560 | + } |
|
| 561 | + if (is_array($query_params['limit'])) { |
|
| 562 | + $limit_parts = $query_params['limit']; |
|
| 563 | + } else { |
|
| 564 | + $limit_parts = explode(',', $query_params['limit']); |
|
| 565 | + if (count($limit_parts) == 1) { |
|
| 566 | + $limit_parts = array(0, $limit_parts[0]); |
|
| 567 | + } |
|
| 568 | + } |
|
| 569 | + // remove the group by and having parts of the query, as those will |
|
| 570 | + // make the sql query return an array of values, instead of just a single value |
|
| 571 | + unset($query_params['group_by'], $query_params['having'], $query_params['limit']); |
|
| 572 | + $count = $model->count($query_params, null, true); |
|
| 573 | + $pages = $count / $limit_parts[1]; |
|
| 574 | + $this->setResponseHeader('Total', $count, false); |
|
| 575 | + $this->setResponseHeader('PageSize', $limit_parts[1], false); |
|
| 576 | + $this->setResponseHeader('TotalPages', ceil($pages), false); |
|
| 577 | + } |
|
| 578 | + |
|
| 579 | + |
|
| 580 | + /** |
|
| 581 | + * Changes database results into REST API entities |
|
| 582 | + * |
|
| 583 | + * @param EEM_Base $model |
|
| 584 | + * @param array $db_row like results from $wpdb->get_results() |
|
| 585 | + * @param WP_REST_Request $rest_request |
|
| 586 | + * @param string $deprecated no longer used |
|
| 587 | + * @return array ready for being converted into json for sending to client |
|
| 588 | + * @throws EE_Error |
|
| 589 | + * @throws RestException |
|
| 590 | + * @throws InvalidDataTypeException |
|
| 591 | + * @throws InvalidInterfaceException |
|
| 592 | + * @throws InvalidArgumentException |
|
| 593 | + * @throws ReflectionException |
|
| 594 | + */ |
|
| 595 | + public function createEntityFromWpdbResult($model, $db_row, $rest_request, $deprecated = null) |
|
| 596 | + { |
|
| 597 | + if (! $rest_request instanceof WP_REST_Request) { |
|
| 598 | + // ok so this was called in the old style, where the 3rd arg was |
|
| 599 | + // $include, and the 4th arg was $context |
|
| 600 | + // now setup the request just to avoid fatal errors, although we won't be able |
|
| 601 | + // to truly make use of it because it's kinda devoid of info |
|
| 602 | + $rest_request = new WP_REST_Request(); |
|
| 603 | + $rest_request->set_param('include', $rest_request); |
|
| 604 | + $rest_request->set_param('caps', $deprecated); |
|
| 605 | + } |
|
| 606 | + if ($rest_request->get_param('caps') == null) { |
|
| 607 | + $rest_request->set_param('caps', EEM_Base::caps_read); |
|
| 608 | + } |
|
| 609 | + $current_user_full_access_to_entity = $model->currentUserCan( |
|
| 610 | + EEM_Base::caps_read_admin, |
|
| 611 | + $model->deduce_fields_n_values_from_cols_n_values($db_row) |
|
| 612 | + ); |
|
| 613 | + $entity_array = $this->createBareEntityFromWpdbResults($model, $db_row); |
|
| 614 | + $entity_array = $this->addExtraFields($model, $db_row, $entity_array); |
|
| 615 | + $entity_array['_links'] = $this->getEntityLinks($model, $db_row, $entity_array); |
|
| 616 | + // when it's a regular read request for a model with a password and the password wasn't provided |
|
| 617 | + // remove the password protected fields |
|
| 618 | + $has_protected_fields = false; |
|
| 619 | + try { |
|
| 620 | + $this->checkPassword( |
|
| 621 | + $model, |
|
| 622 | + $db_row, |
|
| 623 | + $model->alter_query_params_to_restrict_by_ID( |
|
| 624 | + $model->get_index_primary_key_string( |
|
| 625 | + $model->deduce_fields_n_values_from_cols_n_values($db_row) |
|
| 626 | + ) |
|
| 627 | + ), |
|
| 628 | + $rest_request |
|
| 629 | + ); |
|
| 630 | + } catch (RestPasswordRequiredException $e) { |
|
| 631 | + if ($model->hasPassword()) { |
|
| 632 | + // just remove protected fields |
|
| 633 | + $has_protected_fields = true; |
|
| 634 | + $entity_array = Capabilities::filterOutPasswordProtectedFields( |
|
| 635 | + $entity_array, |
|
| 636 | + $model, |
|
| 637 | + $this->getModelVersionInfo() |
|
| 638 | + ); |
|
| 639 | + } else { |
|
| 640 | + // that's a problem. None of this should be accessible if no password was provided |
|
| 641 | + throw $e; |
|
| 642 | + } |
|
| 643 | + } |
|
| 644 | + |
|
| 645 | + $entity_array['_calculated_fields'] = $this->getEntityCalculations($model, $db_row, $rest_request, $has_protected_fields); |
|
| 646 | + $entity_array = apply_filters( |
|
| 647 | + 'FHEE__Read__create_entity_from_wpdb_results__entity_before_including_requested_models', |
|
| 648 | + $entity_array, |
|
| 649 | + $model, |
|
| 650 | + $rest_request->get_param('caps'), |
|
| 651 | + $rest_request, |
|
| 652 | + $this |
|
| 653 | + ); |
|
| 654 | + // add an empty protected property for now. If it's still around after we remove everything the request didn't |
|
| 655 | + // want, we'll populate it then. k? |
|
| 656 | + $entity_array['_protected'] = array(); |
|
| 657 | + // remove any properties the request didn't want. This way _protected won't bother mentioning them |
|
| 658 | + $entity_array = $this->includeOnlyRequestedProperties($model, $rest_request, $entity_array); |
|
| 659 | + $entity_array = $this->includeRequestedModels($model, $rest_request, $entity_array, $db_row, $has_protected_fields); |
|
| 660 | + // if they still wanted the _protected property, add it. |
|
| 661 | + if (isset($entity_array['_protected'])) { |
|
| 662 | + $entity_array = $this->addProtectedProperty($model, $entity_array, $has_protected_fields); |
|
| 663 | + } |
|
| 664 | + $entity_array = apply_filters( |
|
| 665 | + 'FHEE__Read__create_entity_from_wpdb_results__entity_before_inaccessible_field_removal', |
|
| 666 | + $entity_array, |
|
| 667 | + $model, |
|
| 668 | + $rest_request->get_param('caps'), |
|
| 669 | + $rest_request, |
|
| 670 | + $this |
|
| 671 | + ); |
|
| 672 | + if (! $current_user_full_access_to_entity) { |
|
| 673 | + $result_without_inaccessible_fields = Capabilities::filterOutInaccessibleEntityFields( |
|
| 674 | + $entity_array, |
|
| 675 | + $model, |
|
| 676 | + $rest_request->get_param('caps'), |
|
| 677 | + $this->getModelVersionInfo() |
|
| 678 | + ); |
|
| 679 | + } else { |
|
| 680 | + $result_without_inaccessible_fields = $entity_array; |
|
| 681 | + } |
|
| 682 | + $this->setDebugInfo( |
|
| 683 | + 'inaccessible fields', |
|
| 684 | + array_keys(array_diff_key((array) $entity_array, (array) $result_without_inaccessible_fields)) |
|
| 685 | + ); |
|
| 686 | + return apply_filters( |
|
| 687 | + 'FHEE__Read__create_entity_from_wpdb_results__entity_return', |
|
| 688 | + $result_without_inaccessible_fields, |
|
| 689 | + $model, |
|
| 690 | + $rest_request->get_param('caps') |
|
| 691 | + ); |
|
| 692 | + } |
|
| 693 | + |
|
| 694 | + /** |
|
| 695 | + * Returns an array describing which fields can be protected, and which actually were removed this request |
|
| 696 | + * @since 4.9.74.p |
|
| 697 | + * @param $model |
|
| 698 | + * @param $results_so_far |
|
| 699 | + * @param $protected |
|
| 700 | + * @return array results |
|
| 701 | + */ |
|
| 702 | + protected function addProtectedProperty(EEM_Base $model, $results_so_far, $protected) |
|
| 703 | + { |
|
| 704 | + if (! $model->hasPassword() || ! $protected) { |
|
| 705 | + return $results_so_far; |
|
| 706 | + } |
|
| 707 | + $password_field = $model->getPasswordField(); |
|
| 708 | + $all_protected = array_merge( |
|
| 709 | + array($password_field->get_name()), |
|
| 710 | + $password_field->protectedFields() |
|
| 711 | + ); |
|
| 712 | + $fields_included = array_keys($results_so_far); |
|
| 713 | + $fields_included = array_intersect( |
|
| 714 | + $all_protected, |
|
| 715 | + $fields_included |
|
| 716 | + ); |
|
| 717 | + foreach ($fields_included as $field_name) { |
|
| 718 | + $results_so_far['_protected'][] = $field_name ; |
|
| 719 | + } |
|
| 720 | + return $results_so_far; |
|
| 721 | + } |
|
| 722 | + |
|
| 723 | + /** |
|
| 724 | + * Creates a REST entity array (JSON object we're going to return in the response, but |
|
| 725 | + * for now still a PHP array, but soon enough we'll call json_encode on it, don't worry), |
|
| 726 | + * from $wpdb->get_row( $sql, ARRAY_A) |
|
| 727 | + * |
|
| 728 | + * @param EEM_Base $model |
|
| 729 | + * @param array $db_row |
|
| 730 | + * @return array entity mostly ready for converting to JSON and sending in the response |
|
| 731 | + */ |
|
| 732 | + protected function createBareEntityFromWpdbResults(EEM_Base $model, $db_row) |
|
| 733 | + { |
|
| 734 | + $result = $model->deduce_fields_n_values_from_cols_n_values($db_row); |
|
| 735 | + $result = array_intersect_key( |
|
| 736 | + $result, |
|
| 737 | + $this->getModelVersionInfo()->fieldsOnModelInThisVersion($model) |
|
| 738 | + ); |
|
| 739 | + // if this is a CPT, we need to set the global $post to it, |
|
| 740 | + // otherwise shortcodes etc won't work properly while rendering it |
|
| 741 | + if ($model instanceof \EEM_CPT_Base) { |
|
| 742 | + $do_chevy_shuffle = true; |
|
| 743 | + } else { |
|
| 744 | + $do_chevy_shuffle = false; |
|
| 745 | + } |
|
| 746 | + if ($do_chevy_shuffle) { |
|
| 747 | + global $post; |
|
| 748 | + $old_post = $post; |
|
| 749 | + $post = get_post($result[ $model->primary_key_name() ]); |
|
| 750 | + if (! $post instanceof \WP_Post) { |
|
| 751 | + // well that's weird, because $result is what we JUST fetched from the database |
|
| 752 | + throw new RestException( |
|
| 753 | + 'error_fetching_post_from_database_results', |
|
| 754 | + esc_html__( |
|
| 755 | + 'An item was retrieved from the database but it\'s not a WP_Post like it should be.', |
|
| 756 | + 'event_espresso' |
|
| 757 | + ) |
|
| 758 | + ); |
|
| 759 | + } |
|
| 760 | + $model_object_classname = 'EE_' . $model->get_this_model_name(); |
|
| 761 | + $post->{$model_object_classname} = \EE_Registry::instance()->load_class( |
|
| 762 | + $model_object_classname, |
|
| 763 | + $result, |
|
| 764 | + false, |
|
| 765 | + false |
|
| 766 | + ); |
|
| 767 | + } |
|
| 768 | + foreach ($result as $field_name => $field_value) { |
|
| 769 | + $field_obj = $model->field_settings_for($field_name); |
|
| 770 | + if ($this->isSubclassOfOne($field_obj, $this->getModelVersionInfo()->fieldsIgnored())) { |
|
| 771 | + unset($result[ $field_name ]); |
|
| 772 | + } elseif ($this->isSubclassOfOne( |
|
| 773 | + $field_obj, |
|
| 774 | + $this->getModelVersionInfo()->fieldsThatHaveRenderedFormat() |
|
| 775 | + ) |
|
| 776 | + ) { |
|
| 777 | + $result[ $field_name ] = array( |
|
| 778 | + 'raw' => $this->prepareFieldObjValueForJson($field_obj, $field_value), |
|
| 779 | + 'rendered' => $this->prepareFieldObjValueForJson($field_obj, $field_value, 'pretty'), |
|
| 780 | + ); |
|
| 781 | + } elseif ($this->isSubclassOfOne( |
|
| 782 | + $field_obj, |
|
| 783 | + $this->getModelVersionInfo()->fieldsThatHavePrettyFormat() |
|
| 784 | + ) |
|
| 785 | + ) { |
|
| 786 | + $result[ $field_name ] = array( |
|
| 787 | + 'raw' => $this->prepareFieldObjValueForJson($field_obj, $field_value), |
|
| 788 | + 'pretty' => $this->prepareFieldObjValueForJson($field_obj, $field_value, 'pretty'), |
|
| 789 | + ); |
|
| 790 | + } elseif ($field_obj instanceof \EE_Datetime_Field) { |
|
| 791 | + $field_value = $field_obj->prepare_for_set_from_db($field_value); |
|
| 792 | + // if the value is null, but we're not supposed to permit null, then set to the field's default |
|
| 793 | + if (is_null($field_value)) { |
|
| 794 | + $field_value = $field_obj->getDefaultDateTimeObj(); |
|
| 795 | + } |
|
| 796 | + if (is_null($field_value)) { |
|
| 797 | + $gmt_date = $local_date = ModelDataTranslator::prepareFieldValuesForJson( |
|
| 798 | + $field_obj, |
|
| 799 | + $field_value, |
|
| 800 | + $this->getModelVersionInfo()->requestedVersion() |
|
| 801 | + ); |
|
| 802 | + } else { |
|
| 803 | + $timezone = $field_value->getTimezone(); |
|
| 804 | + EEH_DTT_Helper::setTimezone($field_value, new DateTimeZone('UTC')); |
|
| 805 | + $gmt_date = ModelDataTranslator::prepareFieldValuesForJson( |
|
| 806 | + $field_obj, |
|
| 807 | + $field_value, |
|
| 808 | + $this->getModelVersionInfo()->requestedVersion() |
|
| 809 | + ); |
|
| 810 | + EEH_DTT_Helper::setTimezone($field_value, $timezone); |
|
| 811 | + $local_date = ModelDataTranslator::prepareFieldValuesForJson( |
|
| 812 | + $field_obj, |
|
| 813 | + $field_value, |
|
| 814 | + $this->getModelVersionInfo()->requestedVersion() |
|
| 815 | + ); |
|
| 816 | + } |
|
| 817 | + $result[ $field_name . '_gmt' ] = $gmt_date; |
|
| 818 | + $result[ $field_name ] = $local_date; |
|
| 819 | + } else { |
|
| 820 | + $result[ $field_name ] = $this->prepareFieldObjValueForJson($field_obj, $field_value); |
|
| 821 | + } |
|
| 822 | + } |
|
| 823 | + if ($do_chevy_shuffle) { |
|
| 824 | + $post = $old_post; |
|
| 825 | + } |
|
| 826 | + return $result; |
|
| 827 | + } |
|
| 828 | + |
|
| 829 | + |
|
| 830 | + /** |
|
| 831 | + * Takes a value all the way from the DB representation, to the model object's representation, to the |
|
| 832 | + * user-facing PHP representation, to the REST API representation. (Assumes you've already taken from the DB |
|
| 833 | + * representation using $field_obj->prepare_for_set_from_db()) |
|
| 834 | + * |
|
| 835 | + * @param EE_Model_Field_Base $field_obj |
|
| 836 | + * @param mixed $value as it's stored on a model object |
|
| 837 | + * @param string $format valid values are 'normal' (default), 'pretty', 'datetime_obj' |
|
| 838 | + * @return mixed |
|
| 839 | + * @throws ObjectDetectedException if $value contains a PHP object |
|
| 840 | + */ |
|
| 841 | + protected function prepareFieldObjValueForJson(EE_Model_Field_Base $field_obj, $value, $format = 'normal') |
|
| 842 | + { |
|
| 843 | + $value = $field_obj->prepare_for_set_from_db($value); |
|
| 844 | + switch ($format) { |
|
| 845 | + case 'pretty': |
|
| 846 | + $value = $field_obj->prepare_for_pretty_echoing($value); |
|
| 847 | + break; |
|
| 848 | + case 'normal': |
|
| 849 | + default: |
|
| 850 | + $value = $field_obj->prepare_for_get($value); |
|
| 851 | + break; |
|
| 852 | + } |
|
| 853 | + return ModelDataTranslator::prepareFieldValuesForJson( |
|
| 854 | + $field_obj, |
|
| 855 | + $value, |
|
| 856 | + $this->getModelVersionInfo()->requestedVersion() |
|
| 857 | + ); |
|
| 858 | + } |
|
| 859 | + |
|
| 860 | + |
|
| 861 | + /** |
|
| 862 | + * Adds a few extra fields to the entity response |
|
| 863 | + * |
|
| 864 | + * @param EEM_Base $model |
|
| 865 | + * @param array $db_row |
|
| 866 | + * @param array $entity_array |
|
| 867 | + * @return array modified entity |
|
| 868 | + */ |
|
| 869 | + protected function addExtraFields(EEM_Base $model, $db_row, $entity_array) |
|
| 870 | + { |
|
| 871 | + if ($model instanceof EEM_CPT_Base) { |
|
| 872 | + $entity_array['link'] = get_permalink($db_row[ $model->get_primary_key_field()->get_qualified_column() ]); |
|
| 873 | + } |
|
| 874 | + return $entity_array; |
|
| 875 | + } |
|
| 876 | + |
|
| 877 | + |
|
| 878 | + /** |
|
| 879 | + * Gets links we want to add to the response |
|
| 880 | + * |
|
| 881 | + * @global \WP_REST_Server $wp_rest_server |
|
| 882 | + * @param EEM_Base $model |
|
| 883 | + * @param array $db_row |
|
| 884 | + * @param array $entity_array |
|
| 885 | + * @return array the _links item in the entity |
|
| 886 | + */ |
|
| 887 | + protected function getEntityLinks($model, $db_row, $entity_array) |
|
| 888 | + { |
|
| 889 | + // add basic links |
|
| 890 | + $links = array(); |
|
| 891 | + if ($model->has_primary_key_field()) { |
|
| 892 | + $links['self'] = array( |
|
| 893 | + array( |
|
| 894 | + 'href' => $this->getVersionedLinkTo( |
|
| 895 | + EEH_Inflector::pluralize_and_lower($model->get_this_model_name()) |
|
| 896 | + . '/' |
|
| 897 | + . $entity_array[ $model->primary_key_name() ] |
|
| 898 | + ), |
|
| 899 | + ), |
|
| 900 | + ); |
|
| 901 | + } |
|
| 902 | + $links['collection'] = array( |
|
| 903 | + array( |
|
| 904 | + 'href' => $this->getVersionedLinkTo( |
|
| 905 | + EEH_Inflector::pluralize_and_lower($model->get_this_model_name()) |
|
| 906 | + ), |
|
| 907 | + ), |
|
| 908 | + ); |
|
| 909 | + // add links to related models |
|
| 910 | + if ($model->has_primary_key_field()) { |
|
| 911 | + foreach ($this->getModelVersionInfo()->relationSettings($model) as $relation_name => $relation_obj) { |
|
| 912 | + $related_model_part = Read::getRelatedEntityName($relation_name, $relation_obj); |
|
| 913 | + $links[ EED_Core_Rest_Api::ee_api_link_namespace . $related_model_part ] = array( |
|
| 914 | + array( |
|
| 915 | + 'href' => $this->getVersionedLinkTo( |
|
| 916 | + EEH_Inflector::pluralize_and_lower($model->get_this_model_name()) |
|
| 917 | + . '/' |
|
| 918 | + . $entity_array[ $model->primary_key_name() ] |
|
| 919 | + . '/' |
|
| 920 | + . $related_model_part |
|
| 921 | + ), |
|
| 922 | + 'single' => $relation_obj instanceof EE_Belongs_To_Relation ? true : false, |
|
| 923 | + ), |
|
| 924 | + ); |
|
| 925 | + } |
|
| 926 | + } |
|
| 927 | + return $links; |
|
| 928 | + } |
|
| 929 | + |
|
| 930 | + |
|
| 931 | + /** |
|
| 932 | + * Adds the included models indicated in the request to the entity provided |
|
| 933 | + * |
|
| 934 | + * @param EEM_Base $model |
|
| 935 | + * @param WP_REST_Request $rest_request |
|
| 936 | + * @param array $entity_array |
|
| 937 | + * @param array $db_row |
|
| 938 | + * @param boolean $included_items_protected if the original item is password protected, don't include any related models. |
|
| 939 | + * @return array the modified entity |
|
| 940 | + * @throws RestException |
|
| 941 | + */ |
|
| 942 | + protected function includeRequestedModels( |
|
| 943 | + EEM_Base $model, |
|
| 944 | + WP_REST_Request $rest_request, |
|
| 945 | + $entity_array, |
|
| 946 | + $db_row = array(), |
|
| 947 | + $included_items_protected = false |
|
| 948 | + ) { |
|
| 949 | + // if $db_row not included, hope the entity array has what we need |
|
| 950 | + if (! $db_row) { |
|
| 951 | + $db_row = $entity_array; |
|
| 952 | + } |
|
| 953 | + $relation_settings = $this->getModelVersionInfo()->relationSettings($model); |
|
| 954 | + foreach ($relation_settings as $relation_name => $relation_obj) { |
|
| 955 | + $related_fields_to_include = $this->explodeAndGetItemsPrefixedWith( |
|
| 956 | + $rest_request->get_param('include'), |
|
| 957 | + $relation_name |
|
| 958 | + ); |
|
| 959 | + $related_fields_to_calculate = $this->explodeAndGetItemsPrefixedWith( |
|
| 960 | + $rest_request->get_param('calculate'), |
|
| 961 | + $relation_name |
|
| 962 | + ); |
|
| 963 | + // did they specify they wanted to include a related model, or |
|
| 964 | + // specific fields from a related model? |
|
| 965 | + // or did they specify to calculate a field from a related model? |
|
| 966 | + if ($related_fields_to_include || $related_fields_to_calculate) { |
|
| 967 | + // if so, we should include at least some part of the related model |
|
| 968 | + $pretend_related_request = new WP_REST_Request(); |
|
| 969 | + $pretend_related_request->set_query_params( |
|
| 970 | + array( |
|
| 971 | + 'caps' => $rest_request->get_param('caps'), |
|
| 972 | + 'include' => $related_fields_to_include, |
|
| 973 | + 'calculate' => $related_fields_to_calculate, |
|
| 974 | + 'password' => $rest_request->get_param('password') |
|
| 975 | + ) |
|
| 976 | + ); |
|
| 977 | + $pretend_related_request->add_header('no_rest_headers', true); |
|
| 978 | + $primary_model_query_params = $model->alter_query_params_to_restrict_by_ID( |
|
| 979 | + $model->get_index_primary_key_string( |
|
| 980 | + $model->deduce_fields_n_values_from_cols_n_values($db_row) |
|
| 981 | + ) |
|
| 982 | + ); |
|
| 983 | + if (! $included_items_protected) { |
|
| 984 | + try { |
|
| 985 | + $related_results = $this->getEntitiesFromRelationUsingModelQueryParams( |
|
| 986 | + $primary_model_query_params, |
|
| 987 | + $relation_obj, |
|
| 988 | + $pretend_related_request |
|
| 989 | + ); |
|
| 990 | + } catch (RestException $e) { |
|
| 991 | + $related_results = null; |
|
| 992 | + } |
|
| 993 | + } else { |
|
| 994 | + // they're protected, hide them. |
|
| 995 | + $related_results = null; |
|
| 996 | + $entity_array['_protected'][] = Read::getRelatedEntityName($relation_name, $relation_obj); |
|
| 997 | + } |
|
| 998 | + if ($related_results instanceof WP_Error || $related_results === null) { |
|
| 999 | + $related_results = $relation_obj instanceof EE_Belongs_To_Relation ? null : array(); |
|
| 1000 | + } |
|
| 1001 | + $entity_array[ Read::getRelatedEntityName($relation_name, $relation_obj) ] = $related_results; |
|
| 1002 | + } |
|
| 1003 | + } |
|
| 1004 | + return $entity_array; |
|
| 1005 | + } |
|
| 1006 | + |
|
| 1007 | + /** |
|
| 1008 | + * If the user has requested only specific properties (including meta properties like _links or _protected) |
|
| 1009 | + * remove everything else. |
|
| 1010 | + * @since 4.9.74.p |
|
| 1011 | + * @param EEM_Base $model |
|
| 1012 | + * @param WP_REST_Request $rest_request |
|
| 1013 | + * @param $entity_array |
|
| 1014 | + * @return array |
|
| 1015 | + * @throws EE_Error |
|
| 1016 | + */ |
|
| 1017 | + protected function includeOnlyRequestedProperties( |
|
| 1018 | + EEM_Base $model, |
|
| 1019 | + WP_REST_Request $rest_request, |
|
| 1020 | + $entity_array |
|
| 1021 | + ) { |
|
| 1022 | + |
|
| 1023 | + $includes_for_this_model = $this->explodeAndGetItemsPrefixedWith($rest_request->get_param('include'), ''); |
|
| 1024 | + $includes_for_this_model = $this->removeModelNamesFromArray($includes_for_this_model); |
|
| 1025 | + // if they passed in * or didn't specify any includes, return everything |
|
| 1026 | + if (! in_array('*', $includes_for_this_model) |
|
| 1027 | + && ! empty($includes_for_this_model) |
|
| 1028 | + ) { |
|
| 1029 | + if ($model->has_primary_key_field()) { |
|
| 1030 | + // always include the primary key. ya just gotta know that at least |
|
| 1031 | + $includes_for_this_model[] = $model->primary_key_name(); |
|
| 1032 | + } |
|
| 1033 | + if ($this->explodeAndGetItemsPrefixedWith($rest_request->get_param('calculate'), '')) { |
|
| 1034 | + $includes_for_this_model[] = '_calculated_fields'; |
|
| 1035 | + } |
|
| 1036 | + $entity_array = array_intersect_key($entity_array, array_flip($includes_for_this_model)); |
|
| 1037 | + } |
|
| 1038 | + return $entity_array; |
|
| 1039 | + } |
|
| 1040 | + |
|
| 1041 | + |
|
| 1042 | + /** |
|
| 1043 | + * Returns a new array with all the names of models removed. Eg |
|
| 1044 | + * array( 'Event', 'Datetime.*', 'foobar' ) would become array( 'Datetime.*', 'foobar' ) |
|
| 1045 | + * |
|
| 1046 | + * @param array $arr |
|
| 1047 | + * @return array |
|
| 1048 | + */ |
|
| 1049 | + private function removeModelNamesFromArray($arr) |
|
| 1050 | + { |
|
| 1051 | + return array_diff($arr, array_keys(EE_Registry::instance()->non_abstract_db_models)); |
|
| 1052 | + } |
|
| 1053 | + |
|
| 1054 | + |
|
| 1055 | + /** |
|
| 1056 | + * Gets the calculated fields for the response |
|
| 1057 | + * |
|
| 1058 | + * @param EEM_Base $model |
|
| 1059 | + * @param array $wpdb_row |
|
| 1060 | + * @param WP_REST_Request $rest_request |
|
| 1061 | + * @param boolean $row_is_protected whether this row is password protected or not |
|
| 1062 | + * @return \stdClass the _calculations item in the entity |
|
| 1063 | + * @throws ObjectDetectedException if a default value has a PHP object, which should never do (and if we |
|
| 1064 | + * did, let's know about it ASAP, so let the exception bubble up) |
|
| 1065 | + */ |
|
| 1066 | + protected function getEntityCalculations($model, $wpdb_row, $rest_request, $row_is_protected = false) |
|
| 1067 | + { |
|
| 1068 | + $calculated_fields = $this->explodeAndGetItemsPrefixedWith( |
|
| 1069 | + $rest_request->get_param('calculate'), |
|
| 1070 | + '' |
|
| 1071 | + ); |
|
| 1072 | + // note: setting calculate=* doesn't do anything |
|
| 1073 | + $calculated_fields_to_return = new \stdClass(); |
|
| 1074 | + $protected_fields = array(); |
|
| 1075 | + foreach ($calculated_fields as $field_to_calculate) { |
|
| 1076 | + try { |
|
| 1077 | + // it's password protected, so they shouldn't be able to read this. Remove the value |
|
| 1078 | + $schema = $this->fields_calculator->getJsonSchemaForModel($model); |
|
| 1079 | + if ($row_is_protected |
|
| 1080 | + && isset($schema['properties'][ $field_to_calculate ]['protected']) |
|
| 1081 | + && $schema['properties'][ $field_to_calculate ]['protected']) { |
|
| 1082 | + $calculated_value = null; |
|
| 1083 | + $protected_fields[] = $field_to_calculate; |
|
| 1084 | + if ($schema['properties'][ $field_to_calculate ]['type']) { |
|
| 1085 | + switch ($schema['properties'][ $field_to_calculate ]['type']) { |
|
| 1086 | + case 'boolean': |
|
| 1087 | + $calculated_value = false; |
|
| 1088 | + break; |
|
| 1089 | + case 'integer': |
|
| 1090 | + $calculated_value = 0; |
|
| 1091 | + break; |
|
| 1092 | + case 'string': |
|
| 1093 | + $calculated_value = ''; |
|
| 1094 | + break; |
|
| 1095 | + case 'array': |
|
| 1096 | + $calculated_value = array(); |
|
| 1097 | + break; |
|
| 1098 | + case 'object': |
|
| 1099 | + $calculated_value = new stdClass(); |
|
| 1100 | + break; |
|
| 1101 | + } |
|
| 1102 | + } |
|
| 1103 | + } else { |
|
| 1104 | + $calculated_value = ModelDataTranslator::prepareFieldValueForJson( |
|
| 1105 | + null, |
|
| 1106 | + $this->fields_calculator->retrieveCalculatedFieldValue( |
|
| 1107 | + $model, |
|
| 1108 | + $field_to_calculate, |
|
| 1109 | + $wpdb_row, |
|
| 1110 | + $rest_request, |
|
| 1111 | + $this |
|
| 1112 | + ), |
|
| 1113 | + $this->getModelVersionInfo()->requestedVersion() |
|
| 1114 | + ); |
|
| 1115 | + } |
|
| 1116 | + $calculated_fields_to_return->{$field_to_calculate} = $calculated_value; |
|
| 1117 | + } catch (RestException $e) { |
|
| 1118 | + // if we don't have permission to read it, just leave it out. but let devs know about the problem |
|
| 1119 | + $this->setResponseHeader( |
|
| 1120 | + 'Notices-Field-Calculation-Errors[' |
|
| 1121 | + . $e->getStringCode() |
|
| 1122 | + . '][' |
|
| 1123 | + . $model->get_this_model_name() |
|
| 1124 | + . '][' |
|
| 1125 | + . $field_to_calculate |
|
| 1126 | + . ']', |
|
| 1127 | + $e->getMessage(), |
|
| 1128 | + true |
|
| 1129 | + ); |
|
| 1130 | + } |
|
| 1131 | + } |
|
| 1132 | + $calculated_fields_to_return->_protected = $protected_fields; |
|
| 1133 | + return $calculated_fields_to_return; |
|
| 1134 | + } |
|
| 1135 | + |
|
| 1136 | + |
|
| 1137 | + /** |
|
| 1138 | + * Gets the full URL to the resource, taking the requested version into account |
|
| 1139 | + * |
|
| 1140 | + * @param string $link_part_after_version_and_slash eg "events/10/datetimes" |
|
| 1141 | + * @return string url eg "http://mysite.com/wp-json/ee/v4.6/events/10/datetimes" |
|
| 1142 | + */ |
|
| 1143 | + public function getVersionedLinkTo($link_part_after_version_and_slash) |
|
| 1144 | + { |
|
| 1145 | + return rest_url( |
|
| 1146 | + EED_Core_Rest_Api::get_versioned_route_to( |
|
| 1147 | + $link_part_after_version_and_slash, |
|
| 1148 | + $this->getModelVersionInfo()->requestedVersion() |
|
| 1149 | + ) |
|
| 1150 | + ); |
|
| 1151 | + } |
|
| 1152 | + |
|
| 1153 | + |
|
| 1154 | + /** |
|
| 1155 | + * Gets the correct lowercase name for the relation in the API according |
|
| 1156 | + * to the relation's type |
|
| 1157 | + * |
|
| 1158 | + * @param string $relation_name |
|
| 1159 | + * @param \EE_Model_Relation_Base $relation_obj |
|
| 1160 | + * @return string |
|
| 1161 | + */ |
|
| 1162 | + public static function getRelatedEntityName($relation_name, $relation_obj) |
|
| 1163 | + { |
|
| 1164 | + if ($relation_obj instanceof EE_Belongs_To_Relation) { |
|
| 1165 | + return strtolower($relation_name); |
|
| 1166 | + } else { |
|
| 1167 | + return EEH_Inflector::pluralize_and_lower($relation_name); |
|
| 1168 | + } |
|
| 1169 | + } |
|
| 1170 | + |
|
| 1171 | + |
|
| 1172 | + /** |
|
| 1173 | + * Gets the one model object with the specified id for the specified model |
|
| 1174 | + * |
|
| 1175 | + * @param EEM_Base $model |
|
| 1176 | + * @param WP_REST_Request $request |
|
| 1177 | + * @return array |
|
| 1178 | + */ |
|
| 1179 | + public function getEntityFromModel($model, $request) |
|
| 1180 | + { |
|
| 1181 | + $context = $this->validateContext($request->get_param('caps')); |
|
| 1182 | + return $this->getOneOrReportPermissionError($model, $request, $context); |
|
| 1183 | + } |
|
| 1184 | + |
|
| 1185 | + |
|
| 1186 | + /** |
|
| 1187 | + * If a context is provided which isn't valid, maybe it was added in a future |
|
| 1188 | + * version so just treat it as a default read |
|
| 1189 | + * |
|
| 1190 | + * @param string $context |
|
| 1191 | + * @return string array key of EEM_Base::cap_contexts_to_cap_action_map() |
|
| 1192 | + */ |
|
| 1193 | + public function validateContext($context) |
|
| 1194 | + { |
|
| 1195 | + if (! $context) { |
|
| 1196 | + $context = EEM_Base::caps_read; |
|
| 1197 | + } |
|
| 1198 | + $valid_contexts = EEM_Base::valid_cap_contexts(); |
|
| 1199 | + if (in_array($context, $valid_contexts)) { |
|
| 1200 | + return $context; |
|
| 1201 | + } else { |
|
| 1202 | + return EEM_Base::caps_read; |
|
| 1203 | + } |
|
| 1204 | + } |
|
| 1205 | + |
|
| 1206 | + |
|
| 1207 | + /** |
|
| 1208 | + * Verifies the passed in value is an allowable default where conditions value. |
|
| 1209 | + * |
|
| 1210 | + * @param $default_query_params |
|
| 1211 | + * @return string |
|
| 1212 | + */ |
|
| 1213 | + public function validateDefaultQueryParams($default_query_params) |
|
| 1214 | + { |
|
| 1215 | + $valid_default_where_conditions_for_api_calls = array( |
|
| 1216 | + EEM_Base::default_where_conditions_all, |
|
| 1217 | + EEM_Base::default_where_conditions_minimum_all, |
|
| 1218 | + EEM_Base::default_where_conditions_minimum_others, |
|
| 1219 | + ); |
|
| 1220 | + if (! $default_query_params) { |
|
| 1221 | + $default_query_params = EEM_Base::default_where_conditions_all; |
|
| 1222 | + } |
|
| 1223 | + if (in_array( |
|
| 1224 | + $default_query_params, |
|
| 1225 | + $valid_default_where_conditions_for_api_calls, |
|
| 1226 | + true |
|
| 1227 | + )) { |
|
| 1228 | + return $default_query_params; |
|
| 1229 | + } else { |
|
| 1230 | + return EEM_Base::default_where_conditions_all; |
|
| 1231 | + } |
|
| 1232 | + } |
|
| 1233 | + |
|
| 1234 | + |
|
| 1235 | + /** |
|
| 1236 | + * Translates API filter get parameter into model query params @see https://github.com/eventespresso/event-espresso-core/tree/master/docs/G--Model-System/model-query-params.md#0-where-conditions. |
|
| 1237 | + * Note: right now the query parameter keys for fields (and related fields) |
|
| 1238 | + * can be left as-is, but it's quite possible this will change someday. |
|
| 1239 | + * Also, this method's contents might be candidate for moving to Model_Data_Translator |
|
| 1240 | + * |
|
| 1241 | + * @param EEM_Base $model |
|
| 1242 | + * @param array $query_parameters from $_GET parameter @see Read:handle_request_get_all |
|
| 1243 | + * @return array model query params (@see https://github.com/eventespresso/event-espresso-core/tree/master/docs/G--Model-System/model-query-params.md#0-where-conditions) |
|
| 1244 | + * or FALSE to indicate that absolutely no results should be returned |
|
| 1245 | + * @throws EE_Error |
|
| 1246 | + * @throws RestException |
|
| 1247 | + */ |
|
| 1248 | + public function createModelQueryParams($model, $query_params) |
|
| 1249 | + { |
|
| 1250 | + $model_query_params = array(); |
|
| 1251 | + if (isset($query_params['where'])) { |
|
| 1252 | + $model_query_params[0] = ModelDataTranslator::prepareConditionsQueryParamsForModels( |
|
| 1253 | + $query_params['where'], |
|
| 1254 | + $model, |
|
| 1255 | + $this->getModelVersionInfo()->requestedVersion() |
|
| 1256 | + ); |
|
| 1257 | + } |
|
| 1258 | + if (isset($query_params['order_by'])) { |
|
| 1259 | + $order_by = $query_params['order_by']; |
|
| 1260 | + } elseif (isset($query_params['orderby'])) { |
|
| 1261 | + $order_by = $query_params['orderby']; |
|
| 1262 | + } else { |
|
| 1263 | + $order_by = null; |
|
| 1264 | + } |
|
| 1265 | + if ($order_by !== null) { |
|
| 1266 | + if (is_array($order_by)) { |
|
| 1267 | + $order_by = ModelDataTranslator::prepareFieldNamesInArrayKeysFromJson($order_by); |
|
| 1268 | + } else { |
|
| 1269 | + // it's a single item |
|
| 1270 | + $order_by = ModelDataTranslator::prepareFieldNameFromJson($order_by); |
|
| 1271 | + } |
|
| 1272 | + $model_query_params['order_by'] = $order_by; |
|
| 1273 | + } |
|
| 1274 | + if (isset($query_params['group_by'])) { |
|
| 1275 | + $group_by = $query_params['group_by']; |
|
| 1276 | + } elseif (isset($query_params['groupby'])) { |
|
| 1277 | + $group_by = $query_params['groupby']; |
|
| 1278 | + } else { |
|
| 1279 | + $group_by = array_keys($model->get_combined_primary_key_fields()); |
|
| 1280 | + } |
|
| 1281 | + // make sure they're all real names |
|
| 1282 | + if (is_array($group_by)) { |
|
| 1283 | + $group_by = ModelDataTranslator::prepareFieldNamesFromJson($group_by); |
|
| 1284 | + } |
|
| 1285 | + if ($group_by !== null) { |
|
| 1286 | + $model_query_params['group_by'] = $group_by; |
|
| 1287 | + } |
|
| 1288 | + if (isset($query_params['having'])) { |
|
| 1289 | + $model_query_params['having'] = ModelDataTranslator::prepareConditionsQueryParamsForModels( |
|
| 1290 | + $query_params['having'], |
|
| 1291 | + $model, |
|
| 1292 | + $this->getModelVersionInfo()->requestedVersion() |
|
| 1293 | + ); |
|
| 1294 | + } |
|
| 1295 | + if (isset($query_params['order'])) { |
|
| 1296 | + $model_query_params['order'] = $query_params['order']; |
|
| 1297 | + } |
|
| 1298 | + if (isset($query_params['mine'])) { |
|
| 1299 | + $model_query_params = $model->alter_query_params_to_only_include_mine($model_query_params); |
|
| 1300 | + } |
|
| 1301 | + if (isset($query_params['limit'])) { |
|
| 1302 | + // limit should be either a string like '23' or '23,43', or an array with two items in it |
|
| 1303 | + if (! is_array($query_params['limit'])) { |
|
| 1304 | + $limit_array = explode(',', (string) $query_params['limit']); |
|
| 1305 | + } else { |
|
| 1306 | + $limit_array = $query_params['limit']; |
|
| 1307 | + } |
|
| 1308 | + $sanitized_limit = array(); |
|
| 1309 | + foreach ($limit_array as $key => $limit_part) { |
|
| 1310 | + if ($this->debug_mode && (! is_numeric($limit_part) || count($sanitized_limit) > 2)) { |
|
| 1311 | + throw new EE_Error( |
|
| 1312 | + sprintf( |
|
| 1313 | + __( |
|
| 1314 | + // @codingStandardsIgnoreStart |
|
| 1315 | + 'An invalid limit filter was provided. It was: %s. If the EE4 JSON REST API weren\'t in debug mode, this message would not appear.', |
|
| 1316 | + // @codingStandardsIgnoreEnd |
|
| 1317 | + 'event_espresso' |
|
| 1318 | + ), |
|
| 1319 | + wp_json_encode($query_params['limit']) |
|
| 1320 | + ) |
|
| 1321 | + ); |
|
| 1322 | + } |
|
| 1323 | + $sanitized_limit[] = (int) $limit_part; |
|
| 1324 | + } |
|
| 1325 | + $model_query_params['limit'] = implode(',', $sanitized_limit); |
|
| 1326 | + } else { |
|
| 1327 | + $model_query_params['limit'] = EED_Core_Rest_Api::get_default_query_limit(); |
|
| 1328 | + } |
|
| 1329 | + if (isset($query_params['caps'])) { |
|
| 1330 | + $model_query_params['caps'] = $this->validateContext($query_params['caps']); |
|
| 1331 | + } else { |
|
| 1332 | + $model_query_params['caps'] = EEM_Base::caps_read; |
|
| 1333 | + } |
|
| 1334 | + if (isset($query_params['default_where_conditions'])) { |
|
| 1335 | + $model_query_params['default_where_conditions'] = $this->validateDefaultQueryParams( |
|
| 1336 | + $query_params['default_where_conditions'] |
|
| 1337 | + ); |
|
| 1338 | + } |
|
| 1339 | + // if this is a model protected by a password on another model, exclude the password protected |
|
| 1340 | + // entities by default. But if they passed in a password, try to show them all. If the password is wrong, |
|
| 1341 | + // though, they'll get an error (see Read::createEntityFromWpdbResult() which calls Read::checkPassword) |
|
| 1342 | + if (! $model->hasPassword() |
|
| 1343 | + && $model->restrictedByRelatedModelPassword() |
|
| 1344 | + && $model_query_params['caps'] === EEM_Base::caps_read) { |
|
| 1345 | + if (empty($query_params['password'])) { |
|
| 1346 | + $model_query_params['exclude_protected'] = true; |
|
| 1347 | + } |
|
| 1348 | + } |
|
| 1349 | + |
|
| 1350 | + return apply_filters('FHEE__Read__create_model_query_params', $model_query_params, $query_params, $model); |
|
| 1351 | + } |
|
| 1352 | + |
|
| 1353 | + |
|
| 1354 | + /** |
|
| 1355 | + * Changes the REST-style query params for use in the models |
|
| 1356 | + * |
|
| 1357 | + * @deprecated |
|
| 1358 | + * @param EEM_Base $model |
|
| 1359 | + * @param array $query_params sub-array from @see EEM_Base::get_all() |
|
| 1360 | + * @return array |
|
| 1361 | + */ |
|
| 1362 | + public function prepareRestQueryParamsKeyForModels($model, $query_params) |
|
| 1363 | + { |
|
| 1364 | + $model_ready_query_params = array(); |
|
| 1365 | + foreach ($query_params as $key => $value) { |
|
| 1366 | + if (is_array($value)) { |
|
| 1367 | + $model_ready_query_params[ $key ] = $this->prepareRestQueryParamsKeyForModels($model, $value); |
|
| 1368 | + } else { |
|
| 1369 | + $model_ready_query_params[ $key ] = $value; |
|
| 1370 | + } |
|
| 1371 | + } |
|
| 1372 | + return $model_ready_query_params; |
|
| 1373 | + } |
|
| 1374 | + |
|
| 1375 | + |
|
| 1376 | + /** |
|
| 1377 | + * @deprecated instead use ModelDataTranslator::prepareFieldValuesFromJson() |
|
| 1378 | + * @param $model |
|
| 1379 | + * @param $query_params |
|
| 1380 | + * @return array |
|
| 1381 | + */ |
|
| 1382 | + public function prepareRestQueryParamsValuesForModels($model, $query_params) |
|
| 1383 | + { |
|
| 1384 | + $model_ready_query_params = array(); |
|
| 1385 | + foreach ($query_params as $key => $value) { |
|
| 1386 | + if (is_array($value)) { |
|
| 1387 | + $model_ready_query_params[ $key ] = $this->prepareRestQueryParamsValuesForModels($model, $value); |
|
| 1388 | + } else { |
|
| 1389 | + $model_ready_query_params[ $key ] = $value; |
|
| 1390 | + } |
|
| 1391 | + } |
|
| 1392 | + return $model_ready_query_params; |
|
| 1393 | + } |
|
| 1394 | + |
|
| 1395 | + |
|
| 1396 | + /** |
|
| 1397 | + * Explodes the string on commas, and only returns items with $prefix followed by a period. |
|
| 1398 | + * If no prefix is specified, returns items with no period. |
|
| 1399 | + * |
|
| 1400 | + * @param string|array $string_to_explode eg "jibba,jabba, blah, blah, blah" or array('jibba', 'jabba' ) |
|
| 1401 | + * @param string $prefix "Event" or "foobar" |
|
| 1402 | + * @return array $string_to_exploded exploded on COMMAS, and if a prefix was specified |
|
| 1403 | + * we only return strings starting with that and a period; if no prefix was |
|
| 1404 | + * specified we return all items containing NO periods |
|
| 1405 | + */ |
|
| 1406 | + public function explodeAndGetItemsPrefixedWith($string_to_explode, $prefix) |
|
| 1407 | + { |
|
| 1408 | + if (is_string($string_to_explode)) { |
|
| 1409 | + $exploded_contents = explode(',', $string_to_explode); |
|
| 1410 | + } elseif (is_array($string_to_explode)) { |
|
| 1411 | + $exploded_contents = $string_to_explode; |
|
| 1412 | + } else { |
|
| 1413 | + $exploded_contents = array(); |
|
| 1414 | + } |
|
| 1415 | + // if the string was empty, we want an empty array |
|
| 1416 | + $exploded_contents = array_filter($exploded_contents); |
|
| 1417 | + $contents_with_prefix = array(); |
|
| 1418 | + foreach ($exploded_contents as $item) { |
|
| 1419 | + $item = trim($item); |
|
| 1420 | + // if no prefix was provided, so we look for items with no "." in them |
|
| 1421 | + if (! $prefix) { |
|
| 1422 | + // does this item have a period? |
|
| 1423 | + if (strpos($item, '.') === false) { |
|
| 1424 | + // if not, then its what we're looking for |
|
| 1425 | + $contents_with_prefix[] = $item; |
|
| 1426 | + } |
|
| 1427 | + } elseif (strpos($item, $prefix . '.') === 0) { |
|
| 1428 | + // this item has the prefix and a period, grab it |
|
| 1429 | + $contents_with_prefix[] = substr( |
|
| 1430 | + $item, |
|
| 1431 | + strpos($item, $prefix . '.') + strlen($prefix . '.') |
|
| 1432 | + ); |
|
| 1433 | + } elseif ($item === $prefix) { |
|
| 1434 | + // this item is JUST the prefix |
|
| 1435 | + // so let's grab everything after, which is a blank string |
|
| 1436 | + $contents_with_prefix[] = ''; |
|
| 1437 | + } |
|
| 1438 | + } |
|
| 1439 | + return $contents_with_prefix; |
|
| 1440 | + } |
|
| 1441 | + |
|
| 1442 | + |
|
| 1443 | + /** |
|
| 1444 | + * @deprecated since 4.8.36.rc.001 You should instead use Read::explode_and_get_items_prefixed_with. |
|
| 1445 | + * Deprecated because its return values were really quite confusing- sometimes it returned |
|
| 1446 | + * an empty array (when the include string was blank or '*') or sometimes it returned |
|
| 1447 | + * array('*') (when you provided a model and a model of that kind was found). |
|
| 1448 | + * Parses the $include_string so we fetch all the field names relating to THIS model |
|
| 1449 | + * (ie have NO period in them), or for the provided model (ie start with the model |
|
| 1450 | + * name and then a period). |
|
| 1451 | + * @param string $include_string @see Read:handle_request_get_all |
|
| 1452 | + * @param string $model_name |
|
| 1453 | + * @return array of fields for this model. If $model_name is provided, then |
|
| 1454 | + * the fields for that model, with the model's name removed from each. |
|
| 1455 | + * If $include_string was blank or '*' returns an empty array |
|
| 1456 | + */ |
|
| 1457 | + public function extractIncludesForThisModel($include_string, $model_name = null) |
|
| 1458 | + { |
|
| 1459 | + if (is_array($include_string)) { |
|
| 1460 | + $include_string = implode(',', $include_string); |
|
| 1461 | + } |
|
| 1462 | + if ($include_string === '*' || $include_string === '') { |
|
| 1463 | + return array(); |
|
| 1464 | + } |
|
| 1465 | + $includes = explode(',', $include_string); |
|
| 1466 | + $extracted_fields_to_include = array(); |
|
| 1467 | + if ($model_name) { |
|
| 1468 | + foreach ($includes as $field_to_include) { |
|
| 1469 | + $field_to_include = trim($field_to_include); |
|
| 1470 | + if (strpos($field_to_include, $model_name . '.') === 0) { |
|
| 1471 | + // found the model name at the exact start |
|
| 1472 | + $field_sans_model_name = str_replace($model_name . '.', '', $field_to_include); |
|
| 1473 | + $extracted_fields_to_include[] = $field_sans_model_name; |
|
| 1474 | + } elseif ($field_to_include == $model_name) { |
|
| 1475 | + $extracted_fields_to_include[] = '*'; |
|
| 1476 | + } |
|
| 1477 | + } |
|
| 1478 | + } else { |
|
| 1479 | + // look for ones with no period |
|
| 1480 | + foreach ($includes as $field_to_include) { |
|
| 1481 | + $field_to_include = trim($field_to_include); |
|
| 1482 | + if (strpos($field_to_include, '.') === false |
|
| 1483 | + && ! $this->getModelVersionInfo()->isModelNameInThisVersion($field_to_include) |
|
| 1484 | + ) { |
|
| 1485 | + $extracted_fields_to_include[] = $field_to_include; |
|
| 1486 | + } |
|
| 1487 | + } |
|
| 1488 | + } |
|
| 1489 | + return $extracted_fields_to_include; |
|
| 1490 | + } |
|
| 1491 | + |
|
| 1492 | + |
|
| 1493 | + /** |
|
| 1494 | + * Gets the single item using the model according to the request in the context given, otherwise |
|
| 1495 | + * returns that it's inaccessible to the current user |
|
| 1496 | + * |
|
| 1497 | + * @param EEM_Base $model |
|
| 1498 | + * @param WP_REST_Request $request |
|
| 1499 | + * @param null $context |
|
| 1500 | + * @return array |
|
| 1501 | + * @throws EE_Error |
|
| 1502 | + */ |
|
| 1503 | + public function getOneOrReportPermissionError(EEM_Base $model, WP_REST_Request $request, $context = null) |
|
| 1504 | + { |
|
| 1505 | + $query_params = array(array($model->primary_key_name() => $request->get_param('id')), 'limit' => 1); |
|
| 1506 | + if ($model instanceof EEM_Soft_Delete_Base) { |
|
| 1507 | + $query_params = $model->alter_query_params_so_deleted_and_undeleted_items_included($query_params); |
|
| 1508 | + } |
|
| 1509 | + $restricted_query_params = $query_params; |
|
| 1510 | + $restricted_query_params['caps'] = $context; |
|
| 1511 | + $this->setDebugInfo('model query params', $restricted_query_params); |
|
| 1512 | + $model_rows = $model->get_all_wpdb_results($restricted_query_params); |
|
| 1513 | + if (! empty($model_rows)) { |
|
| 1514 | + return $this->createEntityFromWpdbResult( |
|
| 1515 | + $model, |
|
| 1516 | + reset($model_rows), |
|
| 1517 | + $request |
|
| 1518 | + ); |
|
| 1519 | + } else { |
|
| 1520 | + // ok let's test to see if we WOULD have found it, had we not had restrictions from missing capabilities |
|
| 1521 | + $lowercase_model_name = strtolower($model->get_this_model_name()); |
|
| 1522 | + if ($model->exists($query_params)) { |
|
| 1523 | + // you got shafted- it existed but we didn't want to tell you! |
|
| 1524 | + throw new RestException( |
|
| 1525 | + 'rest_user_cannot_' . $context, |
|
| 1526 | + sprintf( |
|
| 1527 | + __('Sorry, you cannot %1$s this %2$s. Missing permissions are: %3$s', 'event_espresso'), |
|
| 1528 | + $context, |
|
| 1529 | + $lowercase_model_name, |
|
| 1530 | + Capabilities::getMissingPermissionsString( |
|
| 1531 | + $model, |
|
| 1532 | + $context |
|
| 1533 | + ) |
|
| 1534 | + ), |
|
| 1535 | + array('status' => 403) |
|
| 1536 | + ); |
|
| 1537 | + } else { |
|
| 1538 | + // it's not you. It just doesn't exist |
|
| 1539 | + throw new RestException( |
|
| 1540 | + sprintf('rest_%s_invalid_id', $lowercase_model_name), |
|
| 1541 | + sprintf(__('Invalid %s ID.', 'event_espresso'), $lowercase_model_name), |
|
| 1542 | + array('status' => 404) |
|
| 1543 | + ); |
|
| 1544 | + } |
|
| 1545 | + } |
|
| 1546 | + } |
|
| 1547 | + |
|
| 1548 | + /** |
|
| 1549 | + * Checks that if this content requires a password to be read, that it's been provided and is correct. |
|
| 1550 | + * @since 4.9.74.p |
|
| 1551 | + * @param EEM_Base $model |
|
| 1552 | + * @param $model_row |
|
| 1553 | + * @param $query_params Adds 'default_where_conditions' => 'minimum' to ensure we don't confuse trashed with |
|
| 1554 | + * password protected. |
|
| 1555 | + * @param WP_REST_Request $request |
|
| 1556 | + * @throws EE_Error |
|
| 1557 | + * @throws InvalidArgumentException |
|
| 1558 | + * @throws InvalidDataTypeException |
|
| 1559 | + * @throws InvalidInterfaceException |
|
| 1560 | + * @throws RestPasswordRequiredException |
|
| 1561 | + * @throws RestPasswordIncorrectException |
|
| 1562 | + * @throws \EventEspresso\core\exceptions\ModelConfigurationException |
|
| 1563 | + * @throws ReflectionException |
|
| 1564 | + */ |
|
| 1565 | + protected function checkPassword(EEM_Base $model, $model_row, $query_params, WP_REST_Request $request) |
|
| 1566 | + { |
|
| 1567 | + $query_params['default_where_conditions'] = 'minimum'; |
|
| 1568 | + // stuff is only "protected" for front-end requests. Elsewhere, you either get full permission to access the object |
|
| 1569 | + // or you don't. |
|
| 1570 | + $request_caps = $request->get_param('caps'); |
|
| 1571 | + if (isset($request_caps) && $request_caps !== EEM_Base::caps_read) { |
|
| 1572 | + return; |
|
| 1573 | + } |
|
| 1574 | + // if this entity requires a password, they better give it and it better be right! |
|
| 1575 | + if ($model->hasPassword() |
|
| 1576 | + && $model_row[ $model->getPasswordField()->get_qualified_column() ] !== '') { |
|
| 1577 | + if (empty($request['password'])) { |
|
| 1578 | + throw new RestPasswordRequiredException(); |
|
| 1579 | + } elseif (!hash_equals( |
|
| 1580 | + $model_row[ $model->getPasswordField()->get_qualified_column() ], |
|
| 1581 | + $request['password'] |
|
| 1582 | + )) { |
|
| 1583 | + throw new RestPasswordIncorrectException(); |
|
| 1584 | + } |
|
| 1585 | + } // wait! maybe this content is password protected |
|
| 1586 | + elseif ($model->restrictedByRelatedModelPassword() |
|
| 1587 | + && $request->get_param('caps') === EEM_Base::caps_read) { |
|
| 1588 | + $password_supplied = $request->get_param('password'); |
|
| 1589 | + if (empty($password_supplied)) { |
|
| 1590 | + $query_params['exclude_protected'] = true; |
|
| 1591 | + if (!$model->exists($query_params)) { |
|
| 1592 | + throw new RestPasswordRequiredException(); |
|
| 1593 | + } |
|
| 1594 | + } else { |
|
| 1595 | + $query_params[0][ $model->modelChainAndPassword() ] = $password_supplied; |
|
| 1596 | + if (!$model->exists($query_params)) { |
|
| 1597 | + throw new RestPasswordIncorrectException(); |
|
| 1598 | + } |
|
| 1599 | + } |
|
| 1600 | + } |
|
| 1601 | + } |
|
| 1602 | 1602 | } |
@@ -18,181 +18,181 @@ |
||
| 18 | 18 | class RequestTypeContextDetector |
| 19 | 19 | { |
| 20 | 20 | |
| 21 | - /** |
|
| 22 | - * @var RequestTypeContextFactoryInterface $factory |
|
| 23 | - */ |
|
| 24 | - private $factory; |
|
| 25 | - |
|
| 26 | - /** |
|
| 27 | - * @var RequestInterface $request |
|
| 28 | - */ |
|
| 29 | - private $request; |
|
| 30 | - |
|
| 31 | - /** |
|
| 32 | - * @var array $globalRouteConditions |
|
| 33 | - */ |
|
| 34 | - private $globalRouteConditions; |
|
| 35 | - |
|
| 36 | - |
|
| 37 | - /** |
|
| 38 | - * RequestTypeContextDetector constructor. |
|
| 39 | - * |
|
| 40 | - * @param RequestInterface $request |
|
| 41 | - * @param RequestTypeContextFactoryInterface $factory |
|
| 42 | - * @param array $globalRouteConditions an array for injecting values that would |
|
| 43 | - * otherwise be defined as global constants |
|
| 44 | - * or other global variables for the current |
|
| 45 | - * request route such as DOING_AJAX |
|
| 46 | - */ |
|
| 47 | - public function __construct( |
|
| 48 | - RequestInterface $request, |
|
| 49 | - RequestTypeContextFactoryInterface $factory, |
|
| 50 | - array $globalRouteConditions = array() |
|
| 51 | - ) { |
|
| 52 | - $this->request = $request; |
|
| 53 | - $this->factory = $factory; |
|
| 54 | - $this->globalRouteConditions = $globalRouteConditions; |
|
| 55 | - } |
|
| 56 | - |
|
| 57 | - |
|
| 58 | - /** |
|
| 59 | - * @return mixed |
|
| 60 | - */ |
|
| 61 | - private function getGlobalRouteCondition($globalRouteCondition, $default) |
|
| 62 | - { |
|
| 63 | - return isset($this->globalRouteConditions[ $globalRouteCondition ]) |
|
| 64 | - ? $this->globalRouteConditions[ $globalRouteCondition ] |
|
| 65 | - : $default; |
|
| 66 | - } |
|
| 67 | - |
|
| 68 | - |
|
| 69 | - /** |
|
| 70 | - * @return RequestTypeContext |
|
| 71 | - * @throws InvalidArgumentException |
|
| 72 | - */ |
|
| 73 | - public function detectRequestTypeContext() |
|
| 74 | - { |
|
| 75 | - // Detect error scrapes |
|
| 76 | - if ($this->request->getRequestParam('wp_scrape_key') !== null |
|
| 77 | - && $this->request->getRequestParam('wp_scrape_nonce') !== null |
|
| 78 | - ) { |
|
| 79 | - return $this->factory->create(RequestTypeContext::WP_SCRAPE); |
|
| 80 | - } |
|
| 81 | - // Detect EE REST API |
|
| 82 | - if ($this->isEspressoRestApiRequest()) { |
|
| 83 | - return $this->factory->create(RequestTypeContext::API); |
|
| 84 | - } |
|
| 85 | - // Detect WP REST API |
|
| 86 | - if ($this->isWordPressRestApiRequest()) { |
|
| 87 | - return $this->factory->create(RequestTypeContext::WP_API); |
|
| 88 | - } |
|
| 89 | - // Detect AJAX |
|
| 90 | - if ($this->getGlobalRouteCondition('DOING_AJAX', false)) { |
|
| 91 | - if (filter_var($this->request->getRequestParam('ee_front_ajax'), FILTER_VALIDATE_BOOLEAN)) { |
|
| 92 | - return $this->factory->create(RequestTypeContext::AJAX_FRONT); |
|
| 93 | - } |
|
| 94 | - if (filter_var($this->request->getRequestParam('ee_admin_ajax'), FILTER_VALIDATE_BOOLEAN)) { |
|
| 95 | - return $this->factory->create(RequestTypeContext::AJAX_ADMIN); |
|
| 96 | - } |
|
| 97 | - if ($this->request->getRequestParam('action') === 'heartbeat') { |
|
| 98 | - return $this->factory->create(RequestTypeContext::AJAX_HEARTBEAT); |
|
| 99 | - } |
|
| 100 | - return $this->factory->create(RequestTypeContext::AJAX_OTHER); |
|
| 101 | - } |
|
| 102 | - // Detect WP_Cron |
|
| 103 | - if ($this->isCronRequest()) { |
|
| 104 | - return $this->factory->create(RequestTypeContext::CRON); |
|
| 105 | - } |
|
| 106 | - // Detect command line requests |
|
| 107 | - if ($this->getGlobalRouteCondition('WP_CLI', false)) { |
|
| 108 | - return $this->factory->create(RequestTypeContext::CLI); |
|
| 109 | - } |
|
| 110 | - // detect WordPress admin (ie: "Dashboard") |
|
| 111 | - if ($this->getGlobalRouteCondition('is_admin', false)) { |
|
| 112 | - return $this->factory->create(RequestTypeContext::ADMIN); |
|
| 113 | - } |
|
| 114 | - // Detect iFrames |
|
| 115 | - if ($this->isIframeRoute()) { |
|
| 116 | - return $this->factory->create(RequestTypeContext::IFRAME); |
|
| 117 | - } |
|
| 118 | - // Detect Feeds |
|
| 119 | - if ($this->isFeedRequest()) { |
|
| 120 | - return $this->factory->create(RequestTypeContext::FEED); |
|
| 121 | - } |
|
| 122 | - // and by process of elimination... |
|
| 123 | - return $this->factory->create(RequestTypeContext::FRONTEND); |
|
| 124 | - } |
|
| 125 | - |
|
| 126 | - |
|
| 127 | - /** |
|
| 128 | - * @return bool |
|
| 129 | - */ |
|
| 130 | - private function isEspressoRestApiRequest() |
|
| 131 | - { |
|
| 132 | - // Check for URLs like http://mysite.com/?rest_route=/ee... and http://mysite.com/wp-json/ee/... |
|
| 133 | - return strpos( |
|
| 134 | - $this->request->getRequestParam('rest_route', false), |
|
| 135 | - '/' . Domain::API_NAMESPACE |
|
| 136 | - ) === 0 |
|
| 137 | - || $this->uriPathMatches(trim(rest_get_url_prefix(), '/') . '/' . Domain::API_NAMESPACE); |
|
| 138 | - } |
|
| 139 | - |
|
| 140 | - |
|
| 141 | - |
|
| 142 | - /** |
|
| 143 | - * @return bool |
|
| 144 | - */ |
|
| 145 | - private function isWordPressRestApiRequest() |
|
| 146 | - { |
|
| 147 | - // Check for URLs like http://mysite.com/?rest_route=/.. and http://mysite.com/wp-json/... |
|
| 148 | - return $this->request->getRequestParam('rest_route', false) |
|
| 149 | - || $this->uriPathMatches(trim(rest_get_url_prefix(), '/')); |
|
| 150 | - } |
|
| 151 | - |
|
| 152 | - |
|
| 153 | - /** |
|
| 154 | - * @return bool |
|
| 155 | - */ |
|
| 156 | - private function isCronRequest() |
|
| 157 | - { |
|
| 158 | - return $this->uriPathMatches('wp-cron.php'); |
|
| 159 | - } |
|
| 160 | - |
|
| 161 | - |
|
| 162 | - /** |
|
| 163 | - * @return bool |
|
| 164 | - */ |
|
| 165 | - private function isFeedRequest() |
|
| 166 | - { |
|
| 167 | - return $this->uriPathMatches('feed'); |
|
| 168 | - } |
|
| 169 | - |
|
| 170 | - |
|
| 171 | - /** |
|
| 172 | - * @param string $component |
|
| 173 | - * @return bool |
|
| 174 | - */ |
|
| 175 | - private function uriPathMatches($component) |
|
| 176 | - { |
|
| 177 | - $request_uri = $this->request->requestUri(); |
|
| 178 | - $parts = explode('?', $request_uri); |
|
| 179 | - $path = trim(reset($parts), '/'); |
|
| 180 | - return strpos($path, $component) === 0; |
|
| 181 | - } |
|
| 182 | - |
|
| 183 | - |
|
| 184 | - /** |
|
| 185 | - * @return bool |
|
| 186 | - */ |
|
| 187 | - private function isIframeRoute() |
|
| 188 | - { |
|
| 189 | - $is_iframe_route = apply_filters( |
|
| 190 | - 'FHEE__EventEspresso_core_domain_services_contexts_RequestTypeContextDetector__isIframeRoute', |
|
| 191 | - $this->request->getRequestParam('event_list', '') === 'iframe' |
|
| 192 | - || $this->request->getRequestParam('ticket_selector', '') === 'iframe' |
|
| 193 | - || $this->request->getRequestParam('calendar', '') === 'iframe', |
|
| 194 | - $this |
|
| 195 | - ); |
|
| 196 | - return filter_var($is_iframe_route, FILTER_VALIDATE_BOOLEAN); |
|
| 197 | - } |
|
| 21 | + /** |
|
| 22 | + * @var RequestTypeContextFactoryInterface $factory |
|
| 23 | + */ |
|
| 24 | + private $factory; |
|
| 25 | + |
|
| 26 | + /** |
|
| 27 | + * @var RequestInterface $request |
|
| 28 | + */ |
|
| 29 | + private $request; |
|
| 30 | + |
|
| 31 | + /** |
|
| 32 | + * @var array $globalRouteConditions |
|
| 33 | + */ |
|
| 34 | + private $globalRouteConditions; |
|
| 35 | + |
|
| 36 | + |
|
| 37 | + /** |
|
| 38 | + * RequestTypeContextDetector constructor. |
|
| 39 | + * |
|
| 40 | + * @param RequestInterface $request |
|
| 41 | + * @param RequestTypeContextFactoryInterface $factory |
|
| 42 | + * @param array $globalRouteConditions an array for injecting values that would |
|
| 43 | + * otherwise be defined as global constants |
|
| 44 | + * or other global variables for the current |
|
| 45 | + * request route such as DOING_AJAX |
|
| 46 | + */ |
|
| 47 | + public function __construct( |
|
| 48 | + RequestInterface $request, |
|
| 49 | + RequestTypeContextFactoryInterface $factory, |
|
| 50 | + array $globalRouteConditions = array() |
|
| 51 | + ) { |
|
| 52 | + $this->request = $request; |
|
| 53 | + $this->factory = $factory; |
|
| 54 | + $this->globalRouteConditions = $globalRouteConditions; |
|
| 55 | + } |
|
| 56 | + |
|
| 57 | + |
|
| 58 | + /** |
|
| 59 | + * @return mixed |
|
| 60 | + */ |
|
| 61 | + private function getGlobalRouteCondition($globalRouteCondition, $default) |
|
| 62 | + { |
|
| 63 | + return isset($this->globalRouteConditions[ $globalRouteCondition ]) |
|
| 64 | + ? $this->globalRouteConditions[ $globalRouteCondition ] |
|
| 65 | + : $default; |
|
| 66 | + } |
|
| 67 | + |
|
| 68 | + |
|
| 69 | + /** |
|
| 70 | + * @return RequestTypeContext |
|
| 71 | + * @throws InvalidArgumentException |
|
| 72 | + */ |
|
| 73 | + public function detectRequestTypeContext() |
|
| 74 | + { |
|
| 75 | + // Detect error scrapes |
|
| 76 | + if ($this->request->getRequestParam('wp_scrape_key') !== null |
|
| 77 | + && $this->request->getRequestParam('wp_scrape_nonce') !== null |
|
| 78 | + ) { |
|
| 79 | + return $this->factory->create(RequestTypeContext::WP_SCRAPE); |
|
| 80 | + } |
|
| 81 | + // Detect EE REST API |
|
| 82 | + if ($this->isEspressoRestApiRequest()) { |
|
| 83 | + return $this->factory->create(RequestTypeContext::API); |
|
| 84 | + } |
|
| 85 | + // Detect WP REST API |
|
| 86 | + if ($this->isWordPressRestApiRequest()) { |
|
| 87 | + return $this->factory->create(RequestTypeContext::WP_API); |
|
| 88 | + } |
|
| 89 | + // Detect AJAX |
|
| 90 | + if ($this->getGlobalRouteCondition('DOING_AJAX', false)) { |
|
| 91 | + if (filter_var($this->request->getRequestParam('ee_front_ajax'), FILTER_VALIDATE_BOOLEAN)) { |
|
| 92 | + return $this->factory->create(RequestTypeContext::AJAX_FRONT); |
|
| 93 | + } |
|
| 94 | + if (filter_var($this->request->getRequestParam('ee_admin_ajax'), FILTER_VALIDATE_BOOLEAN)) { |
|
| 95 | + return $this->factory->create(RequestTypeContext::AJAX_ADMIN); |
|
| 96 | + } |
|
| 97 | + if ($this->request->getRequestParam('action') === 'heartbeat') { |
|
| 98 | + return $this->factory->create(RequestTypeContext::AJAX_HEARTBEAT); |
|
| 99 | + } |
|
| 100 | + return $this->factory->create(RequestTypeContext::AJAX_OTHER); |
|
| 101 | + } |
|
| 102 | + // Detect WP_Cron |
|
| 103 | + if ($this->isCronRequest()) { |
|
| 104 | + return $this->factory->create(RequestTypeContext::CRON); |
|
| 105 | + } |
|
| 106 | + // Detect command line requests |
|
| 107 | + if ($this->getGlobalRouteCondition('WP_CLI', false)) { |
|
| 108 | + return $this->factory->create(RequestTypeContext::CLI); |
|
| 109 | + } |
|
| 110 | + // detect WordPress admin (ie: "Dashboard") |
|
| 111 | + if ($this->getGlobalRouteCondition('is_admin', false)) { |
|
| 112 | + return $this->factory->create(RequestTypeContext::ADMIN); |
|
| 113 | + } |
|
| 114 | + // Detect iFrames |
|
| 115 | + if ($this->isIframeRoute()) { |
|
| 116 | + return $this->factory->create(RequestTypeContext::IFRAME); |
|
| 117 | + } |
|
| 118 | + // Detect Feeds |
|
| 119 | + if ($this->isFeedRequest()) { |
|
| 120 | + return $this->factory->create(RequestTypeContext::FEED); |
|
| 121 | + } |
|
| 122 | + // and by process of elimination... |
|
| 123 | + return $this->factory->create(RequestTypeContext::FRONTEND); |
|
| 124 | + } |
|
| 125 | + |
|
| 126 | + |
|
| 127 | + /** |
|
| 128 | + * @return bool |
|
| 129 | + */ |
|
| 130 | + private function isEspressoRestApiRequest() |
|
| 131 | + { |
|
| 132 | + // Check for URLs like http://mysite.com/?rest_route=/ee... and http://mysite.com/wp-json/ee/... |
|
| 133 | + return strpos( |
|
| 134 | + $this->request->getRequestParam('rest_route', false), |
|
| 135 | + '/' . Domain::API_NAMESPACE |
|
| 136 | + ) === 0 |
|
| 137 | + || $this->uriPathMatches(trim(rest_get_url_prefix(), '/') . '/' . Domain::API_NAMESPACE); |
|
| 138 | + } |
|
| 139 | + |
|
| 140 | + |
|
| 141 | + |
|
| 142 | + /** |
|
| 143 | + * @return bool |
|
| 144 | + */ |
|
| 145 | + private function isWordPressRestApiRequest() |
|
| 146 | + { |
|
| 147 | + // Check for URLs like http://mysite.com/?rest_route=/.. and http://mysite.com/wp-json/... |
|
| 148 | + return $this->request->getRequestParam('rest_route', false) |
|
| 149 | + || $this->uriPathMatches(trim(rest_get_url_prefix(), '/')); |
|
| 150 | + } |
|
| 151 | + |
|
| 152 | + |
|
| 153 | + /** |
|
| 154 | + * @return bool |
|
| 155 | + */ |
|
| 156 | + private function isCronRequest() |
|
| 157 | + { |
|
| 158 | + return $this->uriPathMatches('wp-cron.php'); |
|
| 159 | + } |
|
| 160 | + |
|
| 161 | + |
|
| 162 | + /** |
|
| 163 | + * @return bool |
|
| 164 | + */ |
|
| 165 | + private function isFeedRequest() |
|
| 166 | + { |
|
| 167 | + return $this->uriPathMatches('feed'); |
|
| 168 | + } |
|
| 169 | + |
|
| 170 | + |
|
| 171 | + /** |
|
| 172 | + * @param string $component |
|
| 173 | + * @return bool |
|
| 174 | + */ |
|
| 175 | + private function uriPathMatches($component) |
|
| 176 | + { |
|
| 177 | + $request_uri = $this->request->requestUri(); |
|
| 178 | + $parts = explode('?', $request_uri); |
|
| 179 | + $path = trim(reset($parts), '/'); |
|
| 180 | + return strpos($path, $component) === 0; |
|
| 181 | + } |
|
| 182 | + |
|
| 183 | + |
|
| 184 | + /** |
|
| 185 | + * @return bool |
|
| 186 | + */ |
|
| 187 | + private function isIframeRoute() |
|
| 188 | + { |
|
| 189 | + $is_iframe_route = apply_filters( |
|
| 190 | + 'FHEE__EventEspresso_core_domain_services_contexts_RequestTypeContextDetector__isIframeRoute', |
|
| 191 | + $this->request->getRequestParam('event_list', '') === 'iframe' |
|
| 192 | + || $this->request->getRequestParam('ticket_selector', '') === 'iframe' |
|
| 193 | + || $this->request->getRequestParam('calendar', '') === 'iframe', |
|
| 194 | + $this |
|
| 195 | + ); |
|
| 196 | + return filter_var($is_iframe_route, FILTER_VALIDATE_BOOLEAN); |
|
| 197 | + } |
|
| 198 | 198 | } |
@@ -60,8 +60,8 @@ discard block |
||
| 60 | 60 | */ |
| 61 | 61 | private function getGlobalRouteCondition($globalRouteCondition, $default) |
| 62 | 62 | { |
| 63 | - return isset($this->globalRouteConditions[ $globalRouteCondition ]) |
|
| 64 | - ? $this->globalRouteConditions[ $globalRouteCondition ] |
|
| 63 | + return isset($this->globalRouteConditions[$globalRouteCondition]) |
|
| 64 | + ? $this->globalRouteConditions[$globalRouteCondition] |
|
| 65 | 65 | : $default; |
| 66 | 66 | } |
| 67 | 67 | |
@@ -132,9 +132,9 @@ discard block |
||
| 132 | 132 | // Check for URLs like http://mysite.com/?rest_route=/ee... and http://mysite.com/wp-json/ee/... |
| 133 | 133 | return strpos( |
| 134 | 134 | $this->request->getRequestParam('rest_route', false), |
| 135 | - '/' . Domain::API_NAMESPACE |
|
| 135 | + '/'.Domain::API_NAMESPACE |
|
| 136 | 136 | ) === 0 |
| 137 | - || $this->uriPathMatches(trim(rest_get_url_prefix(), '/') . '/' . Domain::API_NAMESPACE); |
|
| 137 | + || $this->uriPathMatches(trim(rest_get_url_prefix(), '/').'/'.Domain::API_NAMESPACE); |
|
| 138 | 138 | } |
| 139 | 139 | |
| 140 | 140 | |
@@ -112,7 +112,7 @@ discard block |
||
| 112 | 112 | $class = parent::_get_row_class($item); |
| 113 | 113 | // add status class |
| 114 | 114 | $class .= $item instanceof EE_Event |
| 115 | - ? ' ee-status-strip event-status-' . $item->get_active_status() |
|
| 115 | + ? ' ee-status-strip event-status-'.$item->get_active_status() |
|
| 116 | 116 | : ''; |
| 117 | 117 | if ($this->_has_checkbox_column) { |
| 118 | 118 | $class .= ' has-checkbox-column'; |
@@ -141,7 +141,7 @@ discard block |
||
| 141 | 141 | */ |
| 142 | 142 | public function column_cb($item) |
| 143 | 143 | { |
| 144 | - if (! $item instanceof EE_Event) { |
|
| 144 | + if ( ! $item instanceof EE_Event) { |
|
| 145 | 145 | return ''; |
| 146 | 146 | } |
| 147 | 147 | $this->_dtt = $item->primary_datetime(); // set this for use in other columns |
@@ -164,7 +164,7 @@ discard block |
||
| 164 | 164 | public function column_id(EE_Event $item) |
| 165 | 165 | { |
| 166 | 166 | $content = $item->ID(); |
| 167 | - $content .= ' <span class="show-on-mobile-view-only">' . $item->name() . '</span>'; |
|
| 167 | + $content .= ' <span class="show-on-mobile-view-only">'.$item->name().'</span>'; |
|
| 168 | 168 | return $content; |
| 169 | 169 | } |
| 170 | 170 | |
@@ -187,7 +187,7 @@ discard block |
||
| 187 | 187 | $actions = $this->_column_name_action_setup($item); |
| 188 | 188 | $status = ''; // $item->status() !== 'publish' ? ' (' . $item->status() . ')' : ''; |
| 189 | 189 | $content = '<strong><a class="row-title" href="' |
| 190 | - . $edit_link . '">' |
|
| 190 | + . $edit_link.'">' |
|
| 191 | 191 | . $item->name() |
| 192 | 192 | . '</a></strong>' |
| 193 | 193 | . $status; |
@@ -216,7 +216,7 @@ discard block |
||
| 216 | 216 | protected function _column_name_action_setup(EE_Event $item) |
| 217 | 217 | { |
| 218 | 218 | // todo: remove when attendees is active |
| 219 | - if (! defined('REG_ADMIN_URL')) { |
|
| 219 | + if ( ! defined('REG_ADMIN_URL')) { |
|
| 220 | 220 | define('REG_ADMIN_URL', EVENTS_ADMIN_URL); |
| 221 | 221 | } |
| 222 | 222 | $actions = array(); |
@@ -233,8 +233,8 @@ discard block |
||
| 233 | 233 | 'post' => $item->ID(), |
| 234 | 234 | ); |
| 235 | 235 | $edit_link = EE_Admin_Page::add_query_args_and_nonce($edit_query_args, EVENTS_ADMIN_URL); |
| 236 | - $actions['edit'] = '<a href="' . $edit_link . '"' |
|
| 237 | - . ' title="' . esc_attr__('Edit Event', 'event_espresso') . '">' |
|
| 236 | + $actions['edit'] = '<a href="'.$edit_link.'"' |
|
| 237 | + . ' title="'.esc_attr__('Edit Event', 'event_espresso').'">' |
|
| 238 | 238 | . esc_html__('Edit', 'event_espresso') |
| 239 | 239 | . '</a>'; |
| 240 | 240 | } |
@@ -253,8 +253,8 @@ discard block |
||
| 253 | 253 | 'event_id' => $item->ID(), |
| 254 | 254 | ); |
| 255 | 255 | $attendees_link = EE_Admin_Page::add_query_args_and_nonce($attendees_query_args, REG_ADMIN_URL); |
| 256 | - $actions['attendees'] = '<a href="' . $attendees_link . '"' |
|
| 257 | - . ' title="' . esc_attr__('View Registrations', 'event_espresso') . '">' |
|
| 256 | + $actions['attendees'] = '<a href="'.$attendees_link.'"' |
|
| 257 | + . ' title="'.esc_attr__('View Registrations', 'event_espresso').'">' |
|
| 258 | 258 | . esc_html__('Registrations', 'event_espresso') |
| 259 | 259 | . '</a>'; |
| 260 | 260 | } |
@@ -301,8 +301,8 @@ discard block |
||
| 301 | 301 | ); |
| 302 | 302 | } |
| 303 | 303 | $view_link = get_permalink($item->ID()); |
| 304 | - $actions['view'] = '<a href="' . $view_link . '"' |
|
| 305 | - . ' title="' . esc_attr__('View Event', 'event_espresso') . '">' |
|
| 304 | + $actions['view'] = '<a href="'.$view_link.'"' |
|
| 305 | + . ' title="'.esc_attr__('View Event', 'event_espresso').'">' |
|
| 306 | 306 | . esc_html__('View', 'event_espresso') |
| 307 | 307 | . '</a>'; |
| 308 | 308 | if ($item->get('status') === 'trash') { |
@@ -311,8 +311,8 @@ discard block |
||
| 311 | 311 | 'espresso_events_restore_event', |
| 312 | 312 | $item->ID() |
| 313 | 313 | )) { |
| 314 | - $actions['restore_from_trash'] = '<a href="' . $restore_event_link . '"' |
|
| 315 | - . ' title="' . esc_attr__('Restore from Trash', 'event_espresso') |
|
| 314 | + $actions['restore_from_trash'] = '<a href="'.$restore_event_link.'"' |
|
| 315 | + . ' title="'.esc_attr__('Restore from Trash', 'event_espresso') |
|
| 316 | 316 | . '">' |
| 317 | 317 | . esc_html__('Restore from Trash', 'event_espresso') |
| 318 | 318 | . '</a>'; |
@@ -324,8 +324,8 @@ discard block |
||
| 324 | 324 | $item->ID() |
| 325 | 325 | ) |
| 326 | 326 | ) { |
| 327 | - $actions['delete'] = '<a href="' . $delete_event_link . '"' |
|
| 328 | - . ' title="' . esc_attr__('Delete Permanently', 'event_espresso') . '">' |
|
| 327 | + $actions['delete'] = '<a href="'.$delete_event_link.'"' |
|
| 328 | + . ' title="'.esc_attr__('Delete Permanently', 'event_espresso').'">' |
|
| 329 | 329 | . esc_html__('Delete Permanently', 'event_espresso') |
| 330 | 330 | . '</a>'; |
| 331 | 331 | } |
@@ -335,8 +335,8 @@ discard block |
||
| 335 | 335 | 'espresso_events_trash_event', |
| 336 | 336 | $item->ID() |
| 337 | 337 | )) { |
| 338 | - $actions['move to trash'] = '<a href="' . $trash_event_link . '"' |
|
| 339 | - . ' title="' . esc_attr__('Trash Event', 'event_espresso') . '">' |
|
| 338 | + $actions['move to trash'] = '<a href="'.$trash_event_link.'"' |
|
| 339 | + . ' title="'.esc_attr__('Trash Event', 'event_espresso').'">' |
|
| 340 | 340 | . esc_html__('Trash', 'event_espresso') |
| 341 | 341 | . '</a>'; |
| 342 | 342 | } |
@@ -361,8 +361,8 @@ discard block |
||
| 361 | 361 | 'EVT_wp_user' => $item->wp_user(), |
| 362 | 362 | ); |
| 363 | 363 | $filter_url = EE_Admin_Page::add_query_args_and_nonce($query_args, EVENTS_ADMIN_URL); |
| 364 | - return $gravatar . ' <a href="' . $filter_url . '"' |
|
| 365 | - . ' title="' . esc_attr__('Click to filter events by this author.', 'event_espresso') . '">' |
|
| 364 | + return $gravatar.' <a href="'.$filter_url.'"' |
|
| 365 | + . ' title="'.esc_attr__('Click to filter events by this author.', 'event_espresso').'">' |
|
| 366 | 366 | . $event_author->display_name |
| 367 | 367 | . '</a>'; |
| 368 | 368 | } |
@@ -379,7 +379,7 @@ discard block |
||
| 379 | 379 | return implode( |
| 380 | 380 | ', ', |
| 381 | 381 | array_map( |
| 382 | - function (EE_Term $category) { |
|
| 382 | + function(EE_Term $category) { |
|
| 383 | 383 | return $category->name(); |
| 384 | 384 | }, |
| 385 | 385 | $event_categories |
@@ -454,7 +454,7 @@ discard block |
||
| 454 | 454 | 'ee_read_registrations', |
| 455 | 455 | 'espresso_registrations_view_registration' |
| 456 | 456 | ) |
| 457 | - ? '<a href="' . $attendees_link . '">' . $registered_attendees . '</a>' |
|
| 457 | + ? '<a href="'.$attendees_link.'">'.$registered_attendees.'</a>' |
|
| 458 | 458 | : $registered_attendees; |
| 459 | 459 | } |
| 460 | 460 | |
@@ -484,13 +484,13 @@ discard block |
||
| 484 | 484 | public function column_actions(EE_Event $item) |
| 485 | 485 | { |
| 486 | 486 | // todo: remove when attendees is active |
| 487 | - if (! defined('REG_ADMIN_URL')) { |
|
| 487 | + if ( ! defined('REG_ADMIN_URL')) { |
|
| 488 | 488 | define('REG_ADMIN_URL', EVENTS_ADMIN_URL); |
| 489 | 489 | } |
| 490 | 490 | $action_links = array(); |
| 491 | 491 | $view_link = get_permalink($item->ID()); |
| 492 | - $action_links[] = '<a href="' . $view_link . '"' |
|
| 493 | - . ' title="' . esc_attr__('View Event', 'event_espresso') . '" target="_blank">'; |
|
| 492 | + $action_links[] = '<a href="'.$view_link.'"' |
|
| 493 | + . ' title="'.esc_attr__('View Event', 'event_espresso').'" target="_blank">'; |
|
| 494 | 494 | $action_links[] = '<div class="dashicons dashicons-search"></div></a>'; |
| 495 | 495 | if (EE_Registry::instance()->CAP->current_user_can( |
| 496 | 496 | 'ee_edit_event', |
@@ -502,8 +502,8 @@ discard block |
||
| 502 | 502 | 'post' => $item->ID(), |
| 503 | 503 | ); |
| 504 | 504 | $edit_link = EE_Admin_Page::add_query_args_and_nonce($edit_query_args, EVENTS_ADMIN_URL); |
| 505 | - $action_links[] = '<a href="' . $edit_link . '"' |
|
| 506 | - . ' title="' . esc_attr__('Edit Event', 'event_espresso') . '">' |
|
| 505 | + $action_links[] = '<a href="'.$edit_link.'"' |
|
| 506 | + . ' title="'.esc_attr__('Edit Event', 'event_espresso').'">' |
|
| 507 | 507 | . '<div class="ee-icon ee-icon-calendar-edit"></div>' |
| 508 | 508 | . '</a>'; |
| 509 | 509 | } |
@@ -521,8 +521,8 @@ discard block |
||
| 521 | 521 | 'event_id' => $item->ID(), |
| 522 | 522 | ); |
| 523 | 523 | $attendees_link = EE_Admin_Page::add_query_args_and_nonce($attendees_query_args, REG_ADMIN_URL); |
| 524 | - $action_links[] = '<a href="' . $attendees_link . '"' |
|
| 525 | - . ' title="' . esc_attr__('View Registrants', 'event_espresso') . '">' |
|
| 524 | + $action_links[] = '<a href="'.$attendees_link.'"' |
|
| 525 | + . ' title="'.esc_attr__('View Registrants', 'event_espresso').'">' |
|
| 526 | 526 | . '<div class="dashicons dashicons-groups"></div>' |
| 527 | 527 | . '</a>'; |
| 528 | 528 | } |
@@ -557,7 +557,7 @@ discard block |
||
| 557 | 557 | } |
| 558 | 558 | $column_array = []; |
| 559 | 559 | foreach ($this->_columns as $column => $column_label) { |
| 560 | - $column_array[ $column ] = $column_label; |
|
| 560 | + $column_array[$column] = $column_label; |
|
| 561 | 561 | if ($column === 'venue') { |
| 562 | 562 | $column_array['event_category'] = esc_html__('Event Category', 'event_espresso'); |
| 563 | 563 | } |
@@ -15,553 +15,553 @@ |
||
| 15 | 15 | class Events_Admin_List_Table extends EE_Admin_List_Table |
| 16 | 16 | { |
| 17 | 17 | |
| 18 | - /** |
|
| 19 | - * @var EE_Datetime |
|
| 20 | - */ |
|
| 21 | - private $_dtt; |
|
| 22 | - |
|
| 23 | - |
|
| 24 | - /** |
|
| 25 | - * Initial setup of data properties for the list table. |
|
| 26 | - */ |
|
| 27 | - protected function _setup_data() |
|
| 28 | - { |
|
| 29 | - $this->_data = $this->_admin_page->get_events($this->_per_page, $this->_current_page); |
|
| 30 | - $this->_all_data_count = $this->_admin_page->get_events(0, 0, true); |
|
| 31 | - } |
|
| 32 | - |
|
| 33 | - |
|
| 34 | - /** |
|
| 35 | - * Set up of additional properties for the list table. |
|
| 36 | - */ |
|
| 37 | - protected function _set_properties() |
|
| 38 | - { |
|
| 39 | - $this->_wp_list_args = array( |
|
| 40 | - 'singular' => esc_html__('event', 'event_espresso'), |
|
| 41 | - 'plural' => esc_html__('events', 'event_espresso'), |
|
| 42 | - 'ajax' => true, // for now |
|
| 43 | - 'screen' => $this->_admin_page->get_current_screen()->id, |
|
| 44 | - ); |
|
| 45 | - $this->_columns = array( |
|
| 46 | - 'cb' => '<input type="checkbox" />', |
|
| 47 | - 'id' => esc_html__('ID', 'event_espresso'), |
|
| 48 | - 'name' => esc_html__('Name', 'event_espresso'), |
|
| 49 | - 'author' => esc_html__('Author', 'event_espresso'), |
|
| 50 | - 'venue' => esc_html__('Venue', 'event_espresso'), |
|
| 51 | - 'start_date_time' => esc_html__('Event Start', 'event_espresso'), |
|
| 52 | - 'reg_begins' => esc_html__('On Sale', 'event_espresso'), |
|
| 53 | - 'attendees' => '<span class="dashicons dashicons-groups ee-icon-color-ee-green ee-icon-size-20">' |
|
| 54 | - . '<span class="screen-reader-text">' |
|
| 55 | - . esc_html__('Approved Registrations', 'event_espresso') |
|
| 56 | - . '</span>' |
|
| 57 | - . '</span>', |
|
| 58 | - // 'tkts_sold' => esc_html__('Tickets Sold', 'event_espresso'), |
|
| 59 | - 'actions' => esc_html__('Actions', 'event_espresso'), |
|
| 60 | - ); |
|
| 61 | - $this->addConditionalColumns(); |
|
| 62 | - $this->_sortable_columns = array( |
|
| 63 | - 'id' => array('EVT_ID' => true), |
|
| 64 | - 'name' => array('EVT_name' => false), |
|
| 65 | - 'author' => array('EVT_wp_user' => false), |
|
| 66 | - 'venue' => array('Venue.VNU_name' => false), |
|
| 67 | - 'start_date_time' => array('Datetime.DTT_EVT_start' => false), |
|
| 68 | - 'reg_begins' => array('Datetime.Ticket.TKT_start_date' => false), |
|
| 69 | - ); |
|
| 70 | - |
|
| 71 | - $this->_primary_column = 'id'; |
|
| 72 | - $this->_hidden_columns = array('author', 'event_category'); |
|
| 73 | - } |
|
| 74 | - |
|
| 75 | - |
|
| 76 | - /** |
|
| 77 | - * @return array |
|
| 78 | - */ |
|
| 79 | - protected function _get_table_filters() |
|
| 80 | - { |
|
| 81 | - return array(); // no filters with decaf |
|
| 82 | - } |
|
| 83 | - |
|
| 84 | - |
|
| 85 | - /** |
|
| 86 | - * Setup of views properties. |
|
| 87 | - * |
|
| 88 | - * @throws InvalidDataTypeException |
|
| 89 | - * @throws InvalidInterfaceException |
|
| 90 | - * @throws InvalidArgumentException |
|
| 91 | - */ |
|
| 92 | - protected function _add_view_counts() |
|
| 93 | - { |
|
| 94 | - $this->_views['all']['count'] = $this->_admin_page->total_events(); |
|
| 95 | - $this->_views['draft']['count'] = $this->_admin_page->total_events_draft(); |
|
| 96 | - if (EE_Registry::instance()->CAP->current_user_can( |
|
| 97 | - 'ee_delete_events', |
|
| 98 | - 'espresso_events_trash_events' |
|
| 99 | - )) { |
|
| 100 | - $this->_views['trash']['count'] = $this->_admin_page->total_trashed_events(); |
|
| 101 | - } |
|
| 102 | - } |
|
| 103 | - |
|
| 104 | - |
|
| 105 | - /** |
|
| 106 | - * @param EE_Event $item |
|
| 107 | - * @return string |
|
| 108 | - * @throws EE_Error |
|
| 109 | - */ |
|
| 110 | - protected function _get_row_class($item) |
|
| 111 | - { |
|
| 112 | - $class = parent::_get_row_class($item); |
|
| 113 | - // add status class |
|
| 114 | - $class .= $item instanceof EE_Event |
|
| 115 | - ? ' ee-status-strip event-status-' . $item->get_active_status() |
|
| 116 | - : ''; |
|
| 117 | - if ($this->_has_checkbox_column) { |
|
| 118 | - $class .= ' has-checkbox-column'; |
|
| 119 | - } |
|
| 120 | - return $class; |
|
| 121 | - } |
|
| 122 | - |
|
| 123 | - |
|
| 124 | - /** |
|
| 125 | - * @param EE_Event $item |
|
| 126 | - * @return string |
|
| 127 | - * @throws EE_Error |
|
| 128 | - */ |
|
| 129 | - public function column_status(EE_Event $item) |
|
| 130 | - { |
|
| 131 | - return '<span class="ee-status-strip ee-status-strip-td event-status-' |
|
| 132 | - . $item->get_active_status() |
|
| 133 | - . '"></span>'; |
|
| 134 | - } |
|
| 135 | - |
|
| 136 | - |
|
| 137 | - /** |
|
| 138 | - * @param EE_Event $item |
|
| 139 | - * @return string |
|
| 140 | - * @throws EE_Error |
|
| 141 | - */ |
|
| 142 | - public function column_cb($item) |
|
| 143 | - { |
|
| 144 | - if (! $item instanceof EE_Event) { |
|
| 145 | - return ''; |
|
| 146 | - } |
|
| 147 | - $this->_dtt = $item->primary_datetime(); // set this for use in other columns |
|
| 148 | - // does event have any attached registrations? |
|
| 149 | - $regs = $item->count_related('Registration'); |
|
| 150 | - return $regs > 0 && $this->_view === 'trash' |
|
| 151 | - ? '<span class="ee-lock-icon"></span>' |
|
| 152 | - : sprintf( |
|
| 153 | - '<input type="checkbox" name="EVT_IDs[]" value="%s" />', |
|
| 154 | - $item->ID() |
|
| 155 | - ); |
|
| 156 | - } |
|
| 157 | - |
|
| 158 | - |
|
| 159 | - /** |
|
| 160 | - * @param EE_Event $item |
|
| 161 | - * @return mixed|string |
|
| 162 | - * @throws EE_Error |
|
| 163 | - */ |
|
| 164 | - public function column_id(EE_Event $item) |
|
| 165 | - { |
|
| 166 | - $content = $item->ID(); |
|
| 167 | - $content .= ' <span class="show-on-mobile-view-only">' . $item->name() . '</span>'; |
|
| 168 | - return $content; |
|
| 169 | - } |
|
| 170 | - |
|
| 171 | - |
|
| 172 | - /** |
|
| 173 | - * @param EE_Event $item |
|
| 174 | - * @return string |
|
| 175 | - * @throws EE_Error |
|
| 176 | - * @throws InvalidArgumentException |
|
| 177 | - * @throws InvalidDataTypeException |
|
| 178 | - * @throws InvalidInterfaceException |
|
| 179 | - */ |
|
| 180 | - public function column_name(EE_Event $item) |
|
| 181 | - { |
|
| 182 | - $edit_query_args = array( |
|
| 183 | - 'action' => 'edit', |
|
| 184 | - 'post' => $item->ID(), |
|
| 185 | - ); |
|
| 186 | - $edit_link = EE_Admin_Page::add_query_args_and_nonce($edit_query_args, EVENTS_ADMIN_URL); |
|
| 187 | - $actions = $this->_column_name_action_setup($item); |
|
| 188 | - $status = ''; // $item->status() !== 'publish' ? ' (' . $item->status() . ')' : ''; |
|
| 189 | - $content = '<strong><a class="row-title" href="' |
|
| 190 | - . $edit_link . '">' |
|
| 191 | - . $item->name() |
|
| 192 | - . '</a></strong>' |
|
| 193 | - . $status; |
|
| 194 | - $content .= '<br><span class="ee-status-text-small">' |
|
| 195 | - . EEH_Template::pretty_status( |
|
| 196 | - $item->get_active_status(), |
|
| 197 | - false, |
|
| 198 | - 'sentence' |
|
| 199 | - ) |
|
| 200 | - . '</span>'; |
|
| 201 | - $content .= $this->row_actions($actions); |
|
| 202 | - return $content; |
|
| 203 | - } |
|
| 204 | - |
|
| 205 | - |
|
| 206 | - /** |
|
| 207 | - * Just a method for setting up the actions for the name column |
|
| 208 | - * |
|
| 209 | - * @param EE_Event $item |
|
| 210 | - * @return array array of actions |
|
| 211 | - * @throws EE_Error |
|
| 212 | - * @throws InvalidArgumentException |
|
| 213 | - * @throws InvalidDataTypeException |
|
| 214 | - * @throws InvalidInterfaceException |
|
| 215 | - */ |
|
| 216 | - protected function _column_name_action_setup(EE_Event $item) |
|
| 217 | - { |
|
| 218 | - // todo: remove when attendees is active |
|
| 219 | - if (! defined('REG_ADMIN_URL')) { |
|
| 220 | - define('REG_ADMIN_URL', EVENTS_ADMIN_URL); |
|
| 221 | - } |
|
| 222 | - $actions = array(); |
|
| 223 | - $restore_event_link = ''; |
|
| 224 | - $delete_event_link = ''; |
|
| 225 | - $trash_event_link = ''; |
|
| 226 | - if (EE_Registry::instance()->CAP->current_user_can( |
|
| 227 | - 'ee_edit_event', |
|
| 228 | - 'espresso_events_edit', |
|
| 229 | - $item->ID() |
|
| 230 | - )) { |
|
| 231 | - $edit_query_args = array( |
|
| 232 | - 'action' => 'edit', |
|
| 233 | - 'post' => $item->ID(), |
|
| 234 | - ); |
|
| 235 | - $edit_link = EE_Admin_Page::add_query_args_and_nonce($edit_query_args, EVENTS_ADMIN_URL); |
|
| 236 | - $actions['edit'] = '<a href="' . $edit_link . '"' |
|
| 237 | - . ' title="' . esc_attr__('Edit Event', 'event_espresso') . '">' |
|
| 238 | - . esc_html__('Edit', 'event_espresso') |
|
| 239 | - . '</a>'; |
|
| 240 | - } |
|
| 241 | - if (EE_Registry::instance()->CAP->current_user_can( |
|
| 242 | - 'ee_read_registrations', |
|
| 243 | - 'espresso_registrations_view_registration' |
|
| 244 | - ) |
|
| 245 | - && EE_Registry::instance()->CAP->current_user_can( |
|
| 246 | - 'ee_read_event', |
|
| 247 | - 'espresso_registrations_view_registration', |
|
| 248 | - $item->ID() |
|
| 249 | - ) |
|
| 250 | - ) { |
|
| 251 | - $attendees_query_args = array( |
|
| 252 | - 'action' => 'default', |
|
| 253 | - 'event_id' => $item->ID(), |
|
| 254 | - ); |
|
| 255 | - $attendees_link = EE_Admin_Page::add_query_args_and_nonce($attendees_query_args, REG_ADMIN_URL); |
|
| 256 | - $actions['attendees'] = '<a href="' . $attendees_link . '"' |
|
| 257 | - . ' title="' . esc_attr__('View Registrations', 'event_espresso') . '">' |
|
| 258 | - . esc_html__('Registrations', 'event_espresso') |
|
| 259 | - . '</a>'; |
|
| 260 | - } |
|
| 261 | - if (EE_Registry::instance()->CAP->current_user_can( |
|
| 262 | - 'ee_delete_event', |
|
| 263 | - 'espresso_events_trash_event', |
|
| 264 | - $item->ID() |
|
| 265 | - )) { |
|
| 266 | - $trash_event_query_args = array( |
|
| 267 | - 'action' => 'trash_event', |
|
| 268 | - 'EVT_ID' => $item->ID(), |
|
| 269 | - ); |
|
| 270 | - $trash_event_link = EE_Admin_Page::add_query_args_and_nonce( |
|
| 271 | - $trash_event_query_args, |
|
| 272 | - EVENTS_ADMIN_URL |
|
| 273 | - ); |
|
| 274 | - } |
|
| 275 | - if (EE_Registry::instance()->CAP->current_user_can( |
|
| 276 | - 'ee_delete_event', |
|
| 277 | - 'espresso_events_restore_event', |
|
| 278 | - $item->ID() |
|
| 279 | - )) { |
|
| 280 | - $restore_event_query_args = array( |
|
| 281 | - 'action' => 'restore_event', |
|
| 282 | - 'EVT_ID' => $item->ID(), |
|
| 283 | - ); |
|
| 284 | - $restore_event_link = EE_Admin_Page::add_query_args_and_nonce( |
|
| 285 | - $restore_event_query_args, |
|
| 286 | - EVENTS_ADMIN_URL |
|
| 287 | - ); |
|
| 288 | - } |
|
| 289 | - if (EE_Registry::instance()->CAP->current_user_can( |
|
| 290 | - 'ee_delete_event', |
|
| 291 | - 'espresso_events_delete_event', |
|
| 292 | - $item->ID() |
|
| 293 | - )) { |
|
| 294 | - $delete_event_query_args = array( |
|
| 295 | - 'action' => 'delete_event', |
|
| 296 | - 'EVT_ID' => $item->ID(), |
|
| 297 | - ); |
|
| 298 | - $delete_event_link = EE_Admin_Page::add_query_args_and_nonce( |
|
| 299 | - $delete_event_query_args, |
|
| 300 | - EVENTS_ADMIN_URL |
|
| 301 | - ); |
|
| 302 | - } |
|
| 303 | - $view_link = get_permalink($item->ID()); |
|
| 304 | - $actions['view'] = '<a href="' . $view_link . '"' |
|
| 305 | - . ' title="' . esc_attr__('View Event', 'event_espresso') . '">' |
|
| 306 | - . esc_html__('View', 'event_espresso') |
|
| 307 | - . '</a>'; |
|
| 308 | - if ($item->get('status') === 'trash') { |
|
| 309 | - if (EE_Registry::instance()->CAP->current_user_can( |
|
| 310 | - 'ee_delete_event', |
|
| 311 | - 'espresso_events_restore_event', |
|
| 312 | - $item->ID() |
|
| 313 | - )) { |
|
| 314 | - $actions['restore_from_trash'] = '<a href="' . $restore_event_link . '"' |
|
| 315 | - . ' title="' . esc_attr__('Restore from Trash', 'event_espresso') |
|
| 316 | - . '">' |
|
| 317 | - . esc_html__('Restore from Trash', 'event_espresso') |
|
| 318 | - . '</a>'; |
|
| 319 | - } |
|
| 320 | - if ($item->count_related('Registration') === 0 |
|
| 321 | - && EE_Registry::instance()->CAP->current_user_can( |
|
| 322 | - 'ee_delete_event', |
|
| 323 | - 'espresso_events_delete_event', |
|
| 324 | - $item->ID() |
|
| 325 | - ) |
|
| 326 | - ) { |
|
| 327 | - $actions['delete'] = '<a href="' . $delete_event_link . '"' |
|
| 328 | - . ' title="' . esc_attr__('Delete Permanently', 'event_espresso') . '">' |
|
| 329 | - . esc_html__('Delete Permanently', 'event_espresso') |
|
| 330 | - . '</a>'; |
|
| 331 | - } |
|
| 332 | - } else { |
|
| 333 | - if (EE_Registry::instance()->CAP->current_user_can( |
|
| 334 | - 'ee_delete_event', |
|
| 335 | - 'espresso_events_trash_event', |
|
| 336 | - $item->ID() |
|
| 337 | - )) { |
|
| 338 | - $actions['move to trash'] = '<a href="' . $trash_event_link . '"' |
|
| 339 | - . ' title="' . esc_attr__('Trash Event', 'event_espresso') . '">' |
|
| 340 | - . esc_html__('Trash', 'event_espresso') |
|
| 341 | - . '</a>'; |
|
| 342 | - } |
|
| 343 | - } |
|
| 344 | - return $actions; |
|
| 345 | - } |
|
| 346 | - |
|
| 347 | - |
|
| 348 | - /** |
|
| 349 | - * @param EE_Event $item |
|
| 350 | - * @return string |
|
| 351 | - * @throws EE_Error |
|
| 352 | - */ |
|
| 353 | - public function column_author(EE_Event $item) |
|
| 354 | - { |
|
| 355 | - // user author info |
|
| 356 | - $event_author = get_userdata($item->wp_user()); |
|
| 357 | - $gravatar = get_avatar($item->wp_user(), '15'); |
|
| 358 | - // filter link |
|
| 359 | - $query_args = array( |
|
| 360 | - 'action' => 'default', |
|
| 361 | - 'EVT_wp_user' => $item->wp_user(), |
|
| 362 | - ); |
|
| 363 | - $filter_url = EE_Admin_Page::add_query_args_and_nonce($query_args, EVENTS_ADMIN_URL); |
|
| 364 | - return $gravatar . ' <a href="' . $filter_url . '"' |
|
| 365 | - . ' title="' . esc_attr__('Click to filter events by this author.', 'event_espresso') . '">' |
|
| 366 | - . $event_author->display_name |
|
| 367 | - . '</a>'; |
|
| 368 | - } |
|
| 369 | - |
|
| 370 | - |
|
| 371 | - /** |
|
| 372 | - * @param EE_Event $event |
|
| 373 | - * @return string |
|
| 374 | - * @throws EE_Error |
|
| 375 | - */ |
|
| 376 | - public function column_event_category(EE_Event $event) |
|
| 377 | - { |
|
| 378 | - $event_categories = $event->get_all_event_categories(); |
|
| 379 | - return implode( |
|
| 380 | - ', ', |
|
| 381 | - array_map( |
|
| 382 | - function (EE_Term $category) { |
|
| 383 | - return $category->name(); |
|
| 384 | - }, |
|
| 385 | - $event_categories |
|
| 386 | - ) |
|
| 387 | - ); |
|
| 388 | - } |
|
| 389 | - |
|
| 390 | - |
|
| 391 | - /** |
|
| 392 | - * @param EE_Event $item |
|
| 393 | - * @return string |
|
| 394 | - * @throws EE_Error |
|
| 395 | - */ |
|
| 396 | - public function column_venue(EE_Event $item) |
|
| 397 | - { |
|
| 398 | - $venue = $item->get_first_related('Venue'); |
|
| 399 | - return ! empty($venue) |
|
| 400 | - ? $venue->name() |
|
| 401 | - : ''; |
|
| 402 | - } |
|
| 403 | - |
|
| 404 | - |
|
| 405 | - /** |
|
| 406 | - * @param EE_Event $item |
|
| 407 | - * @return string |
|
| 408 | - * @throws EE_Error |
|
| 409 | - */ |
|
| 410 | - public function column_start_date_time(EE_Event $item) |
|
| 411 | - { |
|
| 412 | - return $this->_dtt instanceof EE_Datetime |
|
| 413 | - ? $this->_dtt->get_i18n_datetime('DTT_EVT_start') |
|
| 414 | - : esc_html__('No Date was saved for this Event', 'event_espresso'); |
|
| 415 | - } |
|
| 416 | - |
|
| 417 | - |
|
| 418 | - /** |
|
| 419 | - * @param EE_Event $item |
|
| 420 | - * @return string |
|
| 421 | - * @throws EE_Error |
|
| 422 | - */ |
|
| 423 | - public function column_reg_begins(EE_Event $item) |
|
| 424 | - { |
|
| 425 | - $reg_start = $item->get_ticket_with_earliest_start_time(); |
|
| 426 | - return $reg_start instanceof EE_Ticket |
|
| 427 | - ? $reg_start->get_i18n_datetime('TKT_start_date') |
|
| 428 | - : esc_html__('No Tickets have been setup for this Event', 'event_espresso'); |
|
| 429 | - } |
|
| 430 | - |
|
| 431 | - |
|
| 432 | - /** |
|
| 433 | - * @param EE_Event $item |
|
| 434 | - * @return int|string |
|
| 435 | - * @throws EE_Error |
|
| 436 | - * @throws InvalidArgumentException |
|
| 437 | - * @throws InvalidDataTypeException |
|
| 438 | - * @throws InvalidInterfaceException |
|
| 439 | - */ |
|
| 440 | - public function column_attendees(EE_Event $item) |
|
| 441 | - { |
|
| 442 | - $attendees_query_args = array( |
|
| 443 | - 'action' => 'default', |
|
| 444 | - 'event_id' => $item->ID(), |
|
| 445 | - ); |
|
| 446 | - $attendees_link = EE_Admin_Page::add_query_args_and_nonce($attendees_query_args, REG_ADMIN_URL); |
|
| 447 | - $registered_attendees = EEM_Registration::instance()->get_event_registration_count($item->ID()); |
|
| 448 | - return EE_Registry::instance()->CAP->current_user_can( |
|
| 449 | - 'ee_read_event', |
|
| 450 | - 'espresso_registrations_view_registration', |
|
| 451 | - $item->ID() |
|
| 452 | - ) |
|
| 453 | - && EE_Registry::instance()->CAP->current_user_can( |
|
| 454 | - 'ee_read_registrations', |
|
| 455 | - 'espresso_registrations_view_registration' |
|
| 456 | - ) |
|
| 457 | - ? '<a href="' . $attendees_link . '">' . $registered_attendees . '</a>' |
|
| 458 | - : $registered_attendees; |
|
| 459 | - } |
|
| 460 | - |
|
| 461 | - |
|
| 462 | - /** |
|
| 463 | - * @param EE_Event $item |
|
| 464 | - * @return float |
|
| 465 | - * @throws EE_Error |
|
| 466 | - * @throws InvalidArgumentException |
|
| 467 | - * @throws InvalidDataTypeException |
|
| 468 | - * @throws InvalidInterfaceException |
|
| 469 | - */ |
|
| 470 | - public function column_tkts_sold(EE_Event $item) |
|
| 471 | - { |
|
| 472 | - return EEM_Ticket::instance()->sum(array(array('Datetime.EVT_ID' => $item->ID())), 'TKT_sold'); |
|
| 473 | - } |
|
| 474 | - |
|
| 475 | - |
|
| 476 | - /** |
|
| 477 | - * @param EE_Event $item |
|
| 478 | - * @return string |
|
| 479 | - * @throws EE_Error |
|
| 480 | - * @throws InvalidArgumentException |
|
| 481 | - * @throws InvalidDataTypeException |
|
| 482 | - * @throws InvalidInterfaceException |
|
| 483 | - */ |
|
| 484 | - public function column_actions(EE_Event $item) |
|
| 485 | - { |
|
| 486 | - // todo: remove when attendees is active |
|
| 487 | - if (! defined('REG_ADMIN_URL')) { |
|
| 488 | - define('REG_ADMIN_URL', EVENTS_ADMIN_URL); |
|
| 489 | - } |
|
| 490 | - $action_links = array(); |
|
| 491 | - $view_link = get_permalink($item->ID()); |
|
| 492 | - $action_links[] = '<a href="' . $view_link . '"' |
|
| 493 | - . ' title="' . esc_attr__('View Event', 'event_espresso') . '" target="_blank">'; |
|
| 494 | - $action_links[] = '<div class="dashicons dashicons-search"></div></a>'; |
|
| 495 | - if (EE_Registry::instance()->CAP->current_user_can( |
|
| 496 | - 'ee_edit_event', |
|
| 497 | - 'espresso_events_edit', |
|
| 498 | - $item->ID() |
|
| 499 | - )) { |
|
| 500 | - $edit_query_args = array( |
|
| 501 | - 'action' => 'edit', |
|
| 502 | - 'post' => $item->ID(), |
|
| 503 | - ); |
|
| 504 | - $edit_link = EE_Admin_Page::add_query_args_and_nonce($edit_query_args, EVENTS_ADMIN_URL); |
|
| 505 | - $action_links[] = '<a href="' . $edit_link . '"' |
|
| 506 | - . ' title="' . esc_attr__('Edit Event', 'event_espresso') . '">' |
|
| 507 | - . '<div class="ee-icon ee-icon-calendar-edit"></div>' |
|
| 508 | - . '</a>'; |
|
| 509 | - } |
|
| 510 | - if (EE_Registry::instance()->CAP->current_user_can( |
|
| 511 | - 'ee_read_registrations', |
|
| 512 | - 'espresso_registrations_view_registration' |
|
| 513 | - ) && EE_Registry::instance()->CAP->current_user_can( |
|
| 514 | - 'ee_read_event', |
|
| 515 | - 'espresso_registrations_view_registration', |
|
| 516 | - $item->ID() |
|
| 517 | - ) |
|
| 518 | - ) { |
|
| 519 | - $attendees_query_args = array( |
|
| 520 | - 'action' => 'default', |
|
| 521 | - 'event_id' => $item->ID(), |
|
| 522 | - ); |
|
| 523 | - $attendees_link = EE_Admin_Page::add_query_args_and_nonce($attendees_query_args, REG_ADMIN_URL); |
|
| 524 | - $action_links[] = '<a href="' . $attendees_link . '"' |
|
| 525 | - . ' title="' . esc_attr__('View Registrants', 'event_espresso') . '">' |
|
| 526 | - . '<div class="dashicons dashicons-groups"></div>' |
|
| 527 | - . '</a>'; |
|
| 528 | - } |
|
| 529 | - $action_links = apply_filters( |
|
| 530 | - 'FHEE__Events_Admin_List_Table__column_actions__action_links', |
|
| 531 | - $action_links, |
|
| 532 | - $item |
|
| 533 | - ); |
|
| 534 | - return $this->_action_string( |
|
| 535 | - implode("\n\t", $action_links), |
|
| 536 | - $item, |
|
| 537 | - 'div' |
|
| 538 | - ); |
|
| 539 | - } |
|
| 540 | - |
|
| 541 | - |
|
| 542 | - /** |
|
| 543 | - * Helper for adding columns conditionally |
|
| 544 | - * |
|
| 545 | - * @throws EE_Error |
|
| 546 | - * @throws InvalidArgumentException |
|
| 547 | - * @throws InvalidDataTypeException |
|
| 548 | - * @throws InvalidInterfaceException |
|
| 549 | - */ |
|
| 550 | - private function addConditionalColumns() |
|
| 551 | - { |
|
| 552 | - $event_category_count = EEM_Term::instance()->count( |
|
| 553 | - [['Term_Taxonomy.taxonomy' => EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY]] |
|
| 554 | - ); |
|
| 555 | - if ($event_category_count === 0) { |
|
| 556 | - return; |
|
| 557 | - } |
|
| 558 | - $column_array = []; |
|
| 559 | - foreach ($this->_columns as $column => $column_label) { |
|
| 560 | - $column_array[ $column ] = $column_label; |
|
| 561 | - if ($column === 'venue') { |
|
| 562 | - $column_array['event_category'] = esc_html__('Event Category', 'event_espresso'); |
|
| 563 | - } |
|
| 564 | - } |
|
| 565 | - $this->_columns = $column_array; |
|
| 566 | - } |
|
| 18 | + /** |
|
| 19 | + * @var EE_Datetime |
|
| 20 | + */ |
|
| 21 | + private $_dtt; |
|
| 22 | + |
|
| 23 | + |
|
| 24 | + /** |
|
| 25 | + * Initial setup of data properties for the list table. |
|
| 26 | + */ |
|
| 27 | + protected function _setup_data() |
|
| 28 | + { |
|
| 29 | + $this->_data = $this->_admin_page->get_events($this->_per_page, $this->_current_page); |
|
| 30 | + $this->_all_data_count = $this->_admin_page->get_events(0, 0, true); |
|
| 31 | + } |
|
| 32 | + |
|
| 33 | + |
|
| 34 | + /** |
|
| 35 | + * Set up of additional properties for the list table. |
|
| 36 | + */ |
|
| 37 | + protected function _set_properties() |
|
| 38 | + { |
|
| 39 | + $this->_wp_list_args = array( |
|
| 40 | + 'singular' => esc_html__('event', 'event_espresso'), |
|
| 41 | + 'plural' => esc_html__('events', 'event_espresso'), |
|
| 42 | + 'ajax' => true, // for now |
|
| 43 | + 'screen' => $this->_admin_page->get_current_screen()->id, |
|
| 44 | + ); |
|
| 45 | + $this->_columns = array( |
|
| 46 | + 'cb' => '<input type="checkbox" />', |
|
| 47 | + 'id' => esc_html__('ID', 'event_espresso'), |
|
| 48 | + 'name' => esc_html__('Name', 'event_espresso'), |
|
| 49 | + 'author' => esc_html__('Author', 'event_espresso'), |
|
| 50 | + 'venue' => esc_html__('Venue', 'event_espresso'), |
|
| 51 | + 'start_date_time' => esc_html__('Event Start', 'event_espresso'), |
|
| 52 | + 'reg_begins' => esc_html__('On Sale', 'event_espresso'), |
|
| 53 | + 'attendees' => '<span class="dashicons dashicons-groups ee-icon-color-ee-green ee-icon-size-20">' |
|
| 54 | + . '<span class="screen-reader-text">' |
|
| 55 | + . esc_html__('Approved Registrations', 'event_espresso') |
|
| 56 | + . '</span>' |
|
| 57 | + . '</span>', |
|
| 58 | + // 'tkts_sold' => esc_html__('Tickets Sold', 'event_espresso'), |
|
| 59 | + 'actions' => esc_html__('Actions', 'event_espresso'), |
|
| 60 | + ); |
|
| 61 | + $this->addConditionalColumns(); |
|
| 62 | + $this->_sortable_columns = array( |
|
| 63 | + 'id' => array('EVT_ID' => true), |
|
| 64 | + 'name' => array('EVT_name' => false), |
|
| 65 | + 'author' => array('EVT_wp_user' => false), |
|
| 66 | + 'venue' => array('Venue.VNU_name' => false), |
|
| 67 | + 'start_date_time' => array('Datetime.DTT_EVT_start' => false), |
|
| 68 | + 'reg_begins' => array('Datetime.Ticket.TKT_start_date' => false), |
|
| 69 | + ); |
|
| 70 | + |
|
| 71 | + $this->_primary_column = 'id'; |
|
| 72 | + $this->_hidden_columns = array('author', 'event_category'); |
|
| 73 | + } |
|
| 74 | + |
|
| 75 | + |
|
| 76 | + /** |
|
| 77 | + * @return array |
|
| 78 | + */ |
|
| 79 | + protected function _get_table_filters() |
|
| 80 | + { |
|
| 81 | + return array(); // no filters with decaf |
|
| 82 | + } |
|
| 83 | + |
|
| 84 | + |
|
| 85 | + /** |
|
| 86 | + * Setup of views properties. |
|
| 87 | + * |
|
| 88 | + * @throws InvalidDataTypeException |
|
| 89 | + * @throws InvalidInterfaceException |
|
| 90 | + * @throws InvalidArgumentException |
|
| 91 | + */ |
|
| 92 | + protected function _add_view_counts() |
|
| 93 | + { |
|
| 94 | + $this->_views['all']['count'] = $this->_admin_page->total_events(); |
|
| 95 | + $this->_views['draft']['count'] = $this->_admin_page->total_events_draft(); |
|
| 96 | + if (EE_Registry::instance()->CAP->current_user_can( |
|
| 97 | + 'ee_delete_events', |
|
| 98 | + 'espresso_events_trash_events' |
|
| 99 | + )) { |
|
| 100 | + $this->_views['trash']['count'] = $this->_admin_page->total_trashed_events(); |
|
| 101 | + } |
|
| 102 | + } |
|
| 103 | + |
|
| 104 | + |
|
| 105 | + /** |
|
| 106 | + * @param EE_Event $item |
|
| 107 | + * @return string |
|
| 108 | + * @throws EE_Error |
|
| 109 | + */ |
|
| 110 | + protected function _get_row_class($item) |
|
| 111 | + { |
|
| 112 | + $class = parent::_get_row_class($item); |
|
| 113 | + // add status class |
|
| 114 | + $class .= $item instanceof EE_Event |
|
| 115 | + ? ' ee-status-strip event-status-' . $item->get_active_status() |
|
| 116 | + : ''; |
|
| 117 | + if ($this->_has_checkbox_column) { |
|
| 118 | + $class .= ' has-checkbox-column'; |
|
| 119 | + } |
|
| 120 | + return $class; |
|
| 121 | + } |
|
| 122 | + |
|
| 123 | + |
|
| 124 | + /** |
|
| 125 | + * @param EE_Event $item |
|
| 126 | + * @return string |
|
| 127 | + * @throws EE_Error |
|
| 128 | + */ |
|
| 129 | + public function column_status(EE_Event $item) |
|
| 130 | + { |
|
| 131 | + return '<span class="ee-status-strip ee-status-strip-td event-status-' |
|
| 132 | + . $item->get_active_status() |
|
| 133 | + . '"></span>'; |
|
| 134 | + } |
|
| 135 | + |
|
| 136 | + |
|
| 137 | + /** |
|
| 138 | + * @param EE_Event $item |
|
| 139 | + * @return string |
|
| 140 | + * @throws EE_Error |
|
| 141 | + */ |
|
| 142 | + public function column_cb($item) |
|
| 143 | + { |
|
| 144 | + if (! $item instanceof EE_Event) { |
|
| 145 | + return ''; |
|
| 146 | + } |
|
| 147 | + $this->_dtt = $item->primary_datetime(); // set this for use in other columns |
|
| 148 | + // does event have any attached registrations? |
|
| 149 | + $regs = $item->count_related('Registration'); |
|
| 150 | + return $regs > 0 && $this->_view === 'trash' |
|
| 151 | + ? '<span class="ee-lock-icon"></span>' |
|
| 152 | + : sprintf( |
|
| 153 | + '<input type="checkbox" name="EVT_IDs[]" value="%s" />', |
|
| 154 | + $item->ID() |
|
| 155 | + ); |
|
| 156 | + } |
|
| 157 | + |
|
| 158 | + |
|
| 159 | + /** |
|
| 160 | + * @param EE_Event $item |
|
| 161 | + * @return mixed|string |
|
| 162 | + * @throws EE_Error |
|
| 163 | + */ |
|
| 164 | + public function column_id(EE_Event $item) |
|
| 165 | + { |
|
| 166 | + $content = $item->ID(); |
|
| 167 | + $content .= ' <span class="show-on-mobile-view-only">' . $item->name() . '</span>'; |
|
| 168 | + return $content; |
|
| 169 | + } |
|
| 170 | + |
|
| 171 | + |
|
| 172 | + /** |
|
| 173 | + * @param EE_Event $item |
|
| 174 | + * @return string |
|
| 175 | + * @throws EE_Error |
|
| 176 | + * @throws InvalidArgumentException |
|
| 177 | + * @throws InvalidDataTypeException |
|
| 178 | + * @throws InvalidInterfaceException |
|
| 179 | + */ |
|
| 180 | + public function column_name(EE_Event $item) |
|
| 181 | + { |
|
| 182 | + $edit_query_args = array( |
|
| 183 | + 'action' => 'edit', |
|
| 184 | + 'post' => $item->ID(), |
|
| 185 | + ); |
|
| 186 | + $edit_link = EE_Admin_Page::add_query_args_and_nonce($edit_query_args, EVENTS_ADMIN_URL); |
|
| 187 | + $actions = $this->_column_name_action_setup($item); |
|
| 188 | + $status = ''; // $item->status() !== 'publish' ? ' (' . $item->status() . ')' : ''; |
|
| 189 | + $content = '<strong><a class="row-title" href="' |
|
| 190 | + . $edit_link . '">' |
|
| 191 | + . $item->name() |
|
| 192 | + . '</a></strong>' |
|
| 193 | + . $status; |
|
| 194 | + $content .= '<br><span class="ee-status-text-small">' |
|
| 195 | + . EEH_Template::pretty_status( |
|
| 196 | + $item->get_active_status(), |
|
| 197 | + false, |
|
| 198 | + 'sentence' |
|
| 199 | + ) |
|
| 200 | + . '</span>'; |
|
| 201 | + $content .= $this->row_actions($actions); |
|
| 202 | + return $content; |
|
| 203 | + } |
|
| 204 | + |
|
| 205 | + |
|
| 206 | + /** |
|
| 207 | + * Just a method for setting up the actions for the name column |
|
| 208 | + * |
|
| 209 | + * @param EE_Event $item |
|
| 210 | + * @return array array of actions |
|
| 211 | + * @throws EE_Error |
|
| 212 | + * @throws InvalidArgumentException |
|
| 213 | + * @throws InvalidDataTypeException |
|
| 214 | + * @throws InvalidInterfaceException |
|
| 215 | + */ |
|
| 216 | + protected function _column_name_action_setup(EE_Event $item) |
|
| 217 | + { |
|
| 218 | + // todo: remove when attendees is active |
|
| 219 | + if (! defined('REG_ADMIN_URL')) { |
|
| 220 | + define('REG_ADMIN_URL', EVENTS_ADMIN_URL); |
|
| 221 | + } |
|
| 222 | + $actions = array(); |
|
| 223 | + $restore_event_link = ''; |
|
| 224 | + $delete_event_link = ''; |
|
| 225 | + $trash_event_link = ''; |
|
| 226 | + if (EE_Registry::instance()->CAP->current_user_can( |
|
| 227 | + 'ee_edit_event', |
|
| 228 | + 'espresso_events_edit', |
|
| 229 | + $item->ID() |
|
| 230 | + )) { |
|
| 231 | + $edit_query_args = array( |
|
| 232 | + 'action' => 'edit', |
|
| 233 | + 'post' => $item->ID(), |
|
| 234 | + ); |
|
| 235 | + $edit_link = EE_Admin_Page::add_query_args_and_nonce($edit_query_args, EVENTS_ADMIN_URL); |
|
| 236 | + $actions['edit'] = '<a href="' . $edit_link . '"' |
|
| 237 | + . ' title="' . esc_attr__('Edit Event', 'event_espresso') . '">' |
|
| 238 | + . esc_html__('Edit', 'event_espresso') |
|
| 239 | + . '</a>'; |
|
| 240 | + } |
|
| 241 | + if (EE_Registry::instance()->CAP->current_user_can( |
|
| 242 | + 'ee_read_registrations', |
|
| 243 | + 'espresso_registrations_view_registration' |
|
| 244 | + ) |
|
| 245 | + && EE_Registry::instance()->CAP->current_user_can( |
|
| 246 | + 'ee_read_event', |
|
| 247 | + 'espresso_registrations_view_registration', |
|
| 248 | + $item->ID() |
|
| 249 | + ) |
|
| 250 | + ) { |
|
| 251 | + $attendees_query_args = array( |
|
| 252 | + 'action' => 'default', |
|
| 253 | + 'event_id' => $item->ID(), |
|
| 254 | + ); |
|
| 255 | + $attendees_link = EE_Admin_Page::add_query_args_and_nonce($attendees_query_args, REG_ADMIN_URL); |
|
| 256 | + $actions['attendees'] = '<a href="' . $attendees_link . '"' |
|
| 257 | + . ' title="' . esc_attr__('View Registrations', 'event_espresso') . '">' |
|
| 258 | + . esc_html__('Registrations', 'event_espresso') |
|
| 259 | + . '</a>'; |
|
| 260 | + } |
|
| 261 | + if (EE_Registry::instance()->CAP->current_user_can( |
|
| 262 | + 'ee_delete_event', |
|
| 263 | + 'espresso_events_trash_event', |
|
| 264 | + $item->ID() |
|
| 265 | + )) { |
|
| 266 | + $trash_event_query_args = array( |
|
| 267 | + 'action' => 'trash_event', |
|
| 268 | + 'EVT_ID' => $item->ID(), |
|
| 269 | + ); |
|
| 270 | + $trash_event_link = EE_Admin_Page::add_query_args_and_nonce( |
|
| 271 | + $trash_event_query_args, |
|
| 272 | + EVENTS_ADMIN_URL |
|
| 273 | + ); |
|
| 274 | + } |
|
| 275 | + if (EE_Registry::instance()->CAP->current_user_can( |
|
| 276 | + 'ee_delete_event', |
|
| 277 | + 'espresso_events_restore_event', |
|
| 278 | + $item->ID() |
|
| 279 | + )) { |
|
| 280 | + $restore_event_query_args = array( |
|
| 281 | + 'action' => 'restore_event', |
|
| 282 | + 'EVT_ID' => $item->ID(), |
|
| 283 | + ); |
|
| 284 | + $restore_event_link = EE_Admin_Page::add_query_args_and_nonce( |
|
| 285 | + $restore_event_query_args, |
|
| 286 | + EVENTS_ADMIN_URL |
|
| 287 | + ); |
|
| 288 | + } |
|
| 289 | + if (EE_Registry::instance()->CAP->current_user_can( |
|
| 290 | + 'ee_delete_event', |
|
| 291 | + 'espresso_events_delete_event', |
|
| 292 | + $item->ID() |
|
| 293 | + )) { |
|
| 294 | + $delete_event_query_args = array( |
|
| 295 | + 'action' => 'delete_event', |
|
| 296 | + 'EVT_ID' => $item->ID(), |
|
| 297 | + ); |
|
| 298 | + $delete_event_link = EE_Admin_Page::add_query_args_and_nonce( |
|
| 299 | + $delete_event_query_args, |
|
| 300 | + EVENTS_ADMIN_URL |
|
| 301 | + ); |
|
| 302 | + } |
|
| 303 | + $view_link = get_permalink($item->ID()); |
|
| 304 | + $actions['view'] = '<a href="' . $view_link . '"' |
|
| 305 | + . ' title="' . esc_attr__('View Event', 'event_espresso') . '">' |
|
| 306 | + . esc_html__('View', 'event_espresso') |
|
| 307 | + . '</a>'; |
|
| 308 | + if ($item->get('status') === 'trash') { |
|
| 309 | + if (EE_Registry::instance()->CAP->current_user_can( |
|
| 310 | + 'ee_delete_event', |
|
| 311 | + 'espresso_events_restore_event', |
|
| 312 | + $item->ID() |
|
| 313 | + )) { |
|
| 314 | + $actions['restore_from_trash'] = '<a href="' . $restore_event_link . '"' |
|
| 315 | + . ' title="' . esc_attr__('Restore from Trash', 'event_espresso') |
|
| 316 | + . '">' |
|
| 317 | + . esc_html__('Restore from Trash', 'event_espresso') |
|
| 318 | + . '</a>'; |
|
| 319 | + } |
|
| 320 | + if ($item->count_related('Registration') === 0 |
|
| 321 | + && EE_Registry::instance()->CAP->current_user_can( |
|
| 322 | + 'ee_delete_event', |
|
| 323 | + 'espresso_events_delete_event', |
|
| 324 | + $item->ID() |
|
| 325 | + ) |
|
| 326 | + ) { |
|
| 327 | + $actions['delete'] = '<a href="' . $delete_event_link . '"' |
|
| 328 | + . ' title="' . esc_attr__('Delete Permanently', 'event_espresso') . '">' |
|
| 329 | + . esc_html__('Delete Permanently', 'event_espresso') |
|
| 330 | + . '</a>'; |
|
| 331 | + } |
|
| 332 | + } else { |
|
| 333 | + if (EE_Registry::instance()->CAP->current_user_can( |
|
| 334 | + 'ee_delete_event', |
|
| 335 | + 'espresso_events_trash_event', |
|
| 336 | + $item->ID() |
|
| 337 | + )) { |
|
| 338 | + $actions['move to trash'] = '<a href="' . $trash_event_link . '"' |
|
| 339 | + . ' title="' . esc_attr__('Trash Event', 'event_espresso') . '">' |
|
| 340 | + . esc_html__('Trash', 'event_espresso') |
|
| 341 | + . '</a>'; |
|
| 342 | + } |
|
| 343 | + } |
|
| 344 | + return $actions; |
|
| 345 | + } |
|
| 346 | + |
|
| 347 | + |
|
| 348 | + /** |
|
| 349 | + * @param EE_Event $item |
|
| 350 | + * @return string |
|
| 351 | + * @throws EE_Error |
|
| 352 | + */ |
|
| 353 | + public function column_author(EE_Event $item) |
|
| 354 | + { |
|
| 355 | + // user author info |
|
| 356 | + $event_author = get_userdata($item->wp_user()); |
|
| 357 | + $gravatar = get_avatar($item->wp_user(), '15'); |
|
| 358 | + // filter link |
|
| 359 | + $query_args = array( |
|
| 360 | + 'action' => 'default', |
|
| 361 | + 'EVT_wp_user' => $item->wp_user(), |
|
| 362 | + ); |
|
| 363 | + $filter_url = EE_Admin_Page::add_query_args_and_nonce($query_args, EVENTS_ADMIN_URL); |
|
| 364 | + return $gravatar . ' <a href="' . $filter_url . '"' |
|
| 365 | + . ' title="' . esc_attr__('Click to filter events by this author.', 'event_espresso') . '">' |
|
| 366 | + . $event_author->display_name |
|
| 367 | + . '</a>'; |
|
| 368 | + } |
|
| 369 | + |
|
| 370 | + |
|
| 371 | + /** |
|
| 372 | + * @param EE_Event $event |
|
| 373 | + * @return string |
|
| 374 | + * @throws EE_Error |
|
| 375 | + */ |
|
| 376 | + public function column_event_category(EE_Event $event) |
|
| 377 | + { |
|
| 378 | + $event_categories = $event->get_all_event_categories(); |
|
| 379 | + return implode( |
|
| 380 | + ', ', |
|
| 381 | + array_map( |
|
| 382 | + function (EE_Term $category) { |
|
| 383 | + return $category->name(); |
|
| 384 | + }, |
|
| 385 | + $event_categories |
|
| 386 | + ) |
|
| 387 | + ); |
|
| 388 | + } |
|
| 389 | + |
|
| 390 | + |
|
| 391 | + /** |
|
| 392 | + * @param EE_Event $item |
|
| 393 | + * @return string |
|
| 394 | + * @throws EE_Error |
|
| 395 | + */ |
|
| 396 | + public function column_venue(EE_Event $item) |
|
| 397 | + { |
|
| 398 | + $venue = $item->get_first_related('Venue'); |
|
| 399 | + return ! empty($venue) |
|
| 400 | + ? $venue->name() |
|
| 401 | + : ''; |
|
| 402 | + } |
|
| 403 | + |
|
| 404 | + |
|
| 405 | + /** |
|
| 406 | + * @param EE_Event $item |
|
| 407 | + * @return string |
|
| 408 | + * @throws EE_Error |
|
| 409 | + */ |
|
| 410 | + public function column_start_date_time(EE_Event $item) |
|
| 411 | + { |
|
| 412 | + return $this->_dtt instanceof EE_Datetime |
|
| 413 | + ? $this->_dtt->get_i18n_datetime('DTT_EVT_start') |
|
| 414 | + : esc_html__('No Date was saved for this Event', 'event_espresso'); |
|
| 415 | + } |
|
| 416 | + |
|
| 417 | + |
|
| 418 | + /** |
|
| 419 | + * @param EE_Event $item |
|
| 420 | + * @return string |
|
| 421 | + * @throws EE_Error |
|
| 422 | + */ |
|
| 423 | + public function column_reg_begins(EE_Event $item) |
|
| 424 | + { |
|
| 425 | + $reg_start = $item->get_ticket_with_earliest_start_time(); |
|
| 426 | + return $reg_start instanceof EE_Ticket |
|
| 427 | + ? $reg_start->get_i18n_datetime('TKT_start_date') |
|
| 428 | + : esc_html__('No Tickets have been setup for this Event', 'event_espresso'); |
|
| 429 | + } |
|
| 430 | + |
|
| 431 | + |
|
| 432 | + /** |
|
| 433 | + * @param EE_Event $item |
|
| 434 | + * @return int|string |
|
| 435 | + * @throws EE_Error |
|
| 436 | + * @throws InvalidArgumentException |
|
| 437 | + * @throws InvalidDataTypeException |
|
| 438 | + * @throws InvalidInterfaceException |
|
| 439 | + */ |
|
| 440 | + public function column_attendees(EE_Event $item) |
|
| 441 | + { |
|
| 442 | + $attendees_query_args = array( |
|
| 443 | + 'action' => 'default', |
|
| 444 | + 'event_id' => $item->ID(), |
|
| 445 | + ); |
|
| 446 | + $attendees_link = EE_Admin_Page::add_query_args_and_nonce($attendees_query_args, REG_ADMIN_URL); |
|
| 447 | + $registered_attendees = EEM_Registration::instance()->get_event_registration_count($item->ID()); |
|
| 448 | + return EE_Registry::instance()->CAP->current_user_can( |
|
| 449 | + 'ee_read_event', |
|
| 450 | + 'espresso_registrations_view_registration', |
|
| 451 | + $item->ID() |
|
| 452 | + ) |
|
| 453 | + && EE_Registry::instance()->CAP->current_user_can( |
|
| 454 | + 'ee_read_registrations', |
|
| 455 | + 'espresso_registrations_view_registration' |
|
| 456 | + ) |
|
| 457 | + ? '<a href="' . $attendees_link . '">' . $registered_attendees . '</a>' |
|
| 458 | + : $registered_attendees; |
|
| 459 | + } |
|
| 460 | + |
|
| 461 | + |
|
| 462 | + /** |
|
| 463 | + * @param EE_Event $item |
|
| 464 | + * @return float |
|
| 465 | + * @throws EE_Error |
|
| 466 | + * @throws InvalidArgumentException |
|
| 467 | + * @throws InvalidDataTypeException |
|
| 468 | + * @throws InvalidInterfaceException |
|
| 469 | + */ |
|
| 470 | + public function column_tkts_sold(EE_Event $item) |
|
| 471 | + { |
|
| 472 | + return EEM_Ticket::instance()->sum(array(array('Datetime.EVT_ID' => $item->ID())), 'TKT_sold'); |
|
| 473 | + } |
|
| 474 | + |
|
| 475 | + |
|
| 476 | + /** |
|
| 477 | + * @param EE_Event $item |
|
| 478 | + * @return string |
|
| 479 | + * @throws EE_Error |
|
| 480 | + * @throws InvalidArgumentException |
|
| 481 | + * @throws InvalidDataTypeException |
|
| 482 | + * @throws InvalidInterfaceException |
|
| 483 | + */ |
|
| 484 | + public function column_actions(EE_Event $item) |
|
| 485 | + { |
|
| 486 | + // todo: remove when attendees is active |
|
| 487 | + if (! defined('REG_ADMIN_URL')) { |
|
| 488 | + define('REG_ADMIN_URL', EVENTS_ADMIN_URL); |
|
| 489 | + } |
|
| 490 | + $action_links = array(); |
|
| 491 | + $view_link = get_permalink($item->ID()); |
|
| 492 | + $action_links[] = '<a href="' . $view_link . '"' |
|
| 493 | + . ' title="' . esc_attr__('View Event', 'event_espresso') . '" target="_blank">'; |
|
| 494 | + $action_links[] = '<div class="dashicons dashicons-search"></div></a>'; |
|
| 495 | + if (EE_Registry::instance()->CAP->current_user_can( |
|
| 496 | + 'ee_edit_event', |
|
| 497 | + 'espresso_events_edit', |
|
| 498 | + $item->ID() |
|
| 499 | + )) { |
|
| 500 | + $edit_query_args = array( |
|
| 501 | + 'action' => 'edit', |
|
| 502 | + 'post' => $item->ID(), |
|
| 503 | + ); |
|
| 504 | + $edit_link = EE_Admin_Page::add_query_args_and_nonce($edit_query_args, EVENTS_ADMIN_URL); |
|
| 505 | + $action_links[] = '<a href="' . $edit_link . '"' |
|
| 506 | + . ' title="' . esc_attr__('Edit Event', 'event_espresso') . '">' |
|
| 507 | + . '<div class="ee-icon ee-icon-calendar-edit"></div>' |
|
| 508 | + . '</a>'; |
|
| 509 | + } |
|
| 510 | + if (EE_Registry::instance()->CAP->current_user_can( |
|
| 511 | + 'ee_read_registrations', |
|
| 512 | + 'espresso_registrations_view_registration' |
|
| 513 | + ) && EE_Registry::instance()->CAP->current_user_can( |
|
| 514 | + 'ee_read_event', |
|
| 515 | + 'espresso_registrations_view_registration', |
|
| 516 | + $item->ID() |
|
| 517 | + ) |
|
| 518 | + ) { |
|
| 519 | + $attendees_query_args = array( |
|
| 520 | + 'action' => 'default', |
|
| 521 | + 'event_id' => $item->ID(), |
|
| 522 | + ); |
|
| 523 | + $attendees_link = EE_Admin_Page::add_query_args_and_nonce($attendees_query_args, REG_ADMIN_URL); |
|
| 524 | + $action_links[] = '<a href="' . $attendees_link . '"' |
|
| 525 | + . ' title="' . esc_attr__('View Registrants', 'event_espresso') . '">' |
|
| 526 | + . '<div class="dashicons dashicons-groups"></div>' |
|
| 527 | + . '</a>'; |
|
| 528 | + } |
|
| 529 | + $action_links = apply_filters( |
|
| 530 | + 'FHEE__Events_Admin_List_Table__column_actions__action_links', |
|
| 531 | + $action_links, |
|
| 532 | + $item |
|
| 533 | + ); |
|
| 534 | + return $this->_action_string( |
|
| 535 | + implode("\n\t", $action_links), |
|
| 536 | + $item, |
|
| 537 | + 'div' |
|
| 538 | + ); |
|
| 539 | + } |
|
| 540 | + |
|
| 541 | + |
|
| 542 | + /** |
|
| 543 | + * Helper for adding columns conditionally |
|
| 544 | + * |
|
| 545 | + * @throws EE_Error |
|
| 546 | + * @throws InvalidArgumentException |
|
| 547 | + * @throws InvalidDataTypeException |
|
| 548 | + * @throws InvalidInterfaceException |
|
| 549 | + */ |
|
| 550 | + private function addConditionalColumns() |
|
| 551 | + { |
|
| 552 | + $event_category_count = EEM_Term::instance()->count( |
|
| 553 | + [['Term_Taxonomy.taxonomy' => EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY]] |
|
| 554 | + ); |
|
| 555 | + if ($event_category_count === 0) { |
|
| 556 | + return; |
|
| 557 | + } |
|
| 558 | + $column_array = []; |
|
| 559 | + foreach ($this->_columns as $column => $column_label) { |
|
| 560 | + $column_array[ $column ] = $column_label; |
|
| 561 | + if ($column === 'venue') { |
|
| 562 | + $column_array['event_category'] = esc_html__('Event Category', 'event_espresso'); |
|
| 563 | + } |
|
| 564 | + } |
|
| 565 | + $this->_columns = $column_array; |
|
| 566 | + } |
|
| 567 | 567 | } |
@@ -12,2671 +12,2671 @@ |
||
| 12 | 12 | class Events_Admin_Page extends EE_Admin_Page_CPT |
| 13 | 13 | { |
| 14 | 14 | |
| 15 | - /** |
|
| 16 | - * This will hold the event object for event_details screen. |
|
| 17 | - * |
|
| 18 | - * @access protected |
|
| 19 | - * @var EE_Event $_event |
|
| 20 | - */ |
|
| 21 | - protected $_event; |
|
| 22 | - |
|
| 23 | - |
|
| 24 | - /** |
|
| 25 | - * This will hold the category object for category_details screen. |
|
| 26 | - * |
|
| 27 | - * @var stdClass $_category |
|
| 28 | - */ |
|
| 29 | - protected $_category; |
|
| 30 | - |
|
| 31 | - |
|
| 32 | - /** |
|
| 33 | - * This will hold the event model instance |
|
| 34 | - * |
|
| 35 | - * @var EEM_Event $_event_model |
|
| 36 | - */ |
|
| 37 | - protected $_event_model; |
|
| 38 | - |
|
| 39 | - |
|
| 40 | - /** |
|
| 41 | - * @var EE_Event |
|
| 42 | - */ |
|
| 43 | - protected $_cpt_model_obj = false; |
|
| 44 | - |
|
| 45 | - |
|
| 46 | - /** |
|
| 47 | - * Initialize page props for this admin page group. |
|
| 48 | - */ |
|
| 49 | - protected function _init_page_props() |
|
| 50 | - { |
|
| 51 | - $this->page_slug = EVENTS_PG_SLUG; |
|
| 52 | - $this->page_label = EVENTS_LABEL; |
|
| 53 | - $this->_admin_base_url = EVENTS_ADMIN_URL; |
|
| 54 | - $this->_admin_base_path = EVENTS_ADMIN; |
|
| 55 | - $this->_cpt_model_names = array( |
|
| 56 | - 'create_new' => 'EEM_Event', |
|
| 57 | - 'edit' => 'EEM_Event', |
|
| 58 | - ); |
|
| 59 | - $this->_cpt_edit_routes = array( |
|
| 60 | - 'espresso_events' => 'edit', |
|
| 61 | - ); |
|
| 62 | - add_action( |
|
| 63 | - 'AHEE__EE_Admin_Page_CPT__set_model_object__after_set_object', |
|
| 64 | - array($this, 'verify_event_edit'), |
|
| 65 | - 10, |
|
| 66 | - 2 |
|
| 67 | - ); |
|
| 68 | - } |
|
| 69 | - |
|
| 70 | - |
|
| 71 | - /** |
|
| 72 | - * Sets the ajax hooks used for this admin page group. |
|
| 73 | - */ |
|
| 74 | - protected function _ajax_hooks() |
|
| 75 | - { |
|
| 76 | - add_action('wp_ajax_ee_save_timezone_setting', array($this, 'save_timezonestring_setting')); |
|
| 77 | - } |
|
| 78 | - |
|
| 79 | - |
|
| 80 | - /** |
|
| 81 | - * Sets the page properties for this admin page group. |
|
| 82 | - */ |
|
| 83 | - protected function _define_page_props() |
|
| 84 | - { |
|
| 85 | - $this->_admin_page_title = EVENTS_LABEL; |
|
| 86 | - $this->_labels = array( |
|
| 87 | - 'buttons' => array( |
|
| 88 | - 'add' => esc_html__('Add New Event', 'event_espresso'), |
|
| 89 | - 'edit' => esc_html__('Edit Event', 'event_espresso'), |
|
| 90 | - 'delete' => esc_html__('Delete Event', 'event_espresso'), |
|
| 91 | - 'add_category' => esc_html__('Add New Category', 'event_espresso'), |
|
| 92 | - 'edit_category' => esc_html__('Edit Category', 'event_espresso'), |
|
| 93 | - 'delete_category' => esc_html__('Delete Category', 'event_espresso'), |
|
| 94 | - ), |
|
| 95 | - 'editor_title' => array( |
|
| 96 | - 'espresso_events' => esc_html__('Enter event title here', 'event_espresso'), |
|
| 97 | - ), |
|
| 98 | - 'publishbox' => array( |
|
| 99 | - 'create_new' => esc_html__('Save New Event', 'event_espresso'), |
|
| 100 | - 'edit' => esc_html__('Update Event', 'event_espresso'), |
|
| 101 | - 'add_category' => esc_html__('Save New Category', 'event_espresso'), |
|
| 102 | - 'edit_category' => esc_html__('Update Category', 'event_espresso'), |
|
| 103 | - 'template_settings' => esc_html__('Update Settings', 'event_espresso'), |
|
| 104 | - ), |
|
| 105 | - ); |
|
| 106 | - } |
|
| 107 | - |
|
| 108 | - |
|
| 109 | - /** |
|
| 110 | - * Sets the page routes property for this admin page group. |
|
| 111 | - */ |
|
| 112 | - protected function _set_page_routes() |
|
| 113 | - { |
|
| 114 | - // load formatter helper |
|
| 115 | - // load field generator helper |
|
| 116 | - // is there a evt_id in the request? |
|
| 117 | - $evt_id = ! empty($this->_req_data['EVT_ID']) && ! is_array($this->_req_data['EVT_ID']) |
|
| 118 | - ? $this->_req_data['EVT_ID'] |
|
| 119 | - : 0; |
|
| 120 | - $evt_id = ! empty($this->_req_data['post']) ? $this->_req_data['post'] : $evt_id; |
|
| 121 | - $this->_page_routes = array( |
|
| 122 | - 'default' => array( |
|
| 123 | - 'func' => '_events_overview_list_table', |
|
| 124 | - 'capability' => 'ee_read_events', |
|
| 125 | - ), |
|
| 126 | - 'create_new' => array( |
|
| 127 | - 'func' => '_create_new_cpt_item', |
|
| 128 | - 'capability' => 'ee_edit_events', |
|
| 129 | - ), |
|
| 130 | - 'edit' => array( |
|
| 131 | - 'func' => '_edit_cpt_item', |
|
| 132 | - 'capability' => 'ee_edit_event', |
|
| 133 | - 'obj_id' => $evt_id, |
|
| 134 | - ), |
|
| 135 | - 'copy_event' => array( |
|
| 136 | - 'func' => '_copy_events', |
|
| 137 | - 'capability' => 'ee_edit_event', |
|
| 138 | - 'obj_id' => $evt_id, |
|
| 139 | - 'noheader' => true, |
|
| 140 | - ), |
|
| 141 | - 'trash_event' => array( |
|
| 142 | - 'func' => '_trash_or_restore_event', |
|
| 143 | - 'args' => array('event_status' => 'trash'), |
|
| 144 | - 'capability' => 'ee_delete_event', |
|
| 145 | - 'obj_id' => $evt_id, |
|
| 146 | - 'noheader' => true, |
|
| 147 | - ), |
|
| 148 | - 'trash_events' => array( |
|
| 149 | - 'func' => '_trash_or_restore_events', |
|
| 150 | - 'args' => array('event_status' => 'trash'), |
|
| 151 | - 'capability' => 'ee_delete_events', |
|
| 152 | - 'noheader' => true, |
|
| 153 | - ), |
|
| 154 | - 'restore_event' => array( |
|
| 155 | - 'func' => '_trash_or_restore_event', |
|
| 156 | - 'args' => array('event_status' => 'draft'), |
|
| 157 | - 'capability' => 'ee_delete_event', |
|
| 158 | - 'obj_id' => $evt_id, |
|
| 159 | - 'noheader' => true, |
|
| 160 | - ), |
|
| 161 | - 'restore_events' => array( |
|
| 162 | - 'func' => '_trash_or_restore_events', |
|
| 163 | - 'args' => array('event_status' => 'draft'), |
|
| 164 | - 'capability' => 'ee_delete_events', |
|
| 165 | - 'noheader' => true, |
|
| 166 | - ), |
|
| 167 | - 'delete_event' => array( |
|
| 168 | - 'func' => '_delete_event', |
|
| 169 | - 'capability' => 'ee_delete_event', |
|
| 170 | - 'obj_id' => $evt_id, |
|
| 171 | - 'noheader' => true, |
|
| 172 | - ), |
|
| 173 | - 'delete_events' => array( |
|
| 174 | - 'func' => '_delete_events', |
|
| 175 | - 'capability' => 'ee_delete_events', |
|
| 176 | - 'noheader' => true, |
|
| 177 | - ), |
|
| 178 | - 'view_report' => array( |
|
| 179 | - 'func' => '_view_report', |
|
| 180 | - 'capablity' => 'ee_edit_events', |
|
| 181 | - ), |
|
| 182 | - 'default_event_settings' => array( |
|
| 183 | - 'func' => '_default_event_settings', |
|
| 184 | - 'capability' => 'manage_options', |
|
| 185 | - ), |
|
| 186 | - 'update_default_event_settings' => array( |
|
| 187 | - 'func' => '_update_default_event_settings', |
|
| 188 | - 'capability' => 'manage_options', |
|
| 189 | - 'noheader' => true, |
|
| 190 | - ), |
|
| 191 | - 'template_settings' => array( |
|
| 192 | - 'func' => '_template_settings', |
|
| 193 | - 'capability' => 'manage_options', |
|
| 194 | - ), |
|
| 195 | - // event category tab related |
|
| 196 | - 'add_category' => array( |
|
| 197 | - 'func' => '_category_details', |
|
| 198 | - 'capability' => 'ee_edit_event_category', |
|
| 199 | - 'args' => array('add'), |
|
| 200 | - ), |
|
| 201 | - 'edit_category' => array( |
|
| 202 | - 'func' => '_category_details', |
|
| 203 | - 'capability' => 'ee_edit_event_category', |
|
| 204 | - 'args' => array('edit'), |
|
| 205 | - ), |
|
| 206 | - 'delete_categories' => array( |
|
| 207 | - 'func' => '_delete_categories', |
|
| 208 | - 'capability' => 'ee_delete_event_category', |
|
| 209 | - 'noheader' => true, |
|
| 210 | - ), |
|
| 211 | - 'delete_category' => array( |
|
| 212 | - 'func' => '_delete_categories', |
|
| 213 | - 'capability' => 'ee_delete_event_category', |
|
| 214 | - 'noheader' => true, |
|
| 215 | - ), |
|
| 216 | - 'insert_category' => array( |
|
| 217 | - 'func' => '_insert_or_update_category', |
|
| 218 | - 'args' => array('new_category' => true), |
|
| 219 | - 'capability' => 'ee_edit_event_category', |
|
| 220 | - 'noheader' => true, |
|
| 221 | - ), |
|
| 222 | - 'update_category' => array( |
|
| 223 | - 'func' => '_insert_or_update_category', |
|
| 224 | - 'args' => array('new_category' => false), |
|
| 225 | - 'capability' => 'ee_edit_event_category', |
|
| 226 | - 'noheader' => true, |
|
| 227 | - ), |
|
| 228 | - 'category_list' => array( |
|
| 229 | - 'func' => '_category_list_table', |
|
| 230 | - 'capability' => 'ee_manage_event_categories', |
|
| 231 | - ), |
|
| 232 | - ); |
|
| 233 | - } |
|
| 234 | - |
|
| 235 | - |
|
| 236 | - /** |
|
| 237 | - * Set the _page_config property for this admin page group. |
|
| 238 | - */ |
|
| 239 | - protected function _set_page_config() |
|
| 240 | - { |
|
| 241 | - $this->_page_config = array( |
|
| 242 | - 'default' => array( |
|
| 243 | - 'nav' => array( |
|
| 244 | - 'label' => esc_html__('Overview', 'event_espresso'), |
|
| 245 | - 'order' => 10, |
|
| 246 | - ), |
|
| 247 | - 'list_table' => 'Events_Admin_List_Table', |
|
| 248 | - 'help_tabs' => array( |
|
| 249 | - 'events_overview_help_tab' => array( |
|
| 250 | - 'title' => esc_html__('Events Overview', 'event_espresso'), |
|
| 251 | - 'filename' => 'events_overview', |
|
| 252 | - ), |
|
| 253 | - 'events_overview_table_column_headings_help_tab' => array( |
|
| 254 | - 'title' => esc_html__('Events Overview Table Column Headings', 'event_espresso'), |
|
| 255 | - 'filename' => 'events_overview_table_column_headings', |
|
| 256 | - ), |
|
| 257 | - 'events_overview_filters_help_tab' => array( |
|
| 258 | - 'title' => esc_html__('Events Overview Filters', 'event_espresso'), |
|
| 259 | - 'filename' => 'events_overview_filters', |
|
| 260 | - ), |
|
| 261 | - 'events_overview_view_help_tab' => array( |
|
| 262 | - 'title' => esc_html__('Events Overview Views', 'event_espresso'), |
|
| 263 | - 'filename' => 'events_overview_views', |
|
| 264 | - ), |
|
| 265 | - 'events_overview_other_help_tab' => array( |
|
| 266 | - 'title' => esc_html__('Events Overview Other', 'event_espresso'), |
|
| 267 | - 'filename' => 'events_overview_other', |
|
| 268 | - ), |
|
| 269 | - ), |
|
| 270 | - 'help_tour' => array( |
|
| 271 | - 'Event_Overview_Help_Tour', |
|
| 272 | - // 'New_Features_Test_Help_Tour' for testing multiple help tour |
|
| 273 | - ), |
|
| 274 | - 'qtips' => array( |
|
| 275 | - 'EE_Event_List_Table_Tips', |
|
| 276 | - ), |
|
| 277 | - 'require_nonce' => false, |
|
| 278 | - ), |
|
| 279 | - 'create_new' => array( |
|
| 280 | - 'nav' => array( |
|
| 281 | - 'label' => esc_html__('Add Event', 'event_espresso'), |
|
| 282 | - 'order' => 5, |
|
| 283 | - 'persistent' => false, |
|
| 284 | - ), |
|
| 285 | - 'metaboxes' => array('_register_event_editor_meta_boxes'), |
|
| 286 | - 'help_tabs' => array( |
|
| 287 | - 'event_editor_help_tab' => array( |
|
| 288 | - 'title' => esc_html__('Event Editor', 'event_espresso'), |
|
| 289 | - 'filename' => 'event_editor', |
|
| 290 | - ), |
|
| 291 | - 'event_editor_title_richtexteditor_help_tab' => array( |
|
| 292 | - 'title' => esc_html__('Event Title & Rich Text Editor', 'event_espresso'), |
|
| 293 | - 'filename' => 'event_editor_title_richtexteditor', |
|
| 294 | - ), |
|
| 295 | - 'event_editor_venue_details_help_tab' => array( |
|
| 296 | - 'title' => esc_html__('Event Venue Details', 'event_espresso'), |
|
| 297 | - 'filename' => 'event_editor_venue_details', |
|
| 298 | - ), |
|
| 299 | - 'event_editor_event_datetimes_help_tab' => array( |
|
| 300 | - 'title' => esc_html__('Event Datetimes', 'event_espresso'), |
|
| 301 | - 'filename' => 'event_editor_event_datetimes', |
|
| 302 | - ), |
|
| 303 | - 'event_editor_event_tickets_help_tab' => array( |
|
| 304 | - 'title' => esc_html__('Event Tickets', 'event_espresso'), |
|
| 305 | - 'filename' => 'event_editor_event_tickets', |
|
| 306 | - ), |
|
| 307 | - 'event_editor_event_registration_options_help_tab' => array( |
|
| 308 | - 'title' => esc_html__('Event Registration Options', 'event_espresso'), |
|
| 309 | - 'filename' => 'event_editor_event_registration_options', |
|
| 310 | - ), |
|
| 311 | - 'event_editor_tags_categories_help_tab' => array( |
|
| 312 | - 'title' => esc_html__('Event Tags & Categories', 'event_espresso'), |
|
| 313 | - 'filename' => 'event_editor_tags_categories', |
|
| 314 | - ), |
|
| 315 | - 'event_editor_questions_registrants_help_tab' => array( |
|
| 316 | - 'title' => esc_html__('Questions for Registrants', 'event_espresso'), |
|
| 317 | - 'filename' => 'event_editor_questions_registrants', |
|
| 318 | - ), |
|
| 319 | - 'event_editor_save_new_event_help_tab' => array( |
|
| 320 | - 'title' => esc_html__('Save New Event', 'event_espresso'), |
|
| 321 | - 'filename' => 'event_editor_save_new_event', |
|
| 322 | - ), |
|
| 323 | - 'event_editor_other_help_tab' => array( |
|
| 324 | - 'title' => esc_html__('Event Other', 'event_espresso'), |
|
| 325 | - 'filename' => 'event_editor_other', |
|
| 326 | - ), |
|
| 327 | - ), |
|
| 328 | - 'help_tour' => array( |
|
| 329 | - 'Event_Editor_Help_Tour', |
|
| 330 | - ), |
|
| 331 | - 'qtips' => array('EE_Event_Editor_Decaf_Tips'), |
|
| 332 | - 'require_nonce' => false, |
|
| 333 | - ), |
|
| 334 | - 'edit' => array( |
|
| 335 | - 'nav' => array( |
|
| 336 | - 'label' => esc_html__('Edit Event', 'event_espresso'), |
|
| 337 | - 'order' => 5, |
|
| 338 | - 'persistent' => false, |
|
| 339 | - 'url' => isset($this->_req_data['post']) |
|
| 340 | - ? EE_Admin_Page::add_query_args_and_nonce( |
|
| 341 | - array('post' => $this->_req_data['post'], 'action' => 'edit'), |
|
| 342 | - $this->_current_page_view_url |
|
| 343 | - ) |
|
| 344 | - : $this->_admin_base_url, |
|
| 345 | - ), |
|
| 346 | - 'metaboxes' => array('_register_event_editor_meta_boxes'), |
|
| 347 | - 'help_tabs' => array( |
|
| 348 | - 'event_editor_help_tab' => array( |
|
| 349 | - 'title' => esc_html__('Event Editor', 'event_espresso'), |
|
| 350 | - 'filename' => 'event_editor', |
|
| 351 | - ), |
|
| 352 | - 'event_editor_title_richtexteditor_help_tab' => array( |
|
| 353 | - 'title' => esc_html__('Event Title & Rich Text Editor', 'event_espresso'), |
|
| 354 | - 'filename' => 'event_editor_title_richtexteditor', |
|
| 355 | - ), |
|
| 356 | - 'event_editor_venue_details_help_tab' => array( |
|
| 357 | - 'title' => esc_html__('Event Venue Details', 'event_espresso'), |
|
| 358 | - 'filename' => 'event_editor_venue_details', |
|
| 359 | - ), |
|
| 360 | - 'event_editor_event_datetimes_help_tab' => array( |
|
| 361 | - 'title' => esc_html__('Event Datetimes', 'event_espresso'), |
|
| 362 | - 'filename' => 'event_editor_event_datetimes', |
|
| 363 | - ), |
|
| 364 | - 'event_editor_event_tickets_help_tab' => array( |
|
| 365 | - 'title' => esc_html__('Event Tickets', 'event_espresso'), |
|
| 366 | - 'filename' => 'event_editor_event_tickets', |
|
| 367 | - ), |
|
| 368 | - 'event_editor_event_registration_options_help_tab' => array( |
|
| 369 | - 'title' => esc_html__('Event Registration Options', 'event_espresso'), |
|
| 370 | - 'filename' => 'event_editor_event_registration_options', |
|
| 371 | - ), |
|
| 372 | - 'event_editor_tags_categories_help_tab' => array( |
|
| 373 | - 'title' => esc_html__('Event Tags & Categories', 'event_espresso'), |
|
| 374 | - 'filename' => 'event_editor_tags_categories', |
|
| 375 | - ), |
|
| 376 | - 'event_editor_questions_registrants_help_tab' => array( |
|
| 377 | - 'title' => esc_html__('Questions for Registrants', 'event_espresso'), |
|
| 378 | - 'filename' => 'event_editor_questions_registrants', |
|
| 379 | - ), |
|
| 380 | - 'event_editor_save_new_event_help_tab' => array( |
|
| 381 | - 'title' => esc_html__('Save New Event', 'event_espresso'), |
|
| 382 | - 'filename' => 'event_editor_save_new_event', |
|
| 383 | - ), |
|
| 384 | - 'event_editor_other_help_tab' => array( |
|
| 385 | - 'title' => esc_html__('Event Other', 'event_espresso'), |
|
| 386 | - 'filename' => 'event_editor_other', |
|
| 387 | - ), |
|
| 388 | - ), |
|
| 389 | - 'qtips' => array('EE_Event_Editor_Decaf_Tips'), |
|
| 390 | - 'require_nonce' => false, |
|
| 391 | - ), |
|
| 392 | - 'default_event_settings' => array( |
|
| 393 | - 'nav' => array( |
|
| 394 | - 'label' => esc_html__('Default Settings', 'event_espresso'), |
|
| 395 | - 'order' => 40, |
|
| 396 | - ), |
|
| 397 | - 'metaboxes' => array_merge($this->_default_espresso_metaboxes, array('_publish_post_box')), |
|
| 398 | - 'labels' => array( |
|
| 399 | - 'publishbox' => esc_html__('Update Settings', 'event_espresso'), |
|
| 400 | - ), |
|
| 401 | - 'help_tabs' => array( |
|
| 402 | - 'default_settings_help_tab' => array( |
|
| 403 | - 'title' => esc_html__('Default Event Settings', 'event_espresso'), |
|
| 404 | - 'filename' => 'events_default_settings', |
|
| 405 | - ), |
|
| 406 | - 'default_settings_status_help_tab' => array( |
|
| 407 | - 'title' => esc_html__('Default Registration Status', 'event_espresso'), |
|
| 408 | - 'filename' => 'events_default_settings_status', |
|
| 409 | - ), |
|
| 410 | - 'default_maximum_tickets_help_tab' => array( |
|
| 411 | - 'title' => esc_html__('Default Maximum Tickets Per Order', 'event_espresso'), |
|
| 412 | - 'filename' => 'events_default_settings_max_tickets', |
|
| 413 | - ), |
|
| 414 | - ), |
|
| 415 | - 'help_tour' => array('Event_Default_Settings_Help_Tour'), |
|
| 416 | - 'require_nonce' => false, |
|
| 417 | - ), |
|
| 418 | - // template settings |
|
| 419 | - 'template_settings' => array( |
|
| 420 | - 'nav' => array( |
|
| 421 | - 'label' => esc_html__('Templates', 'event_espresso'), |
|
| 422 | - 'order' => 30, |
|
| 423 | - ), |
|
| 424 | - 'metaboxes' => $this->_default_espresso_metaboxes, |
|
| 425 | - 'help_tabs' => array( |
|
| 426 | - 'general_settings_templates_help_tab' => array( |
|
| 427 | - 'title' => esc_html__('Templates', 'event_espresso'), |
|
| 428 | - 'filename' => 'general_settings_templates', |
|
| 429 | - ), |
|
| 430 | - ), |
|
| 431 | - 'help_tour' => array('Templates_Help_Tour'), |
|
| 432 | - 'require_nonce' => false, |
|
| 433 | - ), |
|
| 434 | - // event category stuff |
|
| 435 | - 'add_category' => array( |
|
| 436 | - 'nav' => array( |
|
| 437 | - 'label' => esc_html__('Add Category', 'event_espresso'), |
|
| 438 | - 'order' => 15, |
|
| 439 | - 'persistent' => false, |
|
| 440 | - ), |
|
| 441 | - 'help_tabs' => array( |
|
| 442 | - 'add_category_help_tab' => array( |
|
| 443 | - 'title' => esc_html__('Add New Event Category', 'event_espresso'), |
|
| 444 | - 'filename' => 'events_add_category', |
|
| 445 | - ), |
|
| 446 | - ), |
|
| 447 | - 'help_tour' => array('Event_Add_Category_Help_Tour'), |
|
| 448 | - 'metaboxes' => array('_publish_post_box'), |
|
| 449 | - 'require_nonce' => false, |
|
| 450 | - ), |
|
| 451 | - 'edit_category' => array( |
|
| 452 | - 'nav' => array( |
|
| 453 | - 'label' => esc_html__('Edit Category', 'event_espresso'), |
|
| 454 | - 'order' => 15, |
|
| 455 | - 'persistent' => false, |
|
| 456 | - 'url' => isset($this->_req_data['EVT_CAT_ID']) |
|
| 457 | - ? add_query_arg( |
|
| 458 | - array('EVT_CAT_ID' => $this->_req_data['EVT_CAT_ID']), |
|
| 459 | - $this->_current_page_view_url |
|
| 460 | - ) |
|
| 461 | - : $this->_admin_base_url, |
|
| 462 | - ), |
|
| 463 | - 'help_tabs' => array( |
|
| 464 | - 'edit_category_help_tab' => array( |
|
| 465 | - 'title' => esc_html__('Edit Event Category', 'event_espresso'), |
|
| 466 | - 'filename' => 'events_edit_category', |
|
| 467 | - ), |
|
| 468 | - ), |
|
| 469 | - /*'help_tour' => array('Event_Edit_Category_Help_Tour'),*/ |
|
| 470 | - 'metaboxes' => array('_publish_post_box'), |
|
| 471 | - 'require_nonce' => false, |
|
| 472 | - ), |
|
| 473 | - 'category_list' => array( |
|
| 474 | - 'nav' => array( |
|
| 475 | - 'label' => esc_html__('Categories', 'event_espresso'), |
|
| 476 | - 'order' => 20, |
|
| 477 | - ), |
|
| 478 | - 'list_table' => 'Event_Categories_Admin_List_Table', |
|
| 479 | - 'help_tabs' => array( |
|
| 480 | - 'events_categories_help_tab' => array( |
|
| 481 | - 'title' => esc_html__('Event Categories', 'event_espresso'), |
|
| 482 | - 'filename' => 'events_categories', |
|
| 483 | - ), |
|
| 484 | - 'events_categories_table_column_headings_help_tab' => array( |
|
| 485 | - 'title' => esc_html__('Event Categories Table Column Headings', 'event_espresso'), |
|
| 486 | - 'filename' => 'events_categories_table_column_headings', |
|
| 487 | - ), |
|
| 488 | - 'events_categories_view_help_tab' => array( |
|
| 489 | - 'title' => esc_html__('Event Categories Views', 'event_espresso'), |
|
| 490 | - 'filename' => 'events_categories_views', |
|
| 491 | - ), |
|
| 492 | - 'events_categories_other_help_tab' => array( |
|
| 493 | - 'title' => esc_html__('Event Categories Other', 'event_espresso'), |
|
| 494 | - 'filename' => 'events_categories_other', |
|
| 495 | - ), |
|
| 496 | - ), |
|
| 497 | - 'help_tour' => array( |
|
| 498 | - 'Event_Categories_Help_Tour', |
|
| 499 | - ), |
|
| 500 | - 'metaboxes' => $this->_default_espresso_metaboxes, |
|
| 501 | - 'require_nonce' => false, |
|
| 502 | - ), |
|
| 503 | - ); |
|
| 504 | - } |
|
| 505 | - |
|
| 506 | - |
|
| 507 | - /** |
|
| 508 | - * Used to register any global screen options if necessary for every route in this admin page group. |
|
| 509 | - */ |
|
| 510 | - protected function _add_screen_options() |
|
| 511 | - { |
|
| 512 | - } |
|
| 513 | - |
|
| 514 | - |
|
| 515 | - /** |
|
| 516 | - * Implementing the screen options for the 'default' route. |
|
| 517 | - */ |
|
| 518 | - protected function _add_screen_options_default() |
|
| 519 | - { |
|
| 520 | - $this->_per_page_screen_option(); |
|
| 521 | - } |
|
| 522 | - |
|
| 523 | - |
|
| 524 | - /** |
|
| 525 | - * Implementing screen options for the category list route. |
|
| 526 | - */ |
|
| 527 | - protected function _add_screen_options_category_list() |
|
| 528 | - { |
|
| 529 | - $page_title = $this->_admin_page_title; |
|
| 530 | - $this->_admin_page_title = esc_html__('Categories', 'event_espresso'); |
|
| 531 | - $this->_per_page_screen_option(); |
|
| 532 | - $this->_admin_page_title = $page_title; |
|
| 533 | - } |
|
| 534 | - |
|
| 535 | - |
|
| 536 | - /** |
|
| 537 | - * Used to register any global feature pointers for the admin page group. |
|
| 538 | - */ |
|
| 539 | - protected function _add_feature_pointers() |
|
| 540 | - { |
|
| 541 | - } |
|
| 542 | - |
|
| 543 | - |
|
| 544 | - /** |
|
| 545 | - * Registers and enqueues any global scripts and styles for the entire admin page group. |
|
| 546 | - */ |
|
| 547 | - public function load_scripts_styles() |
|
| 548 | - { |
|
| 549 | - wp_register_style( |
|
| 550 | - 'events-admin-css', |
|
| 551 | - EVENTS_ASSETS_URL . 'events-admin-page.css', |
|
| 552 | - array(), |
|
| 553 | - EVENT_ESPRESSO_VERSION |
|
| 554 | - ); |
|
| 555 | - wp_register_style('ee-cat-admin', EVENTS_ASSETS_URL . 'ee-cat-admin.css', array(), EVENT_ESPRESSO_VERSION); |
|
| 556 | - wp_enqueue_style('events-admin-css'); |
|
| 557 | - wp_enqueue_style('ee-cat-admin'); |
|
| 558 | - // todo note: we also need to load_scripts_styles per view (i.e. default/view_report/event_details |
|
| 559 | - // registers for all views |
|
| 560 | - // scripts |
|
| 561 | - wp_register_script( |
|
| 562 | - 'event_editor_js', |
|
| 563 | - EVENTS_ASSETS_URL . 'event_editor.js', |
|
| 564 | - array('ee_admin_js', 'jquery-ui-slider', 'jquery-ui-timepicker-addon'), |
|
| 565 | - EVENT_ESPRESSO_VERSION, |
|
| 566 | - true |
|
| 567 | - ); |
|
| 568 | - } |
|
| 569 | - |
|
| 570 | - |
|
| 571 | - /** |
|
| 572 | - * Enqueuing scripts and styles specific to this view |
|
| 573 | - */ |
|
| 574 | - public function load_scripts_styles_create_new() |
|
| 575 | - { |
|
| 576 | - $this->load_scripts_styles_edit(); |
|
| 577 | - } |
|
| 578 | - |
|
| 579 | - |
|
| 580 | - /** |
|
| 581 | - * Enqueuing scripts and styles specific to this view |
|
| 582 | - */ |
|
| 583 | - public function load_scripts_styles_edit() |
|
| 584 | - { |
|
| 585 | - // styles |
|
| 586 | - wp_enqueue_style('espresso-ui-theme'); |
|
| 587 | - wp_register_style( |
|
| 588 | - 'event-editor-css', |
|
| 589 | - EVENTS_ASSETS_URL . 'event-editor.css', |
|
| 590 | - array('ee-admin-css'), |
|
| 591 | - EVENT_ESPRESSO_VERSION |
|
| 592 | - ); |
|
| 593 | - wp_enqueue_style('event-editor-css'); |
|
| 594 | - // scripts |
|
| 595 | - wp_register_script( |
|
| 596 | - 'event-datetime-metabox', |
|
| 597 | - EVENTS_ASSETS_URL . 'event-datetime-metabox.js', |
|
| 598 | - array('event_editor_js', 'ee-datepicker'), |
|
| 599 | - EVENT_ESPRESSO_VERSION |
|
| 600 | - ); |
|
| 601 | - wp_enqueue_script('event-datetime-metabox'); |
|
| 602 | - } |
|
| 603 | - |
|
| 604 | - |
|
| 605 | - /** |
|
| 606 | - * Populating the _views property for the category list table view. |
|
| 607 | - */ |
|
| 608 | - protected function _set_list_table_views_category_list() |
|
| 609 | - { |
|
| 610 | - $this->_views = array( |
|
| 611 | - 'all' => array( |
|
| 612 | - 'slug' => 'all', |
|
| 613 | - 'label' => esc_html__('All', 'event_espresso'), |
|
| 614 | - 'count' => 0, |
|
| 615 | - 'bulk_action' => array( |
|
| 616 | - 'delete_categories' => esc_html__('Delete Permanently', 'event_espresso'), |
|
| 617 | - ), |
|
| 618 | - ), |
|
| 619 | - ); |
|
| 620 | - } |
|
| 621 | - |
|
| 622 | - |
|
| 623 | - /** |
|
| 624 | - * For adding anything that fires on the admin_init hook for any route within this admin page group. |
|
| 625 | - */ |
|
| 626 | - public function admin_init() |
|
| 627 | - { |
|
| 628 | - EE_Registry::$i18n_js_strings['image_confirm'] = esc_html__( |
|
| 629 | - 'Do you really want to delete this image? Please remember to update your event to complete the removal.', |
|
| 630 | - 'event_espresso' |
|
| 631 | - ); |
|
| 632 | - } |
|
| 633 | - |
|
| 634 | - |
|
| 635 | - /** |
|
| 636 | - * For adding anything that should be triggered on the admin_notices hook for any route within this admin page |
|
| 637 | - * group. |
|
| 638 | - */ |
|
| 639 | - public function admin_notices() |
|
| 640 | - { |
|
| 641 | - } |
|
| 642 | - |
|
| 643 | - |
|
| 644 | - /** |
|
| 645 | - * For adding anything that should be triggered on the `admin_print_footer_scripts` hook for any route within |
|
| 646 | - * this admin page group. |
|
| 647 | - */ |
|
| 648 | - public function admin_footer_scripts() |
|
| 649 | - { |
|
| 650 | - } |
|
| 651 | - |
|
| 652 | - |
|
| 653 | - /** |
|
| 654 | - * Call this function to verify if an event is public and has tickets for sale. If it does, then we need to show a |
|
| 655 | - * warning (via EE_Error::add_error()); |
|
| 656 | - * |
|
| 657 | - * @param EE_Event $event Event object |
|
| 658 | - * @param string $req_type |
|
| 659 | - * @return void |
|
| 660 | - * @throws EE_Error |
|
| 661 | - * @access public |
|
| 662 | - */ |
|
| 663 | - public function verify_event_edit($event = null, $req_type = '') |
|
| 664 | - { |
|
| 665 | - // don't need to do this when processing |
|
| 666 | - if (! empty($req_type)) { |
|
| 667 | - return; |
|
| 668 | - } |
|
| 669 | - // no event? |
|
| 670 | - if (empty($event)) { |
|
| 671 | - // set event |
|
| 672 | - $event = $this->_cpt_model_obj; |
|
| 673 | - } |
|
| 674 | - // STILL no event? |
|
| 675 | - if (! $event instanceof EE_Event) { |
|
| 676 | - return; |
|
| 677 | - } |
|
| 678 | - $orig_status = $event->status(); |
|
| 679 | - // first check if event is active. |
|
| 680 | - if ($orig_status === EEM_Event::cancelled |
|
| 681 | - || $orig_status === EEM_Event::postponed |
|
| 682 | - || $event->is_expired() |
|
| 683 | - || $event->is_inactive() |
|
| 684 | - ) { |
|
| 685 | - return; |
|
| 686 | - } |
|
| 687 | - // made it here so it IS active... next check that any of the tickets are sold. |
|
| 688 | - if ($event->is_sold_out(true)) { |
|
| 689 | - if ($orig_status !== EEM_Event::sold_out && $event->status() !== $orig_status) { |
|
| 690 | - EE_Error::add_attention( |
|
| 691 | - sprintf( |
|
| 692 | - esc_html__( |
|
| 693 | - 'Please note that the Event Status has automatically been changed to %s because there are no more spaces available for this event. However, this change is not permanent until you update the event. You can change the status back to something else before updating if you wish.', |
|
| 694 | - 'event_espresso' |
|
| 695 | - ), |
|
| 696 | - EEH_Template::pretty_status(EEM_Event::sold_out, false, 'sentence') |
|
| 697 | - ) |
|
| 698 | - ); |
|
| 699 | - } |
|
| 700 | - return; |
|
| 701 | - } elseif ($orig_status === EEM_Event::sold_out) { |
|
| 702 | - EE_Error::add_attention( |
|
| 703 | - sprintf( |
|
| 704 | - esc_html__( |
|
| 705 | - 'Please note that the Event Status has automatically been changed to %s because more spaces have become available for this event, most likely due to abandoned transactions freeing up reserved tickets. However, this change is not permanent until you update the event. If you wish, you can change the status back to something else before updating.', |
|
| 706 | - 'event_espresso' |
|
| 707 | - ), |
|
| 708 | - EEH_Template::pretty_status($event->status(), false, 'sentence') |
|
| 709 | - ) |
|
| 710 | - ); |
|
| 711 | - } |
|
| 712 | - // now we need to determine if the event has any tickets on sale. If not then we dont' show the error |
|
| 713 | - if (! $event->tickets_on_sale()) { |
|
| 714 | - return; |
|
| 715 | - } |
|
| 716 | - // made it here so show warning |
|
| 717 | - $this->_edit_event_warning(); |
|
| 718 | - } |
|
| 719 | - |
|
| 720 | - |
|
| 721 | - /** |
|
| 722 | - * This is the text used for when an event is being edited that is public and has tickets for sale. |
|
| 723 | - * When needed, hook this into a EE_Error::add_error() notice. |
|
| 724 | - * |
|
| 725 | - * @access protected |
|
| 726 | - * @return void |
|
| 727 | - */ |
|
| 728 | - protected function _edit_event_warning() |
|
| 729 | - { |
|
| 730 | - // we don't want to add warnings during these requests |
|
| 731 | - if (isset($this->_req_data['action']) && $this->_req_data['action'] === 'editpost') { |
|
| 732 | - return; |
|
| 733 | - } |
|
| 734 | - EE_Error::add_attention( |
|
| 735 | - sprintf( |
|
| 736 | - esc_html__( |
|
| 737 | - 'Your event is open for registration. Making changes may disrupt any transactions in progress. %sLearn more%s', |
|
| 738 | - 'event_espresso' |
|
| 739 | - ), |
|
| 740 | - '<a class="espresso-help-tab-lnk">', |
|
| 741 | - '</a>' |
|
| 742 | - ) |
|
| 743 | - ); |
|
| 744 | - } |
|
| 745 | - |
|
| 746 | - |
|
| 747 | - /** |
|
| 748 | - * When a user is creating a new event, notify them if they haven't set their timezone. |
|
| 749 | - * Otherwise, do the normal logic |
|
| 750 | - * |
|
| 751 | - * @return string |
|
| 752 | - * @throws \EE_Error |
|
| 753 | - */ |
|
| 754 | - protected function _create_new_cpt_item() |
|
| 755 | - { |
|
| 756 | - $has_timezone_string = get_option('timezone_string'); |
|
| 757 | - // only nag them about setting their timezone if it's their first event, and they haven't already done it |
|
| 758 | - if (! $has_timezone_string && ! EEM_Event::instance()->exists(array())) { |
|
| 759 | - EE_Error::add_attention( |
|
| 760 | - sprintf( |
|
| 761 | - __( |
|
| 762 | - 'Your website\'s timezone is currently set to a UTC offset. We recommend updating your timezone to a city or region near you before you create an event. Change your timezone now:%1$s%2$s%3$sChange Timezone%4$s', |
|
| 763 | - 'event_espresso' |
|
| 764 | - ), |
|
| 765 | - '<br>', |
|
| 766 | - '<select id="timezone_string" name="timezone_string" aria-describedby="timezone-description">' |
|
| 767 | - . EEH_DTT_Helper::wp_timezone_choice('', EEH_DTT_Helper::get_user_locale()) |
|
| 768 | - . '</select>', |
|
| 769 | - '<button class="button button-secondary timezone-submit">', |
|
| 770 | - '</button><span class="spinner"></span>' |
|
| 771 | - ), |
|
| 772 | - __FILE__, |
|
| 773 | - __FUNCTION__, |
|
| 774 | - __LINE__ |
|
| 775 | - ); |
|
| 776 | - } |
|
| 777 | - return parent::_create_new_cpt_item(); |
|
| 778 | - } |
|
| 779 | - |
|
| 780 | - |
|
| 781 | - /** |
|
| 782 | - * Sets the _views property for the default route in this admin page group. |
|
| 783 | - */ |
|
| 784 | - protected function _set_list_table_views_default() |
|
| 785 | - { |
|
| 786 | - $this->_views = array( |
|
| 787 | - 'all' => array( |
|
| 788 | - 'slug' => 'all', |
|
| 789 | - 'label' => esc_html__('View All Events', 'event_espresso'), |
|
| 790 | - 'count' => 0, |
|
| 791 | - 'bulk_action' => array( |
|
| 792 | - 'trash_events' => esc_html__('Move to Trash', 'event_espresso'), |
|
| 793 | - ), |
|
| 794 | - ), |
|
| 795 | - 'draft' => array( |
|
| 796 | - 'slug' => 'draft', |
|
| 797 | - 'label' => esc_html__('Draft', 'event_espresso'), |
|
| 798 | - 'count' => 0, |
|
| 799 | - 'bulk_action' => array( |
|
| 800 | - 'trash_events' => esc_html__('Move to Trash', 'event_espresso'), |
|
| 801 | - ), |
|
| 802 | - ), |
|
| 803 | - ); |
|
| 804 | - if (EE_Registry::instance()->CAP->current_user_can('ee_delete_events', 'espresso_events_trash_events')) { |
|
| 805 | - $this->_views['trash'] = array( |
|
| 806 | - 'slug' => 'trash', |
|
| 807 | - 'label' => esc_html__('Trash', 'event_espresso'), |
|
| 808 | - 'count' => 0, |
|
| 809 | - 'bulk_action' => array( |
|
| 810 | - 'restore_events' => esc_html__('Restore From Trash', 'event_espresso'), |
|
| 811 | - 'delete_events' => esc_html__('Delete Permanently', 'event_espresso'), |
|
| 812 | - ), |
|
| 813 | - ); |
|
| 814 | - } |
|
| 815 | - } |
|
| 816 | - |
|
| 817 | - |
|
| 818 | - /** |
|
| 819 | - * Provides the legend item array for the default list table view. |
|
| 820 | - * |
|
| 821 | - * @return array |
|
| 822 | - */ |
|
| 823 | - protected function _event_legend_items() |
|
| 824 | - { |
|
| 825 | - $items = array( |
|
| 826 | - 'view_details' => array( |
|
| 827 | - 'class' => 'dashicons dashicons-search', |
|
| 828 | - 'desc' => esc_html__('View Event', 'event_espresso'), |
|
| 829 | - ), |
|
| 830 | - 'edit_event' => array( |
|
| 831 | - 'class' => 'ee-icon ee-icon-calendar-edit', |
|
| 832 | - 'desc' => esc_html__('Edit Event Details', 'event_espresso'), |
|
| 833 | - ), |
|
| 834 | - 'view_attendees' => array( |
|
| 835 | - 'class' => 'dashicons dashicons-groups', |
|
| 836 | - 'desc' => esc_html__('View Registrations for Event', 'event_espresso'), |
|
| 837 | - ), |
|
| 838 | - ); |
|
| 839 | - $items = apply_filters('FHEE__Events_Admin_Page___event_legend_items__items', $items); |
|
| 840 | - $statuses = array( |
|
| 841 | - 'sold_out_status' => array( |
|
| 842 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::sold_out, |
|
| 843 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::sold_out, false, 'sentence'), |
|
| 844 | - ), |
|
| 845 | - 'active_status' => array( |
|
| 846 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::active, |
|
| 847 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::active, false, 'sentence'), |
|
| 848 | - ), |
|
| 849 | - 'upcoming_status' => array( |
|
| 850 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::upcoming, |
|
| 851 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::upcoming, false, 'sentence'), |
|
| 852 | - ), |
|
| 853 | - 'postponed_status' => array( |
|
| 854 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::postponed, |
|
| 855 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::postponed, false, 'sentence'), |
|
| 856 | - ), |
|
| 857 | - 'cancelled_status' => array( |
|
| 858 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::cancelled, |
|
| 859 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::cancelled, false, 'sentence'), |
|
| 860 | - ), |
|
| 861 | - 'expired_status' => array( |
|
| 862 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::expired, |
|
| 863 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::expired, false, 'sentence'), |
|
| 864 | - ), |
|
| 865 | - 'inactive_status' => array( |
|
| 866 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::inactive, |
|
| 867 | - 'desc' => EEH_Template::pretty_status(EE_Datetime::inactive, false, 'sentence'), |
|
| 868 | - ), |
|
| 869 | - ); |
|
| 870 | - $statuses = apply_filters('FHEE__Events_Admin_Page__event_legend_items__statuses', $statuses); |
|
| 871 | - return array_merge($items, $statuses); |
|
| 872 | - } |
|
| 873 | - |
|
| 874 | - |
|
| 875 | - /** |
|
| 876 | - * @return EEM_Event |
|
| 877 | - */ |
|
| 878 | - private function _event_model() |
|
| 879 | - { |
|
| 880 | - if (! $this->_event_model instanceof EEM_Event) { |
|
| 881 | - $this->_event_model = EE_Registry::instance()->load_model('Event'); |
|
| 882 | - } |
|
| 883 | - return $this->_event_model; |
|
| 884 | - } |
|
| 885 | - |
|
| 886 | - |
|
| 887 | - /** |
|
| 888 | - * Adds extra buttons to the WP CPT permalink field row. |
|
| 889 | - * Method is called from parent and is hooked into the wp 'get_sample_permalink_html' filter. |
|
| 890 | - * |
|
| 891 | - * @param string $return the current html |
|
| 892 | - * @param int $id the post id for the page |
|
| 893 | - * @param string $new_title What the title is |
|
| 894 | - * @param string $new_slug what the slug is |
|
| 895 | - * @return string The new html string for the permalink area |
|
| 896 | - */ |
|
| 897 | - public function extra_permalink_field_buttons($return, $id, $new_title, $new_slug) |
|
| 898 | - { |
|
| 899 | - // make sure this is only when editing |
|
| 900 | - if (! empty($id)) { |
|
| 901 | - $post = get_post($id); |
|
| 902 | - $return .= '<a class="button button-small" onclick="prompt(\'Shortcode:\', jQuery(\'#shortcode\').val()); return false;" href="#" tabindex="-1">' |
|
| 903 | - . esc_html__('Shortcode', 'event_espresso') |
|
| 904 | - . '</a> '; |
|
| 905 | - $return .= '<input id="shortcode" type="hidden" value="[ESPRESSO_TICKET_SELECTOR event_id=' |
|
| 906 | - . $post->ID |
|
| 907 | - . ']">'; |
|
| 908 | - } |
|
| 909 | - return $return; |
|
| 910 | - } |
|
| 911 | - |
|
| 912 | - |
|
| 913 | - /** |
|
| 914 | - * _events_overview_list_table |
|
| 915 | - * This contains the logic for showing the events_overview list |
|
| 916 | - * |
|
| 917 | - * @access protected |
|
| 918 | - * @return void |
|
| 919 | - * @throws \EE_Error |
|
| 920 | - */ |
|
| 921 | - protected function _events_overview_list_table() |
|
| 922 | - { |
|
| 923 | - do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
| 924 | - $this->_template_args['after_list_table'] = ! empty($this->_template_args['after_list_table']) |
|
| 925 | - ? (array) $this->_template_args['after_list_table'] |
|
| 926 | - : array(); |
|
| 927 | - $this->_template_args['after_list_table']['view_event_list_button'] = EEH_HTML::br() |
|
| 928 | - . EEH_Template::get_button_or_link( |
|
| 929 | - get_post_type_archive_link('espresso_events'), |
|
| 930 | - esc_html__("View Event Archive Page", "event_espresso"), |
|
| 931 | - 'button' |
|
| 932 | - ); |
|
| 933 | - $this->_template_args['after_list_table']['legend'] = $this->_display_legend($this->_event_legend_items()); |
|
| 934 | - $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
| 935 | - 'create_new', |
|
| 936 | - 'add', |
|
| 937 | - array(), |
|
| 938 | - 'add-new-h2' |
|
| 939 | - ); |
|
| 940 | - $this->display_admin_list_table_page_with_no_sidebar(); |
|
| 941 | - } |
|
| 942 | - |
|
| 943 | - |
|
| 944 | - /** |
|
| 945 | - * this allows for extra misc actions in the default WP publish box |
|
| 946 | - * |
|
| 947 | - * @return void |
|
| 948 | - */ |
|
| 949 | - public function extra_misc_actions_publish_box() |
|
| 950 | - { |
|
| 951 | - $this->_generate_publish_box_extra_content(); |
|
| 952 | - } |
|
| 953 | - |
|
| 954 | - |
|
| 955 | - /** |
|
| 956 | - * This is hooked into the WordPress do_action('save_post') hook and runs after the custom post type has been |
|
| 957 | - * saved. |
|
| 958 | - * Typically you would use this to save any additional data. |
|
| 959 | - * Keep in mind also that "save_post" runs on EVERY post update to the database. |
|
| 960 | - * ALSO very important. When a post transitions from scheduled to published, |
|
| 961 | - * the save_post action is fired but you will NOT have any _POST data containing any extra info you may have from |
|
| 962 | - * other meta saves. So MAKE sure that you handle this accordingly. |
|
| 963 | - * |
|
| 964 | - * @access protected |
|
| 965 | - * @abstract |
|
| 966 | - * @param string $post_id The ID of the cpt that was saved (so you can link relationally) |
|
| 967 | - * @param object $post The post object of the cpt that was saved. |
|
| 968 | - * @return void |
|
| 969 | - * @throws \EE_Error |
|
| 970 | - */ |
|
| 971 | - protected function _insert_update_cpt_item($post_id, $post) |
|
| 972 | - { |
|
| 973 | - if ($post instanceof WP_Post && $post->post_type !== 'espresso_events') { |
|
| 974 | - // get out we're not processing an event save. |
|
| 975 | - return; |
|
| 976 | - } |
|
| 977 | - $event_values = array( |
|
| 978 | - 'EVT_display_desc' => ! empty($this->_req_data['display_desc']) ? 1 : 0, |
|
| 979 | - 'EVT_display_ticket_selector' => ! empty($this->_req_data['display_ticket_selector']) ? 1 : 0, |
|
| 980 | - 'EVT_additional_limit' => min( |
|
| 981 | - apply_filters('FHEE__EE_Events_Admin__insert_update_cpt_item__EVT_additional_limit_max', 255), |
|
| 982 | - ! empty($this->_req_data['additional_limit']) ? $this->_req_data['additional_limit'] : null |
|
| 983 | - ), |
|
| 984 | - 'EVT_default_registration_status' => ! empty($this->_req_data['EVT_default_registration_status']) |
|
| 985 | - ? $this->_req_data['EVT_default_registration_status'] |
|
| 986 | - : EE_Registry::instance()->CFG->registration->default_STS_ID, |
|
| 987 | - 'EVT_member_only' => ! empty($this->_req_data['member_only']) ? 1 : 0, |
|
| 988 | - 'EVT_allow_overflow' => ! empty($this->_req_data['EVT_allow_overflow']) ? 1 : 0, |
|
| 989 | - 'EVT_timezone_string' => ! empty($this->_req_data['timezone_string']) |
|
| 990 | - ? $this->_req_data['timezone_string'] : null, |
|
| 991 | - 'EVT_external_URL' => ! empty($this->_req_data['externalURL']) |
|
| 992 | - ? $this->_req_data['externalURL'] : null, |
|
| 993 | - 'EVT_phone' => ! empty($this->_req_data['event_phone']) |
|
| 994 | - ? $this->_req_data['event_phone'] : null, |
|
| 995 | - ); |
|
| 996 | - // update event |
|
| 997 | - $success = $this->_event_model()->update_by_ID($event_values, $post_id); |
|
| 998 | - // get event_object for other metaboxes... though it would seem to make sense to just use $this->_event_model()->get_one_by_ID( $post_id ).. i have to setup where conditions to override the filters in the model that filter out autodraft and inherit statuses so we GET the inherit id! |
|
| 999 | - $get_one_where = array( |
|
| 1000 | - $this->_event_model()->primary_key_name() => $post_id, |
|
| 1001 | - 'OR' => array( |
|
| 1002 | - 'status' => $post->post_status, |
|
| 1003 | - // if trying to "Publish" a sold out event, it's status will get switched back to "sold_out" in the db, |
|
| 1004 | - // but the returned object here has a status of "publish", so use the original post status as well |
|
| 1005 | - 'status*1' => $this->_req_data['original_post_status'], |
|
| 1006 | - ), |
|
| 1007 | - ); |
|
| 1008 | - $event = $this->_event_model()->get_one(array($get_one_where)); |
|
| 1009 | - // the following are default callbacks for event attachment updates that can be overridden by caffeinated functionality and/or addons. |
|
| 1010 | - $event_update_callbacks = apply_filters( |
|
| 1011 | - 'FHEE__Events_Admin_Page___insert_update_cpt_item__event_update_callbacks', |
|
| 1012 | - array( |
|
| 1013 | - array($this, '_default_venue_update'), |
|
| 1014 | - array($this, '_default_tickets_update'), |
|
| 1015 | - ) |
|
| 1016 | - ); |
|
| 1017 | - $att_success = true; |
|
| 1018 | - foreach ($event_update_callbacks as $e_callback) { |
|
| 1019 | - $_success = is_callable($e_callback) |
|
| 1020 | - ? call_user_func($e_callback, $event, $this->_req_data) |
|
| 1021 | - : false; |
|
| 1022 | - // if ANY of these updates fail then we want the appropriate global error message |
|
| 1023 | - $att_success = ! $att_success ? $att_success : $_success; |
|
| 1024 | - } |
|
| 1025 | - // any errors? |
|
| 1026 | - if ($success && false === $att_success) { |
|
| 1027 | - EE_Error::add_error( |
|
| 1028 | - esc_html__( |
|
| 1029 | - 'Event Details saved successfully but something went wrong with saving attachments.', |
|
| 1030 | - 'event_espresso' |
|
| 1031 | - ), |
|
| 1032 | - __FILE__, |
|
| 1033 | - __FUNCTION__, |
|
| 1034 | - __LINE__ |
|
| 1035 | - ); |
|
| 1036 | - } elseif ($success === false) { |
|
| 1037 | - EE_Error::add_error( |
|
| 1038 | - esc_html__('Event Details did not save successfully.', 'event_espresso'), |
|
| 1039 | - __FILE__, |
|
| 1040 | - __FUNCTION__, |
|
| 1041 | - __LINE__ |
|
| 1042 | - ); |
|
| 1043 | - } |
|
| 1044 | - } |
|
| 1045 | - |
|
| 1046 | - |
|
| 1047 | - /** |
|
| 1048 | - * @see parent::restore_item() |
|
| 1049 | - * @param int $post_id |
|
| 1050 | - * @param int $revision_id |
|
| 1051 | - */ |
|
| 1052 | - protected function _restore_cpt_item($post_id, $revision_id) |
|
| 1053 | - { |
|
| 1054 | - // copy existing event meta to new post |
|
| 1055 | - $post_evt = $this->_event_model()->get_one_by_ID($post_id); |
|
| 1056 | - if ($post_evt instanceof EE_Event) { |
|
| 1057 | - // meta revision restore |
|
| 1058 | - $post_evt->restore_revision($revision_id); |
|
| 1059 | - // related objs restore |
|
| 1060 | - $post_evt->restore_revision($revision_id, array('Venue', 'Datetime', 'Price')); |
|
| 1061 | - } |
|
| 1062 | - } |
|
| 1063 | - |
|
| 1064 | - |
|
| 1065 | - /** |
|
| 1066 | - * Attach the venue to the Event |
|
| 1067 | - * |
|
| 1068 | - * @param \EE_Event $evtobj Event Object to add the venue to |
|
| 1069 | - * @param array $data The request data from the form |
|
| 1070 | - * @return bool Success or fail. |
|
| 1071 | - */ |
|
| 1072 | - protected function _default_venue_update(\EE_Event $evtobj, $data) |
|
| 1073 | - { |
|
| 1074 | - require_once(EE_MODELS . 'EEM_Venue.model.php'); |
|
| 1075 | - $venue_model = EE_Registry::instance()->load_model('Venue'); |
|
| 1076 | - $rows_affected = null; |
|
| 1077 | - $venue_id = ! empty($data['venue_id']) ? $data['venue_id'] : null; |
|
| 1078 | - // very important. If we don't have a venue name... |
|
| 1079 | - // then we'll get out because not necessary to create empty venue |
|
| 1080 | - if (empty($data['venue_title'])) { |
|
| 1081 | - return false; |
|
| 1082 | - } |
|
| 1083 | - $venue_array = array( |
|
| 1084 | - 'VNU_wp_user' => $evtobj->get('EVT_wp_user'), |
|
| 1085 | - 'VNU_name' => ! empty($data['venue_title']) ? $data['venue_title'] : null, |
|
| 1086 | - 'VNU_desc' => ! empty($data['venue_description']) ? $data['venue_description'] : null, |
|
| 1087 | - 'VNU_identifier' => ! empty($data['venue_identifier']) ? $data['venue_identifier'] : null, |
|
| 1088 | - 'VNU_short_desc' => ! empty($data['venue_short_description']) ? $data['venue_short_description'] |
|
| 1089 | - : null, |
|
| 1090 | - 'VNU_address' => ! empty($data['address']) ? $data['address'] : null, |
|
| 1091 | - 'VNU_address2' => ! empty($data['address2']) ? $data['address2'] : null, |
|
| 1092 | - 'VNU_city' => ! empty($data['city']) ? $data['city'] : null, |
|
| 1093 | - 'STA_ID' => ! empty($data['state']) ? $data['state'] : null, |
|
| 1094 | - 'CNT_ISO' => ! empty($data['countries']) ? $data['countries'] : null, |
|
| 1095 | - 'VNU_zip' => ! empty($data['zip']) ? $data['zip'] : null, |
|
| 1096 | - 'VNU_phone' => ! empty($data['venue_phone']) ? $data['venue_phone'] : null, |
|
| 1097 | - 'VNU_capacity' => ! empty($data['venue_capacity']) ? $data['venue_capacity'] : null, |
|
| 1098 | - 'VNU_url' => ! empty($data['venue_url']) ? $data['venue_url'] : null, |
|
| 1099 | - 'VNU_virtual_phone' => ! empty($data['virtual_phone']) ? $data['virtual_phone'] : null, |
|
| 1100 | - 'VNU_virtual_url' => ! empty($data['virtual_url']) ? $data['virtual_url'] : null, |
|
| 1101 | - 'VNU_enable_for_gmap' => isset($data['enable_for_gmap']) ? 1 : 0, |
|
| 1102 | - 'status' => 'publish', |
|
| 1103 | - ); |
|
| 1104 | - // if we've got the venue_id then we're just updating the existing venue so let's do that and then get out. |
|
| 1105 | - if (! empty($venue_id)) { |
|
| 1106 | - $update_where = array($venue_model->primary_key_name() => $venue_id); |
|
| 1107 | - $rows_affected = $venue_model->update($venue_array, array($update_where)); |
|
| 1108 | - // we've gotta make sure that the venue is always attached to a revision.. add_relation_to should take care of making sure that the relation is already present. |
|
| 1109 | - $evtobj->_add_relation_to($venue_id, 'Venue'); |
|
| 1110 | - return $rows_affected > 0 ? true : false; |
|
| 1111 | - } else { |
|
| 1112 | - // we insert the venue |
|
| 1113 | - $venue_id = $venue_model->insert($venue_array); |
|
| 1114 | - $evtobj->_add_relation_to($venue_id, 'Venue'); |
|
| 1115 | - return ! empty($venue_id) ? true : false; |
|
| 1116 | - } |
|
| 1117 | - // when we have the ancestor come in it's already been handled by the revision save. |
|
| 1118 | - } |
|
| 1119 | - |
|
| 1120 | - |
|
| 1121 | - /** |
|
| 1122 | - * Handles saving everything related to Tickets (datetimes, tickets, prices) |
|
| 1123 | - * |
|
| 1124 | - * @param EE_Event $evtobj The Event object we're attaching data to |
|
| 1125 | - * @param array $data The request data from the form |
|
| 1126 | - * @return array |
|
| 1127 | - */ |
|
| 1128 | - protected function _default_tickets_update(EE_Event $evtobj, $data) |
|
| 1129 | - { |
|
| 1130 | - $success = true; |
|
| 1131 | - $saved_dtt = null; |
|
| 1132 | - $saved_tickets = array(); |
|
| 1133 | - $incoming_date_formats = array('Y-m-d', 'h:i a'); |
|
| 1134 | - foreach ($data['edit_event_datetimes'] as $row => $dtt) { |
|
| 1135 | - // trim all values to ensure any excess whitespace is removed. |
|
| 1136 | - $dtt = array_map('trim', $dtt); |
|
| 1137 | - $dtt['DTT_EVT_end'] = isset($dtt['DTT_EVT_end']) && ! empty($dtt['DTT_EVT_end']) ? $dtt['DTT_EVT_end'] |
|
| 1138 | - : $dtt['DTT_EVT_start']; |
|
| 1139 | - $datetime_values = array( |
|
| 1140 | - 'DTT_ID' => ! empty($dtt['DTT_ID']) ? $dtt['DTT_ID'] : null, |
|
| 1141 | - 'DTT_EVT_start' => $dtt['DTT_EVT_start'], |
|
| 1142 | - 'DTT_EVT_end' => $dtt['DTT_EVT_end'], |
|
| 1143 | - 'DTT_reg_limit' => empty($dtt['DTT_reg_limit']) ? EE_INF : $dtt['DTT_reg_limit'], |
|
| 1144 | - 'DTT_order' => $row, |
|
| 1145 | - ); |
|
| 1146 | - // if we have an id then let's get existing object first and then set the new values. Otherwise we instantiate a new object for save. |
|
| 1147 | - if (! empty($dtt['DTT_ID'])) { |
|
| 1148 | - $DTM = EE_Registry::instance() |
|
| 1149 | - ->load_model('Datetime', array($evtobj->get_timezone())) |
|
| 1150 | - ->get_one_by_ID($dtt['DTT_ID']); |
|
| 1151 | - $DTM->set_date_format($incoming_date_formats[0]); |
|
| 1152 | - $DTM->set_time_format($incoming_date_formats[1]); |
|
| 1153 | - foreach ($datetime_values as $field => $value) { |
|
| 1154 | - $DTM->set($field, $value); |
|
| 1155 | - } |
|
| 1156 | - // make sure the $dtt_id here is saved just in case after the add_relation_to() the autosave replaces it. We need to do this so we dont' TRASH the parent DTT. |
|
| 1157 | - $saved_dtts[ $DTM->ID() ] = $DTM; |
|
| 1158 | - } else { |
|
| 1159 | - $DTM = EE_Registry::instance()->load_class( |
|
| 1160 | - 'Datetime', |
|
| 1161 | - array($datetime_values, $evtobj->get_timezone(), $incoming_date_formats), |
|
| 1162 | - false, |
|
| 1163 | - false |
|
| 1164 | - ); |
|
| 1165 | - foreach ($datetime_values as $field => $value) { |
|
| 1166 | - $DTM->set($field, $value); |
|
| 1167 | - } |
|
| 1168 | - } |
|
| 1169 | - $DTM->save(); |
|
| 1170 | - $DTT = $evtobj->_add_relation_to($DTM, 'Datetime'); |
|
| 1171 | - // load DTT helper |
|
| 1172 | - // before going any further make sure our dates are setup correctly so that the end date is always equal or greater than the start date. |
|
| 1173 | - if ($DTT->get_raw('DTT_EVT_start') > $DTT->get_raw('DTT_EVT_end')) { |
|
| 1174 | - $DTT->set('DTT_EVT_end', $DTT->get('DTT_EVT_start')); |
|
| 1175 | - $DTT = EEH_DTT_Helper::date_time_add($DTT, 'DTT_EVT_end', 'days'); |
|
| 1176 | - $DTT->save(); |
|
| 1177 | - } |
|
| 1178 | - // now we got to make sure we add the new DTT_ID to the $saved_dtts array because it is possible there was a new one created for the autosave. |
|
| 1179 | - $saved_dtt = $DTT; |
|
| 1180 | - $success = ! $success ? $success : $DTT; |
|
| 1181 | - // if ANY of these updates fail then we want the appropriate global error message. |
|
| 1182 | - // //todo this is actually sucky we need a better error message but this is what it is for now. |
|
| 1183 | - } |
|
| 1184 | - // no dtts get deleted so we don't do any of that logic here. |
|
| 1185 | - // update tickets next |
|
| 1186 | - $old_tickets = isset($data['ticket_IDs']) ? explode(',', $data['ticket_IDs']) : array(); |
|
| 1187 | - foreach ($data['edit_tickets'] as $row => $tkt) { |
|
| 1188 | - $incoming_date_formats = array('Y-m-d', 'h:i a'); |
|
| 1189 | - $update_prices = false; |
|
| 1190 | - $ticket_price = isset($data['edit_prices'][ $row ][1]['PRC_amount']) |
|
| 1191 | - ? $data['edit_prices'][ $row ][1]['PRC_amount'] : 0; |
|
| 1192 | - // trim inputs to ensure any excess whitespace is removed. |
|
| 1193 | - $tkt = array_map('trim', $tkt); |
|
| 1194 | - if (empty($tkt['TKT_start_date'])) { |
|
| 1195 | - // let's use now in the set timezone. |
|
| 1196 | - $now = new DateTime('now', new DateTimeZone($evtobj->get_timezone())); |
|
| 1197 | - $tkt['TKT_start_date'] = $now->format($incoming_date_formats[0] . ' ' . $incoming_date_formats[1]); |
|
| 1198 | - } |
|
| 1199 | - if (empty($tkt['TKT_end_date'])) { |
|
| 1200 | - // use the start date of the first datetime |
|
| 1201 | - $dtt = $evtobj->first_datetime(); |
|
| 1202 | - $tkt['TKT_end_date'] = $dtt->start_date_and_time( |
|
| 1203 | - $incoming_date_formats[0], |
|
| 1204 | - $incoming_date_formats[1] |
|
| 1205 | - ); |
|
| 1206 | - } |
|
| 1207 | - $TKT_values = array( |
|
| 1208 | - 'TKT_ID' => ! empty($tkt['TKT_ID']) ? $tkt['TKT_ID'] : null, |
|
| 1209 | - 'TTM_ID' => ! empty($tkt['TTM_ID']) ? $tkt['TTM_ID'] : 0, |
|
| 1210 | - 'TKT_name' => ! empty($tkt['TKT_name']) ? $tkt['TKT_name'] : '', |
|
| 1211 | - 'TKT_description' => ! empty($tkt['TKT_description']) ? $tkt['TKT_description'] : '', |
|
| 1212 | - 'TKT_start_date' => $tkt['TKT_start_date'], |
|
| 1213 | - 'TKT_end_date' => $tkt['TKT_end_date'], |
|
| 1214 | - 'TKT_qty' => ! isset($tkt['TKT_qty']) || $tkt['TKT_qty'] === '' ? EE_INF : $tkt['TKT_qty'], |
|
| 1215 | - 'TKT_uses' => ! isset($tkt['TKT_uses']) || $tkt['TKT_uses'] === '' ? EE_INF : $tkt['TKT_uses'], |
|
| 1216 | - 'TKT_min' => empty($tkt['TKT_min']) ? 0 : $tkt['TKT_min'], |
|
| 1217 | - 'TKT_max' => empty($tkt['TKT_max']) ? EE_INF : $tkt['TKT_max'], |
|
| 1218 | - 'TKT_row' => $row, |
|
| 1219 | - 'TKT_order' => isset($tkt['TKT_order']) ? $tkt['TKT_order'] : $row, |
|
| 1220 | - 'TKT_price' => $ticket_price, |
|
| 1221 | - ); |
|
| 1222 | - // if this is a default TKT, then we need to set the TKT_ID to 0 and update accordingly, which means in turn that the prices will become new prices as well. |
|
| 1223 | - if (isset($tkt['TKT_is_default']) && $tkt['TKT_is_default']) { |
|
| 1224 | - $TKT_values['TKT_ID'] = 0; |
|
| 1225 | - $TKT_values['TKT_is_default'] = 0; |
|
| 1226 | - $TKT_values['TKT_price'] = $ticket_price; |
|
| 1227 | - $update_prices = true; |
|
| 1228 | - } |
|
| 1229 | - // if we have a TKT_ID then we need to get that existing TKT_obj and update it |
|
| 1230 | - // we actually do our saves a head of doing any add_relations to because its entirely possible that this ticket didn't removed or added to any datetime in the session but DID have it's items modified. |
|
| 1231 | - // keep in mind that if the TKT has been sold (and we have changed pricing information), then we won't be updating the tkt but instead a new tkt will be created and the old one archived. |
|
| 1232 | - if (! empty($tkt['TKT_ID'])) { |
|
| 1233 | - $TKT = EE_Registry::instance() |
|
| 1234 | - ->load_model('Ticket', array($evtobj->get_timezone())) |
|
| 1235 | - ->get_one_by_ID($tkt['TKT_ID']); |
|
| 1236 | - if ($TKT instanceof EE_Ticket) { |
|
| 1237 | - $ticket_sold = $TKT->count_related( |
|
| 1238 | - 'Registration', |
|
| 1239 | - array( |
|
| 1240 | - array( |
|
| 1241 | - 'STS_ID' => array( |
|
| 1242 | - 'NOT IN', |
|
| 1243 | - array(EEM_Registration::status_id_incomplete), |
|
| 1244 | - ), |
|
| 1245 | - ), |
|
| 1246 | - ) |
|
| 1247 | - ) > 0 ? true : false; |
|
| 1248 | - // let's just check the total price for the existing ticket and determine if it matches the new total price. if they are different then we create a new ticket (if tkts sold) if they aren't different then we go ahead and modify existing ticket. |
|
| 1249 | - $create_new_TKT = $ticket_sold && $ticket_price != $TKT->get('TKT_price') |
|
| 1250 | - && ! $TKT->get('TKT_deleted'); |
|
| 1251 | - $TKT->set_date_format($incoming_date_formats[0]); |
|
| 1252 | - $TKT->set_time_format($incoming_date_formats[1]); |
|
| 1253 | - // set new values |
|
| 1254 | - foreach ($TKT_values as $field => $value) { |
|
| 1255 | - if ($field == 'TKT_qty') { |
|
| 1256 | - $TKT->set_qty($value); |
|
| 1257 | - } else { |
|
| 1258 | - $TKT->set($field, $value); |
|
| 1259 | - } |
|
| 1260 | - } |
|
| 1261 | - // if $create_new_TKT is false then we can safely update the existing ticket. Otherwise we have to create a new ticket. |
|
| 1262 | - if ($create_new_TKT) { |
|
| 1263 | - // archive the old ticket first |
|
| 1264 | - $TKT->set('TKT_deleted', 1); |
|
| 1265 | - $TKT->save(); |
|
| 1266 | - // make sure this ticket is still recorded in our saved_tkts so we don't run it through the regular trash routine. |
|
| 1267 | - $saved_tickets[ $TKT->ID() ] = $TKT; |
|
| 1268 | - // create new ticket that's a copy of the existing except a new id of course (and not archived) AND has the new TKT_price associated with it. |
|
| 1269 | - $TKT = clone $TKT; |
|
| 1270 | - $TKT->set('TKT_ID', 0); |
|
| 1271 | - $TKT->set('TKT_deleted', 0); |
|
| 1272 | - $TKT->set('TKT_price', $ticket_price); |
|
| 1273 | - $TKT->set('TKT_sold', 0); |
|
| 1274 | - // now we need to make sure that $new prices are created as well and attached to new ticket. |
|
| 1275 | - $update_prices = true; |
|
| 1276 | - } |
|
| 1277 | - // make sure price is set if it hasn't been already |
|
| 1278 | - $TKT->set('TKT_price', $ticket_price); |
|
| 1279 | - } |
|
| 1280 | - } else { |
|
| 1281 | - // no TKT_id so a new TKT |
|
| 1282 | - $TKT_values['TKT_price'] = $ticket_price; |
|
| 1283 | - $TKT = EE_Registry::instance()->load_class('Ticket', array($TKT_values), false, false); |
|
| 1284 | - if ($TKT instanceof EE_Ticket) { |
|
| 1285 | - // need to reset values to properly account for the date formats |
|
| 1286 | - $TKT->set_date_format($incoming_date_formats[0]); |
|
| 1287 | - $TKT->set_time_format($incoming_date_formats[1]); |
|
| 1288 | - $TKT->set_timezone($evtobj->get_timezone()); |
|
| 1289 | - // set new values |
|
| 1290 | - foreach ($TKT_values as $field => $value) { |
|
| 1291 | - if ($field == 'TKT_qty') { |
|
| 1292 | - $TKT->set_qty($value); |
|
| 1293 | - } else { |
|
| 1294 | - $TKT->set($field, $value); |
|
| 1295 | - } |
|
| 1296 | - } |
|
| 1297 | - $update_prices = true; |
|
| 1298 | - } |
|
| 1299 | - } |
|
| 1300 | - // cap ticket qty by datetime reg limits |
|
| 1301 | - $TKT->set_qty(min($TKT->qty(), $TKT->qty('reg_limit'))); |
|
| 1302 | - // update ticket. |
|
| 1303 | - $TKT->save(); |
|
| 1304 | - // before going any further make sure our dates are setup correctly so that the end date is always equal or greater than the start date. |
|
| 1305 | - if ($TKT->get_raw('TKT_start_date') > $TKT->get_raw('TKT_end_date')) { |
|
| 1306 | - $TKT->set('TKT_end_date', $TKT->get('TKT_start_date')); |
|
| 1307 | - $TKT = EEH_DTT_Helper::date_time_add($TKT, 'TKT_end_date', 'days'); |
|
| 1308 | - $TKT->save(); |
|
| 1309 | - } |
|
| 1310 | - // initially let's add the ticket to the dtt |
|
| 1311 | - $saved_dtt->_add_relation_to($TKT, 'Ticket'); |
|
| 1312 | - $saved_tickets[ $TKT->ID() ] = $TKT; |
|
| 1313 | - // add prices to ticket |
|
| 1314 | - $this->_add_prices_to_ticket($data['edit_prices'][ $row ], $TKT, $update_prices); |
|
| 1315 | - } |
|
| 1316 | - // however now we need to handle permanently deleting tickets via the ui. Keep in mind that the ui does not allow deleting/archiving tickets that have ticket sold. However, it does allow for deleting tickets that have no tickets sold, in which case we want to get rid of permanently because there is no need to save in db. |
|
| 1317 | - $old_tickets = isset($old_tickets[0]) && $old_tickets[0] == '' ? array() : $old_tickets; |
|
| 1318 | - $tickets_removed = array_diff($old_tickets, array_keys($saved_tickets)); |
|
| 1319 | - foreach ($tickets_removed as $id) { |
|
| 1320 | - $id = absint($id); |
|
| 1321 | - // get the ticket for this id |
|
| 1322 | - $tkt_to_remove = EE_Registry::instance()->load_model('Ticket')->get_one_by_ID($id); |
|
| 1323 | - // need to get all the related datetimes on this ticket and remove from every single one of them (remember this process can ONLY kick off if there are NO tkts_sold) |
|
| 1324 | - $dtts = $tkt_to_remove->get_many_related('Datetime'); |
|
| 1325 | - foreach ($dtts as $dtt) { |
|
| 1326 | - $tkt_to_remove->_remove_relation_to($dtt, 'Datetime'); |
|
| 1327 | - } |
|
| 1328 | - // need to do the same for prices (except these prices can also be deleted because again, tickets can only be trashed if they don't have any TKTs sold (otherwise they are just archived)) |
|
| 1329 | - $tkt_to_remove->delete_related_permanently('Price'); |
|
| 1330 | - // finally let's delete this ticket (which should not be blocked at this point b/c we've removed all our relationships) |
|
| 1331 | - $tkt_to_remove->delete_permanently(); |
|
| 1332 | - } |
|
| 1333 | - return array($saved_dtt, $saved_tickets); |
|
| 1334 | - } |
|
| 1335 | - |
|
| 1336 | - |
|
| 1337 | - /** |
|
| 1338 | - * This attaches a list of given prices to a ticket. |
|
| 1339 | - * Note we dont' have to worry about ever removing relationships (or archiving prices) because if there is a change |
|
| 1340 | - * in price information on a ticket, a new ticket is created anyways so the archived ticket will retain the old |
|
| 1341 | - * price info and prices are automatically "archived" via the ticket. |
|
| 1342 | - * |
|
| 1343 | - * @access private |
|
| 1344 | - * @param array $prices Array of prices from the form. |
|
| 1345 | - * @param EE_Ticket $ticket EE_Ticket object that prices are being attached to. |
|
| 1346 | - * @param bool $new_prices Whether attach existing incoming prices or create new ones. |
|
| 1347 | - * @return void |
|
| 1348 | - */ |
|
| 1349 | - private function _add_prices_to_ticket($prices, EE_Ticket $ticket, $new_prices = false) |
|
| 1350 | - { |
|
| 1351 | - foreach ($prices as $row => $prc) { |
|
| 1352 | - $PRC_values = array( |
|
| 1353 | - 'PRC_ID' => ! empty($prc['PRC_ID']) ? $prc['PRC_ID'] : null, |
|
| 1354 | - 'PRT_ID' => ! empty($prc['PRT_ID']) ? $prc['PRT_ID'] : null, |
|
| 1355 | - 'PRC_amount' => ! empty($prc['PRC_amount']) ? $prc['PRC_amount'] : 0, |
|
| 1356 | - 'PRC_name' => ! empty($prc['PRC_name']) ? $prc['PRC_name'] : '', |
|
| 1357 | - 'PRC_desc' => ! empty($prc['PRC_desc']) ? $prc['PRC_desc'] : '', |
|
| 1358 | - 'PRC_is_default' => 0, // make sure prices are NOT set as default from this context |
|
| 1359 | - 'PRC_order' => $row, |
|
| 1360 | - ); |
|
| 1361 | - if ($new_prices || empty($PRC_values['PRC_ID'])) { |
|
| 1362 | - $PRC_values['PRC_ID'] = 0; |
|
| 1363 | - $PRC = EE_Registry::instance()->load_class('Price', array($PRC_values), false, false); |
|
| 1364 | - } else { |
|
| 1365 | - $PRC = EE_Registry::instance()->load_model('Price')->get_one_by_ID($prc['PRC_ID']); |
|
| 1366 | - // update this price with new values |
|
| 1367 | - foreach ($PRC_values as $field => $newprc) { |
|
| 1368 | - $PRC->set($field, $newprc); |
|
| 1369 | - } |
|
| 1370 | - $PRC->save(); |
|
| 1371 | - } |
|
| 1372 | - $ticket->_add_relation_to($PRC, 'Price'); |
|
| 1373 | - } |
|
| 1374 | - } |
|
| 1375 | - |
|
| 1376 | - |
|
| 1377 | - /** |
|
| 1378 | - * Add in our autosave ajax handlers |
|
| 1379 | - * |
|
| 1380 | - */ |
|
| 1381 | - protected function _ee_autosave_create_new() |
|
| 1382 | - { |
|
| 1383 | - } |
|
| 1384 | - |
|
| 1385 | - |
|
| 1386 | - /** |
|
| 1387 | - * More autosave handlers. |
|
| 1388 | - */ |
|
| 1389 | - protected function _ee_autosave_edit() |
|
| 1390 | - { |
|
| 1391 | - return; // TEMPORARILY EXITING CAUSE THIS IS A TODO |
|
| 1392 | - } |
|
| 1393 | - |
|
| 1394 | - |
|
| 1395 | - /** |
|
| 1396 | - * _generate_publish_box_extra_content |
|
| 1397 | - */ |
|
| 1398 | - private function _generate_publish_box_extra_content() |
|
| 1399 | - { |
|
| 1400 | - // load formatter helper |
|
| 1401 | - // args for getting related registrations |
|
| 1402 | - $approved_query_args = array( |
|
| 1403 | - array( |
|
| 1404 | - 'REG_deleted' => 0, |
|
| 1405 | - 'STS_ID' => EEM_Registration::status_id_approved, |
|
| 1406 | - ), |
|
| 1407 | - ); |
|
| 1408 | - $not_approved_query_args = array( |
|
| 1409 | - array( |
|
| 1410 | - 'REG_deleted' => 0, |
|
| 1411 | - 'STS_ID' => EEM_Registration::status_id_not_approved, |
|
| 1412 | - ), |
|
| 1413 | - ); |
|
| 1414 | - $pending_payment_query_args = array( |
|
| 1415 | - array( |
|
| 1416 | - 'REG_deleted' => 0, |
|
| 1417 | - 'STS_ID' => EEM_Registration::status_id_pending_payment, |
|
| 1418 | - ), |
|
| 1419 | - ); |
|
| 1420 | - // publish box |
|
| 1421 | - $publish_box_extra_args = array( |
|
| 1422 | - 'view_approved_reg_url' => add_query_arg( |
|
| 1423 | - array( |
|
| 1424 | - 'action' => 'default', |
|
| 1425 | - 'event_id' => $this->_cpt_model_obj->ID(), |
|
| 1426 | - '_reg_status' => EEM_Registration::status_id_approved, |
|
| 1427 | - ), |
|
| 1428 | - REG_ADMIN_URL |
|
| 1429 | - ), |
|
| 1430 | - 'view_not_approved_reg_url' => add_query_arg( |
|
| 1431 | - array( |
|
| 1432 | - 'action' => 'default', |
|
| 1433 | - 'event_id' => $this->_cpt_model_obj->ID(), |
|
| 1434 | - '_reg_status' => EEM_Registration::status_id_not_approved, |
|
| 1435 | - ), |
|
| 1436 | - REG_ADMIN_URL |
|
| 1437 | - ), |
|
| 1438 | - 'view_pending_payment_reg_url' => add_query_arg( |
|
| 1439 | - array( |
|
| 1440 | - 'action' => 'default', |
|
| 1441 | - 'event_id' => $this->_cpt_model_obj->ID(), |
|
| 1442 | - '_reg_status' => EEM_Registration::status_id_pending_payment, |
|
| 1443 | - ), |
|
| 1444 | - REG_ADMIN_URL |
|
| 1445 | - ), |
|
| 1446 | - 'approved_regs' => $this->_cpt_model_obj->count_related( |
|
| 1447 | - 'Registration', |
|
| 1448 | - $approved_query_args |
|
| 1449 | - ), |
|
| 1450 | - 'not_approved_regs' => $this->_cpt_model_obj->count_related( |
|
| 1451 | - 'Registration', |
|
| 1452 | - $not_approved_query_args |
|
| 1453 | - ), |
|
| 1454 | - 'pending_payment_regs' => $this->_cpt_model_obj->count_related( |
|
| 1455 | - 'Registration', |
|
| 1456 | - $pending_payment_query_args |
|
| 1457 | - ), |
|
| 1458 | - 'misc_pub_section_class' => apply_filters( |
|
| 1459 | - 'FHEE_Events_Admin_Page___generate_publish_box_extra_content__misc_pub_section_class', |
|
| 1460 | - 'misc-pub-section' |
|
| 1461 | - ), |
|
| 1462 | - ); |
|
| 1463 | - ob_start(); |
|
| 1464 | - do_action( |
|
| 1465 | - 'AHEE__Events_Admin_Page___generate_publish_box_extra_content__event_editor_overview_add', |
|
| 1466 | - $this->_cpt_model_obj |
|
| 1467 | - ); |
|
| 1468 | - $publish_box_extra_args['event_editor_overview_add'] = ob_get_clean(); |
|
| 1469 | - // load template |
|
| 1470 | - EEH_Template::display_template( |
|
| 1471 | - EVENTS_TEMPLATE_PATH . 'event_publish_box_extras.template.php', |
|
| 1472 | - $publish_box_extra_args |
|
| 1473 | - ); |
|
| 1474 | - } |
|
| 1475 | - |
|
| 1476 | - |
|
| 1477 | - /** |
|
| 1478 | - * @return EE_Event |
|
| 1479 | - */ |
|
| 1480 | - public function get_event_object() |
|
| 1481 | - { |
|
| 1482 | - return $this->_cpt_model_obj; |
|
| 1483 | - } |
|
| 1484 | - |
|
| 1485 | - |
|
| 1486 | - |
|
| 1487 | - |
|
| 1488 | - /** METABOXES * */ |
|
| 1489 | - /** |
|
| 1490 | - * _register_event_editor_meta_boxes |
|
| 1491 | - * add all metaboxes related to the event_editor |
|
| 1492 | - * |
|
| 1493 | - * @return void |
|
| 1494 | - */ |
|
| 1495 | - protected function _register_event_editor_meta_boxes() |
|
| 1496 | - { |
|
| 1497 | - $this->verify_cpt_object(); |
|
| 1498 | - add_meta_box( |
|
| 1499 | - 'espresso_event_editor_tickets', |
|
| 1500 | - esc_html__('Event Datetime & Ticket', 'event_espresso'), |
|
| 1501 | - array($this, 'ticket_metabox'), |
|
| 1502 | - $this->page_slug, |
|
| 1503 | - 'normal', |
|
| 1504 | - 'high' |
|
| 1505 | - ); |
|
| 1506 | - add_meta_box( |
|
| 1507 | - 'espresso_event_editor_event_options', |
|
| 1508 | - esc_html__('Event Registration Options', 'event_espresso'), |
|
| 1509 | - array($this, 'registration_options_meta_box'), |
|
| 1510 | - $this->page_slug, |
|
| 1511 | - 'side', |
|
| 1512 | - 'default' |
|
| 1513 | - ); |
|
| 1514 | - // NOTE: if you're looking for other metaboxes in here, |
|
| 1515 | - // where a metabox has a related management page in the admin |
|
| 1516 | - // you will find it setup in the related management page's "_Hooks" file. |
|
| 1517 | - // i.e. messages metabox is found in "espresso_events_Messages_Hooks.class.php". |
|
| 1518 | - } |
|
| 1519 | - |
|
| 1520 | - |
|
| 1521 | - /** |
|
| 1522 | - * @throws DomainException |
|
| 1523 | - * @throws EE_Error |
|
| 1524 | - */ |
|
| 1525 | - public function ticket_metabox() |
|
| 1526 | - { |
|
| 1527 | - $existing_datetime_ids = $existing_ticket_ids = array(); |
|
| 1528 | - // defaults for template args |
|
| 1529 | - $template_args = array( |
|
| 1530 | - 'existing_datetime_ids' => '', |
|
| 1531 | - 'event_datetime_help_link' => '', |
|
| 1532 | - 'ticket_options_help_link' => '', |
|
| 1533 | - 'time' => null, |
|
| 1534 | - 'ticket_rows' => '', |
|
| 1535 | - 'existing_ticket_ids' => '', |
|
| 1536 | - 'total_ticket_rows' => 1, |
|
| 1537 | - 'ticket_js_structure' => '', |
|
| 1538 | - 'trash_icon' => 'ee-lock-icon', |
|
| 1539 | - 'disabled' => '', |
|
| 1540 | - ); |
|
| 1541 | - $event_id = is_object($this->_cpt_model_obj) ? $this->_cpt_model_obj->ID() : null; |
|
| 1542 | - do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
| 1543 | - /** |
|
| 1544 | - * 1. Start with retrieving Datetimes |
|
| 1545 | - * 2. Fore each datetime get related tickets |
|
| 1546 | - * 3. For each ticket get related prices |
|
| 1547 | - */ |
|
| 1548 | - $times = EE_Registry::instance()->load_model('Datetime')->get_all_event_dates($event_id); |
|
| 1549 | - /** @type EE_Datetime $first_datetime */ |
|
| 1550 | - $first_datetime = reset($times); |
|
| 1551 | - // do we get related tickets? |
|
| 1552 | - if ($first_datetime instanceof EE_Datetime |
|
| 1553 | - && $first_datetime->ID() !== 0 |
|
| 1554 | - ) { |
|
| 1555 | - $existing_datetime_ids[] = $first_datetime->get('DTT_ID'); |
|
| 1556 | - $template_args['time'] = $first_datetime; |
|
| 1557 | - $related_tickets = $first_datetime->tickets( |
|
| 1558 | - array( |
|
| 1559 | - array('OR' => array('TKT_deleted' => 1, 'TKT_deleted*' => 0)), |
|
| 1560 | - 'default_where_conditions' => 'none', |
|
| 1561 | - ) |
|
| 1562 | - ); |
|
| 1563 | - if (! empty($related_tickets)) { |
|
| 1564 | - $template_args['total_ticket_rows'] = count($related_tickets); |
|
| 1565 | - $row = 0; |
|
| 1566 | - foreach ($related_tickets as $ticket) { |
|
| 1567 | - $existing_ticket_ids[] = $ticket->get('TKT_ID'); |
|
| 1568 | - $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket, false, $row); |
|
| 1569 | - $row++; |
|
| 1570 | - } |
|
| 1571 | - } else { |
|
| 1572 | - $template_args['total_ticket_rows'] = 1; |
|
| 1573 | - /** @type EE_Ticket $ticket */ |
|
| 1574 | - $ticket = EE_Registry::instance()->load_model('Ticket')->create_default_object(); |
|
| 1575 | - $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket); |
|
| 1576 | - } |
|
| 1577 | - } else { |
|
| 1578 | - $template_args['time'] = $times[0]; |
|
| 1579 | - /** @type EE_Ticket $ticket */ |
|
| 1580 | - $ticket = EE_Registry::instance()->load_model('Ticket')->get_all_default_tickets(); |
|
| 1581 | - $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket[1]); |
|
| 1582 | - // NOTE: we're just sending the first default row |
|
| 1583 | - // (decaf can't manage default tickets so this should be sufficient); |
|
| 1584 | - } |
|
| 1585 | - $template_args['event_datetime_help_link'] = $this->_get_help_tab_link( |
|
| 1586 | - 'event_editor_event_datetimes_help_tab' |
|
| 1587 | - ); |
|
| 1588 | - $template_args['ticket_options_help_link'] = $this->_get_help_tab_link('ticket_options_info'); |
|
| 1589 | - $template_args['existing_datetime_ids'] = implode(',', $existing_datetime_ids); |
|
| 1590 | - $template_args['existing_ticket_ids'] = implode(',', $existing_ticket_ids); |
|
| 1591 | - $template_args['ticket_js_structure'] = $this->_get_ticket_row( |
|
| 1592 | - EE_Registry::instance()->load_model('Ticket')->create_default_object(), |
|
| 1593 | - true |
|
| 1594 | - ); |
|
| 1595 | - $template = apply_filters( |
|
| 1596 | - 'FHEE__Events_Admin_Page__ticket_metabox__template', |
|
| 1597 | - EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_main.template.php' |
|
| 1598 | - ); |
|
| 1599 | - EEH_Template::display_template($template, $template_args); |
|
| 1600 | - } |
|
| 1601 | - |
|
| 1602 | - |
|
| 1603 | - /** |
|
| 1604 | - * Setup an individual ticket form for the decaf event editor page |
|
| 1605 | - * |
|
| 1606 | - * @access private |
|
| 1607 | - * @param EE_Ticket $ticket the ticket object |
|
| 1608 | - * @param boolean $skeleton whether we're generating a skeleton for js manipulation |
|
| 1609 | - * @param int $row |
|
| 1610 | - * @return string generated html for the ticket row. |
|
| 1611 | - */ |
|
| 1612 | - private function _get_ticket_row($ticket, $skeleton = false, $row = 0) |
|
| 1613 | - { |
|
| 1614 | - $template_args = array( |
|
| 1615 | - 'tkt_status_class' => ' tkt-status-' . $ticket->ticket_status(), |
|
| 1616 | - 'tkt_archive_class' => $ticket->ticket_status() === EE_Ticket::archived && ! $skeleton ? ' tkt-archived' |
|
| 1617 | - : '', |
|
| 1618 | - 'ticketrow' => $skeleton ? 'TICKETNUM' : $row, |
|
| 1619 | - 'TKT_ID' => $ticket->get('TKT_ID'), |
|
| 1620 | - 'TKT_name' => $ticket->get('TKT_name'), |
|
| 1621 | - 'TKT_start_date' => $skeleton ? '' : $ticket->get_date('TKT_start_date', 'Y-m-d h:i a'), |
|
| 1622 | - 'TKT_end_date' => $skeleton ? '' : $ticket->get_date('TKT_end_date', 'Y-m-d h:i a'), |
|
| 1623 | - 'TKT_is_default' => $ticket->get('TKT_is_default'), |
|
| 1624 | - 'TKT_qty' => $ticket->get_pretty('TKT_qty', 'input'), |
|
| 1625 | - 'edit_ticketrow_name' => $skeleton ? 'TICKETNAMEATTR' : 'edit_tickets', |
|
| 1626 | - 'TKT_sold' => $skeleton ? 0 : $ticket->get('TKT_sold'), |
|
| 1627 | - 'trash_icon' => ($skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted'))) |
|
| 1628 | - && (! empty($ticket) && $ticket->get('TKT_sold') === 0) |
|
| 1629 | - ? 'trash-icon dashicons dashicons-post-trash clickable' : 'ee-lock-icon', |
|
| 1630 | - 'disabled' => $skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted')) ? '' |
|
| 1631 | - : ' disabled=disabled', |
|
| 1632 | - ); |
|
| 1633 | - $price = $ticket->ID() !== 0 |
|
| 1634 | - ? $ticket->get_first_related('Price', array('default_where_conditions' => 'none')) |
|
| 1635 | - : EE_Registry::instance()->load_model('Price')->create_default_object(); |
|
| 1636 | - $price_args = array( |
|
| 1637 | - 'price_currency_symbol' => EE_Registry::instance()->CFG->currency->sign, |
|
| 1638 | - 'PRC_amount' => $price->get('PRC_amount'), |
|
| 1639 | - 'PRT_ID' => $price->get('PRT_ID'), |
|
| 1640 | - 'PRC_ID' => $price->get('PRC_ID'), |
|
| 1641 | - 'PRC_is_default' => $price->get('PRC_is_default'), |
|
| 1642 | - ); |
|
| 1643 | - // make sure we have default start and end dates if skeleton |
|
| 1644 | - // handle rows that should NOT be empty |
|
| 1645 | - if (empty($template_args['TKT_start_date'])) { |
|
| 1646 | - // if empty then the start date will be now. |
|
| 1647 | - $template_args['TKT_start_date'] = date('Y-m-d h:i a', current_time('timestamp')); |
|
| 1648 | - } |
|
| 1649 | - if (empty($template_args['TKT_end_date'])) { |
|
| 1650 | - // get the earliest datetime (if present); |
|
| 1651 | - $earliest_dtt = $this->_cpt_model_obj->ID() > 0 |
|
| 1652 | - ? $this->_cpt_model_obj->get_first_related( |
|
| 1653 | - 'Datetime', |
|
| 1654 | - array('order_by' => array('DTT_EVT_start' => 'ASC')) |
|
| 1655 | - ) |
|
| 1656 | - : null; |
|
| 1657 | - if (! empty($earliest_dtt)) { |
|
| 1658 | - $template_args['TKT_end_date'] = $earliest_dtt->get_datetime('DTT_EVT_start', 'Y-m-d', 'h:i a'); |
|
| 1659 | - } else { |
|
| 1660 | - $template_args['TKT_end_date'] = date( |
|
| 1661 | - 'Y-m-d h:i a', |
|
| 1662 | - mktime(0, 0, 0, date("m"), date("d") + 7, date("Y")) |
|
| 1663 | - ); |
|
| 1664 | - } |
|
| 1665 | - } |
|
| 1666 | - $template_args = array_merge($template_args, $price_args); |
|
| 1667 | - $template = apply_filters( |
|
| 1668 | - 'FHEE__Events_Admin_Page__get_ticket_row__template', |
|
| 1669 | - EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_ticket_row.template.php', |
|
| 1670 | - $ticket |
|
| 1671 | - ); |
|
| 1672 | - return EEH_Template::display_template($template, $template_args, true); |
|
| 1673 | - } |
|
| 1674 | - |
|
| 1675 | - |
|
| 1676 | - /** |
|
| 1677 | - * @throws DomainException |
|
| 1678 | - */ |
|
| 1679 | - public function registration_options_meta_box() |
|
| 1680 | - { |
|
| 1681 | - $yes_no_values = array( |
|
| 1682 | - array('id' => true, 'text' => esc_html__('Yes', 'event_espresso')), |
|
| 1683 | - array('id' => false, 'text' => esc_html__('No', 'event_espresso')), |
|
| 1684 | - ); |
|
| 1685 | - $default_reg_status_values = EEM_Registration::reg_status_array( |
|
| 1686 | - array( |
|
| 1687 | - EEM_Registration::status_id_cancelled, |
|
| 1688 | - EEM_Registration::status_id_declined, |
|
| 1689 | - EEM_Registration::status_id_incomplete, |
|
| 1690 | - ), |
|
| 1691 | - true |
|
| 1692 | - ); |
|
| 1693 | - // $template_args['is_active_select'] = EEH_Form_Fields::select_input('is_active', $yes_no_values, $this->_cpt_model_obj->is_active()); |
|
| 1694 | - $template_args['_event'] = $this->_cpt_model_obj; |
|
| 1695 | - $template_args['active_status'] = $this->_cpt_model_obj->pretty_active_status(false); |
|
| 1696 | - $template_args['additional_limit'] = $this->_cpt_model_obj->additional_limit(); |
|
| 1697 | - $template_args['default_registration_status'] = EEH_Form_Fields::select_input( |
|
| 1698 | - 'default_reg_status', |
|
| 1699 | - $default_reg_status_values, |
|
| 1700 | - $this->_cpt_model_obj->default_registration_status() |
|
| 1701 | - ); |
|
| 1702 | - $template_args['display_description'] = EEH_Form_Fields::select_input( |
|
| 1703 | - 'display_desc', |
|
| 1704 | - $yes_no_values, |
|
| 1705 | - $this->_cpt_model_obj->display_description() |
|
| 1706 | - ); |
|
| 1707 | - $template_args['display_ticket_selector'] = EEH_Form_Fields::select_input( |
|
| 1708 | - 'display_ticket_selector', |
|
| 1709 | - $yes_no_values, |
|
| 1710 | - $this->_cpt_model_obj->display_ticket_selector(), |
|
| 1711 | - '', |
|
| 1712 | - '', |
|
| 1713 | - false |
|
| 1714 | - ); |
|
| 1715 | - $template_args['additional_registration_options'] = apply_filters( |
|
| 1716 | - 'FHEE__Events_Admin_Page__registration_options_meta_box__additional_registration_options', |
|
| 1717 | - '', |
|
| 1718 | - $template_args, |
|
| 1719 | - $yes_no_values, |
|
| 1720 | - $default_reg_status_values |
|
| 1721 | - ); |
|
| 1722 | - EEH_Template::display_template( |
|
| 1723 | - EVENTS_TEMPLATE_PATH . 'event_registration_options.template.php', |
|
| 1724 | - $template_args |
|
| 1725 | - ); |
|
| 1726 | - } |
|
| 1727 | - |
|
| 1728 | - |
|
| 1729 | - /** |
|
| 1730 | - * _get_events() |
|
| 1731 | - * This method simply returns all the events (for the given _view and paging) |
|
| 1732 | - * |
|
| 1733 | - * @access public |
|
| 1734 | - * @param int $per_page count of items per page (20 default); |
|
| 1735 | - * @param int $current_page what is the current page being viewed. |
|
| 1736 | - * @param bool $count if TRUE then we just return a count of ALL events matching the given _view. |
|
| 1737 | - * If FALSE then we return an array of event objects |
|
| 1738 | - * that match the given _view and paging parameters. |
|
| 1739 | - * @return array an array of event objects. |
|
| 1740 | - */ |
|
| 1741 | - public function get_events($per_page = 10, $current_page = 1, $count = false) |
|
| 1742 | - { |
|
| 1743 | - $EEME = $this->_event_model(); |
|
| 1744 | - $offset = ($current_page - 1) * $per_page; |
|
| 1745 | - $limit = $count ? null : $offset . ',' . $per_page; |
|
| 1746 | - $orderby = isset($this->_req_data['orderby']) ? $this->_req_data['orderby'] : 'EVT_ID'; |
|
| 1747 | - $order = isset($this->_req_data['order']) ? $this->_req_data['order'] : "DESC"; |
|
| 1748 | - if (isset($this->_req_data['month_range'])) { |
|
| 1749 | - $pieces = explode(' ', $this->_req_data['month_range'], 3); |
|
| 1750 | - // simulate the FIRST day of the month, that fixes issues for months like February |
|
| 1751 | - // where PHP doesn't know what to assume for date. |
|
| 1752 | - // @see https://events.codebasehq.com/projects/event-espresso/tickets/10437 |
|
| 1753 | - $month_r = ! empty($pieces[0]) ? date('m', \EEH_DTT_Helper::first_of_month_timestamp($pieces[0])) : ''; |
|
| 1754 | - $year_r = ! empty($pieces[1]) ? $pieces[1] : ''; |
|
| 1755 | - } |
|
| 1756 | - $where = array(); |
|
| 1757 | - $status = isset($this->_req_data['status']) ? $this->_req_data['status'] : null; |
|
| 1758 | - // determine what post_status our condition will have for the query. |
|
| 1759 | - switch ($status) { |
|
| 1760 | - case 'month': |
|
| 1761 | - case 'today': |
|
| 1762 | - case null: |
|
| 1763 | - case 'all': |
|
| 1764 | - break; |
|
| 1765 | - case 'draft': |
|
| 1766 | - $where['status'] = array('IN', array('draft', 'auto-draft')); |
|
| 1767 | - break; |
|
| 1768 | - default: |
|
| 1769 | - $where['status'] = $status; |
|
| 1770 | - } |
|
| 1771 | - // categories? |
|
| 1772 | - $category = isset($this->_req_data['EVT_CAT']) && $this->_req_data['EVT_CAT'] > 0 |
|
| 1773 | - ? $this->_req_data['EVT_CAT'] : null; |
|
| 1774 | - if (! empty($category)) { |
|
| 1775 | - $where['Term_Taxonomy.taxonomy'] = EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY; |
|
| 1776 | - $where['Term_Taxonomy.term_id'] = $category; |
|
| 1777 | - } |
|
| 1778 | - // date where conditions |
|
| 1779 | - $start_formats = EEM_Datetime::instance()->get_formats_for('DTT_EVT_start'); |
|
| 1780 | - if (isset($this->_req_data['month_range']) && $this->_req_data['month_range'] != '') { |
|
| 1781 | - $DateTime = new DateTime( |
|
| 1782 | - $year_r . '-' . $month_r . '-01 00:00:00', |
|
| 1783 | - new DateTimeZone(EEM_Datetime::instance()->get_timezone()) |
|
| 1784 | - ); |
|
| 1785 | - $start = $DateTime->format(implode(' ', $start_formats)); |
|
| 1786 | - $end = $DateTime->setDate( |
|
| 1787 | - $year_r, |
|
| 1788 | - $month_r, |
|
| 1789 | - $DateTime |
|
| 1790 | - ->format('t') |
|
| 1791 | - )->setTime(23, 59, 59) |
|
| 1792 | - ->format(implode(' ', $start_formats)); |
|
| 1793 | - $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
| 1794 | - } elseif (isset($this->_req_data['status']) && $this->_req_data['status'] == 'today') { |
|
| 1795 | - $DateTime = new DateTime('now', new DateTimeZone(EEM_Event::instance()->get_timezone())); |
|
| 1796 | - $start = $DateTime->setTime(0, 0, 0)->format(implode(' ', $start_formats)); |
|
| 1797 | - $end = $DateTime->setTime(23, 59, 59)->format(implode(' ', $start_formats)); |
|
| 1798 | - $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
| 1799 | - } elseif (isset($this->_req_data['status']) && $this->_req_data['status'] == 'month') { |
|
| 1800 | - $now = date('Y-m-01'); |
|
| 1801 | - $DateTime = new DateTime($now, new DateTimeZone(EEM_Event::instance()->get_timezone())); |
|
| 1802 | - $start = $DateTime->setTime(0, 0, 0)->format(implode(' ', $start_formats)); |
|
| 1803 | - $end = $DateTime->setDate(date('Y'), date('m'), $DateTime->format('t')) |
|
| 1804 | - ->setTime(23, 59, 59) |
|
| 1805 | - ->format(implode(' ', $start_formats)); |
|
| 1806 | - $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
| 1807 | - } |
|
| 1808 | - if (! EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events')) { |
|
| 1809 | - $where['EVT_wp_user'] = get_current_user_id(); |
|
| 1810 | - } else { |
|
| 1811 | - if (! isset($where['status'])) { |
|
| 1812 | - if (! EE_Registry::instance()->CAP->current_user_can('ee_read_private_events', 'get_events')) { |
|
| 1813 | - $where['OR'] = array( |
|
| 1814 | - 'status*restrict_private' => array('!=', 'private'), |
|
| 1815 | - 'AND' => array( |
|
| 1816 | - 'status*inclusive' => array('=', 'private'), |
|
| 1817 | - 'EVT_wp_user' => get_current_user_id(), |
|
| 1818 | - ), |
|
| 1819 | - ); |
|
| 1820 | - } |
|
| 1821 | - } |
|
| 1822 | - } |
|
| 1823 | - if (isset($this->_req_data['EVT_wp_user'])) { |
|
| 1824 | - if ($this->_req_data['EVT_wp_user'] != get_current_user_id() |
|
| 1825 | - && EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events') |
|
| 1826 | - ) { |
|
| 1827 | - $where['EVT_wp_user'] = $this->_req_data['EVT_wp_user']; |
|
| 1828 | - } |
|
| 1829 | - } |
|
| 1830 | - // search query handling |
|
| 1831 | - if (isset($this->_req_data['s'])) { |
|
| 1832 | - $search_string = '%' . $this->_req_data['s'] . '%'; |
|
| 1833 | - $where['OR'] = array( |
|
| 1834 | - 'EVT_name' => array('LIKE', $search_string), |
|
| 1835 | - 'EVT_desc' => array('LIKE', $search_string), |
|
| 1836 | - 'EVT_short_desc' => array('LIKE', $search_string), |
|
| 1837 | - ); |
|
| 1838 | - } |
|
| 1839 | - $where = apply_filters('FHEE__Events_Admin_Page__get_events__where', $where, $this->_req_data); |
|
| 1840 | - $query_params = apply_filters( |
|
| 1841 | - 'FHEE__Events_Admin_Page__get_events__query_params', |
|
| 1842 | - array( |
|
| 1843 | - $where, |
|
| 1844 | - 'limit' => $limit, |
|
| 1845 | - 'order_by' => $orderby, |
|
| 1846 | - 'order' => $order, |
|
| 1847 | - 'group_by' => 'EVT_ID', |
|
| 1848 | - ), |
|
| 1849 | - $this->_req_data |
|
| 1850 | - ); |
|
| 1851 | - // let's first check if we have special requests coming in. |
|
| 1852 | - if (isset($this->_req_data['active_status'])) { |
|
| 1853 | - switch ($this->_req_data['active_status']) { |
|
| 1854 | - case 'upcoming': |
|
| 1855 | - return $EEME->get_upcoming_events($query_params, $count); |
|
| 1856 | - break; |
|
| 1857 | - case 'expired': |
|
| 1858 | - return $EEME->get_expired_events($query_params, $count); |
|
| 1859 | - break; |
|
| 1860 | - case 'active': |
|
| 1861 | - return $EEME->get_active_events($query_params, $count); |
|
| 1862 | - break; |
|
| 1863 | - case 'inactive': |
|
| 1864 | - return $EEME->get_inactive_events($query_params, $count); |
|
| 1865 | - break; |
|
| 1866 | - } |
|
| 1867 | - } |
|
| 1868 | - $events = $count ? $EEME->count(array($where), 'EVT_ID', true) : $EEME->get_all($query_params); |
|
| 1869 | - return $events; |
|
| 1870 | - } |
|
| 1871 | - |
|
| 1872 | - |
|
| 1873 | - /** |
|
| 1874 | - * handling for WordPress CPT actions (trash, restore, delete) |
|
| 1875 | - * |
|
| 1876 | - * @param string $post_id |
|
| 1877 | - */ |
|
| 1878 | - public function trash_cpt_item($post_id) |
|
| 1879 | - { |
|
| 1880 | - $this->_req_data['EVT_ID'] = $post_id; |
|
| 1881 | - $this->_trash_or_restore_event('trash', false); |
|
| 1882 | - } |
|
| 1883 | - |
|
| 1884 | - |
|
| 1885 | - /** |
|
| 1886 | - * @param string $post_id |
|
| 1887 | - */ |
|
| 1888 | - public function restore_cpt_item($post_id) |
|
| 1889 | - { |
|
| 1890 | - $this->_req_data['EVT_ID'] = $post_id; |
|
| 1891 | - $this->_trash_or_restore_event('draft', false); |
|
| 1892 | - } |
|
| 1893 | - |
|
| 1894 | - |
|
| 1895 | - /** |
|
| 1896 | - * @param string $post_id |
|
| 1897 | - */ |
|
| 1898 | - public function delete_cpt_item($post_id) |
|
| 1899 | - { |
|
| 1900 | - $this->_req_data['EVT_ID'] = $post_id; |
|
| 1901 | - $this->_delete_event(false); |
|
| 1902 | - } |
|
| 1903 | - |
|
| 1904 | - |
|
| 1905 | - /** |
|
| 1906 | - * _trash_or_restore_event |
|
| 1907 | - * |
|
| 1908 | - * @access protected |
|
| 1909 | - * @param string $event_status |
|
| 1910 | - * @param bool $redirect_after |
|
| 1911 | - */ |
|
| 1912 | - protected function _trash_or_restore_event($event_status = 'trash', $redirect_after = true) |
|
| 1913 | - { |
|
| 1914 | - // determine the event id and set to array. |
|
| 1915 | - $EVT_ID = isset($this->_req_data['EVT_ID']) ? absint($this->_req_data['EVT_ID']) : false; |
|
| 1916 | - // loop thru events |
|
| 1917 | - if ($EVT_ID) { |
|
| 1918 | - // clean status |
|
| 1919 | - $event_status = sanitize_key($event_status); |
|
| 1920 | - // grab status |
|
| 1921 | - if (! empty($event_status)) { |
|
| 1922 | - $success = $this->_change_event_status($EVT_ID, $event_status); |
|
| 1923 | - } else { |
|
| 1924 | - $success = false; |
|
| 1925 | - $msg = esc_html__( |
|
| 1926 | - 'An error occurred. The event could not be moved to the trash because a valid event status was not not supplied.', |
|
| 1927 | - 'event_espresso' |
|
| 1928 | - ); |
|
| 1929 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 1930 | - } |
|
| 1931 | - } else { |
|
| 1932 | - $success = false; |
|
| 1933 | - $msg = esc_html__( |
|
| 1934 | - 'An error occurred. The event could not be moved to the trash because a valid event ID was not not supplied.', |
|
| 1935 | - 'event_espresso' |
|
| 1936 | - ); |
|
| 1937 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 1938 | - } |
|
| 1939 | - $action = $event_status == 'trash' ? 'moved to the trash' : 'restored from the trash'; |
|
| 1940 | - if ($redirect_after) { |
|
| 1941 | - $this->_redirect_after_action($success, 'Event', $action, array('action' => 'default')); |
|
| 1942 | - } |
|
| 1943 | - } |
|
| 1944 | - |
|
| 1945 | - |
|
| 1946 | - /** |
|
| 1947 | - * _trash_or_restore_events |
|
| 1948 | - * |
|
| 1949 | - * @access protected |
|
| 1950 | - * @param string $event_status |
|
| 1951 | - * @return void |
|
| 1952 | - */ |
|
| 1953 | - protected function _trash_or_restore_events($event_status = 'trash') |
|
| 1954 | - { |
|
| 1955 | - // clean status |
|
| 1956 | - $event_status = sanitize_key($event_status); |
|
| 1957 | - // grab status |
|
| 1958 | - if (! empty($event_status)) { |
|
| 1959 | - $success = true; |
|
| 1960 | - // determine the event id and set to array. |
|
| 1961 | - $EVT_IDs = isset($this->_req_data['EVT_IDs']) ? (array) $this->_req_data['EVT_IDs'] : array(); |
|
| 1962 | - // loop thru events |
|
| 1963 | - foreach ($EVT_IDs as $EVT_ID) { |
|
| 1964 | - if ($EVT_ID = absint($EVT_ID)) { |
|
| 1965 | - $results = $this->_change_event_status($EVT_ID, $event_status); |
|
| 1966 | - $success = $results !== false ? $success : false; |
|
| 1967 | - } else { |
|
| 1968 | - $msg = sprintf( |
|
| 1969 | - esc_html__( |
|
| 1970 | - 'An error occurred. Event #%d could not be moved to the trash because a valid event ID was not not supplied.', |
|
| 1971 | - 'event_espresso' |
|
| 1972 | - ), |
|
| 1973 | - $EVT_ID |
|
| 1974 | - ); |
|
| 1975 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 1976 | - $success = false; |
|
| 1977 | - } |
|
| 1978 | - } |
|
| 1979 | - } else { |
|
| 1980 | - $success = false; |
|
| 1981 | - $msg = esc_html__( |
|
| 1982 | - 'An error occurred. The event could not be moved to the trash because a valid event status was not not supplied.', |
|
| 1983 | - 'event_espresso' |
|
| 1984 | - ); |
|
| 1985 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 1986 | - } |
|
| 1987 | - // in order to force a pluralized result message we need to send back a success status greater than 1 |
|
| 1988 | - $success = $success ? 2 : false; |
|
| 1989 | - $action = $event_status == 'trash' ? 'moved to the trash' : 'restored from the trash'; |
|
| 1990 | - $this->_redirect_after_action($success, 'Events', $action, array('action' => 'default')); |
|
| 1991 | - } |
|
| 1992 | - |
|
| 1993 | - |
|
| 1994 | - /** |
|
| 1995 | - * _trash_or_restore_events |
|
| 1996 | - * |
|
| 1997 | - * @access private |
|
| 1998 | - * @param int $EVT_ID |
|
| 1999 | - * @param string $event_status |
|
| 2000 | - * @return bool |
|
| 2001 | - */ |
|
| 2002 | - private function _change_event_status($EVT_ID = 0, $event_status = '') |
|
| 2003 | - { |
|
| 2004 | - // grab event id |
|
| 2005 | - if (! $EVT_ID) { |
|
| 2006 | - $msg = esc_html__( |
|
| 2007 | - 'An error occurred. No Event ID or an invalid Event ID was received.', |
|
| 2008 | - 'event_espresso' |
|
| 2009 | - ); |
|
| 2010 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2011 | - return false; |
|
| 2012 | - } |
|
| 2013 | - $this->_cpt_model_obj = EEM_Event::instance()->get_one_by_ID($EVT_ID); |
|
| 2014 | - // clean status |
|
| 2015 | - $event_status = sanitize_key($event_status); |
|
| 2016 | - // grab status |
|
| 2017 | - if (empty($event_status)) { |
|
| 2018 | - $msg = esc_html__( |
|
| 2019 | - 'An error occurred. No Event Status or an invalid Event Status was received.', |
|
| 2020 | - 'event_espresso' |
|
| 2021 | - ); |
|
| 2022 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2023 | - return false; |
|
| 2024 | - } |
|
| 2025 | - // was event trashed or restored ? |
|
| 2026 | - switch ($event_status) { |
|
| 2027 | - case 'draft': |
|
| 2028 | - $action = 'restored from the trash'; |
|
| 2029 | - $hook = 'AHEE_event_restored_from_trash'; |
|
| 2030 | - break; |
|
| 2031 | - case 'trash': |
|
| 2032 | - $action = 'moved to the trash'; |
|
| 2033 | - $hook = 'AHEE_event_moved_to_trash'; |
|
| 2034 | - break; |
|
| 2035 | - default: |
|
| 2036 | - $action = 'updated'; |
|
| 2037 | - $hook = false; |
|
| 2038 | - } |
|
| 2039 | - // use class to change status |
|
| 2040 | - $this->_cpt_model_obj->set_status($event_status); |
|
| 2041 | - $success = $this->_cpt_model_obj->save(); |
|
| 2042 | - if ($success === false) { |
|
| 2043 | - $msg = sprintf(esc_html__('An error occurred. The event could not be %s.', 'event_espresso'), $action); |
|
| 2044 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2045 | - return false; |
|
| 2046 | - } |
|
| 2047 | - if ($hook) { |
|
| 2048 | - do_action($hook); |
|
| 2049 | - } |
|
| 2050 | - return true; |
|
| 2051 | - } |
|
| 2052 | - |
|
| 2053 | - |
|
| 2054 | - /** |
|
| 2055 | - * _delete_event |
|
| 2056 | - * |
|
| 2057 | - * @access protected |
|
| 2058 | - * @param bool $redirect_after |
|
| 2059 | - */ |
|
| 2060 | - protected function _delete_event($redirect_after = true) |
|
| 2061 | - { |
|
| 2062 | - // determine the event id and set to array. |
|
| 2063 | - $EVT_ID = isset($this->_req_data['EVT_ID']) ? absint($this->_req_data['EVT_ID']) : null; |
|
| 2064 | - $EVT_ID = isset($this->_req_data['post']) ? absint($this->_req_data['post']) : $EVT_ID; |
|
| 2065 | - // loop thru events |
|
| 2066 | - if ($EVT_ID) { |
|
| 2067 | - $success = $this->_permanently_delete_event($EVT_ID); |
|
| 2068 | - // get list of events with no prices |
|
| 2069 | - $espresso_no_ticket_prices = get_option('ee_no_ticket_prices', array()); |
|
| 2070 | - // remove this event from the list of events with no prices |
|
| 2071 | - if (isset($espresso_no_ticket_prices[ $EVT_ID ])) { |
|
| 2072 | - unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
| 2073 | - } |
|
| 2074 | - update_option('ee_no_ticket_prices', $espresso_no_ticket_prices); |
|
| 2075 | - } else { |
|
| 2076 | - $success = false; |
|
| 2077 | - $msg = esc_html__( |
|
| 2078 | - 'An error occurred. An event could not be deleted because a valid event ID was not not supplied.', |
|
| 2079 | - 'event_espresso' |
|
| 2080 | - ); |
|
| 2081 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2082 | - } |
|
| 2083 | - if ($redirect_after) { |
|
| 2084 | - $this->_redirect_after_action( |
|
| 2085 | - $success, |
|
| 2086 | - 'Event', |
|
| 2087 | - 'deleted', |
|
| 2088 | - array('action' => 'default', 'status' => 'trash') |
|
| 2089 | - ); |
|
| 2090 | - } |
|
| 2091 | - } |
|
| 2092 | - |
|
| 2093 | - |
|
| 2094 | - /** |
|
| 2095 | - * _delete_events |
|
| 2096 | - * |
|
| 2097 | - * @access protected |
|
| 2098 | - * @return void |
|
| 2099 | - */ |
|
| 2100 | - protected function _delete_events() |
|
| 2101 | - { |
|
| 2102 | - $success = true; |
|
| 2103 | - // get list of events with no prices |
|
| 2104 | - $espresso_no_ticket_prices = get_option('ee_no_ticket_prices', array()); |
|
| 2105 | - // determine the event id and set to array. |
|
| 2106 | - $EVT_IDs = isset($this->_req_data['EVT_IDs']) ? (array) $this->_req_data['EVT_IDs'] : array(); |
|
| 2107 | - // loop thru events |
|
| 2108 | - foreach ($EVT_IDs as $EVT_ID) { |
|
| 2109 | - $EVT_ID = absint($EVT_ID); |
|
| 2110 | - if ($EVT_ID) { |
|
| 2111 | - $results = $this->_permanently_delete_event($EVT_ID); |
|
| 2112 | - $success = $results !== false ? $success : false; |
|
| 2113 | - // remove this event from the list of events with no prices |
|
| 2114 | - unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
| 2115 | - } else { |
|
| 2116 | - $success = false; |
|
| 2117 | - $msg = esc_html__( |
|
| 2118 | - 'An error occurred. An event could not be deleted because a valid event ID was not not supplied.', |
|
| 2119 | - 'event_espresso' |
|
| 2120 | - ); |
|
| 2121 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2122 | - } |
|
| 2123 | - } |
|
| 2124 | - update_option('ee_no_ticket_prices', $espresso_no_ticket_prices); |
|
| 2125 | - // in order to force a pluralized result message we need to send back a success status greater than 1 |
|
| 2126 | - $success = $success ? 2 : false; |
|
| 2127 | - $this->_redirect_after_action($success, 'Events', 'deleted', array('action' => 'default')); |
|
| 2128 | - } |
|
| 2129 | - |
|
| 2130 | - |
|
| 2131 | - /** |
|
| 2132 | - * _permanently_delete_event |
|
| 2133 | - * |
|
| 2134 | - * @access private |
|
| 2135 | - * @param int $EVT_ID |
|
| 2136 | - * @return bool |
|
| 2137 | - */ |
|
| 2138 | - private function _permanently_delete_event($EVT_ID = 0) |
|
| 2139 | - { |
|
| 2140 | - // grab event id |
|
| 2141 | - if (! $EVT_ID) { |
|
| 2142 | - $msg = esc_html__( |
|
| 2143 | - 'An error occurred. No Event ID or an invalid Event ID was received.', |
|
| 2144 | - 'event_espresso' |
|
| 2145 | - ); |
|
| 2146 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2147 | - return false; |
|
| 2148 | - } |
|
| 2149 | - if (! $this->_cpt_model_obj instanceof EE_Event |
|
| 2150 | - || $this->_cpt_model_obj->ID() !== $EVT_ID |
|
| 2151 | - ) { |
|
| 2152 | - $this->_cpt_model_obj = EEM_Event::instance()->get_one_by_ID($EVT_ID); |
|
| 2153 | - } |
|
| 2154 | - if (! $this->_cpt_model_obj instanceof EE_Event) { |
|
| 2155 | - return false; |
|
| 2156 | - } |
|
| 2157 | - // need to delete related tickets and prices first. |
|
| 2158 | - $datetimes = $this->_cpt_model_obj->get_many_related('Datetime'); |
|
| 2159 | - foreach ($datetimes as $datetime) { |
|
| 2160 | - $this->_cpt_model_obj->_remove_relation_to($datetime, 'Datetime'); |
|
| 2161 | - $tickets = $datetime->get_many_related('Ticket'); |
|
| 2162 | - foreach ($tickets as $ticket) { |
|
| 2163 | - $ticket->_remove_relation_to($datetime, 'Datetime'); |
|
| 2164 | - $ticket->delete_related_permanently('Price'); |
|
| 2165 | - $ticket->delete_permanently(); |
|
| 2166 | - } |
|
| 2167 | - $datetime->delete(); |
|
| 2168 | - } |
|
| 2169 | - // what about related venues or terms? |
|
| 2170 | - $venues = $this->_cpt_model_obj->get_many_related('Venue'); |
|
| 2171 | - foreach ($venues as $venue) { |
|
| 2172 | - $this->_cpt_model_obj->_remove_relation_to($venue, 'Venue'); |
|
| 2173 | - } |
|
| 2174 | - // any attached question groups? |
|
| 2175 | - $question_groups = $this->_cpt_model_obj->get_many_related('Question_Group'); |
|
| 2176 | - if (! empty($question_groups)) { |
|
| 2177 | - foreach ($question_groups as $question_group) { |
|
| 2178 | - $this->_cpt_model_obj->_remove_relation_to($question_group, 'Question_Group'); |
|
| 2179 | - } |
|
| 2180 | - } |
|
| 2181 | - // Message Template Groups |
|
| 2182 | - $this->_cpt_model_obj->_remove_relations('Message_Template_Group'); |
|
| 2183 | - /** @type EE_Term_Taxonomy[] $term_taxonomies */ |
|
| 2184 | - $term_taxonomies = $this->_cpt_model_obj->term_taxonomies(); |
|
| 2185 | - foreach ($term_taxonomies as $term_taxonomy) { |
|
| 2186 | - $this->_cpt_model_obj->remove_relation_to_term_taxonomy($term_taxonomy); |
|
| 2187 | - } |
|
| 2188 | - $success = $this->_cpt_model_obj->delete_permanently(); |
|
| 2189 | - // did it all go as planned ? |
|
| 2190 | - if ($success) { |
|
| 2191 | - $msg = sprintf(esc_html__('Event ID # %d has been deleted.', 'event_espresso'), $EVT_ID); |
|
| 2192 | - EE_Error::add_success($msg); |
|
| 2193 | - } else { |
|
| 2194 | - $msg = sprintf( |
|
| 2195 | - esc_html__('An error occurred. Event ID # %d could not be deleted.', 'event_espresso'), |
|
| 2196 | - $EVT_ID |
|
| 2197 | - ); |
|
| 2198 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2199 | - return false; |
|
| 2200 | - } |
|
| 2201 | - do_action('AHEE__Events_Admin_Page___permanently_delete_event__after_event_deleted', $EVT_ID); |
|
| 2202 | - return true; |
|
| 2203 | - } |
|
| 2204 | - |
|
| 2205 | - |
|
| 2206 | - /** |
|
| 2207 | - * get total number of events |
|
| 2208 | - * |
|
| 2209 | - * @access public |
|
| 2210 | - * @return int |
|
| 2211 | - */ |
|
| 2212 | - public function total_events() |
|
| 2213 | - { |
|
| 2214 | - $count = EEM_Event::instance()->count(array('caps' => 'read_admin'), 'EVT_ID', true); |
|
| 2215 | - return $count; |
|
| 2216 | - } |
|
| 2217 | - |
|
| 2218 | - |
|
| 2219 | - /** |
|
| 2220 | - * get total number of draft events |
|
| 2221 | - * |
|
| 2222 | - * @access public |
|
| 2223 | - * @return int |
|
| 2224 | - */ |
|
| 2225 | - public function total_events_draft() |
|
| 2226 | - { |
|
| 2227 | - $where = array( |
|
| 2228 | - 'status' => array('IN', array('draft', 'auto-draft')), |
|
| 2229 | - ); |
|
| 2230 | - $count = EEM_Event::instance()->count(array($where, 'caps' => 'read_admin'), 'EVT_ID', true); |
|
| 2231 | - return $count; |
|
| 2232 | - } |
|
| 2233 | - |
|
| 2234 | - |
|
| 2235 | - /** |
|
| 2236 | - * get total number of trashed events |
|
| 2237 | - * |
|
| 2238 | - * @access public |
|
| 2239 | - * @return int |
|
| 2240 | - */ |
|
| 2241 | - public function total_trashed_events() |
|
| 2242 | - { |
|
| 2243 | - $where = array( |
|
| 2244 | - 'status' => 'trash', |
|
| 2245 | - ); |
|
| 2246 | - $count = EEM_Event::instance()->count(array($where, 'caps' => 'read_admin'), 'EVT_ID', true); |
|
| 2247 | - return $count; |
|
| 2248 | - } |
|
| 2249 | - |
|
| 2250 | - |
|
| 2251 | - /** |
|
| 2252 | - * _default_event_settings |
|
| 2253 | - * This generates the Default Settings Tab |
|
| 2254 | - * |
|
| 2255 | - * @return void |
|
| 2256 | - * @throws EE_Error |
|
| 2257 | - */ |
|
| 2258 | - protected function _default_event_settings() |
|
| 2259 | - { |
|
| 2260 | - $this->_set_add_edit_form_tags('update_default_event_settings'); |
|
| 2261 | - $this->_set_publish_post_box_vars(null, false, false, null, false); |
|
| 2262 | - $this->_template_args['admin_page_content'] = $this->_default_event_settings_form()->get_html(); |
|
| 2263 | - $this->display_admin_page_with_sidebar(); |
|
| 2264 | - } |
|
| 2265 | - |
|
| 2266 | - |
|
| 2267 | - /** |
|
| 2268 | - * Return the form for event settings. |
|
| 2269 | - * |
|
| 2270 | - * @return EE_Form_Section_Proper |
|
| 2271 | - * @throws EE_Error |
|
| 2272 | - */ |
|
| 2273 | - protected function _default_event_settings_form() |
|
| 2274 | - { |
|
| 2275 | - $registration_config = EE_Registry::instance()->CFG->registration; |
|
| 2276 | - $registration_stati_for_selection = EEM_Registration::reg_status_array( |
|
| 2277 | - // exclude |
|
| 2278 | - array( |
|
| 2279 | - EEM_Registration::status_id_cancelled, |
|
| 2280 | - EEM_Registration::status_id_declined, |
|
| 2281 | - EEM_Registration::status_id_incomplete, |
|
| 2282 | - EEM_Registration::status_id_wait_list, |
|
| 2283 | - ), |
|
| 2284 | - true |
|
| 2285 | - ); |
|
| 2286 | - return new EE_Form_Section_Proper( |
|
| 2287 | - array( |
|
| 2288 | - 'name' => 'update_default_event_settings', |
|
| 2289 | - 'html_id' => 'update_default_event_settings', |
|
| 2290 | - 'html_class' => 'form-table', |
|
| 2291 | - 'layout_strategy' => new EE_Admin_Two_Column_Layout(), |
|
| 2292 | - 'subsections' => apply_filters( |
|
| 2293 | - 'FHEE__Events_Admin_Page___default_event_settings_form__form_subsections', |
|
| 2294 | - array( |
|
| 2295 | - 'default_reg_status' => new EE_Select_Input( |
|
| 2296 | - $registration_stati_for_selection, |
|
| 2297 | - array( |
|
| 2298 | - 'default' => isset($registration_config->default_STS_ID) |
|
| 2299 | - && array_key_exists( |
|
| 2300 | - $registration_config->default_STS_ID, |
|
| 2301 | - $registration_stati_for_selection |
|
| 2302 | - ) |
|
| 2303 | - ? sanitize_text_field($registration_config->default_STS_ID) |
|
| 2304 | - : EEM_Registration::status_id_pending_payment, |
|
| 2305 | - 'html_label_text' => esc_html__('Default Registration Status', 'event_espresso') |
|
| 2306 | - . EEH_Template::get_help_tab_link( |
|
| 2307 | - 'default_settings_status_help_tab' |
|
| 2308 | - ), |
|
| 2309 | - 'html_help_text' => esc_html__( |
|
| 2310 | - 'This setting allows you to preselect what the default registration status setting is when creating an event. Note that changing this setting does NOT retroactively apply it to existing events.', |
|
| 2311 | - 'event_espresso' |
|
| 2312 | - ), |
|
| 2313 | - ) |
|
| 2314 | - ), |
|
| 2315 | - 'default_max_tickets' => new EE_Integer_Input( |
|
| 2316 | - array( |
|
| 2317 | - 'default' => isset($registration_config->default_maximum_number_of_tickets) |
|
| 2318 | - ? $registration_config->default_maximum_number_of_tickets |
|
| 2319 | - : EEM_Event::get_default_additional_limit(), |
|
| 2320 | - 'html_label_text' => esc_html__( |
|
| 2321 | - 'Default Maximum Tickets Allowed Per Order:', |
|
| 2322 | - 'event_espresso' |
|
| 2323 | - ) |
|
| 2324 | - . EEH_Template::get_help_tab_link( |
|
| 2325 | - 'default_maximum_tickets_help_tab"' |
|
| 2326 | - ), |
|
| 2327 | - 'html_help_text' => esc_html__( |
|
| 2328 | - 'This setting allows you to indicate what will be the default for the maximum number of tickets per order when creating new events.', |
|
| 2329 | - 'event_espresso' |
|
| 2330 | - ), |
|
| 2331 | - ) |
|
| 2332 | - ), |
|
| 2333 | - ) |
|
| 2334 | - ), |
|
| 2335 | - ) |
|
| 2336 | - ); |
|
| 2337 | - } |
|
| 2338 | - |
|
| 2339 | - |
|
| 2340 | - /** |
|
| 2341 | - * _update_default_event_settings |
|
| 2342 | - * |
|
| 2343 | - * @access protected |
|
| 2344 | - * @return void |
|
| 2345 | - * @throws EE_Error |
|
| 2346 | - */ |
|
| 2347 | - protected function _update_default_event_settings() |
|
| 2348 | - { |
|
| 2349 | - $registration_config = EE_Registry::instance()->CFG->registration; |
|
| 2350 | - $form = $this->_default_event_settings_form(); |
|
| 2351 | - if ($form->was_submitted()) { |
|
| 2352 | - $form->receive_form_submission(); |
|
| 2353 | - if ($form->is_valid()) { |
|
| 2354 | - $valid_data = $form->valid_data(); |
|
| 2355 | - if (isset($valid_data['default_reg_status'])) { |
|
| 2356 | - $registration_config->default_STS_ID = $valid_data['default_reg_status']; |
|
| 2357 | - } |
|
| 2358 | - if (isset($valid_data['default_max_tickets'])) { |
|
| 2359 | - $registration_config->default_maximum_number_of_tickets = $valid_data['default_max_tickets']; |
|
| 2360 | - } |
|
| 2361 | - // update because data was valid! |
|
| 2362 | - EE_Registry::instance()->CFG->update_espresso_config(); |
|
| 2363 | - EE_Error::overwrite_success(); |
|
| 2364 | - EE_Error::add_success( |
|
| 2365 | - __('Default Event Settings were updated', 'event_espresso') |
|
| 2366 | - ); |
|
| 2367 | - } |
|
| 2368 | - } |
|
| 2369 | - $this->_redirect_after_action(0, '', '', array('action' => 'default_event_settings'), true); |
|
| 2370 | - } |
|
| 2371 | - |
|
| 2372 | - |
|
| 2373 | - /************* Templates *************/ |
|
| 2374 | - protected function _template_settings() |
|
| 2375 | - { |
|
| 2376 | - $this->_admin_page_title = esc_html__('Template Settings (Preview)', 'event_espresso'); |
|
| 2377 | - $this->_template_args['preview_img'] = '<img src="' |
|
| 2378 | - . EVENTS_ASSETS_URL |
|
| 2379 | - . DS |
|
| 2380 | - . 'images' |
|
| 2381 | - . DS |
|
| 2382 | - . 'caffeinated_template_features.jpg" alt="' |
|
| 2383 | - . esc_attr__('Template Settings Preview screenshot', 'event_espresso') |
|
| 2384 | - . '" />'; |
|
| 2385 | - $this->_template_args['preview_text'] = '<strong>' |
|
| 2386 | - . esc_html__( |
|
| 2387 | - 'Template Settings is a feature that is only available in the premium version of Event Espresso 4 which is available with a support license purchase on EventEspresso.com. Template Settings allow you to configure some of the appearance options for both the Event List and Event Details pages.', |
|
| 2388 | - 'event_espresso' |
|
| 2389 | - ) . '</strong>'; |
|
| 2390 | - $this->display_admin_caf_preview_page('template_settings_tab'); |
|
| 2391 | - } |
|
| 2392 | - |
|
| 2393 | - |
|
| 2394 | - /** Event Category Stuff **/ |
|
| 2395 | - /** |
|
| 2396 | - * set the _category property with the category object for the loaded page. |
|
| 2397 | - * |
|
| 2398 | - * @access private |
|
| 2399 | - * @return void |
|
| 2400 | - */ |
|
| 2401 | - private function _set_category_object() |
|
| 2402 | - { |
|
| 2403 | - if (isset($this->_category->id) && ! empty($this->_category->id)) { |
|
| 2404 | - return; |
|
| 2405 | - } //already have the category object so get out. |
|
| 2406 | - // set default category object |
|
| 2407 | - $this->_set_empty_category_object(); |
|
| 2408 | - // only set if we've got an id |
|
| 2409 | - if (! isset($this->_req_data['EVT_CAT_ID'])) { |
|
| 2410 | - return; |
|
| 2411 | - } |
|
| 2412 | - $category_id = absint($this->_req_data['EVT_CAT_ID']); |
|
| 2413 | - $term = get_term($category_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
| 2414 | - if (! empty($term)) { |
|
| 2415 | - $this->_category->category_name = $term->name; |
|
| 2416 | - $this->_category->category_identifier = $term->slug; |
|
| 2417 | - $this->_category->category_desc = $term->description; |
|
| 2418 | - $this->_category->id = $term->term_id; |
|
| 2419 | - $this->_category->parent = $term->parent; |
|
| 2420 | - } |
|
| 2421 | - } |
|
| 2422 | - |
|
| 2423 | - |
|
| 2424 | - /** |
|
| 2425 | - * Clears out category properties. |
|
| 2426 | - */ |
|
| 2427 | - private function _set_empty_category_object() |
|
| 2428 | - { |
|
| 2429 | - $this->_category = new stdClass(); |
|
| 2430 | - $this->_category->category_name = $this->_category->category_identifier = $this->_category->category_desc = ''; |
|
| 2431 | - $this->_category->id = $this->_category->parent = 0; |
|
| 2432 | - } |
|
| 2433 | - |
|
| 2434 | - |
|
| 2435 | - /** |
|
| 2436 | - * @throws EE_Error |
|
| 2437 | - */ |
|
| 2438 | - protected function _category_list_table() |
|
| 2439 | - { |
|
| 2440 | - do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
| 2441 | - $this->_search_btn_label = esc_html__('Categories', 'event_espresso'); |
|
| 2442 | - $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
| 2443 | - 'add_category', |
|
| 2444 | - 'add_category', |
|
| 2445 | - array(), |
|
| 2446 | - 'add-new-h2' |
|
| 2447 | - ); |
|
| 2448 | - $this->display_admin_list_table_page_with_sidebar(); |
|
| 2449 | - } |
|
| 2450 | - |
|
| 2451 | - |
|
| 2452 | - /** |
|
| 2453 | - * Output category details view. |
|
| 2454 | - */ |
|
| 2455 | - protected function _category_details($view) |
|
| 2456 | - { |
|
| 2457 | - // load formatter helper |
|
| 2458 | - // load field generator helper |
|
| 2459 | - $route = $view == 'edit' ? 'update_category' : 'insert_category'; |
|
| 2460 | - $this->_set_add_edit_form_tags($route); |
|
| 2461 | - $this->_set_category_object(); |
|
| 2462 | - $id = ! empty($this->_category->id) ? $this->_category->id : ''; |
|
| 2463 | - $delete_action = 'delete_category'; |
|
| 2464 | - // custom redirect |
|
| 2465 | - $redirect = EE_Admin_Page::add_query_args_and_nonce( |
|
| 2466 | - array('action' => 'category_list'), |
|
| 2467 | - $this->_admin_base_url |
|
| 2468 | - ); |
|
| 2469 | - $this->_set_publish_post_box_vars('EVT_CAT_ID', $id, $delete_action, $redirect); |
|
| 2470 | - // take care of contents |
|
| 2471 | - $this->_template_args['admin_page_content'] = $this->_category_details_content(); |
|
| 2472 | - $this->display_admin_page_with_sidebar(); |
|
| 2473 | - } |
|
| 2474 | - |
|
| 2475 | - |
|
| 2476 | - /** |
|
| 2477 | - * Output category details content. |
|
| 2478 | - */ |
|
| 2479 | - protected function _category_details_content() |
|
| 2480 | - { |
|
| 2481 | - $editor_args['category_desc'] = array( |
|
| 2482 | - 'type' => 'wp_editor', |
|
| 2483 | - 'value' => EEH_Formatter::admin_format_content($this->_category->category_desc), |
|
| 2484 | - 'class' => 'my_editor_custom', |
|
| 2485 | - 'wpeditor_args' => array('media_buttons' => false), |
|
| 2486 | - ); |
|
| 2487 | - $_wp_editor = $this->_generate_admin_form_fields($editor_args, 'array'); |
|
| 2488 | - $all_terms = get_terms( |
|
| 2489 | - array(EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY), |
|
| 2490 | - array('hide_empty' => 0, 'exclude' => array($this->_category->id)) |
|
| 2491 | - ); |
|
| 2492 | - // setup category select for term parents. |
|
| 2493 | - $category_select_values[] = array( |
|
| 2494 | - 'text' => esc_html__('No Parent', 'event_espresso'), |
|
| 2495 | - 'id' => 0, |
|
| 2496 | - ); |
|
| 2497 | - foreach ($all_terms as $term) { |
|
| 2498 | - $category_select_values[] = array( |
|
| 2499 | - 'text' => $term->name, |
|
| 2500 | - 'id' => $term->term_id, |
|
| 2501 | - ); |
|
| 2502 | - } |
|
| 2503 | - $category_select = EEH_Form_Fields::select_input( |
|
| 2504 | - 'category_parent', |
|
| 2505 | - $category_select_values, |
|
| 2506 | - $this->_category->parent |
|
| 2507 | - ); |
|
| 2508 | - $template_args = array( |
|
| 2509 | - 'category' => $this->_category, |
|
| 2510 | - 'category_select' => $category_select, |
|
| 2511 | - 'unique_id_info_help_link' => $this->_get_help_tab_link('unique_id_info'), |
|
| 2512 | - 'category_desc_editor' => $_wp_editor['category_desc']['field'], |
|
| 2513 | - 'disable' => '', |
|
| 2514 | - 'disabled_message' => false, |
|
| 2515 | - ); |
|
| 2516 | - $template = EVENTS_TEMPLATE_PATH . 'event_category_details.template.php'; |
|
| 2517 | - return EEH_Template::display_template($template, $template_args, true); |
|
| 2518 | - } |
|
| 2519 | - |
|
| 2520 | - |
|
| 2521 | - /** |
|
| 2522 | - * Handles deleting categories. |
|
| 2523 | - */ |
|
| 2524 | - protected function _delete_categories() |
|
| 2525 | - { |
|
| 2526 | - $cat_ids = isset($this->_req_data['EVT_CAT_ID']) ? (array) $this->_req_data['EVT_CAT_ID'] |
|
| 2527 | - : (array) $this->_req_data['category_id']; |
|
| 2528 | - foreach ($cat_ids as $cat_id) { |
|
| 2529 | - $this->_delete_category($cat_id); |
|
| 2530 | - } |
|
| 2531 | - // doesn't matter what page we're coming from... we're going to the same place after delete. |
|
| 2532 | - $query_args = array( |
|
| 2533 | - 'action' => 'category_list', |
|
| 2534 | - ); |
|
| 2535 | - $this->_redirect_after_action(0, '', '', $query_args); |
|
| 2536 | - } |
|
| 2537 | - |
|
| 2538 | - |
|
| 2539 | - /** |
|
| 2540 | - * Handles deleting specific category. |
|
| 2541 | - * |
|
| 2542 | - * @param int $cat_id |
|
| 2543 | - */ |
|
| 2544 | - protected function _delete_category($cat_id) |
|
| 2545 | - { |
|
| 2546 | - $cat_id = absint($cat_id); |
|
| 2547 | - wp_delete_term($cat_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
| 2548 | - } |
|
| 2549 | - |
|
| 2550 | - |
|
| 2551 | - /** |
|
| 2552 | - * Handles triggering the update or insertion of a new category. |
|
| 2553 | - * |
|
| 2554 | - * @param bool $new_category true means we're triggering the insert of a new category. |
|
| 2555 | - */ |
|
| 2556 | - protected function _insert_or_update_category($new_category) |
|
| 2557 | - { |
|
| 2558 | - $cat_id = $new_category ? $this->_insert_category() : $this->_insert_category(true); |
|
| 2559 | - $success = 0; // we already have a success message so lets not send another. |
|
| 2560 | - if ($cat_id) { |
|
| 2561 | - $query_args = array( |
|
| 2562 | - 'action' => 'edit_category', |
|
| 2563 | - 'EVT_CAT_ID' => $cat_id, |
|
| 2564 | - ); |
|
| 2565 | - } else { |
|
| 2566 | - $query_args = array('action' => 'add_category'); |
|
| 2567 | - } |
|
| 2568 | - $this->_redirect_after_action($success, '', '', $query_args, true); |
|
| 2569 | - } |
|
| 2570 | - |
|
| 2571 | - |
|
| 2572 | - /** |
|
| 2573 | - * Inserts or updates category |
|
| 2574 | - * |
|
| 2575 | - * @param bool $update (true indicates we're updating a category). |
|
| 2576 | - * @return bool|mixed|string |
|
| 2577 | - */ |
|
| 2578 | - private function _insert_category($update = false) |
|
| 2579 | - { |
|
| 2580 | - $cat_id = $update ? $this->_req_data['EVT_CAT_ID'] : ''; |
|
| 2581 | - $category_name = isset($this->_req_data['category_name']) ? $this->_req_data['category_name'] : ''; |
|
| 2582 | - $category_desc = isset($this->_req_data['category_desc']) ? $this->_req_data['category_desc'] : ''; |
|
| 2583 | - $category_parent = isset($this->_req_data['category_parent']) ? $this->_req_data['category_parent'] : 0; |
|
| 2584 | - if (empty($category_name)) { |
|
| 2585 | - $msg = esc_html__('You must add a name for the category.', 'event_espresso'); |
|
| 2586 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2587 | - return false; |
|
| 2588 | - } |
|
| 2589 | - $term_args = array( |
|
| 2590 | - 'name' => $category_name, |
|
| 2591 | - 'description' => $category_desc, |
|
| 2592 | - 'parent' => $category_parent, |
|
| 2593 | - ); |
|
| 2594 | - // was the category_identifier input disabled? |
|
| 2595 | - if (isset($this->_req_data['category_identifier'])) { |
|
| 2596 | - $term_args['slug'] = $this->_req_data['category_identifier']; |
|
| 2597 | - } |
|
| 2598 | - $insert_ids = $update |
|
| 2599 | - ? wp_update_term($cat_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args) |
|
| 2600 | - : wp_insert_term($category_name, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args); |
|
| 2601 | - if (! is_array($insert_ids)) { |
|
| 2602 | - $msg = esc_html__( |
|
| 2603 | - 'An error occurred and the category has not been saved to the database.', |
|
| 2604 | - 'event_espresso' |
|
| 2605 | - ); |
|
| 2606 | - EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2607 | - } else { |
|
| 2608 | - $cat_id = $insert_ids['term_id']; |
|
| 2609 | - $msg = sprintf(esc_html__('The category %s was successfully saved', 'event_espresso'), $category_name); |
|
| 2610 | - EE_Error::add_success($msg); |
|
| 2611 | - } |
|
| 2612 | - return $cat_id; |
|
| 2613 | - } |
|
| 2614 | - |
|
| 2615 | - |
|
| 2616 | - /** |
|
| 2617 | - * Gets categories or count of categories matching the arguments in the request. |
|
| 2618 | - * |
|
| 2619 | - * @param int $per_page |
|
| 2620 | - * @param int $current_page |
|
| 2621 | - * @param bool $count |
|
| 2622 | - * @return EE_Base_Class[]|EE_Term_Taxonomy[]|int |
|
| 2623 | - */ |
|
| 2624 | - public function get_categories($per_page = 10, $current_page = 1, $count = false) |
|
| 2625 | - { |
|
| 2626 | - // testing term stuff |
|
| 2627 | - $orderby = isset($this->_req_data['orderby']) ? $this->_req_data['orderby'] : 'Term.term_id'; |
|
| 2628 | - $order = isset($this->_req_data['order']) ? $this->_req_data['order'] : 'DESC'; |
|
| 2629 | - $limit = ($current_page - 1) * $per_page; |
|
| 2630 | - $where = array('taxonomy' => EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
| 2631 | - if (isset($this->_req_data['s'])) { |
|
| 2632 | - $sstr = '%' . $this->_req_data['s'] . '%'; |
|
| 2633 | - $where['OR'] = array( |
|
| 2634 | - 'Term.name' => array('LIKE', $sstr), |
|
| 2635 | - 'description' => array('LIKE', $sstr), |
|
| 2636 | - ); |
|
| 2637 | - } |
|
| 2638 | - $query_params = array( |
|
| 2639 | - $where, |
|
| 2640 | - 'order_by' => array($orderby => $order), |
|
| 2641 | - 'limit' => $limit . ',' . $per_page, |
|
| 2642 | - 'force_join' => array('Term'), |
|
| 2643 | - ); |
|
| 2644 | - $categories = $count |
|
| 2645 | - ? EEM_Term_Taxonomy::instance()->count($query_params, 'term_id') |
|
| 2646 | - : EEM_Term_Taxonomy::instance()->get_all($query_params); |
|
| 2647 | - return $categories; |
|
| 2648 | - } |
|
| 2649 | - |
|
| 2650 | - /* end category stuff */ |
|
| 2651 | - /**************/ |
|
| 2652 | - |
|
| 2653 | - |
|
| 2654 | - /** |
|
| 2655 | - * Callback for the `ee_save_timezone_setting` ajax action. |
|
| 2656 | - * |
|
| 2657 | - * @throws EE_Error |
|
| 2658 | - */ |
|
| 2659 | - public function save_timezonestring_setting() |
|
| 2660 | - { |
|
| 2661 | - $timezone_string = isset($this->_req_data['timezone_selected']) |
|
| 2662 | - ? $this->_req_data['timezone_selected'] |
|
| 2663 | - : ''; |
|
| 2664 | - if (empty($timezone_string) || ! EEH_DTT_Helper::validate_timezone($timezone_string, false)) { |
|
| 2665 | - EE_Error::add_error( |
|
| 2666 | - esc_html__('An invalid timezone string submitted.', 'event_espresso'), |
|
| 2667 | - __FILE__, |
|
| 2668 | - __FUNCTION__, |
|
| 2669 | - __LINE__ |
|
| 2670 | - ); |
|
| 2671 | - $this->_template_args['error'] = true; |
|
| 2672 | - $this->_return_json(); |
|
| 2673 | - } |
|
| 2674 | - |
|
| 2675 | - update_option('timezone_string', $timezone_string); |
|
| 2676 | - EE_Error::add_success( |
|
| 2677 | - esc_html__('Your timezone string was updated.', 'event_espresso') |
|
| 2678 | - ); |
|
| 2679 | - $this->_template_args['success'] = true; |
|
| 2680 | - $this->_return_json(true, array('action' => 'create_new')); |
|
| 2681 | - } |
|
| 15 | + /** |
|
| 16 | + * This will hold the event object for event_details screen. |
|
| 17 | + * |
|
| 18 | + * @access protected |
|
| 19 | + * @var EE_Event $_event |
|
| 20 | + */ |
|
| 21 | + protected $_event; |
|
| 22 | + |
|
| 23 | + |
|
| 24 | + /** |
|
| 25 | + * This will hold the category object for category_details screen. |
|
| 26 | + * |
|
| 27 | + * @var stdClass $_category |
|
| 28 | + */ |
|
| 29 | + protected $_category; |
|
| 30 | + |
|
| 31 | + |
|
| 32 | + /** |
|
| 33 | + * This will hold the event model instance |
|
| 34 | + * |
|
| 35 | + * @var EEM_Event $_event_model |
|
| 36 | + */ |
|
| 37 | + protected $_event_model; |
|
| 38 | + |
|
| 39 | + |
|
| 40 | + /** |
|
| 41 | + * @var EE_Event |
|
| 42 | + */ |
|
| 43 | + protected $_cpt_model_obj = false; |
|
| 44 | + |
|
| 45 | + |
|
| 46 | + /** |
|
| 47 | + * Initialize page props for this admin page group. |
|
| 48 | + */ |
|
| 49 | + protected function _init_page_props() |
|
| 50 | + { |
|
| 51 | + $this->page_slug = EVENTS_PG_SLUG; |
|
| 52 | + $this->page_label = EVENTS_LABEL; |
|
| 53 | + $this->_admin_base_url = EVENTS_ADMIN_URL; |
|
| 54 | + $this->_admin_base_path = EVENTS_ADMIN; |
|
| 55 | + $this->_cpt_model_names = array( |
|
| 56 | + 'create_new' => 'EEM_Event', |
|
| 57 | + 'edit' => 'EEM_Event', |
|
| 58 | + ); |
|
| 59 | + $this->_cpt_edit_routes = array( |
|
| 60 | + 'espresso_events' => 'edit', |
|
| 61 | + ); |
|
| 62 | + add_action( |
|
| 63 | + 'AHEE__EE_Admin_Page_CPT__set_model_object__after_set_object', |
|
| 64 | + array($this, 'verify_event_edit'), |
|
| 65 | + 10, |
|
| 66 | + 2 |
|
| 67 | + ); |
|
| 68 | + } |
|
| 69 | + |
|
| 70 | + |
|
| 71 | + /** |
|
| 72 | + * Sets the ajax hooks used for this admin page group. |
|
| 73 | + */ |
|
| 74 | + protected function _ajax_hooks() |
|
| 75 | + { |
|
| 76 | + add_action('wp_ajax_ee_save_timezone_setting', array($this, 'save_timezonestring_setting')); |
|
| 77 | + } |
|
| 78 | + |
|
| 79 | + |
|
| 80 | + /** |
|
| 81 | + * Sets the page properties for this admin page group. |
|
| 82 | + */ |
|
| 83 | + protected function _define_page_props() |
|
| 84 | + { |
|
| 85 | + $this->_admin_page_title = EVENTS_LABEL; |
|
| 86 | + $this->_labels = array( |
|
| 87 | + 'buttons' => array( |
|
| 88 | + 'add' => esc_html__('Add New Event', 'event_espresso'), |
|
| 89 | + 'edit' => esc_html__('Edit Event', 'event_espresso'), |
|
| 90 | + 'delete' => esc_html__('Delete Event', 'event_espresso'), |
|
| 91 | + 'add_category' => esc_html__('Add New Category', 'event_espresso'), |
|
| 92 | + 'edit_category' => esc_html__('Edit Category', 'event_espresso'), |
|
| 93 | + 'delete_category' => esc_html__('Delete Category', 'event_espresso'), |
|
| 94 | + ), |
|
| 95 | + 'editor_title' => array( |
|
| 96 | + 'espresso_events' => esc_html__('Enter event title here', 'event_espresso'), |
|
| 97 | + ), |
|
| 98 | + 'publishbox' => array( |
|
| 99 | + 'create_new' => esc_html__('Save New Event', 'event_espresso'), |
|
| 100 | + 'edit' => esc_html__('Update Event', 'event_espresso'), |
|
| 101 | + 'add_category' => esc_html__('Save New Category', 'event_espresso'), |
|
| 102 | + 'edit_category' => esc_html__('Update Category', 'event_espresso'), |
|
| 103 | + 'template_settings' => esc_html__('Update Settings', 'event_espresso'), |
|
| 104 | + ), |
|
| 105 | + ); |
|
| 106 | + } |
|
| 107 | + |
|
| 108 | + |
|
| 109 | + /** |
|
| 110 | + * Sets the page routes property for this admin page group. |
|
| 111 | + */ |
|
| 112 | + protected function _set_page_routes() |
|
| 113 | + { |
|
| 114 | + // load formatter helper |
|
| 115 | + // load field generator helper |
|
| 116 | + // is there a evt_id in the request? |
|
| 117 | + $evt_id = ! empty($this->_req_data['EVT_ID']) && ! is_array($this->_req_data['EVT_ID']) |
|
| 118 | + ? $this->_req_data['EVT_ID'] |
|
| 119 | + : 0; |
|
| 120 | + $evt_id = ! empty($this->_req_data['post']) ? $this->_req_data['post'] : $evt_id; |
|
| 121 | + $this->_page_routes = array( |
|
| 122 | + 'default' => array( |
|
| 123 | + 'func' => '_events_overview_list_table', |
|
| 124 | + 'capability' => 'ee_read_events', |
|
| 125 | + ), |
|
| 126 | + 'create_new' => array( |
|
| 127 | + 'func' => '_create_new_cpt_item', |
|
| 128 | + 'capability' => 'ee_edit_events', |
|
| 129 | + ), |
|
| 130 | + 'edit' => array( |
|
| 131 | + 'func' => '_edit_cpt_item', |
|
| 132 | + 'capability' => 'ee_edit_event', |
|
| 133 | + 'obj_id' => $evt_id, |
|
| 134 | + ), |
|
| 135 | + 'copy_event' => array( |
|
| 136 | + 'func' => '_copy_events', |
|
| 137 | + 'capability' => 'ee_edit_event', |
|
| 138 | + 'obj_id' => $evt_id, |
|
| 139 | + 'noheader' => true, |
|
| 140 | + ), |
|
| 141 | + 'trash_event' => array( |
|
| 142 | + 'func' => '_trash_or_restore_event', |
|
| 143 | + 'args' => array('event_status' => 'trash'), |
|
| 144 | + 'capability' => 'ee_delete_event', |
|
| 145 | + 'obj_id' => $evt_id, |
|
| 146 | + 'noheader' => true, |
|
| 147 | + ), |
|
| 148 | + 'trash_events' => array( |
|
| 149 | + 'func' => '_trash_or_restore_events', |
|
| 150 | + 'args' => array('event_status' => 'trash'), |
|
| 151 | + 'capability' => 'ee_delete_events', |
|
| 152 | + 'noheader' => true, |
|
| 153 | + ), |
|
| 154 | + 'restore_event' => array( |
|
| 155 | + 'func' => '_trash_or_restore_event', |
|
| 156 | + 'args' => array('event_status' => 'draft'), |
|
| 157 | + 'capability' => 'ee_delete_event', |
|
| 158 | + 'obj_id' => $evt_id, |
|
| 159 | + 'noheader' => true, |
|
| 160 | + ), |
|
| 161 | + 'restore_events' => array( |
|
| 162 | + 'func' => '_trash_or_restore_events', |
|
| 163 | + 'args' => array('event_status' => 'draft'), |
|
| 164 | + 'capability' => 'ee_delete_events', |
|
| 165 | + 'noheader' => true, |
|
| 166 | + ), |
|
| 167 | + 'delete_event' => array( |
|
| 168 | + 'func' => '_delete_event', |
|
| 169 | + 'capability' => 'ee_delete_event', |
|
| 170 | + 'obj_id' => $evt_id, |
|
| 171 | + 'noheader' => true, |
|
| 172 | + ), |
|
| 173 | + 'delete_events' => array( |
|
| 174 | + 'func' => '_delete_events', |
|
| 175 | + 'capability' => 'ee_delete_events', |
|
| 176 | + 'noheader' => true, |
|
| 177 | + ), |
|
| 178 | + 'view_report' => array( |
|
| 179 | + 'func' => '_view_report', |
|
| 180 | + 'capablity' => 'ee_edit_events', |
|
| 181 | + ), |
|
| 182 | + 'default_event_settings' => array( |
|
| 183 | + 'func' => '_default_event_settings', |
|
| 184 | + 'capability' => 'manage_options', |
|
| 185 | + ), |
|
| 186 | + 'update_default_event_settings' => array( |
|
| 187 | + 'func' => '_update_default_event_settings', |
|
| 188 | + 'capability' => 'manage_options', |
|
| 189 | + 'noheader' => true, |
|
| 190 | + ), |
|
| 191 | + 'template_settings' => array( |
|
| 192 | + 'func' => '_template_settings', |
|
| 193 | + 'capability' => 'manage_options', |
|
| 194 | + ), |
|
| 195 | + // event category tab related |
|
| 196 | + 'add_category' => array( |
|
| 197 | + 'func' => '_category_details', |
|
| 198 | + 'capability' => 'ee_edit_event_category', |
|
| 199 | + 'args' => array('add'), |
|
| 200 | + ), |
|
| 201 | + 'edit_category' => array( |
|
| 202 | + 'func' => '_category_details', |
|
| 203 | + 'capability' => 'ee_edit_event_category', |
|
| 204 | + 'args' => array('edit'), |
|
| 205 | + ), |
|
| 206 | + 'delete_categories' => array( |
|
| 207 | + 'func' => '_delete_categories', |
|
| 208 | + 'capability' => 'ee_delete_event_category', |
|
| 209 | + 'noheader' => true, |
|
| 210 | + ), |
|
| 211 | + 'delete_category' => array( |
|
| 212 | + 'func' => '_delete_categories', |
|
| 213 | + 'capability' => 'ee_delete_event_category', |
|
| 214 | + 'noheader' => true, |
|
| 215 | + ), |
|
| 216 | + 'insert_category' => array( |
|
| 217 | + 'func' => '_insert_or_update_category', |
|
| 218 | + 'args' => array('new_category' => true), |
|
| 219 | + 'capability' => 'ee_edit_event_category', |
|
| 220 | + 'noheader' => true, |
|
| 221 | + ), |
|
| 222 | + 'update_category' => array( |
|
| 223 | + 'func' => '_insert_or_update_category', |
|
| 224 | + 'args' => array('new_category' => false), |
|
| 225 | + 'capability' => 'ee_edit_event_category', |
|
| 226 | + 'noheader' => true, |
|
| 227 | + ), |
|
| 228 | + 'category_list' => array( |
|
| 229 | + 'func' => '_category_list_table', |
|
| 230 | + 'capability' => 'ee_manage_event_categories', |
|
| 231 | + ), |
|
| 232 | + ); |
|
| 233 | + } |
|
| 234 | + |
|
| 235 | + |
|
| 236 | + /** |
|
| 237 | + * Set the _page_config property for this admin page group. |
|
| 238 | + */ |
|
| 239 | + protected function _set_page_config() |
|
| 240 | + { |
|
| 241 | + $this->_page_config = array( |
|
| 242 | + 'default' => array( |
|
| 243 | + 'nav' => array( |
|
| 244 | + 'label' => esc_html__('Overview', 'event_espresso'), |
|
| 245 | + 'order' => 10, |
|
| 246 | + ), |
|
| 247 | + 'list_table' => 'Events_Admin_List_Table', |
|
| 248 | + 'help_tabs' => array( |
|
| 249 | + 'events_overview_help_tab' => array( |
|
| 250 | + 'title' => esc_html__('Events Overview', 'event_espresso'), |
|
| 251 | + 'filename' => 'events_overview', |
|
| 252 | + ), |
|
| 253 | + 'events_overview_table_column_headings_help_tab' => array( |
|
| 254 | + 'title' => esc_html__('Events Overview Table Column Headings', 'event_espresso'), |
|
| 255 | + 'filename' => 'events_overview_table_column_headings', |
|
| 256 | + ), |
|
| 257 | + 'events_overview_filters_help_tab' => array( |
|
| 258 | + 'title' => esc_html__('Events Overview Filters', 'event_espresso'), |
|
| 259 | + 'filename' => 'events_overview_filters', |
|
| 260 | + ), |
|
| 261 | + 'events_overview_view_help_tab' => array( |
|
| 262 | + 'title' => esc_html__('Events Overview Views', 'event_espresso'), |
|
| 263 | + 'filename' => 'events_overview_views', |
|
| 264 | + ), |
|
| 265 | + 'events_overview_other_help_tab' => array( |
|
| 266 | + 'title' => esc_html__('Events Overview Other', 'event_espresso'), |
|
| 267 | + 'filename' => 'events_overview_other', |
|
| 268 | + ), |
|
| 269 | + ), |
|
| 270 | + 'help_tour' => array( |
|
| 271 | + 'Event_Overview_Help_Tour', |
|
| 272 | + // 'New_Features_Test_Help_Tour' for testing multiple help tour |
|
| 273 | + ), |
|
| 274 | + 'qtips' => array( |
|
| 275 | + 'EE_Event_List_Table_Tips', |
|
| 276 | + ), |
|
| 277 | + 'require_nonce' => false, |
|
| 278 | + ), |
|
| 279 | + 'create_new' => array( |
|
| 280 | + 'nav' => array( |
|
| 281 | + 'label' => esc_html__('Add Event', 'event_espresso'), |
|
| 282 | + 'order' => 5, |
|
| 283 | + 'persistent' => false, |
|
| 284 | + ), |
|
| 285 | + 'metaboxes' => array('_register_event_editor_meta_boxes'), |
|
| 286 | + 'help_tabs' => array( |
|
| 287 | + 'event_editor_help_tab' => array( |
|
| 288 | + 'title' => esc_html__('Event Editor', 'event_espresso'), |
|
| 289 | + 'filename' => 'event_editor', |
|
| 290 | + ), |
|
| 291 | + 'event_editor_title_richtexteditor_help_tab' => array( |
|
| 292 | + 'title' => esc_html__('Event Title & Rich Text Editor', 'event_espresso'), |
|
| 293 | + 'filename' => 'event_editor_title_richtexteditor', |
|
| 294 | + ), |
|
| 295 | + 'event_editor_venue_details_help_tab' => array( |
|
| 296 | + 'title' => esc_html__('Event Venue Details', 'event_espresso'), |
|
| 297 | + 'filename' => 'event_editor_venue_details', |
|
| 298 | + ), |
|
| 299 | + 'event_editor_event_datetimes_help_tab' => array( |
|
| 300 | + 'title' => esc_html__('Event Datetimes', 'event_espresso'), |
|
| 301 | + 'filename' => 'event_editor_event_datetimes', |
|
| 302 | + ), |
|
| 303 | + 'event_editor_event_tickets_help_tab' => array( |
|
| 304 | + 'title' => esc_html__('Event Tickets', 'event_espresso'), |
|
| 305 | + 'filename' => 'event_editor_event_tickets', |
|
| 306 | + ), |
|
| 307 | + 'event_editor_event_registration_options_help_tab' => array( |
|
| 308 | + 'title' => esc_html__('Event Registration Options', 'event_espresso'), |
|
| 309 | + 'filename' => 'event_editor_event_registration_options', |
|
| 310 | + ), |
|
| 311 | + 'event_editor_tags_categories_help_tab' => array( |
|
| 312 | + 'title' => esc_html__('Event Tags & Categories', 'event_espresso'), |
|
| 313 | + 'filename' => 'event_editor_tags_categories', |
|
| 314 | + ), |
|
| 315 | + 'event_editor_questions_registrants_help_tab' => array( |
|
| 316 | + 'title' => esc_html__('Questions for Registrants', 'event_espresso'), |
|
| 317 | + 'filename' => 'event_editor_questions_registrants', |
|
| 318 | + ), |
|
| 319 | + 'event_editor_save_new_event_help_tab' => array( |
|
| 320 | + 'title' => esc_html__('Save New Event', 'event_espresso'), |
|
| 321 | + 'filename' => 'event_editor_save_new_event', |
|
| 322 | + ), |
|
| 323 | + 'event_editor_other_help_tab' => array( |
|
| 324 | + 'title' => esc_html__('Event Other', 'event_espresso'), |
|
| 325 | + 'filename' => 'event_editor_other', |
|
| 326 | + ), |
|
| 327 | + ), |
|
| 328 | + 'help_tour' => array( |
|
| 329 | + 'Event_Editor_Help_Tour', |
|
| 330 | + ), |
|
| 331 | + 'qtips' => array('EE_Event_Editor_Decaf_Tips'), |
|
| 332 | + 'require_nonce' => false, |
|
| 333 | + ), |
|
| 334 | + 'edit' => array( |
|
| 335 | + 'nav' => array( |
|
| 336 | + 'label' => esc_html__('Edit Event', 'event_espresso'), |
|
| 337 | + 'order' => 5, |
|
| 338 | + 'persistent' => false, |
|
| 339 | + 'url' => isset($this->_req_data['post']) |
|
| 340 | + ? EE_Admin_Page::add_query_args_and_nonce( |
|
| 341 | + array('post' => $this->_req_data['post'], 'action' => 'edit'), |
|
| 342 | + $this->_current_page_view_url |
|
| 343 | + ) |
|
| 344 | + : $this->_admin_base_url, |
|
| 345 | + ), |
|
| 346 | + 'metaboxes' => array('_register_event_editor_meta_boxes'), |
|
| 347 | + 'help_tabs' => array( |
|
| 348 | + 'event_editor_help_tab' => array( |
|
| 349 | + 'title' => esc_html__('Event Editor', 'event_espresso'), |
|
| 350 | + 'filename' => 'event_editor', |
|
| 351 | + ), |
|
| 352 | + 'event_editor_title_richtexteditor_help_tab' => array( |
|
| 353 | + 'title' => esc_html__('Event Title & Rich Text Editor', 'event_espresso'), |
|
| 354 | + 'filename' => 'event_editor_title_richtexteditor', |
|
| 355 | + ), |
|
| 356 | + 'event_editor_venue_details_help_tab' => array( |
|
| 357 | + 'title' => esc_html__('Event Venue Details', 'event_espresso'), |
|
| 358 | + 'filename' => 'event_editor_venue_details', |
|
| 359 | + ), |
|
| 360 | + 'event_editor_event_datetimes_help_tab' => array( |
|
| 361 | + 'title' => esc_html__('Event Datetimes', 'event_espresso'), |
|
| 362 | + 'filename' => 'event_editor_event_datetimes', |
|
| 363 | + ), |
|
| 364 | + 'event_editor_event_tickets_help_tab' => array( |
|
| 365 | + 'title' => esc_html__('Event Tickets', 'event_espresso'), |
|
| 366 | + 'filename' => 'event_editor_event_tickets', |
|
| 367 | + ), |
|
| 368 | + 'event_editor_event_registration_options_help_tab' => array( |
|
| 369 | + 'title' => esc_html__('Event Registration Options', 'event_espresso'), |
|
| 370 | + 'filename' => 'event_editor_event_registration_options', |
|
| 371 | + ), |
|
| 372 | + 'event_editor_tags_categories_help_tab' => array( |
|
| 373 | + 'title' => esc_html__('Event Tags & Categories', 'event_espresso'), |
|
| 374 | + 'filename' => 'event_editor_tags_categories', |
|
| 375 | + ), |
|
| 376 | + 'event_editor_questions_registrants_help_tab' => array( |
|
| 377 | + 'title' => esc_html__('Questions for Registrants', 'event_espresso'), |
|
| 378 | + 'filename' => 'event_editor_questions_registrants', |
|
| 379 | + ), |
|
| 380 | + 'event_editor_save_new_event_help_tab' => array( |
|
| 381 | + 'title' => esc_html__('Save New Event', 'event_espresso'), |
|
| 382 | + 'filename' => 'event_editor_save_new_event', |
|
| 383 | + ), |
|
| 384 | + 'event_editor_other_help_tab' => array( |
|
| 385 | + 'title' => esc_html__('Event Other', 'event_espresso'), |
|
| 386 | + 'filename' => 'event_editor_other', |
|
| 387 | + ), |
|
| 388 | + ), |
|
| 389 | + 'qtips' => array('EE_Event_Editor_Decaf_Tips'), |
|
| 390 | + 'require_nonce' => false, |
|
| 391 | + ), |
|
| 392 | + 'default_event_settings' => array( |
|
| 393 | + 'nav' => array( |
|
| 394 | + 'label' => esc_html__('Default Settings', 'event_espresso'), |
|
| 395 | + 'order' => 40, |
|
| 396 | + ), |
|
| 397 | + 'metaboxes' => array_merge($this->_default_espresso_metaboxes, array('_publish_post_box')), |
|
| 398 | + 'labels' => array( |
|
| 399 | + 'publishbox' => esc_html__('Update Settings', 'event_espresso'), |
|
| 400 | + ), |
|
| 401 | + 'help_tabs' => array( |
|
| 402 | + 'default_settings_help_tab' => array( |
|
| 403 | + 'title' => esc_html__('Default Event Settings', 'event_espresso'), |
|
| 404 | + 'filename' => 'events_default_settings', |
|
| 405 | + ), |
|
| 406 | + 'default_settings_status_help_tab' => array( |
|
| 407 | + 'title' => esc_html__('Default Registration Status', 'event_espresso'), |
|
| 408 | + 'filename' => 'events_default_settings_status', |
|
| 409 | + ), |
|
| 410 | + 'default_maximum_tickets_help_tab' => array( |
|
| 411 | + 'title' => esc_html__('Default Maximum Tickets Per Order', 'event_espresso'), |
|
| 412 | + 'filename' => 'events_default_settings_max_tickets', |
|
| 413 | + ), |
|
| 414 | + ), |
|
| 415 | + 'help_tour' => array('Event_Default_Settings_Help_Tour'), |
|
| 416 | + 'require_nonce' => false, |
|
| 417 | + ), |
|
| 418 | + // template settings |
|
| 419 | + 'template_settings' => array( |
|
| 420 | + 'nav' => array( |
|
| 421 | + 'label' => esc_html__('Templates', 'event_espresso'), |
|
| 422 | + 'order' => 30, |
|
| 423 | + ), |
|
| 424 | + 'metaboxes' => $this->_default_espresso_metaboxes, |
|
| 425 | + 'help_tabs' => array( |
|
| 426 | + 'general_settings_templates_help_tab' => array( |
|
| 427 | + 'title' => esc_html__('Templates', 'event_espresso'), |
|
| 428 | + 'filename' => 'general_settings_templates', |
|
| 429 | + ), |
|
| 430 | + ), |
|
| 431 | + 'help_tour' => array('Templates_Help_Tour'), |
|
| 432 | + 'require_nonce' => false, |
|
| 433 | + ), |
|
| 434 | + // event category stuff |
|
| 435 | + 'add_category' => array( |
|
| 436 | + 'nav' => array( |
|
| 437 | + 'label' => esc_html__('Add Category', 'event_espresso'), |
|
| 438 | + 'order' => 15, |
|
| 439 | + 'persistent' => false, |
|
| 440 | + ), |
|
| 441 | + 'help_tabs' => array( |
|
| 442 | + 'add_category_help_tab' => array( |
|
| 443 | + 'title' => esc_html__('Add New Event Category', 'event_espresso'), |
|
| 444 | + 'filename' => 'events_add_category', |
|
| 445 | + ), |
|
| 446 | + ), |
|
| 447 | + 'help_tour' => array('Event_Add_Category_Help_Tour'), |
|
| 448 | + 'metaboxes' => array('_publish_post_box'), |
|
| 449 | + 'require_nonce' => false, |
|
| 450 | + ), |
|
| 451 | + 'edit_category' => array( |
|
| 452 | + 'nav' => array( |
|
| 453 | + 'label' => esc_html__('Edit Category', 'event_espresso'), |
|
| 454 | + 'order' => 15, |
|
| 455 | + 'persistent' => false, |
|
| 456 | + 'url' => isset($this->_req_data['EVT_CAT_ID']) |
|
| 457 | + ? add_query_arg( |
|
| 458 | + array('EVT_CAT_ID' => $this->_req_data['EVT_CAT_ID']), |
|
| 459 | + $this->_current_page_view_url |
|
| 460 | + ) |
|
| 461 | + : $this->_admin_base_url, |
|
| 462 | + ), |
|
| 463 | + 'help_tabs' => array( |
|
| 464 | + 'edit_category_help_tab' => array( |
|
| 465 | + 'title' => esc_html__('Edit Event Category', 'event_espresso'), |
|
| 466 | + 'filename' => 'events_edit_category', |
|
| 467 | + ), |
|
| 468 | + ), |
|
| 469 | + /*'help_tour' => array('Event_Edit_Category_Help_Tour'),*/ |
|
| 470 | + 'metaboxes' => array('_publish_post_box'), |
|
| 471 | + 'require_nonce' => false, |
|
| 472 | + ), |
|
| 473 | + 'category_list' => array( |
|
| 474 | + 'nav' => array( |
|
| 475 | + 'label' => esc_html__('Categories', 'event_espresso'), |
|
| 476 | + 'order' => 20, |
|
| 477 | + ), |
|
| 478 | + 'list_table' => 'Event_Categories_Admin_List_Table', |
|
| 479 | + 'help_tabs' => array( |
|
| 480 | + 'events_categories_help_tab' => array( |
|
| 481 | + 'title' => esc_html__('Event Categories', 'event_espresso'), |
|
| 482 | + 'filename' => 'events_categories', |
|
| 483 | + ), |
|
| 484 | + 'events_categories_table_column_headings_help_tab' => array( |
|
| 485 | + 'title' => esc_html__('Event Categories Table Column Headings', 'event_espresso'), |
|
| 486 | + 'filename' => 'events_categories_table_column_headings', |
|
| 487 | + ), |
|
| 488 | + 'events_categories_view_help_tab' => array( |
|
| 489 | + 'title' => esc_html__('Event Categories Views', 'event_espresso'), |
|
| 490 | + 'filename' => 'events_categories_views', |
|
| 491 | + ), |
|
| 492 | + 'events_categories_other_help_tab' => array( |
|
| 493 | + 'title' => esc_html__('Event Categories Other', 'event_espresso'), |
|
| 494 | + 'filename' => 'events_categories_other', |
|
| 495 | + ), |
|
| 496 | + ), |
|
| 497 | + 'help_tour' => array( |
|
| 498 | + 'Event_Categories_Help_Tour', |
|
| 499 | + ), |
|
| 500 | + 'metaboxes' => $this->_default_espresso_metaboxes, |
|
| 501 | + 'require_nonce' => false, |
|
| 502 | + ), |
|
| 503 | + ); |
|
| 504 | + } |
|
| 505 | + |
|
| 506 | + |
|
| 507 | + /** |
|
| 508 | + * Used to register any global screen options if necessary for every route in this admin page group. |
|
| 509 | + */ |
|
| 510 | + protected function _add_screen_options() |
|
| 511 | + { |
|
| 512 | + } |
|
| 513 | + |
|
| 514 | + |
|
| 515 | + /** |
|
| 516 | + * Implementing the screen options for the 'default' route. |
|
| 517 | + */ |
|
| 518 | + protected function _add_screen_options_default() |
|
| 519 | + { |
|
| 520 | + $this->_per_page_screen_option(); |
|
| 521 | + } |
|
| 522 | + |
|
| 523 | + |
|
| 524 | + /** |
|
| 525 | + * Implementing screen options for the category list route. |
|
| 526 | + */ |
|
| 527 | + protected function _add_screen_options_category_list() |
|
| 528 | + { |
|
| 529 | + $page_title = $this->_admin_page_title; |
|
| 530 | + $this->_admin_page_title = esc_html__('Categories', 'event_espresso'); |
|
| 531 | + $this->_per_page_screen_option(); |
|
| 532 | + $this->_admin_page_title = $page_title; |
|
| 533 | + } |
|
| 534 | + |
|
| 535 | + |
|
| 536 | + /** |
|
| 537 | + * Used to register any global feature pointers for the admin page group. |
|
| 538 | + */ |
|
| 539 | + protected function _add_feature_pointers() |
|
| 540 | + { |
|
| 541 | + } |
|
| 542 | + |
|
| 543 | + |
|
| 544 | + /** |
|
| 545 | + * Registers and enqueues any global scripts and styles for the entire admin page group. |
|
| 546 | + */ |
|
| 547 | + public function load_scripts_styles() |
|
| 548 | + { |
|
| 549 | + wp_register_style( |
|
| 550 | + 'events-admin-css', |
|
| 551 | + EVENTS_ASSETS_URL . 'events-admin-page.css', |
|
| 552 | + array(), |
|
| 553 | + EVENT_ESPRESSO_VERSION |
|
| 554 | + ); |
|
| 555 | + wp_register_style('ee-cat-admin', EVENTS_ASSETS_URL . 'ee-cat-admin.css', array(), EVENT_ESPRESSO_VERSION); |
|
| 556 | + wp_enqueue_style('events-admin-css'); |
|
| 557 | + wp_enqueue_style('ee-cat-admin'); |
|
| 558 | + // todo note: we also need to load_scripts_styles per view (i.e. default/view_report/event_details |
|
| 559 | + // registers for all views |
|
| 560 | + // scripts |
|
| 561 | + wp_register_script( |
|
| 562 | + 'event_editor_js', |
|
| 563 | + EVENTS_ASSETS_URL . 'event_editor.js', |
|
| 564 | + array('ee_admin_js', 'jquery-ui-slider', 'jquery-ui-timepicker-addon'), |
|
| 565 | + EVENT_ESPRESSO_VERSION, |
|
| 566 | + true |
|
| 567 | + ); |
|
| 568 | + } |
|
| 569 | + |
|
| 570 | + |
|
| 571 | + /** |
|
| 572 | + * Enqueuing scripts and styles specific to this view |
|
| 573 | + */ |
|
| 574 | + public function load_scripts_styles_create_new() |
|
| 575 | + { |
|
| 576 | + $this->load_scripts_styles_edit(); |
|
| 577 | + } |
|
| 578 | + |
|
| 579 | + |
|
| 580 | + /** |
|
| 581 | + * Enqueuing scripts and styles specific to this view |
|
| 582 | + */ |
|
| 583 | + public function load_scripts_styles_edit() |
|
| 584 | + { |
|
| 585 | + // styles |
|
| 586 | + wp_enqueue_style('espresso-ui-theme'); |
|
| 587 | + wp_register_style( |
|
| 588 | + 'event-editor-css', |
|
| 589 | + EVENTS_ASSETS_URL . 'event-editor.css', |
|
| 590 | + array('ee-admin-css'), |
|
| 591 | + EVENT_ESPRESSO_VERSION |
|
| 592 | + ); |
|
| 593 | + wp_enqueue_style('event-editor-css'); |
|
| 594 | + // scripts |
|
| 595 | + wp_register_script( |
|
| 596 | + 'event-datetime-metabox', |
|
| 597 | + EVENTS_ASSETS_URL . 'event-datetime-metabox.js', |
|
| 598 | + array('event_editor_js', 'ee-datepicker'), |
|
| 599 | + EVENT_ESPRESSO_VERSION |
|
| 600 | + ); |
|
| 601 | + wp_enqueue_script('event-datetime-metabox'); |
|
| 602 | + } |
|
| 603 | + |
|
| 604 | + |
|
| 605 | + /** |
|
| 606 | + * Populating the _views property for the category list table view. |
|
| 607 | + */ |
|
| 608 | + protected function _set_list_table_views_category_list() |
|
| 609 | + { |
|
| 610 | + $this->_views = array( |
|
| 611 | + 'all' => array( |
|
| 612 | + 'slug' => 'all', |
|
| 613 | + 'label' => esc_html__('All', 'event_espresso'), |
|
| 614 | + 'count' => 0, |
|
| 615 | + 'bulk_action' => array( |
|
| 616 | + 'delete_categories' => esc_html__('Delete Permanently', 'event_espresso'), |
|
| 617 | + ), |
|
| 618 | + ), |
|
| 619 | + ); |
|
| 620 | + } |
|
| 621 | + |
|
| 622 | + |
|
| 623 | + /** |
|
| 624 | + * For adding anything that fires on the admin_init hook for any route within this admin page group. |
|
| 625 | + */ |
|
| 626 | + public function admin_init() |
|
| 627 | + { |
|
| 628 | + EE_Registry::$i18n_js_strings['image_confirm'] = esc_html__( |
|
| 629 | + 'Do you really want to delete this image? Please remember to update your event to complete the removal.', |
|
| 630 | + 'event_espresso' |
|
| 631 | + ); |
|
| 632 | + } |
|
| 633 | + |
|
| 634 | + |
|
| 635 | + /** |
|
| 636 | + * For adding anything that should be triggered on the admin_notices hook for any route within this admin page |
|
| 637 | + * group. |
|
| 638 | + */ |
|
| 639 | + public function admin_notices() |
|
| 640 | + { |
|
| 641 | + } |
|
| 642 | + |
|
| 643 | + |
|
| 644 | + /** |
|
| 645 | + * For adding anything that should be triggered on the `admin_print_footer_scripts` hook for any route within |
|
| 646 | + * this admin page group. |
|
| 647 | + */ |
|
| 648 | + public function admin_footer_scripts() |
|
| 649 | + { |
|
| 650 | + } |
|
| 651 | + |
|
| 652 | + |
|
| 653 | + /** |
|
| 654 | + * Call this function to verify if an event is public and has tickets for sale. If it does, then we need to show a |
|
| 655 | + * warning (via EE_Error::add_error()); |
|
| 656 | + * |
|
| 657 | + * @param EE_Event $event Event object |
|
| 658 | + * @param string $req_type |
|
| 659 | + * @return void |
|
| 660 | + * @throws EE_Error |
|
| 661 | + * @access public |
|
| 662 | + */ |
|
| 663 | + public function verify_event_edit($event = null, $req_type = '') |
|
| 664 | + { |
|
| 665 | + // don't need to do this when processing |
|
| 666 | + if (! empty($req_type)) { |
|
| 667 | + return; |
|
| 668 | + } |
|
| 669 | + // no event? |
|
| 670 | + if (empty($event)) { |
|
| 671 | + // set event |
|
| 672 | + $event = $this->_cpt_model_obj; |
|
| 673 | + } |
|
| 674 | + // STILL no event? |
|
| 675 | + if (! $event instanceof EE_Event) { |
|
| 676 | + return; |
|
| 677 | + } |
|
| 678 | + $orig_status = $event->status(); |
|
| 679 | + // first check if event is active. |
|
| 680 | + if ($orig_status === EEM_Event::cancelled |
|
| 681 | + || $orig_status === EEM_Event::postponed |
|
| 682 | + || $event->is_expired() |
|
| 683 | + || $event->is_inactive() |
|
| 684 | + ) { |
|
| 685 | + return; |
|
| 686 | + } |
|
| 687 | + // made it here so it IS active... next check that any of the tickets are sold. |
|
| 688 | + if ($event->is_sold_out(true)) { |
|
| 689 | + if ($orig_status !== EEM_Event::sold_out && $event->status() !== $orig_status) { |
|
| 690 | + EE_Error::add_attention( |
|
| 691 | + sprintf( |
|
| 692 | + esc_html__( |
|
| 693 | + 'Please note that the Event Status has automatically been changed to %s because there are no more spaces available for this event. However, this change is not permanent until you update the event. You can change the status back to something else before updating if you wish.', |
|
| 694 | + 'event_espresso' |
|
| 695 | + ), |
|
| 696 | + EEH_Template::pretty_status(EEM_Event::sold_out, false, 'sentence') |
|
| 697 | + ) |
|
| 698 | + ); |
|
| 699 | + } |
|
| 700 | + return; |
|
| 701 | + } elseif ($orig_status === EEM_Event::sold_out) { |
|
| 702 | + EE_Error::add_attention( |
|
| 703 | + sprintf( |
|
| 704 | + esc_html__( |
|
| 705 | + 'Please note that the Event Status has automatically been changed to %s because more spaces have become available for this event, most likely due to abandoned transactions freeing up reserved tickets. However, this change is not permanent until you update the event. If you wish, you can change the status back to something else before updating.', |
|
| 706 | + 'event_espresso' |
|
| 707 | + ), |
|
| 708 | + EEH_Template::pretty_status($event->status(), false, 'sentence') |
|
| 709 | + ) |
|
| 710 | + ); |
|
| 711 | + } |
|
| 712 | + // now we need to determine if the event has any tickets on sale. If not then we dont' show the error |
|
| 713 | + if (! $event->tickets_on_sale()) { |
|
| 714 | + return; |
|
| 715 | + } |
|
| 716 | + // made it here so show warning |
|
| 717 | + $this->_edit_event_warning(); |
|
| 718 | + } |
|
| 719 | + |
|
| 720 | + |
|
| 721 | + /** |
|
| 722 | + * This is the text used for when an event is being edited that is public and has tickets for sale. |
|
| 723 | + * When needed, hook this into a EE_Error::add_error() notice. |
|
| 724 | + * |
|
| 725 | + * @access protected |
|
| 726 | + * @return void |
|
| 727 | + */ |
|
| 728 | + protected function _edit_event_warning() |
|
| 729 | + { |
|
| 730 | + // we don't want to add warnings during these requests |
|
| 731 | + if (isset($this->_req_data['action']) && $this->_req_data['action'] === 'editpost') { |
|
| 732 | + return; |
|
| 733 | + } |
|
| 734 | + EE_Error::add_attention( |
|
| 735 | + sprintf( |
|
| 736 | + esc_html__( |
|
| 737 | + 'Your event is open for registration. Making changes may disrupt any transactions in progress. %sLearn more%s', |
|
| 738 | + 'event_espresso' |
|
| 739 | + ), |
|
| 740 | + '<a class="espresso-help-tab-lnk">', |
|
| 741 | + '</a>' |
|
| 742 | + ) |
|
| 743 | + ); |
|
| 744 | + } |
|
| 745 | + |
|
| 746 | + |
|
| 747 | + /** |
|
| 748 | + * When a user is creating a new event, notify them if they haven't set their timezone. |
|
| 749 | + * Otherwise, do the normal logic |
|
| 750 | + * |
|
| 751 | + * @return string |
|
| 752 | + * @throws \EE_Error |
|
| 753 | + */ |
|
| 754 | + protected function _create_new_cpt_item() |
|
| 755 | + { |
|
| 756 | + $has_timezone_string = get_option('timezone_string'); |
|
| 757 | + // only nag them about setting their timezone if it's their first event, and they haven't already done it |
|
| 758 | + if (! $has_timezone_string && ! EEM_Event::instance()->exists(array())) { |
|
| 759 | + EE_Error::add_attention( |
|
| 760 | + sprintf( |
|
| 761 | + __( |
|
| 762 | + 'Your website\'s timezone is currently set to a UTC offset. We recommend updating your timezone to a city or region near you before you create an event. Change your timezone now:%1$s%2$s%3$sChange Timezone%4$s', |
|
| 763 | + 'event_espresso' |
|
| 764 | + ), |
|
| 765 | + '<br>', |
|
| 766 | + '<select id="timezone_string" name="timezone_string" aria-describedby="timezone-description">' |
|
| 767 | + . EEH_DTT_Helper::wp_timezone_choice('', EEH_DTT_Helper::get_user_locale()) |
|
| 768 | + . '</select>', |
|
| 769 | + '<button class="button button-secondary timezone-submit">', |
|
| 770 | + '</button><span class="spinner"></span>' |
|
| 771 | + ), |
|
| 772 | + __FILE__, |
|
| 773 | + __FUNCTION__, |
|
| 774 | + __LINE__ |
|
| 775 | + ); |
|
| 776 | + } |
|
| 777 | + return parent::_create_new_cpt_item(); |
|
| 778 | + } |
|
| 779 | + |
|
| 780 | + |
|
| 781 | + /** |
|
| 782 | + * Sets the _views property for the default route in this admin page group. |
|
| 783 | + */ |
|
| 784 | + protected function _set_list_table_views_default() |
|
| 785 | + { |
|
| 786 | + $this->_views = array( |
|
| 787 | + 'all' => array( |
|
| 788 | + 'slug' => 'all', |
|
| 789 | + 'label' => esc_html__('View All Events', 'event_espresso'), |
|
| 790 | + 'count' => 0, |
|
| 791 | + 'bulk_action' => array( |
|
| 792 | + 'trash_events' => esc_html__('Move to Trash', 'event_espresso'), |
|
| 793 | + ), |
|
| 794 | + ), |
|
| 795 | + 'draft' => array( |
|
| 796 | + 'slug' => 'draft', |
|
| 797 | + 'label' => esc_html__('Draft', 'event_espresso'), |
|
| 798 | + 'count' => 0, |
|
| 799 | + 'bulk_action' => array( |
|
| 800 | + 'trash_events' => esc_html__('Move to Trash', 'event_espresso'), |
|
| 801 | + ), |
|
| 802 | + ), |
|
| 803 | + ); |
|
| 804 | + if (EE_Registry::instance()->CAP->current_user_can('ee_delete_events', 'espresso_events_trash_events')) { |
|
| 805 | + $this->_views['trash'] = array( |
|
| 806 | + 'slug' => 'trash', |
|
| 807 | + 'label' => esc_html__('Trash', 'event_espresso'), |
|
| 808 | + 'count' => 0, |
|
| 809 | + 'bulk_action' => array( |
|
| 810 | + 'restore_events' => esc_html__('Restore From Trash', 'event_espresso'), |
|
| 811 | + 'delete_events' => esc_html__('Delete Permanently', 'event_espresso'), |
|
| 812 | + ), |
|
| 813 | + ); |
|
| 814 | + } |
|
| 815 | + } |
|
| 816 | + |
|
| 817 | + |
|
| 818 | + /** |
|
| 819 | + * Provides the legend item array for the default list table view. |
|
| 820 | + * |
|
| 821 | + * @return array |
|
| 822 | + */ |
|
| 823 | + protected function _event_legend_items() |
|
| 824 | + { |
|
| 825 | + $items = array( |
|
| 826 | + 'view_details' => array( |
|
| 827 | + 'class' => 'dashicons dashicons-search', |
|
| 828 | + 'desc' => esc_html__('View Event', 'event_espresso'), |
|
| 829 | + ), |
|
| 830 | + 'edit_event' => array( |
|
| 831 | + 'class' => 'ee-icon ee-icon-calendar-edit', |
|
| 832 | + 'desc' => esc_html__('Edit Event Details', 'event_espresso'), |
|
| 833 | + ), |
|
| 834 | + 'view_attendees' => array( |
|
| 835 | + 'class' => 'dashicons dashicons-groups', |
|
| 836 | + 'desc' => esc_html__('View Registrations for Event', 'event_espresso'), |
|
| 837 | + ), |
|
| 838 | + ); |
|
| 839 | + $items = apply_filters('FHEE__Events_Admin_Page___event_legend_items__items', $items); |
|
| 840 | + $statuses = array( |
|
| 841 | + 'sold_out_status' => array( |
|
| 842 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::sold_out, |
|
| 843 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::sold_out, false, 'sentence'), |
|
| 844 | + ), |
|
| 845 | + 'active_status' => array( |
|
| 846 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::active, |
|
| 847 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::active, false, 'sentence'), |
|
| 848 | + ), |
|
| 849 | + 'upcoming_status' => array( |
|
| 850 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::upcoming, |
|
| 851 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::upcoming, false, 'sentence'), |
|
| 852 | + ), |
|
| 853 | + 'postponed_status' => array( |
|
| 854 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::postponed, |
|
| 855 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::postponed, false, 'sentence'), |
|
| 856 | + ), |
|
| 857 | + 'cancelled_status' => array( |
|
| 858 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::cancelled, |
|
| 859 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::cancelled, false, 'sentence'), |
|
| 860 | + ), |
|
| 861 | + 'expired_status' => array( |
|
| 862 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::expired, |
|
| 863 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::expired, false, 'sentence'), |
|
| 864 | + ), |
|
| 865 | + 'inactive_status' => array( |
|
| 866 | + 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::inactive, |
|
| 867 | + 'desc' => EEH_Template::pretty_status(EE_Datetime::inactive, false, 'sentence'), |
|
| 868 | + ), |
|
| 869 | + ); |
|
| 870 | + $statuses = apply_filters('FHEE__Events_Admin_Page__event_legend_items__statuses', $statuses); |
|
| 871 | + return array_merge($items, $statuses); |
|
| 872 | + } |
|
| 873 | + |
|
| 874 | + |
|
| 875 | + /** |
|
| 876 | + * @return EEM_Event |
|
| 877 | + */ |
|
| 878 | + private function _event_model() |
|
| 879 | + { |
|
| 880 | + if (! $this->_event_model instanceof EEM_Event) { |
|
| 881 | + $this->_event_model = EE_Registry::instance()->load_model('Event'); |
|
| 882 | + } |
|
| 883 | + return $this->_event_model; |
|
| 884 | + } |
|
| 885 | + |
|
| 886 | + |
|
| 887 | + /** |
|
| 888 | + * Adds extra buttons to the WP CPT permalink field row. |
|
| 889 | + * Method is called from parent and is hooked into the wp 'get_sample_permalink_html' filter. |
|
| 890 | + * |
|
| 891 | + * @param string $return the current html |
|
| 892 | + * @param int $id the post id for the page |
|
| 893 | + * @param string $new_title What the title is |
|
| 894 | + * @param string $new_slug what the slug is |
|
| 895 | + * @return string The new html string for the permalink area |
|
| 896 | + */ |
|
| 897 | + public function extra_permalink_field_buttons($return, $id, $new_title, $new_slug) |
|
| 898 | + { |
|
| 899 | + // make sure this is only when editing |
|
| 900 | + if (! empty($id)) { |
|
| 901 | + $post = get_post($id); |
|
| 902 | + $return .= '<a class="button button-small" onclick="prompt(\'Shortcode:\', jQuery(\'#shortcode\').val()); return false;" href="#" tabindex="-1">' |
|
| 903 | + . esc_html__('Shortcode', 'event_espresso') |
|
| 904 | + . '</a> '; |
|
| 905 | + $return .= '<input id="shortcode" type="hidden" value="[ESPRESSO_TICKET_SELECTOR event_id=' |
|
| 906 | + . $post->ID |
|
| 907 | + . ']">'; |
|
| 908 | + } |
|
| 909 | + return $return; |
|
| 910 | + } |
|
| 911 | + |
|
| 912 | + |
|
| 913 | + /** |
|
| 914 | + * _events_overview_list_table |
|
| 915 | + * This contains the logic for showing the events_overview list |
|
| 916 | + * |
|
| 917 | + * @access protected |
|
| 918 | + * @return void |
|
| 919 | + * @throws \EE_Error |
|
| 920 | + */ |
|
| 921 | + protected function _events_overview_list_table() |
|
| 922 | + { |
|
| 923 | + do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
| 924 | + $this->_template_args['after_list_table'] = ! empty($this->_template_args['after_list_table']) |
|
| 925 | + ? (array) $this->_template_args['after_list_table'] |
|
| 926 | + : array(); |
|
| 927 | + $this->_template_args['after_list_table']['view_event_list_button'] = EEH_HTML::br() |
|
| 928 | + . EEH_Template::get_button_or_link( |
|
| 929 | + get_post_type_archive_link('espresso_events'), |
|
| 930 | + esc_html__("View Event Archive Page", "event_espresso"), |
|
| 931 | + 'button' |
|
| 932 | + ); |
|
| 933 | + $this->_template_args['after_list_table']['legend'] = $this->_display_legend($this->_event_legend_items()); |
|
| 934 | + $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
| 935 | + 'create_new', |
|
| 936 | + 'add', |
|
| 937 | + array(), |
|
| 938 | + 'add-new-h2' |
|
| 939 | + ); |
|
| 940 | + $this->display_admin_list_table_page_with_no_sidebar(); |
|
| 941 | + } |
|
| 942 | + |
|
| 943 | + |
|
| 944 | + /** |
|
| 945 | + * this allows for extra misc actions in the default WP publish box |
|
| 946 | + * |
|
| 947 | + * @return void |
|
| 948 | + */ |
|
| 949 | + public function extra_misc_actions_publish_box() |
|
| 950 | + { |
|
| 951 | + $this->_generate_publish_box_extra_content(); |
|
| 952 | + } |
|
| 953 | + |
|
| 954 | + |
|
| 955 | + /** |
|
| 956 | + * This is hooked into the WordPress do_action('save_post') hook and runs after the custom post type has been |
|
| 957 | + * saved. |
|
| 958 | + * Typically you would use this to save any additional data. |
|
| 959 | + * Keep in mind also that "save_post" runs on EVERY post update to the database. |
|
| 960 | + * ALSO very important. When a post transitions from scheduled to published, |
|
| 961 | + * the save_post action is fired but you will NOT have any _POST data containing any extra info you may have from |
|
| 962 | + * other meta saves. So MAKE sure that you handle this accordingly. |
|
| 963 | + * |
|
| 964 | + * @access protected |
|
| 965 | + * @abstract |
|
| 966 | + * @param string $post_id The ID of the cpt that was saved (so you can link relationally) |
|
| 967 | + * @param object $post The post object of the cpt that was saved. |
|
| 968 | + * @return void |
|
| 969 | + * @throws \EE_Error |
|
| 970 | + */ |
|
| 971 | + protected function _insert_update_cpt_item($post_id, $post) |
|
| 972 | + { |
|
| 973 | + if ($post instanceof WP_Post && $post->post_type !== 'espresso_events') { |
|
| 974 | + // get out we're not processing an event save. |
|
| 975 | + return; |
|
| 976 | + } |
|
| 977 | + $event_values = array( |
|
| 978 | + 'EVT_display_desc' => ! empty($this->_req_data['display_desc']) ? 1 : 0, |
|
| 979 | + 'EVT_display_ticket_selector' => ! empty($this->_req_data['display_ticket_selector']) ? 1 : 0, |
|
| 980 | + 'EVT_additional_limit' => min( |
|
| 981 | + apply_filters('FHEE__EE_Events_Admin__insert_update_cpt_item__EVT_additional_limit_max', 255), |
|
| 982 | + ! empty($this->_req_data['additional_limit']) ? $this->_req_data['additional_limit'] : null |
|
| 983 | + ), |
|
| 984 | + 'EVT_default_registration_status' => ! empty($this->_req_data['EVT_default_registration_status']) |
|
| 985 | + ? $this->_req_data['EVT_default_registration_status'] |
|
| 986 | + : EE_Registry::instance()->CFG->registration->default_STS_ID, |
|
| 987 | + 'EVT_member_only' => ! empty($this->_req_data['member_only']) ? 1 : 0, |
|
| 988 | + 'EVT_allow_overflow' => ! empty($this->_req_data['EVT_allow_overflow']) ? 1 : 0, |
|
| 989 | + 'EVT_timezone_string' => ! empty($this->_req_data['timezone_string']) |
|
| 990 | + ? $this->_req_data['timezone_string'] : null, |
|
| 991 | + 'EVT_external_URL' => ! empty($this->_req_data['externalURL']) |
|
| 992 | + ? $this->_req_data['externalURL'] : null, |
|
| 993 | + 'EVT_phone' => ! empty($this->_req_data['event_phone']) |
|
| 994 | + ? $this->_req_data['event_phone'] : null, |
|
| 995 | + ); |
|
| 996 | + // update event |
|
| 997 | + $success = $this->_event_model()->update_by_ID($event_values, $post_id); |
|
| 998 | + // get event_object for other metaboxes... though it would seem to make sense to just use $this->_event_model()->get_one_by_ID( $post_id ).. i have to setup where conditions to override the filters in the model that filter out autodraft and inherit statuses so we GET the inherit id! |
|
| 999 | + $get_one_where = array( |
|
| 1000 | + $this->_event_model()->primary_key_name() => $post_id, |
|
| 1001 | + 'OR' => array( |
|
| 1002 | + 'status' => $post->post_status, |
|
| 1003 | + // if trying to "Publish" a sold out event, it's status will get switched back to "sold_out" in the db, |
|
| 1004 | + // but the returned object here has a status of "publish", so use the original post status as well |
|
| 1005 | + 'status*1' => $this->_req_data['original_post_status'], |
|
| 1006 | + ), |
|
| 1007 | + ); |
|
| 1008 | + $event = $this->_event_model()->get_one(array($get_one_where)); |
|
| 1009 | + // the following are default callbacks for event attachment updates that can be overridden by caffeinated functionality and/or addons. |
|
| 1010 | + $event_update_callbacks = apply_filters( |
|
| 1011 | + 'FHEE__Events_Admin_Page___insert_update_cpt_item__event_update_callbacks', |
|
| 1012 | + array( |
|
| 1013 | + array($this, '_default_venue_update'), |
|
| 1014 | + array($this, '_default_tickets_update'), |
|
| 1015 | + ) |
|
| 1016 | + ); |
|
| 1017 | + $att_success = true; |
|
| 1018 | + foreach ($event_update_callbacks as $e_callback) { |
|
| 1019 | + $_success = is_callable($e_callback) |
|
| 1020 | + ? call_user_func($e_callback, $event, $this->_req_data) |
|
| 1021 | + : false; |
|
| 1022 | + // if ANY of these updates fail then we want the appropriate global error message |
|
| 1023 | + $att_success = ! $att_success ? $att_success : $_success; |
|
| 1024 | + } |
|
| 1025 | + // any errors? |
|
| 1026 | + if ($success && false === $att_success) { |
|
| 1027 | + EE_Error::add_error( |
|
| 1028 | + esc_html__( |
|
| 1029 | + 'Event Details saved successfully but something went wrong with saving attachments.', |
|
| 1030 | + 'event_espresso' |
|
| 1031 | + ), |
|
| 1032 | + __FILE__, |
|
| 1033 | + __FUNCTION__, |
|
| 1034 | + __LINE__ |
|
| 1035 | + ); |
|
| 1036 | + } elseif ($success === false) { |
|
| 1037 | + EE_Error::add_error( |
|
| 1038 | + esc_html__('Event Details did not save successfully.', 'event_espresso'), |
|
| 1039 | + __FILE__, |
|
| 1040 | + __FUNCTION__, |
|
| 1041 | + __LINE__ |
|
| 1042 | + ); |
|
| 1043 | + } |
|
| 1044 | + } |
|
| 1045 | + |
|
| 1046 | + |
|
| 1047 | + /** |
|
| 1048 | + * @see parent::restore_item() |
|
| 1049 | + * @param int $post_id |
|
| 1050 | + * @param int $revision_id |
|
| 1051 | + */ |
|
| 1052 | + protected function _restore_cpt_item($post_id, $revision_id) |
|
| 1053 | + { |
|
| 1054 | + // copy existing event meta to new post |
|
| 1055 | + $post_evt = $this->_event_model()->get_one_by_ID($post_id); |
|
| 1056 | + if ($post_evt instanceof EE_Event) { |
|
| 1057 | + // meta revision restore |
|
| 1058 | + $post_evt->restore_revision($revision_id); |
|
| 1059 | + // related objs restore |
|
| 1060 | + $post_evt->restore_revision($revision_id, array('Venue', 'Datetime', 'Price')); |
|
| 1061 | + } |
|
| 1062 | + } |
|
| 1063 | + |
|
| 1064 | + |
|
| 1065 | + /** |
|
| 1066 | + * Attach the venue to the Event |
|
| 1067 | + * |
|
| 1068 | + * @param \EE_Event $evtobj Event Object to add the venue to |
|
| 1069 | + * @param array $data The request data from the form |
|
| 1070 | + * @return bool Success or fail. |
|
| 1071 | + */ |
|
| 1072 | + protected function _default_venue_update(\EE_Event $evtobj, $data) |
|
| 1073 | + { |
|
| 1074 | + require_once(EE_MODELS . 'EEM_Venue.model.php'); |
|
| 1075 | + $venue_model = EE_Registry::instance()->load_model('Venue'); |
|
| 1076 | + $rows_affected = null; |
|
| 1077 | + $venue_id = ! empty($data['venue_id']) ? $data['venue_id'] : null; |
|
| 1078 | + // very important. If we don't have a venue name... |
|
| 1079 | + // then we'll get out because not necessary to create empty venue |
|
| 1080 | + if (empty($data['venue_title'])) { |
|
| 1081 | + return false; |
|
| 1082 | + } |
|
| 1083 | + $venue_array = array( |
|
| 1084 | + 'VNU_wp_user' => $evtobj->get('EVT_wp_user'), |
|
| 1085 | + 'VNU_name' => ! empty($data['venue_title']) ? $data['venue_title'] : null, |
|
| 1086 | + 'VNU_desc' => ! empty($data['venue_description']) ? $data['venue_description'] : null, |
|
| 1087 | + 'VNU_identifier' => ! empty($data['venue_identifier']) ? $data['venue_identifier'] : null, |
|
| 1088 | + 'VNU_short_desc' => ! empty($data['venue_short_description']) ? $data['venue_short_description'] |
|
| 1089 | + : null, |
|
| 1090 | + 'VNU_address' => ! empty($data['address']) ? $data['address'] : null, |
|
| 1091 | + 'VNU_address2' => ! empty($data['address2']) ? $data['address2'] : null, |
|
| 1092 | + 'VNU_city' => ! empty($data['city']) ? $data['city'] : null, |
|
| 1093 | + 'STA_ID' => ! empty($data['state']) ? $data['state'] : null, |
|
| 1094 | + 'CNT_ISO' => ! empty($data['countries']) ? $data['countries'] : null, |
|
| 1095 | + 'VNU_zip' => ! empty($data['zip']) ? $data['zip'] : null, |
|
| 1096 | + 'VNU_phone' => ! empty($data['venue_phone']) ? $data['venue_phone'] : null, |
|
| 1097 | + 'VNU_capacity' => ! empty($data['venue_capacity']) ? $data['venue_capacity'] : null, |
|
| 1098 | + 'VNU_url' => ! empty($data['venue_url']) ? $data['venue_url'] : null, |
|
| 1099 | + 'VNU_virtual_phone' => ! empty($data['virtual_phone']) ? $data['virtual_phone'] : null, |
|
| 1100 | + 'VNU_virtual_url' => ! empty($data['virtual_url']) ? $data['virtual_url'] : null, |
|
| 1101 | + 'VNU_enable_for_gmap' => isset($data['enable_for_gmap']) ? 1 : 0, |
|
| 1102 | + 'status' => 'publish', |
|
| 1103 | + ); |
|
| 1104 | + // if we've got the venue_id then we're just updating the existing venue so let's do that and then get out. |
|
| 1105 | + if (! empty($venue_id)) { |
|
| 1106 | + $update_where = array($venue_model->primary_key_name() => $venue_id); |
|
| 1107 | + $rows_affected = $venue_model->update($venue_array, array($update_where)); |
|
| 1108 | + // we've gotta make sure that the venue is always attached to a revision.. add_relation_to should take care of making sure that the relation is already present. |
|
| 1109 | + $evtobj->_add_relation_to($venue_id, 'Venue'); |
|
| 1110 | + return $rows_affected > 0 ? true : false; |
|
| 1111 | + } else { |
|
| 1112 | + // we insert the venue |
|
| 1113 | + $venue_id = $venue_model->insert($venue_array); |
|
| 1114 | + $evtobj->_add_relation_to($venue_id, 'Venue'); |
|
| 1115 | + return ! empty($venue_id) ? true : false; |
|
| 1116 | + } |
|
| 1117 | + // when we have the ancestor come in it's already been handled by the revision save. |
|
| 1118 | + } |
|
| 1119 | + |
|
| 1120 | + |
|
| 1121 | + /** |
|
| 1122 | + * Handles saving everything related to Tickets (datetimes, tickets, prices) |
|
| 1123 | + * |
|
| 1124 | + * @param EE_Event $evtobj The Event object we're attaching data to |
|
| 1125 | + * @param array $data The request data from the form |
|
| 1126 | + * @return array |
|
| 1127 | + */ |
|
| 1128 | + protected function _default_tickets_update(EE_Event $evtobj, $data) |
|
| 1129 | + { |
|
| 1130 | + $success = true; |
|
| 1131 | + $saved_dtt = null; |
|
| 1132 | + $saved_tickets = array(); |
|
| 1133 | + $incoming_date_formats = array('Y-m-d', 'h:i a'); |
|
| 1134 | + foreach ($data['edit_event_datetimes'] as $row => $dtt) { |
|
| 1135 | + // trim all values to ensure any excess whitespace is removed. |
|
| 1136 | + $dtt = array_map('trim', $dtt); |
|
| 1137 | + $dtt['DTT_EVT_end'] = isset($dtt['DTT_EVT_end']) && ! empty($dtt['DTT_EVT_end']) ? $dtt['DTT_EVT_end'] |
|
| 1138 | + : $dtt['DTT_EVT_start']; |
|
| 1139 | + $datetime_values = array( |
|
| 1140 | + 'DTT_ID' => ! empty($dtt['DTT_ID']) ? $dtt['DTT_ID'] : null, |
|
| 1141 | + 'DTT_EVT_start' => $dtt['DTT_EVT_start'], |
|
| 1142 | + 'DTT_EVT_end' => $dtt['DTT_EVT_end'], |
|
| 1143 | + 'DTT_reg_limit' => empty($dtt['DTT_reg_limit']) ? EE_INF : $dtt['DTT_reg_limit'], |
|
| 1144 | + 'DTT_order' => $row, |
|
| 1145 | + ); |
|
| 1146 | + // if we have an id then let's get existing object first and then set the new values. Otherwise we instantiate a new object for save. |
|
| 1147 | + if (! empty($dtt['DTT_ID'])) { |
|
| 1148 | + $DTM = EE_Registry::instance() |
|
| 1149 | + ->load_model('Datetime', array($evtobj->get_timezone())) |
|
| 1150 | + ->get_one_by_ID($dtt['DTT_ID']); |
|
| 1151 | + $DTM->set_date_format($incoming_date_formats[0]); |
|
| 1152 | + $DTM->set_time_format($incoming_date_formats[1]); |
|
| 1153 | + foreach ($datetime_values as $field => $value) { |
|
| 1154 | + $DTM->set($field, $value); |
|
| 1155 | + } |
|
| 1156 | + // make sure the $dtt_id here is saved just in case after the add_relation_to() the autosave replaces it. We need to do this so we dont' TRASH the parent DTT. |
|
| 1157 | + $saved_dtts[ $DTM->ID() ] = $DTM; |
|
| 1158 | + } else { |
|
| 1159 | + $DTM = EE_Registry::instance()->load_class( |
|
| 1160 | + 'Datetime', |
|
| 1161 | + array($datetime_values, $evtobj->get_timezone(), $incoming_date_formats), |
|
| 1162 | + false, |
|
| 1163 | + false |
|
| 1164 | + ); |
|
| 1165 | + foreach ($datetime_values as $field => $value) { |
|
| 1166 | + $DTM->set($field, $value); |
|
| 1167 | + } |
|
| 1168 | + } |
|
| 1169 | + $DTM->save(); |
|
| 1170 | + $DTT = $evtobj->_add_relation_to($DTM, 'Datetime'); |
|
| 1171 | + // load DTT helper |
|
| 1172 | + // before going any further make sure our dates are setup correctly so that the end date is always equal or greater than the start date. |
|
| 1173 | + if ($DTT->get_raw('DTT_EVT_start') > $DTT->get_raw('DTT_EVT_end')) { |
|
| 1174 | + $DTT->set('DTT_EVT_end', $DTT->get('DTT_EVT_start')); |
|
| 1175 | + $DTT = EEH_DTT_Helper::date_time_add($DTT, 'DTT_EVT_end', 'days'); |
|
| 1176 | + $DTT->save(); |
|
| 1177 | + } |
|
| 1178 | + // now we got to make sure we add the new DTT_ID to the $saved_dtts array because it is possible there was a new one created for the autosave. |
|
| 1179 | + $saved_dtt = $DTT; |
|
| 1180 | + $success = ! $success ? $success : $DTT; |
|
| 1181 | + // if ANY of these updates fail then we want the appropriate global error message. |
|
| 1182 | + // //todo this is actually sucky we need a better error message but this is what it is for now. |
|
| 1183 | + } |
|
| 1184 | + // no dtts get deleted so we don't do any of that logic here. |
|
| 1185 | + // update tickets next |
|
| 1186 | + $old_tickets = isset($data['ticket_IDs']) ? explode(',', $data['ticket_IDs']) : array(); |
|
| 1187 | + foreach ($data['edit_tickets'] as $row => $tkt) { |
|
| 1188 | + $incoming_date_formats = array('Y-m-d', 'h:i a'); |
|
| 1189 | + $update_prices = false; |
|
| 1190 | + $ticket_price = isset($data['edit_prices'][ $row ][1]['PRC_amount']) |
|
| 1191 | + ? $data['edit_prices'][ $row ][1]['PRC_amount'] : 0; |
|
| 1192 | + // trim inputs to ensure any excess whitespace is removed. |
|
| 1193 | + $tkt = array_map('trim', $tkt); |
|
| 1194 | + if (empty($tkt['TKT_start_date'])) { |
|
| 1195 | + // let's use now in the set timezone. |
|
| 1196 | + $now = new DateTime('now', new DateTimeZone($evtobj->get_timezone())); |
|
| 1197 | + $tkt['TKT_start_date'] = $now->format($incoming_date_formats[0] . ' ' . $incoming_date_formats[1]); |
|
| 1198 | + } |
|
| 1199 | + if (empty($tkt['TKT_end_date'])) { |
|
| 1200 | + // use the start date of the first datetime |
|
| 1201 | + $dtt = $evtobj->first_datetime(); |
|
| 1202 | + $tkt['TKT_end_date'] = $dtt->start_date_and_time( |
|
| 1203 | + $incoming_date_formats[0], |
|
| 1204 | + $incoming_date_formats[1] |
|
| 1205 | + ); |
|
| 1206 | + } |
|
| 1207 | + $TKT_values = array( |
|
| 1208 | + 'TKT_ID' => ! empty($tkt['TKT_ID']) ? $tkt['TKT_ID'] : null, |
|
| 1209 | + 'TTM_ID' => ! empty($tkt['TTM_ID']) ? $tkt['TTM_ID'] : 0, |
|
| 1210 | + 'TKT_name' => ! empty($tkt['TKT_name']) ? $tkt['TKT_name'] : '', |
|
| 1211 | + 'TKT_description' => ! empty($tkt['TKT_description']) ? $tkt['TKT_description'] : '', |
|
| 1212 | + 'TKT_start_date' => $tkt['TKT_start_date'], |
|
| 1213 | + 'TKT_end_date' => $tkt['TKT_end_date'], |
|
| 1214 | + 'TKT_qty' => ! isset($tkt['TKT_qty']) || $tkt['TKT_qty'] === '' ? EE_INF : $tkt['TKT_qty'], |
|
| 1215 | + 'TKT_uses' => ! isset($tkt['TKT_uses']) || $tkt['TKT_uses'] === '' ? EE_INF : $tkt['TKT_uses'], |
|
| 1216 | + 'TKT_min' => empty($tkt['TKT_min']) ? 0 : $tkt['TKT_min'], |
|
| 1217 | + 'TKT_max' => empty($tkt['TKT_max']) ? EE_INF : $tkt['TKT_max'], |
|
| 1218 | + 'TKT_row' => $row, |
|
| 1219 | + 'TKT_order' => isset($tkt['TKT_order']) ? $tkt['TKT_order'] : $row, |
|
| 1220 | + 'TKT_price' => $ticket_price, |
|
| 1221 | + ); |
|
| 1222 | + // if this is a default TKT, then we need to set the TKT_ID to 0 and update accordingly, which means in turn that the prices will become new prices as well. |
|
| 1223 | + if (isset($tkt['TKT_is_default']) && $tkt['TKT_is_default']) { |
|
| 1224 | + $TKT_values['TKT_ID'] = 0; |
|
| 1225 | + $TKT_values['TKT_is_default'] = 0; |
|
| 1226 | + $TKT_values['TKT_price'] = $ticket_price; |
|
| 1227 | + $update_prices = true; |
|
| 1228 | + } |
|
| 1229 | + // if we have a TKT_ID then we need to get that existing TKT_obj and update it |
|
| 1230 | + // we actually do our saves a head of doing any add_relations to because its entirely possible that this ticket didn't removed or added to any datetime in the session but DID have it's items modified. |
|
| 1231 | + // keep in mind that if the TKT has been sold (and we have changed pricing information), then we won't be updating the tkt but instead a new tkt will be created and the old one archived. |
|
| 1232 | + if (! empty($tkt['TKT_ID'])) { |
|
| 1233 | + $TKT = EE_Registry::instance() |
|
| 1234 | + ->load_model('Ticket', array($evtobj->get_timezone())) |
|
| 1235 | + ->get_one_by_ID($tkt['TKT_ID']); |
|
| 1236 | + if ($TKT instanceof EE_Ticket) { |
|
| 1237 | + $ticket_sold = $TKT->count_related( |
|
| 1238 | + 'Registration', |
|
| 1239 | + array( |
|
| 1240 | + array( |
|
| 1241 | + 'STS_ID' => array( |
|
| 1242 | + 'NOT IN', |
|
| 1243 | + array(EEM_Registration::status_id_incomplete), |
|
| 1244 | + ), |
|
| 1245 | + ), |
|
| 1246 | + ) |
|
| 1247 | + ) > 0 ? true : false; |
|
| 1248 | + // let's just check the total price for the existing ticket and determine if it matches the new total price. if they are different then we create a new ticket (if tkts sold) if they aren't different then we go ahead and modify existing ticket. |
|
| 1249 | + $create_new_TKT = $ticket_sold && $ticket_price != $TKT->get('TKT_price') |
|
| 1250 | + && ! $TKT->get('TKT_deleted'); |
|
| 1251 | + $TKT->set_date_format($incoming_date_formats[0]); |
|
| 1252 | + $TKT->set_time_format($incoming_date_formats[1]); |
|
| 1253 | + // set new values |
|
| 1254 | + foreach ($TKT_values as $field => $value) { |
|
| 1255 | + if ($field == 'TKT_qty') { |
|
| 1256 | + $TKT->set_qty($value); |
|
| 1257 | + } else { |
|
| 1258 | + $TKT->set($field, $value); |
|
| 1259 | + } |
|
| 1260 | + } |
|
| 1261 | + // if $create_new_TKT is false then we can safely update the existing ticket. Otherwise we have to create a new ticket. |
|
| 1262 | + if ($create_new_TKT) { |
|
| 1263 | + // archive the old ticket first |
|
| 1264 | + $TKT->set('TKT_deleted', 1); |
|
| 1265 | + $TKT->save(); |
|
| 1266 | + // make sure this ticket is still recorded in our saved_tkts so we don't run it through the regular trash routine. |
|
| 1267 | + $saved_tickets[ $TKT->ID() ] = $TKT; |
|
| 1268 | + // create new ticket that's a copy of the existing except a new id of course (and not archived) AND has the new TKT_price associated with it. |
|
| 1269 | + $TKT = clone $TKT; |
|
| 1270 | + $TKT->set('TKT_ID', 0); |
|
| 1271 | + $TKT->set('TKT_deleted', 0); |
|
| 1272 | + $TKT->set('TKT_price', $ticket_price); |
|
| 1273 | + $TKT->set('TKT_sold', 0); |
|
| 1274 | + // now we need to make sure that $new prices are created as well and attached to new ticket. |
|
| 1275 | + $update_prices = true; |
|
| 1276 | + } |
|
| 1277 | + // make sure price is set if it hasn't been already |
|
| 1278 | + $TKT->set('TKT_price', $ticket_price); |
|
| 1279 | + } |
|
| 1280 | + } else { |
|
| 1281 | + // no TKT_id so a new TKT |
|
| 1282 | + $TKT_values['TKT_price'] = $ticket_price; |
|
| 1283 | + $TKT = EE_Registry::instance()->load_class('Ticket', array($TKT_values), false, false); |
|
| 1284 | + if ($TKT instanceof EE_Ticket) { |
|
| 1285 | + // need to reset values to properly account for the date formats |
|
| 1286 | + $TKT->set_date_format($incoming_date_formats[0]); |
|
| 1287 | + $TKT->set_time_format($incoming_date_formats[1]); |
|
| 1288 | + $TKT->set_timezone($evtobj->get_timezone()); |
|
| 1289 | + // set new values |
|
| 1290 | + foreach ($TKT_values as $field => $value) { |
|
| 1291 | + if ($field == 'TKT_qty') { |
|
| 1292 | + $TKT->set_qty($value); |
|
| 1293 | + } else { |
|
| 1294 | + $TKT->set($field, $value); |
|
| 1295 | + } |
|
| 1296 | + } |
|
| 1297 | + $update_prices = true; |
|
| 1298 | + } |
|
| 1299 | + } |
|
| 1300 | + // cap ticket qty by datetime reg limits |
|
| 1301 | + $TKT->set_qty(min($TKT->qty(), $TKT->qty('reg_limit'))); |
|
| 1302 | + // update ticket. |
|
| 1303 | + $TKT->save(); |
|
| 1304 | + // before going any further make sure our dates are setup correctly so that the end date is always equal or greater than the start date. |
|
| 1305 | + if ($TKT->get_raw('TKT_start_date') > $TKT->get_raw('TKT_end_date')) { |
|
| 1306 | + $TKT->set('TKT_end_date', $TKT->get('TKT_start_date')); |
|
| 1307 | + $TKT = EEH_DTT_Helper::date_time_add($TKT, 'TKT_end_date', 'days'); |
|
| 1308 | + $TKT->save(); |
|
| 1309 | + } |
|
| 1310 | + // initially let's add the ticket to the dtt |
|
| 1311 | + $saved_dtt->_add_relation_to($TKT, 'Ticket'); |
|
| 1312 | + $saved_tickets[ $TKT->ID() ] = $TKT; |
|
| 1313 | + // add prices to ticket |
|
| 1314 | + $this->_add_prices_to_ticket($data['edit_prices'][ $row ], $TKT, $update_prices); |
|
| 1315 | + } |
|
| 1316 | + // however now we need to handle permanently deleting tickets via the ui. Keep in mind that the ui does not allow deleting/archiving tickets that have ticket sold. However, it does allow for deleting tickets that have no tickets sold, in which case we want to get rid of permanently because there is no need to save in db. |
|
| 1317 | + $old_tickets = isset($old_tickets[0]) && $old_tickets[0] == '' ? array() : $old_tickets; |
|
| 1318 | + $tickets_removed = array_diff($old_tickets, array_keys($saved_tickets)); |
|
| 1319 | + foreach ($tickets_removed as $id) { |
|
| 1320 | + $id = absint($id); |
|
| 1321 | + // get the ticket for this id |
|
| 1322 | + $tkt_to_remove = EE_Registry::instance()->load_model('Ticket')->get_one_by_ID($id); |
|
| 1323 | + // need to get all the related datetimes on this ticket and remove from every single one of them (remember this process can ONLY kick off if there are NO tkts_sold) |
|
| 1324 | + $dtts = $tkt_to_remove->get_many_related('Datetime'); |
|
| 1325 | + foreach ($dtts as $dtt) { |
|
| 1326 | + $tkt_to_remove->_remove_relation_to($dtt, 'Datetime'); |
|
| 1327 | + } |
|
| 1328 | + // need to do the same for prices (except these prices can also be deleted because again, tickets can only be trashed if they don't have any TKTs sold (otherwise they are just archived)) |
|
| 1329 | + $tkt_to_remove->delete_related_permanently('Price'); |
|
| 1330 | + // finally let's delete this ticket (which should not be blocked at this point b/c we've removed all our relationships) |
|
| 1331 | + $tkt_to_remove->delete_permanently(); |
|
| 1332 | + } |
|
| 1333 | + return array($saved_dtt, $saved_tickets); |
|
| 1334 | + } |
|
| 1335 | + |
|
| 1336 | + |
|
| 1337 | + /** |
|
| 1338 | + * This attaches a list of given prices to a ticket. |
|
| 1339 | + * Note we dont' have to worry about ever removing relationships (or archiving prices) because if there is a change |
|
| 1340 | + * in price information on a ticket, a new ticket is created anyways so the archived ticket will retain the old |
|
| 1341 | + * price info and prices are automatically "archived" via the ticket. |
|
| 1342 | + * |
|
| 1343 | + * @access private |
|
| 1344 | + * @param array $prices Array of prices from the form. |
|
| 1345 | + * @param EE_Ticket $ticket EE_Ticket object that prices are being attached to. |
|
| 1346 | + * @param bool $new_prices Whether attach existing incoming prices or create new ones. |
|
| 1347 | + * @return void |
|
| 1348 | + */ |
|
| 1349 | + private function _add_prices_to_ticket($prices, EE_Ticket $ticket, $new_prices = false) |
|
| 1350 | + { |
|
| 1351 | + foreach ($prices as $row => $prc) { |
|
| 1352 | + $PRC_values = array( |
|
| 1353 | + 'PRC_ID' => ! empty($prc['PRC_ID']) ? $prc['PRC_ID'] : null, |
|
| 1354 | + 'PRT_ID' => ! empty($prc['PRT_ID']) ? $prc['PRT_ID'] : null, |
|
| 1355 | + 'PRC_amount' => ! empty($prc['PRC_amount']) ? $prc['PRC_amount'] : 0, |
|
| 1356 | + 'PRC_name' => ! empty($prc['PRC_name']) ? $prc['PRC_name'] : '', |
|
| 1357 | + 'PRC_desc' => ! empty($prc['PRC_desc']) ? $prc['PRC_desc'] : '', |
|
| 1358 | + 'PRC_is_default' => 0, // make sure prices are NOT set as default from this context |
|
| 1359 | + 'PRC_order' => $row, |
|
| 1360 | + ); |
|
| 1361 | + if ($new_prices || empty($PRC_values['PRC_ID'])) { |
|
| 1362 | + $PRC_values['PRC_ID'] = 0; |
|
| 1363 | + $PRC = EE_Registry::instance()->load_class('Price', array($PRC_values), false, false); |
|
| 1364 | + } else { |
|
| 1365 | + $PRC = EE_Registry::instance()->load_model('Price')->get_one_by_ID($prc['PRC_ID']); |
|
| 1366 | + // update this price with new values |
|
| 1367 | + foreach ($PRC_values as $field => $newprc) { |
|
| 1368 | + $PRC->set($field, $newprc); |
|
| 1369 | + } |
|
| 1370 | + $PRC->save(); |
|
| 1371 | + } |
|
| 1372 | + $ticket->_add_relation_to($PRC, 'Price'); |
|
| 1373 | + } |
|
| 1374 | + } |
|
| 1375 | + |
|
| 1376 | + |
|
| 1377 | + /** |
|
| 1378 | + * Add in our autosave ajax handlers |
|
| 1379 | + * |
|
| 1380 | + */ |
|
| 1381 | + protected function _ee_autosave_create_new() |
|
| 1382 | + { |
|
| 1383 | + } |
|
| 1384 | + |
|
| 1385 | + |
|
| 1386 | + /** |
|
| 1387 | + * More autosave handlers. |
|
| 1388 | + */ |
|
| 1389 | + protected function _ee_autosave_edit() |
|
| 1390 | + { |
|
| 1391 | + return; // TEMPORARILY EXITING CAUSE THIS IS A TODO |
|
| 1392 | + } |
|
| 1393 | + |
|
| 1394 | + |
|
| 1395 | + /** |
|
| 1396 | + * _generate_publish_box_extra_content |
|
| 1397 | + */ |
|
| 1398 | + private function _generate_publish_box_extra_content() |
|
| 1399 | + { |
|
| 1400 | + // load formatter helper |
|
| 1401 | + // args for getting related registrations |
|
| 1402 | + $approved_query_args = array( |
|
| 1403 | + array( |
|
| 1404 | + 'REG_deleted' => 0, |
|
| 1405 | + 'STS_ID' => EEM_Registration::status_id_approved, |
|
| 1406 | + ), |
|
| 1407 | + ); |
|
| 1408 | + $not_approved_query_args = array( |
|
| 1409 | + array( |
|
| 1410 | + 'REG_deleted' => 0, |
|
| 1411 | + 'STS_ID' => EEM_Registration::status_id_not_approved, |
|
| 1412 | + ), |
|
| 1413 | + ); |
|
| 1414 | + $pending_payment_query_args = array( |
|
| 1415 | + array( |
|
| 1416 | + 'REG_deleted' => 0, |
|
| 1417 | + 'STS_ID' => EEM_Registration::status_id_pending_payment, |
|
| 1418 | + ), |
|
| 1419 | + ); |
|
| 1420 | + // publish box |
|
| 1421 | + $publish_box_extra_args = array( |
|
| 1422 | + 'view_approved_reg_url' => add_query_arg( |
|
| 1423 | + array( |
|
| 1424 | + 'action' => 'default', |
|
| 1425 | + 'event_id' => $this->_cpt_model_obj->ID(), |
|
| 1426 | + '_reg_status' => EEM_Registration::status_id_approved, |
|
| 1427 | + ), |
|
| 1428 | + REG_ADMIN_URL |
|
| 1429 | + ), |
|
| 1430 | + 'view_not_approved_reg_url' => add_query_arg( |
|
| 1431 | + array( |
|
| 1432 | + 'action' => 'default', |
|
| 1433 | + 'event_id' => $this->_cpt_model_obj->ID(), |
|
| 1434 | + '_reg_status' => EEM_Registration::status_id_not_approved, |
|
| 1435 | + ), |
|
| 1436 | + REG_ADMIN_URL |
|
| 1437 | + ), |
|
| 1438 | + 'view_pending_payment_reg_url' => add_query_arg( |
|
| 1439 | + array( |
|
| 1440 | + 'action' => 'default', |
|
| 1441 | + 'event_id' => $this->_cpt_model_obj->ID(), |
|
| 1442 | + '_reg_status' => EEM_Registration::status_id_pending_payment, |
|
| 1443 | + ), |
|
| 1444 | + REG_ADMIN_URL |
|
| 1445 | + ), |
|
| 1446 | + 'approved_regs' => $this->_cpt_model_obj->count_related( |
|
| 1447 | + 'Registration', |
|
| 1448 | + $approved_query_args |
|
| 1449 | + ), |
|
| 1450 | + 'not_approved_regs' => $this->_cpt_model_obj->count_related( |
|
| 1451 | + 'Registration', |
|
| 1452 | + $not_approved_query_args |
|
| 1453 | + ), |
|
| 1454 | + 'pending_payment_regs' => $this->_cpt_model_obj->count_related( |
|
| 1455 | + 'Registration', |
|
| 1456 | + $pending_payment_query_args |
|
| 1457 | + ), |
|
| 1458 | + 'misc_pub_section_class' => apply_filters( |
|
| 1459 | + 'FHEE_Events_Admin_Page___generate_publish_box_extra_content__misc_pub_section_class', |
|
| 1460 | + 'misc-pub-section' |
|
| 1461 | + ), |
|
| 1462 | + ); |
|
| 1463 | + ob_start(); |
|
| 1464 | + do_action( |
|
| 1465 | + 'AHEE__Events_Admin_Page___generate_publish_box_extra_content__event_editor_overview_add', |
|
| 1466 | + $this->_cpt_model_obj |
|
| 1467 | + ); |
|
| 1468 | + $publish_box_extra_args['event_editor_overview_add'] = ob_get_clean(); |
|
| 1469 | + // load template |
|
| 1470 | + EEH_Template::display_template( |
|
| 1471 | + EVENTS_TEMPLATE_PATH . 'event_publish_box_extras.template.php', |
|
| 1472 | + $publish_box_extra_args |
|
| 1473 | + ); |
|
| 1474 | + } |
|
| 1475 | + |
|
| 1476 | + |
|
| 1477 | + /** |
|
| 1478 | + * @return EE_Event |
|
| 1479 | + */ |
|
| 1480 | + public function get_event_object() |
|
| 1481 | + { |
|
| 1482 | + return $this->_cpt_model_obj; |
|
| 1483 | + } |
|
| 1484 | + |
|
| 1485 | + |
|
| 1486 | + |
|
| 1487 | + |
|
| 1488 | + /** METABOXES * */ |
|
| 1489 | + /** |
|
| 1490 | + * _register_event_editor_meta_boxes |
|
| 1491 | + * add all metaboxes related to the event_editor |
|
| 1492 | + * |
|
| 1493 | + * @return void |
|
| 1494 | + */ |
|
| 1495 | + protected function _register_event_editor_meta_boxes() |
|
| 1496 | + { |
|
| 1497 | + $this->verify_cpt_object(); |
|
| 1498 | + add_meta_box( |
|
| 1499 | + 'espresso_event_editor_tickets', |
|
| 1500 | + esc_html__('Event Datetime & Ticket', 'event_espresso'), |
|
| 1501 | + array($this, 'ticket_metabox'), |
|
| 1502 | + $this->page_slug, |
|
| 1503 | + 'normal', |
|
| 1504 | + 'high' |
|
| 1505 | + ); |
|
| 1506 | + add_meta_box( |
|
| 1507 | + 'espresso_event_editor_event_options', |
|
| 1508 | + esc_html__('Event Registration Options', 'event_espresso'), |
|
| 1509 | + array($this, 'registration_options_meta_box'), |
|
| 1510 | + $this->page_slug, |
|
| 1511 | + 'side', |
|
| 1512 | + 'default' |
|
| 1513 | + ); |
|
| 1514 | + // NOTE: if you're looking for other metaboxes in here, |
|
| 1515 | + // where a metabox has a related management page in the admin |
|
| 1516 | + // you will find it setup in the related management page's "_Hooks" file. |
|
| 1517 | + // i.e. messages metabox is found in "espresso_events_Messages_Hooks.class.php". |
|
| 1518 | + } |
|
| 1519 | + |
|
| 1520 | + |
|
| 1521 | + /** |
|
| 1522 | + * @throws DomainException |
|
| 1523 | + * @throws EE_Error |
|
| 1524 | + */ |
|
| 1525 | + public function ticket_metabox() |
|
| 1526 | + { |
|
| 1527 | + $existing_datetime_ids = $existing_ticket_ids = array(); |
|
| 1528 | + // defaults for template args |
|
| 1529 | + $template_args = array( |
|
| 1530 | + 'existing_datetime_ids' => '', |
|
| 1531 | + 'event_datetime_help_link' => '', |
|
| 1532 | + 'ticket_options_help_link' => '', |
|
| 1533 | + 'time' => null, |
|
| 1534 | + 'ticket_rows' => '', |
|
| 1535 | + 'existing_ticket_ids' => '', |
|
| 1536 | + 'total_ticket_rows' => 1, |
|
| 1537 | + 'ticket_js_structure' => '', |
|
| 1538 | + 'trash_icon' => 'ee-lock-icon', |
|
| 1539 | + 'disabled' => '', |
|
| 1540 | + ); |
|
| 1541 | + $event_id = is_object($this->_cpt_model_obj) ? $this->_cpt_model_obj->ID() : null; |
|
| 1542 | + do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
| 1543 | + /** |
|
| 1544 | + * 1. Start with retrieving Datetimes |
|
| 1545 | + * 2. Fore each datetime get related tickets |
|
| 1546 | + * 3. For each ticket get related prices |
|
| 1547 | + */ |
|
| 1548 | + $times = EE_Registry::instance()->load_model('Datetime')->get_all_event_dates($event_id); |
|
| 1549 | + /** @type EE_Datetime $first_datetime */ |
|
| 1550 | + $first_datetime = reset($times); |
|
| 1551 | + // do we get related tickets? |
|
| 1552 | + if ($first_datetime instanceof EE_Datetime |
|
| 1553 | + && $first_datetime->ID() !== 0 |
|
| 1554 | + ) { |
|
| 1555 | + $existing_datetime_ids[] = $first_datetime->get('DTT_ID'); |
|
| 1556 | + $template_args['time'] = $first_datetime; |
|
| 1557 | + $related_tickets = $first_datetime->tickets( |
|
| 1558 | + array( |
|
| 1559 | + array('OR' => array('TKT_deleted' => 1, 'TKT_deleted*' => 0)), |
|
| 1560 | + 'default_where_conditions' => 'none', |
|
| 1561 | + ) |
|
| 1562 | + ); |
|
| 1563 | + if (! empty($related_tickets)) { |
|
| 1564 | + $template_args['total_ticket_rows'] = count($related_tickets); |
|
| 1565 | + $row = 0; |
|
| 1566 | + foreach ($related_tickets as $ticket) { |
|
| 1567 | + $existing_ticket_ids[] = $ticket->get('TKT_ID'); |
|
| 1568 | + $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket, false, $row); |
|
| 1569 | + $row++; |
|
| 1570 | + } |
|
| 1571 | + } else { |
|
| 1572 | + $template_args['total_ticket_rows'] = 1; |
|
| 1573 | + /** @type EE_Ticket $ticket */ |
|
| 1574 | + $ticket = EE_Registry::instance()->load_model('Ticket')->create_default_object(); |
|
| 1575 | + $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket); |
|
| 1576 | + } |
|
| 1577 | + } else { |
|
| 1578 | + $template_args['time'] = $times[0]; |
|
| 1579 | + /** @type EE_Ticket $ticket */ |
|
| 1580 | + $ticket = EE_Registry::instance()->load_model('Ticket')->get_all_default_tickets(); |
|
| 1581 | + $template_args['ticket_rows'] .= $this->_get_ticket_row($ticket[1]); |
|
| 1582 | + // NOTE: we're just sending the first default row |
|
| 1583 | + // (decaf can't manage default tickets so this should be sufficient); |
|
| 1584 | + } |
|
| 1585 | + $template_args['event_datetime_help_link'] = $this->_get_help_tab_link( |
|
| 1586 | + 'event_editor_event_datetimes_help_tab' |
|
| 1587 | + ); |
|
| 1588 | + $template_args['ticket_options_help_link'] = $this->_get_help_tab_link('ticket_options_info'); |
|
| 1589 | + $template_args['existing_datetime_ids'] = implode(',', $existing_datetime_ids); |
|
| 1590 | + $template_args['existing_ticket_ids'] = implode(',', $existing_ticket_ids); |
|
| 1591 | + $template_args['ticket_js_structure'] = $this->_get_ticket_row( |
|
| 1592 | + EE_Registry::instance()->load_model('Ticket')->create_default_object(), |
|
| 1593 | + true |
|
| 1594 | + ); |
|
| 1595 | + $template = apply_filters( |
|
| 1596 | + 'FHEE__Events_Admin_Page__ticket_metabox__template', |
|
| 1597 | + EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_main.template.php' |
|
| 1598 | + ); |
|
| 1599 | + EEH_Template::display_template($template, $template_args); |
|
| 1600 | + } |
|
| 1601 | + |
|
| 1602 | + |
|
| 1603 | + /** |
|
| 1604 | + * Setup an individual ticket form for the decaf event editor page |
|
| 1605 | + * |
|
| 1606 | + * @access private |
|
| 1607 | + * @param EE_Ticket $ticket the ticket object |
|
| 1608 | + * @param boolean $skeleton whether we're generating a skeleton for js manipulation |
|
| 1609 | + * @param int $row |
|
| 1610 | + * @return string generated html for the ticket row. |
|
| 1611 | + */ |
|
| 1612 | + private function _get_ticket_row($ticket, $skeleton = false, $row = 0) |
|
| 1613 | + { |
|
| 1614 | + $template_args = array( |
|
| 1615 | + 'tkt_status_class' => ' tkt-status-' . $ticket->ticket_status(), |
|
| 1616 | + 'tkt_archive_class' => $ticket->ticket_status() === EE_Ticket::archived && ! $skeleton ? ' tkt-archived' |
|
| 1617 | + : '', |
|
| 1618 | + 'ticketrow' => $skeleton ? 'TICKETNUM' : $row, |
|
| 1619 | + 'TKT_ID' => $ticket->get('TKT_ID'), |
|
| 1620 | + 'TKT_name' => $ticket->get('TKT_name'), |
|
| 1621 | + 'TKT_start_date' => $skeleton ? '' : $ticket->get_date('TKT_start_date', 'Y-m-d h:i a'), |
|
| 1622 | + 'TKT_end_date' => $skeleton ? '' : $ticket->get_date('TKT_end_date', 'Y-m-d h:i a'), |
|
| 1623 | + 'TKT_is_default' => $ticket->get('TKT_is_default'), |
|
| 1624 | + 'TKT_qty' => $ticket->get_pretty('TKT_qty', 'input'), |
|
| 1625 | + 'edit_ticketrow_name' => $skeleton ? 'TICKETNAMEATTR' : 'edit_tickets', |
|
| 1626 | + 'TKT_sold' => $skeleton ? 0 : $ticket->get('TKT_sold'), |
|
| 1627 | + 'trash_icon' => ($skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted'))) |
|
| 1628 | + && (! empty($ticket) && $ticket->get('TKT_sold') === 0) |
|
| 1629 | + ? 'trash-icon dashicons dashicons-post-trash clickable' : 'ee-lock-icon', |
|
| 1630 | + 'disabled' => $skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted')) ? '' |
|
| 1631 | + : ' disabled=disabled', |
|
| 1632 | + ); |
|
| 1633 | + $price = $ticket->ID() !== 0 |
|
| 1634 | + ? $ticket->get_first_related('Price', array('default_where_conditions' => 'none')) |
|
| 1635 | + : EE_Registry::instance()->load_model('Price')->create_default_object(); |
|
| 1636 | + $price_args = array( |
|
| 1637 | + 'price_currency_symbol' => EE_Registry::instance()->CFG->currency->sign, |
|
| 1638 | + 'PRC_amount' => $price->get('PRC_amount'), |
|
| 1639 | + 'PRT_ID' => $price->get('PRT_ID'), |
|
| 1640 | + 'PRC_ID' => $price->get('PRC_ID'), |
|
| 1641 | + 'PRC_is_default' => $price->get('PRC_is_default'), |
|
| 1642 | + ); |
|
| 1643 | + // make sure we have default start and end dates if skeleton |
|
| 1644 | + // handle rows that should NOT be empty |
|
| 1645 | + if (empty($template_args['TKT_start_date'])) { |
|
| 1646 | + // if empty then the start date will be now. |
|
| 1647 | + $template_args['TKT_start_date'] = date('Y-m-d h:i a', current_time('timestamp')); |
|
| 1648 | + } |
|
| 1649 | + if (empty($template_args['TKT_end_date'])) { |
|
| 1650 | + // get the earliest datetime (if present); |
|
| 1651 | + $earliest_dtt = $this->_cpt_model_obj->ID() > 0 |
|
| 1652 | + ? $this->_cpt_model_obj->get_first_related( |
|
| 1653 | + 'Datetime', |
|
| 1654 | + array('order_by' => array('DTT_EVT_start' => 'ASC')) |
|
| 1655 | + ) |
|
| 1656 | + : null; |
|
| 1657 | + if (! empty($earliest_dtt)) { |
|
| 1658 | + $template_args['TKT_end_date'] = $earliest_dtt->get_datetime('DTT_EVT_start', 'Y-m-d', 'h:i a'); |
|
| 1659 | + } else { |
|
| 1660 | + $template_args['TKT_end_date'] = date( |
|
| 1661 | + 'Y-m-d h:i a', |
|
| 1662 | + mktime(0, 0, 0, date("m"), date("d") + 7, date("Y")) |
|
| 1663 | + ); |
|
| 1664 | + } |
|
| 1665 | + } |
|
| 1666 | + $template_args = array_merge($template_args, $price_args); |
|
| 1667 | + $template = apply_filters( |
|
| 1668 | + 'FHEE__Events_Admin_Page__get_ticket_row__template', |
|
| 1669 | + EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_ticket_row.template.php', |
|
| 1670 | + $ticket |
|
| 1671 | + ); |
|
| 1672 | + return EEH_Template::display_template($template, $template_args, true); |
|
| 1673 | + } |
|
| 1674 | + |
|
| 1675 | + |
|
| 1676 | + /** |
|
| 1677 | + * @throws DomainException |
|
| 1678 | + */ |
|
| 1679 | + public function registration_options_meta_box() |
|
| 1680 | + { |
|
| 1681 | + $yes_no_values = array( |
|
| 1682 | + array('id' => true, 'text' => esc_html__('Yes', 'event_espresso')), |
|
| 1683 | + array('id' => false, 'text' => esc_html__('No', 'event_espresso')), |
|
| 1684 | + ); |
|
| 1685 | + $default_reg_status_values = EEM_Registration::reg_status_array( |
|
| 1686 | + array( |
|
| 1687 | + EEM_Registration::status_id_cancelled, |
|
| 1688 | + EEM_Registration::status_id_declined, |
|
| 1689 | + EEM_Registration::status_id_incomplete, |
|
| 1690 | + ), |
|
| 1691 | + true |
|
| 1692 | + ); |
|
| 1693 | + // $template_args['is_active_select'] = EEH_Form_Fields::select_input('is_active', $yes_no_values, $this->_cpt_model_obj->is_active()); |
|
| 1694 | + $template_args['_event'] = $this->_cpt_model_obj; |
|
| 1695 | + $template_args['active_status'] = $this->_cpt_model_obj->pretty_active_status(false); |
|
| 1696 | + $template_args['additional_limit'] = $this->_cpt_model_obj->additional_limit(); |
|
| 1697 | + $template_args['default_registration_status'] = EEH_Form_Fields::select_input( |
|
| 1698 | + 'default_reg_status', |
|
| 1699 | + $default_reg_status_values, |
|
| 1700 | + $this->_cpt_model_obj->default_registration_status() |
|
| 1701 | + ); |
|
| 1702 | + $template_args['display_description'] = EEH_Form_Fields::select_input( |
|
| 1703 | + 'display_desc', |
|
| 1704 | + $yes_no_values, |
|
| 1705 | + $this->_cpt_model_obj->display_description() |
|
| 1706 | + ); |
|
| 1707 | + $template_args['display_ticket_selector'] = EEH_Form_Fields::select_input( |
|
| 1708 | + 'display_ticket_selector', |
|
| 1709 | + $yes_no_values, |
|
| 1710 | + $this->_cpt_model_obj->display_ticket_selector(), |
|
| 1711 | + '', |
|
| 1712 | + '', |
|
| 1713 | + false |
|
| 1714 | + ); |
|
| 1715 | + $template_args['additional_registration_options'] = apply_filters( |
|
| 1716 | + 'FHEE__Events_Admin_Page__registration_options_meta_box__additional_registration_options', |
|
| 1717 | + '', |
|
| 1718 | + $template_args, |
|
| 1719 | + $yes_no_values, |
|
| 1720 | + $default_reg_status_values |
|
| 1721 | + ); |
|
| 1722 | + EEH_Template::display_template( |
|
| 1723 | + EVENTS_TEMPLATE_PATH . 'event_registration_options.template.php', |
|
| 1724 | + $template_args |
|
| 1725 | + ); |
|
| 1726 | + } |
|
| 1727 | + |
|
| 1728 | + |
|
| 1729 | + /** |
|
| 1730 | + * _get_events() |
|
| 1731 | + * This method simply returns all the events (for the given _view and paging) |
|
| 1732 | + * |
|
| 1733 | + * @access public |
|
| 1734 | + * @param int $per_page count of items per page (20 default); |
|
| 1735 | + * @param int $current_page what is the current page being viewed. |
|
| 1736 | + * @param bool $count if TRUE then we just return a count of ALL events matching the given _view. |
|
| 1737 | + * If FALSE then we return an array of event objects |
|
| 1738 | + * that match the given _view and paging parameters. |
|
| 1739 | + * @return array an array of event objects. |
|
| 1740 | + */ |
|
| 1741 | + public function get_events($per_page = 10, $current_page = 1, $count = false) |
|
| 1742 | + { |
|
| 1743 | + $EEME = $this->_event_model(); |
|
| 1744 | + $offset = ($current_page - 1) * $per_page; |
|
| 1745 | + $limit = $count ? null : $offset . ',' . $per_page; |
|
| 1746 | + $orderby = isset($this->_req_data['orderby']) ? $this->_req_data['orderby'] : 'EVT_ID'; |
|
| 1747 | + $order = isset($this->_req_data['order']) ? $this->_req_data['order'] : "DESC"; |
|
| 1748 | + if (isset($this->_req_data['month_range'])) { |
|
| 1749 | + $pieces = explode(' ', $this->_req_data['month_range'], 3); |
|
| 1750 | + // simulate the FIRST day of the month, that fixes issues for months like February |
|
| 1751 | + // where PHP doesn't know what to assume for date. |
|
| 1752 | + // @see https://events.codebasehq.com/projects/event-espresso/tickets/10437 |
|
| 1753 | + $month_r = ! empty($pieces[0]) ? date('m', \EEH_DTT_Helper::first_of_month_timestamp($pieces[0])) : ''; |
|
| 1754 | + $year_r = ! empty($pieces[1]) ? $pieces[1] : ''; |
|
| 1755 | + } |
|
| 1756 | + $where = array(); |
|
| 1757 | + $status = isset($this->_req_data['status']) ? $this->_req_data['status'] : null; |
|
| 1758 | + // determine what post_status our condition will have for the query. |
|
| 1759 | + switch ($status) { |
|
| 1760 | + case 'month': |
|
| 1761 | + case 'today': |
|
| 1762 | + case null: |
|
| 1763 | + case 'all': |
|
| 1764 | + break; |
|
| 1765 | + case 'draft': |
|
| 1766 | + $where['status'] = array('IN', array('draft', 'auto-draft')); |
|
| 1767 | + break; |
|
| 1768 | + default: |
|
| 1769 | + $where['status'] = $status; |
|
| 1770 | + } |
|
| 1771 | + // categories? |
|
| 1772 | + $category = isset($this->_req_data['EVT_CAT']) && $this->_req_data['EVT_CAT'] > 0 |
|
| 1773 | + ? $this->_req_data['EVT_CAT'] : null; |
|
| 1774 | + if (! empty($category)) { |
|
| 1775 | + $where['Term_Taxonomy.taxonomy'] = EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY; |
|
| 1776 | + $where['Term_Taxonomy.term_id'] = $category; |
|
| 1777 | + } |
|
| 1778 | + // date where conditions |
|
| 1779 | + $start_formats = EEM_Datetime::instance()->get_formats_for('DTT_EVT_start'); |
|
| 1780 | + if (isset($this->_req_data['month_range']) && $this->_req_data['month_range'] != '') { |
|
| 1781 | + $DateTime = new DateTime( |
|
| 1782 | + $year_r . '-' . $month_r . '-01 00:00:00', |
|
| 1783 | + new DateTimeZone(EEM_Datetime::instance()->get_timezone()) |
|
| 1784 | + ); |
|
| 1785 | + $start = $DateTime->format(implode(' ', $start_formats)); |
|
| 1786 | + $end = $DateTime->setDate( |
|
| 1787 | + $year_r, |
|
| 1788 | + $month_r, |
|
| 1789 | + $DateTime |
|
| 1790 | + ->format('t') |
|
| 1791 | + )->setTime(23, 59, 59) |
|
| 1792 | + ->format(implode(' ', $start_formats)); |
|
| 1793 | + $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
| 1794 | + } elseif (isset($this->_req_data['status']) && $this->_req_data['status'] == 'today') { |
|
| 1795 | + $DateTime = new DateTime('now', new DateTimeZone(EEM_Event::instance()->get_timezone())); |
|
| 1796 | + $start = $DateTime->setTime(0, 0, 0)->format(implode(' ', $start_formats)); |
|
| 1797 | + $end = $DateTime->setTime(23, 59, 59)->format(implode(' ', $start_formats)); |
|
| 1798 | + $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
| 1799 | + } elseif (isset($this->_req_data['status']) && $this->_req_data['status'] == 'month') { |
|
| 1800 | + $now = date('Y-m-01'); |
|
| 1801 | + $DateTime = new DateTime($now, new DateTimeZone(EEM_Event::instance()->get_timezone())); |
|
| 1802 | + $start = $DateTime->setTime(0, 0, 0)->format(implode(' ', $start_formats)); |
|
| 1803 | + $end = $DateTime->setDate(date('Y'), date('m'), $DateTime->format('t')) |
|
| 1804 | + ->setTime(23, 59, 59) |
|
| 1805 | + ->format(implode(' ', $start_formats)); |
|
| 1806 | + $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
|
| 1807 | + } |
|
| 1808 | + if (! EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events')) { |
|
| 1809 | + $where['EVT_wp_user'] = get_current_user_id(); |
|
| 1810 | + } else { |
|
| 1811 | + if (! isset($where['status'])) { |
|
| 1812 | + if (! EE_Registry::instance()->CAP->current_user_can('ee_read_private_events', 'get_events')) { |
|
| 1813 | + $where['OR'] = array( |
|
| 1814 | + 'status*restrict_private' => array('!=', 'private'), |
|
| 1815 | + 'AND' => array( |
|
| 1816 | + 'status*inclusive' => array('=', 'private'), |
|
| 1817 | + 'EVT_wp_user' => get_current_user_id(), |
|
| 1818 | + ), |
|
| 1819 | + ); |
|
| 1820 | + } |
|
| 1821 | + } |
|
| 1822 | + } |
|
| 1823 | + if (isset($this->_req_data['EVT_wp_user'])) { |
|
| 1824 | + if ($this->_req_data['EVT_wp_user'] != get_current_user_id() |
|
| 1825 | + && EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events') |
|
| 1826 | + ) { |
|
| 1827 | + $where['EVT_wp_user'] = $this->_req_data['EVT_wp_user']; |
|
| 1828 | + } |
|
| 1829 | + } |
|
| 1830 | + // search query handling |
|
| 1831 | + if (isset($this->_req_data['s'])) { |
|
| 1832 | + $search_string = '%' . $this->_req_data['s'] . '%'; |
|
| 1833 | + $where['OR'] = array( |
|
| 1834 | + 'EVT_name' => array('LIKE', $search_string), |
|
| 1835 | + 'EVT_desc' => array('LIKE', $search_string), |
|
| 1836 | + 'EVT_short_desc' => array('LIKE', $search_string), |
|
| 1837 | + ); |
|
| 1838 | + } |
|
| 1839 | + $where = apply_filters('FHEE__Events_Admin_Page__get_events__where', $where, $this->_req_data); |
|
| 1840 | + $query_params = apply_filters( |
|
| 1841 | + 'FHEE__Events_Admin_Page__get_events__query_params', |
|
| 1842 | + array( |
|
| 1843 | + $where, |
|
| 1844 | + 'limit' => $limit, |
|
| 1845 | + 'order_by' => $orderby, |
|
| 1846 | + 'order' => $order, |
|
| 1847 | + 'group_by' => 'EVT_ID', |
|
| 1848 | + ), |
|
| 1849 | + $this->_req_data |
|
| 1850 | + ); |
|
| 1851 | + // let's first check if we have special requests coming in. |
|
| 1852 | + if (isset($this->_req_data['active_status'])) { |
|
| 1853 | + switch ($this->_req_data['active_status']) { |
|
| 1854 | + case 'upcoming': |
|
| 1855 | + return $EEME->get_upcoming_events($query_params, $count); |
|
| 1856 | + break; |
|
| 1857 | + case 'expired': |
|
| 1858 | + return $EEME->get_expired_events($query_params, $count); |
|
| 1859 | + break; |
|
| 1860 | + case 'active': |
|
| 1861 | + return $EEME->get_active_events($query_params, $count); |
|
| 1862 | + break; |
|
| 1863 | + case 'inactive': |
|
| 1864 | + return $EEME->get_inactive_events($query_params, $count); |
|
| 1865 | + break; |
|
| 1866 | + } |
|
| 1867 | + } |
|
| 1868 | + $events = $count ? $EEME->count(array($where), 'EVT_ID', true) : $EEME->get_all($query_params); |
|
| 1869 | + return $events; |
|
| 1870 | + } |
|
| 1871 | + |
|
| 1872 | + |
|
| 1873 | + /** |
|
| 1874 | + * handling for WordPress CPT actions (trash, restore, delete) |
|
| 1875 | + * |
|
| 1876 | + * @param string $post_id |
|
| 1877 | + */ |
|
| 1878 | + public function trash_cpt_item($post_id) |
|
| 1879 | + { |
|
| 1880 | + $this->_req_data['EVT_ID'] = $post_id; |
|
| 1881 | + $this->_trash_or_restore_event('trash', false); |
|
| 1882 | + } |
|
| 1883 | + |
|
| 1884 | + |
|
| 1885 | + /** |
|
| 1886 | + * @param string $post_id |
|
| 1887 | + */ |
|
| 1888 | + public function restore_cpt_item($post_id) |
|
| 1889 | + { |
|
| 1890 | + $this->_req_data['EVT_ID'] = $post_id; |
|
| 1891 | + $this->_trash_or_restore_event('draft', false); |
|
| 1892 | + } |
|
| 1893 | + |
|
| 1894 | + |
|
| 1895 | + /** |
|
| 1896 | + * @param string $post_id |
|
| 1897 | + */ |
|
| 1898 | + public function delete_cpt_item($post_id) |
|
| 1899 | + { |
|
| 1900 | + $this->_req_data['EVT_ID'] = $post_id; |
|
| 1901 | + $this->_delete_event(false); |
|
| 1902 | + } |
|
| 1903 | + |
|
| 1904 | + |
|
| 1905 | + /** |
|
| 1906 | + * _trash_or_restore_event |
|
| 1907 | + * |
|
| 1908 | + * @access protected |
|
| 1909 | + * @param string $event_status |
|
| 1910 | + * @param bool $redirect_after |
|
| 1911 | + */ |
|
| 1912 | + protected function _trash_or_restore_event($event_status = 'trash', $redirect_after = true) |
|
| 1913 | + { |
|
| 1914 | + // determine the event id and set to array. |
|
| 1915 | + $EVT_ID = isset($this->_req_data['EVT_ID']) ? absint($this->_req_data['EVT_ID']) : false; |
|
| 1916 | + // loop thru events |
|
| 1917 | + if ($EVT_ID) { |
|
| 1918 | + // clean status |
|
| 1919 | + $event_status = sanitize_key($event_status); |
|
| 1920 | + // grab status |
|
| 1921 | + if (! empty($event_status)) { |
|
| 1922 | + $success = $this->_change_event_status($EVT_ID, $event_status); |
|
| 1923 | + } else { |
|
| 1924 | + $success = false; |
|
| 1925 | + $msg = esc_html__( |
|
| 1926 | + 'An error occurred. The event could not be moved to the trash because a valid event status was not not supplied.', |
|
| 1927 | + 'event_espresso' |
|
| 1928 | + ); |
|
| 1929 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 1930 | + } |
|
| 1931 | + } else { |
|
| 1932 | + $success = false; |
|
| 1933 | + $msg = esc_html__( |
|
| 1934 | + 'An error occurred. The event could not be moved to the trash because a valid event ID was not not supplied.', |
|
| 1935 | + 'event_espresso' |
|
| 1936 | + ); |
|
| 1937 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 1938 | + } |
|
| 1939 | + $action = $event_status == 'trash' ? 'moved to the trash' : 'restored from the trash'; |
|
| 1940 | + if ($redirect_after) { |
|
| 1941 | + $this->_redirect_after_action($success, 'Event', $action, array('action' => 'default')); |
|
| 1942 | + } |
|
| 1943 | + } |
|
| 1944 | + |
|
| 1945 | + |
|
| 1946 | + /** |
|
| 1947 | + * _trash_or_restore_events |
|
| 1948 | + * |
|
| 1949 | + * @access protected |
|
| 1950 | + * @param string $event_status |
|
| 1951 | + * @return void |
|
| 1952 | + */ |
|
| 1953 | + protected function _trash_or_restore_events($event_status = 'trash') |
|
| 1954 | + { |
|
| 1955 | + // clean status |
|
| 1956 | + $event_status = sanitize_key($event_status); |
|
| 1957 | + // grab status |
|
| 1958 | + if (! empty($event_status)) { |
|
| 1959 | + $success = true; |
|
| 1960 | + // determine the event id and set to array. |
|
| 1961 | + $EVT_IDs = isset($this->_req_data['EVT_IDs']) ? (array) $this->_req_data['EVT_IDs'] : array(); |
|
| 1962 | + // loop thru events |
|
| 1963 | + foreach ($EVT_IDs as $EVT_ID) { |
|
| 1964 | + if ($EVT_ID = absint($EVT_ID)) { |
|
| 1965 | + $results = $this->_change_event_status($EVT_ID, $event_status); |
|
| 1966 | + $success = $results !== false ? $success : false; |
|
| 1967 | + } else { |
|
| 1968 | + $msg = sprintf( |
|
| 1969 | + esc_html__( |
|
| 1970 | + 'An error occurred. Event #%d could not be moved to the trash because a valid event ID was not not supplied.', |
|
| 1971 | + 'event_espresso' |
|
| 1972 | + ), |
|
| 1973 | + $EVT_ID |
|
| 1974 | + ); |
|
| 1975 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 1976 | + $success = false; |
|
| 1977 | + } |
|
| 1978 | + } |
|
| 1979 | + } else { |
|
| 1980 | + $success = false; |
|
| 1981 | + $msg = esc_html__( |
|
| 1982 | + 'An error occurred. The event could not be moved to the trash because a valid event status was not not supplied.', |
|
| 1983 | + 'event_espresso' |
|
| 1984 | + ); |
|
| 1985 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 1986 | + } |
|
| 1987 | + // in order to force a pluralized result message we need to send back a success status greater than 1 |
|
| 1988 | + $success = $success ? 2 : false; |
|
| 1989 | + $action = $event_status == 'trash' ? 'moved to the trash' : 'restored from the trash'; |
|
| 1990 | + $this->_redirect_after_action($success, 'Events', $action, array('action' => 'default')); |
|
| 1991 | + } |
|
| 1992 | + |
|
| 1993 | + |
|
| 1994 | + /** |
|
| 1995 | + * _trash_or_restore_events |
|
| 1996 | + * |
|
| 1997 | + * @access private |
|
| 1998 | + * @param int $EVT_ID |
|
| 1999 | + * @param string $event_status |
|
| 2000 | + * @return bool |
|
| 2001 | + */ |
|
| 2002 | + private function _change_event_status($EVT_ID = 0, $event_status = '') |
|
| 2003 | + { |
|
| 2004 | + // grab event id |
|
| 2005 | + if (! $EVT_ID) { |
|
| 2006 | + $msg = esc_html__( |
|
| 2007 | + 'An error occurred. No Event ID or an invalid Event ID was received.', |
|
| 2008 | + 'event_espresso' |
|
| 2009 | + ); |
|
| 2010 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2011 | + return false; |
|
| 2012 | + } |
|
| 2013 | + $this->_cpt_model_obj = EEM_Event::instance()->get_one_by_ID($EVT_ID); |
|
| 2014 | + // clean status |
|
| 2015 | + $event_status = sanitize_key($event_status); |
|
| 2016 | + // grab status |
|
| 2017 | + if (empty($event_status)) { |
|
| 2018 | + $msg = esc_html__( |
|
| 2019 | + 'An error occurred. No Event Status or an invalid Event Status was received.', |
|
| 2020 | + 'event_espresso' |
|
| 2021 | + ); |
|
| 2022 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2023 | + return false; |
|
| 2024 | + } |
|
| 2025 | + // was event trashed or restored ? |
|
| 2026 | + switch ($event_status) { |
|
| 2027 | + case 'draft': |
|
| 2028 | + $action = 'restored from the trash'; |
|
| 2029 | + $hook = 'AHEE_event_restored_from_trash'; |
|
| 2030 | + break; |
|
| 2031 | + case 'trash': |
|
| 2032 | + $action = 'moved to the trash'; |
|
| 2033 | + $hook = 'AHEE_event_moved_to_trash'; |
|
| 2034 | + break; |
|
| 2035 | + default: |
|
| 2036 | + $action = 'updated'; |
|
| 2037 | + $hook = false; |
|
| 2038 | + } |
|
| 2039 | + // use class to change status |
|
| 2040 | + $this->_cpt_model_obj->set_status($event_status); |
|
| 2041 | + $success = $this->_cpt_model_obj->save(); |
|
| 2042 | + if ($success === false) { |
|
| 2043 | + $msg = sprintf(esc_html__('An error occurred. The event could not be %s.', 'event_espresso'), $action); |
|
| 2044 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2045 | + return false; |
|
| 2046 | + } |
|
| 2047 | + if ($hook) { |
|
| 2048 | + do_action($hook); |
|
| 2049 | + } |
|
| 2050 | + return true; |
|
| 2051 | + } |
|
| 2052 | + |
|
| 2053 | + |
|
| 2054 | + /** |
|
| 2055 | + * _delete_event |
|
| 2056 | + * |
|
| 2057 | + * @access protected |
|
| 2058 | + * @param bool $redirect_after |
|
| 2059 | + */ |
|
| 2060 | + protected function _delete_event($redirect_after = true) |
|
| 2061 | + { |
|
| 2062 | + // determine the event id and set to array. |
|
| 2063 | + $EVT_ID = isset($this->_req_data['EVT_ID']) ? absint($this->_req_data['EVT_ID']) : null; |
|
| 2064 | + $EVT_ID = isset($this->_req_data['post']) ? absint($this->_req_data['post']) : $EVT_ID; |
|
| 2065 | + // loop thru events |
|
| 2066 | + if ($EVT_ID) { |
|
| 2067 | + $success = $this->_permanently_delete_event($EVT_ID); |
|
| 2068 | + // get list of events with no prices |
|
| 2069 | + $espresso_no_ticket_prices = get_option('ee_no_ticket_prices', array()); |
|
| 2070 | + // remove this event from the list of events with no prices |
|
| 2071 | + if (isset($espresso_no_ticket_prices[ $EVT_ID ])) { |
|
| 2072 | + unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
| 2073 | + } |
|
| 2074 | + update_option('ee_no_ticket_prices', $espresso_no_ticket_prices); |
|
| 2075 | + } else { |
|
| 2076 | + $success = false; |
|
| 2077 | + $msg = esc_html__( |
|
| 2078 | + 'An error occurred. An event could not be deleted because a valid event ID was not not supplied.', |
|
| 2079 | + 'event_espresso' |
|
| 2080 | + ); |
|
| 2081 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2082 | + } |
|
| 2083 | + if ($redirect_after) { |
|
| 2084 | + $this->_redirect_after_action( |
|
| 2085 | + $success, |
|
| 2086 | + 'Event', |
|
| 2087 | + 'deleted', |
|
| 2088 | + array('action' => 'default', 'status' => 'trash') |
|
| 2089 | + ); |
|
| 2090 | + } |
|
| 2091 | + } |
|
| 2092 | + |
|
| 2093 | + |
|
| 2094 | + /** |
|
| 2095 | + * _delete_events |
|
| 2096 | + * |
|
| 2097 | + * @access protected |
|
| 2098 | + * @return void |
|
| 2099 | + */ |
|
| 2100 | + protected function _delete_events() |
|
| 2101 | + { |
|
| 2102 | + $success = true; |
|
| 2103 | + // get list of events with no prices |
|
| 2104 | + $espresso_no_ticket_prices = get_option('ee_no_ticket_prices', array()); |
|
| 2105 | + // determine the event id and set to array. |
|
| 2106 | + $EVT_IDs = isset($this->_req_data['EVT_IDs']) ? (array) $this->_req_data['EVT_IDs'] : array(); |
|
| 2107 | + // loop thru events |
|
| 2108 | + foreach ($EVT_IDs as $EVT_ID) { |
|
| 2109 | + $EVT_ID = absint($EVT_ID); |
|
| 2110 | + if ($EVT_ID) { |
|
| 2111 | + $results = $this->_permanently_delete_event($EVT_ID); |
|
| 2112 | + $success = $results !== false ? $success : false; |
|
| 2113 | + // remove this event from the list of events with no prices |
|
| 2114 | + unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
| 2115 | + } else { |
|
| 2116 | + $success = false; |
|
| 2117 | + $msg = esc_html__( |
|
| 2118 | + 'An error occurred. An event could not be deleted because a valid event ID was not not supplied.', |
|
| 2119 | + 'event_espresso' |
|
| 2120 | + ); |
|
| 2121 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2122 | + } |
|
| 2123 | + } |
|
| 2124 | + update_option('ee_no_ticket_prices', $espresso_no_ticket_prices); |
|
| 2125 | + // in order to force a pluralized result message we need to send back a success status greater than 1 |
|
| 2126 | + $success = $success ? 2 : false; |
|
| 2127 | + $this->_redirect_after_action($success, 'Events', 'deleted', array('action' => 'default')); |
|
| 2128 | + } |
|
| 2129 | + |
|
| 2130 | + |
|
| 2131 | + /** |
|
| 2132 | + * _permanently_delete_event |
|
| 2133 | + * |
|
| 2134 | + * @access private |
|
| 2135 | + * @param int $EVT_ID |
|
| 2136 | + * @return bool |
|
| 2137 | + */ |
|
| 2138 | + private function _permanently_delete_event($EVT_ID = 0) |
|
| 2139 | + { |
|
| 2140 | + // grab event id |
|
| 2141 | + if (! $EVT_ID) { |
|
| 2142 | + $msg = esc_html__( |
|
| 2143 | + 'An error occurred. No Event ID or an invalid Event ID was received.', |
|
| 2144 | + 'event_espresso' |
|
| 2145 | + ); |
|
| 2146 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2147 | + return false; |
|
| 2148 | + } |
|
| 2149 | + if (! $this->_cpt_model_obj instanceof EE_Event |
|
| 2150 | + || $this->_cpt_model_obj->ID() !== $EVT_ID |
|
| 2151 | + ) { |
|
| 2152 | + $this->_cpt_model_obj = EEM_Event::instance()->get_one_by_ID($EVT_ID); |
|
| 2153 | + } |
|
| 2154 | + if (! $this->_cpt_model_obj instanceof EE_Event) { |
|
| 2155 | + return false; |
|
| 2156 | + } |
|
| 2157 | + // need to delete related tickets and prices first. |
|
| 2158 | + $datetimes = $this->_cpt_model_obj->get_many_related('Datetime'); |
|
| 2159 | + foreach ($datetimes as $datetime) { |
|
| 2160 | + $this->_cpt_model_obj->_remove_relation_to($datetime, 'Datetime'); |
|
| 2161 | + $tickets = $datetime->get_many_related('Ticket'); |
|
| 2162 | + foreach ($tickets as $ticket) { |
|
| 2163 | + $ticket->_remove_relation_to($datetime, 'Datetime'); |
|
| 2164 | + $ticket->delete_related_permanently('Price'); |
|
| 2165 | + $ticket->delete_permanently(); |
|
| 2166 | + } |
|
| 2167 | + $datetime->delete(); |
|
| 2168 | + } |
|
| 2169 | + // what about related venues or terms? |
|
| 2170 | + $venues = $this->_cpt_model_obj->get_many_related('Venue'); |
|
| 2171 | + foreach ($venues as $venue) { |
|
| 2172 | + $this->_cpt_model_obj->_remove_relation_to($venue, 'Venue'); |
|
| 2173 | + } |
|
| 2174 | + // any attached question groups? |
|
| 2175 | + $question_groups = $this->_cpt_model_obj->get_many_related('Question_Group'); |
|
| 2176 | + if (! empty($question_groups)) { |
|
| 2177 | + foreach ($question_groups as $question_group) { |
|
| 2178 | + $this->_cpt_model_obj->_remove_relation_to($question_group, 'Question_Group'); |
|
| 2179 | + } |
|
| 2180 | + } |
|
| 2181 | + // Message Template Groups |
|
| 2182 | + $this->_cpt_model_obj->_remove_relations('Message_Template_Group'); |
|
| 2183 | + /** @type EE_Term_Taxonomy[] $term_taxonomies */ |
|
| 2184 | + $term_taxonomies = $this->_cpt_model_obj->term_taxonomies(); |
|
| 2185 | + foreach ($term_taxonomies as $term_taxonomy) { |
|
| 2186 | + $this->_cpt_model_obj->remove_relation_to_term_taxonomy($term_taxonomy); |
|
| 2187 | + } |
|
| 2188 | + $success = $this->_cpt_model_obj->delete_permanently(); |
|
| 2189 | + // did it all go as planned ? |
|
| 2190 | + if ($success) { |
|
| 2191 | + $msg = sprintf(esc_html__('Event ID # %d has been deleted.', 'event_espresso'), $EVT_ID); |
|
| 2192 | + EE_Error::add_success($msg); |
|
| 2193 | + } else { |
|
| 2194 | + $msg = sprintf( |
|
| 2195 | + esc_html__('An error occurred. Event ID # %d could not be deleted.', 'event_espresso'), |
|
| 2196 | + $EVT_ID |
|
| 2197 | + ); |
|
| 2198 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2199 | + return false; |
|
| 2200 | + } |
|
| 2201 | + do_action('AHEE__Events_Admin_Page___permanently_delete_event__after_event_deleted', $EVT_ID); |
|
| 2202 | + return true; |
|
| 2203 | + } |
|
| 2204 | + |
|
| 2205 | + |
|
| 2206 | + /** |
|
| 2207 | + * get total number of events |
|
| 2208 | + * |
|
| 2209 | + * @access public |
|
| 2210 | + * @return int |
|
| 2211 | + */ |
|
| 2212 | + public function total_events() |
|
| 2213 | + { |
|
| 2214 | + $count = EEM_Event::instance()->count(array('caps' => 'read_admin'), 'EVT_ID', true); |
|
| 2215 | + return $count; |
|
| 2216 | + } |
|
| 2217 | + |
|
| 2218 | + |
|
| 2219 | + /** |
|
| 2220 | + * get total number of draft events |
|
| 2221 | + * |
|
| 2222 | + * @access public |
|
| 2223 | + * @return int |
|
| 2224 | + */ |
|
| 2225 | + public function total_events_draft() |
|
| 2226 | + { |
|
| 2227 | + $where = array( |
|
| 2228 | + 'status' => array('IN', array('draft', 'auto-draft')), |
|
| 2229 | + ); |
|
| 2230 | + $count = EEM_Event::instance()->count(array($where, 'caps' => 'read_admin'), 'EVT_ID', true); |
|
| 2231 | + return $count; |
|
| 2232 | + } |
|
| 2233 | + |
|
| 2234 | + |
|
| 2235 | + /** |
|
| 2236 | + * get total number of trashed events |
|
| 2237 | + * |
|
| 2238 | + * @access public |
|
| 2239 | + * @return int |
|
| 2240 | + */ |
|
| 2241 | + public function total_trashed_events() |
|
| 2242 | + { |
|
| 2243 | + $where = array( |
|
| 2244 | + 'status' => 'trash', |
|
| 2245 | + ); |
|
| 2246 | + $count = EEM_Event::instance()->count(array($where, 'caps' => 'read_admin'), 'EVT_ID', true); |
|
| 2247 | + return $count; |
|
| 2248 | + } |
|
| 2249 | + |
|
| 2250 | + |
|
| 2251 | + /** |
|
| 2252 | + * _default_event_settings |
|
| 2253 | + * This generates the Default Settings Tab |
|
| 2254 | + * |
|
| 2255 | + * @return void |
|
| 2256 | + * @throws EE_Error |
|
| 2257 | + */ |
|
| 2258 | + protected function _default_event_settings() |
|
| 2259 | + { |
|
| 2260 | + $this->_set_add_edit_form_tags('update_default_event_settings'); |
|
| 2261 | + $this->_set_publish_post_box_vars(null, false, false, null, false); |
|
| 2262 | + $this->_template_args['admin_page_content'] = $this->_default_event_settings_form()->get_html(); |
|
| 2263 | + $this->display_admin_page_with_sidebar(); |
|
| 2264 | + } |
|
| 2265 | + |
|
| 2266 | + |
|
| 2267 | + /** |
|
| 2268 | + * Return the form for event settings. |
|
| 2269 | + * |
|
| 2270 | + * @return EE_Form_Section_Proper |
|
| 2271 | + * @throws EE_Error |
|
| 2272 | + */ |
|
| 2273 | + protected function _default_event_settings_form() |
|
| 2274 | + { |
|
| 2275 | + $registration_config = EE_Registry::instance()->CFG->registration; |
|
| 2276 | + $registration_stati_for_selection = EEM_Registration::reg_status_array( |
|
| 2277 | + // exclude |
|
| 2278 | + array( |
|
| 2279 | + EEM_Registration::status_id_cancelled, |
|
| 2280 | + EEM_Registration::status_id_declined, |
|
| 2281 | + EEM_Registration::status_id_incomplete, |
|
| 2282 | + EEM_Registration::status_id_wait_list, |
|
| 2283 | + ), |
|
| 2284 | + true |
|
| 2285 | + ); |
|
| 2286 | + return new EE_Form_Section_Proper( |
|
| 2287 | + array( |
|
| 2288 | + 'name' => 'update_default_event_settings', |
|
| 2289 | + 'html_id' => 'update_default_event_settings', |
|
| 2290 | + 'html_class' => 'form-table', |
|
| 2291 | + 'layout_strategy' => new EE_Admin_Two_Column_Layout(), |
|
| 2292 | + 'subsections' => apply_filters( |
|
| 2293 | + 'FHEE__Events_Admin_Page___default_event_settings_form__form_subsections', |
|
| 2294 | + array( |
|
| 2295 | + 'default_reg_status' => new EE_Select_Input( |
|
| 2296 | + $registration_stati_for_selection, |
|
| 2297 | + array( |
|
| 2298 | + 'default' => isset($registration_config->default_STS_ID) |
|
| 2299 | + && array_key_exists( |
|
| 2300 | + $registration_config->default_STS_ID, |
|
| 2301 | + $registration_stati_for_selection |
|
| 2302 | + ) |
|
| 2303 | + ? sanitize_text_field($registration_config->default_STS_ID) |
|
| 2304 | + : EEM_Registration::status_id_pending_payment, |
|
| 2305 | + 'html_label_text' => esc_html__('Default Registration Status', 'event_espresso') |
|
| 2306 | + . EEH_Template::get_help_tab_link( |
|
| 2307 | + 'default_settings_status_help_tab' |
|
| 2308 | + ), |
|
| 2309 | + 'html_help_text' => esc_html__( |
|
| 2310 | + 'This setting allows you to preselect what the default registration status setting is when creating an event. Note that changing this setting does NOT retroactively apply it to existing events.', |
|
| 2311 | + 'event_espresso' |
|
| 2312 | + ), |
|
| 2313 | + ) |
|
| 2314 | + ), |
|
| 2315 | + 'default_max_tickets' => new EE_Integer_Input( |
|
| 2316 | + array( |
|
| 2317 | + 'default' => isset($registration_config->default_maximum_number_of_tickets) |
|
| 2318 | + ? $registration_config->default_maximum_number_of_tickets |
|
| 2319 | + : EEM_Event::get_default_additional_limit(), |
|
| 2320 | + 'html_label_text' => esc_html__( |
|
| 2321 | + 'Default Maximum Tickets Allowed Per Order:', |
|
| 2322 | + 'event_espresso' |
|
| 2323 | + ) |
|
| 2324 | + . EEH_Template::get_help_tab_link( |
|
| 2325 | + 'default_maximum_tickets_help_tab"' |
|
| 2326 | + ), |
|
| 2327 | + 'html_help_text' => esc_html__( |
|
| 2328 | + 'This setting allows you to indicate what will be the default for the maximum number of tickets per order when creating new events.', |
|
| 2329 | + 'event_espresso' |
|
| 2330 | + ), |
|
| 2331 | + ) |
|
| 2332 | + ), |
|
| 2333 | + ) |
|
| 2334 | + ), |
|
| 2335 | + ) |
|
| 2336 | + ); |
|
| 2337 | + } |
|
| 2338 | + |
|
| 2339 | + |
|
| 2340 | + /** |
|
| 2341 | + * _update_default_event_settings |
|
| 2342 | + * |
|
| 2343 | + * @access protected |
|
| 2344 | + * @return void |
|
| 2345 | + * @throws EE_Error |
|
| 2346 | + */ |
|
| 2347 | + protected function _update_default_event_settings() |
|
| 2348 | + { |
|
| 2349 | + $registration_config = EE_Registry::instance()->CFG->registration; |
|
| 2350 | + $form = $this->_default_event_settings_form(); |
|
| 2351 | + if ($form->was_submitted()) { |
|
| 2352 | + $form->receive_form_submission(); |
|
| 2353 | + if ($form->is_valid()) { |
|
| 2354 | + $valid_data = $form->valid_data(); |
|
| 2355 | + if (isset($valid_data['default_reg_status'])) { |
|
| 2356 | + $registration_config->default_STS_ID = $valid_data['default_reg_status']; |
|
| 2357 | + } |
|
| 2358 | + if (isset($valid_data['default_max_tickets'])) { |
|
| 2359 | + $registration_config->default_maximum_number_of_tickets = $valid_data['default_max_tickets']; |
|
| 2360 | + } |
|
| 2361 | + // update because data was valid! |
|
| 2362 | + EE_Registry::instance()->CFG->update_espresso_config(); |
|
| 2363 | + EE_Error::overwrite_success(); |
|
| 2364 | + EE_Error::add_success( |
|
| 2365 | + __('Default Event Settings were updated', 'event_espresso') |
|
| 2366 | + ); |
|
| 2367 | + } |
|
| 2368 | + } |
|
| 2369 | + $this->_redirect_after_action(0, '', '', array('action' => 'default_event_settings'), true); |
|
| 2370 | + } |
|
| 2371 | + |
|
| 2372 | + |
|
| 2373 | + /************* Templates *************/ |
|
| 2374 | + protected function _template_settings() |
|
| 2375 | + { |
|
| 2376 | + $this->_admin_page_title = esc_html__('Template Settings (Preview)', 'event_espresso'); |
|
| 2377 | + $this->_template_args['preview_img'] = '<img src="' |
|
| 2378 | + . EVENTS_ASSETS_URL |
|
| 2379 | + . DS |
|
| 2380 | + . 'images' |
|
| 2381 | + . DS |
|
| 2382 | + . 'caffeinated_template_features.jpg" alt="' |
|
| 2383 | + . esc_attr__('Template Settings Preview screenshot', 'event_espresso') |
|
| 2384 | + . '" />'; |
|
| 2385 | + $this->_template_args['preview_text'] = '<strong>' |
|
| 2386 | + . esc_html__( |
|
| 2387 | + 'Template Settings is a feature that is only available in the premium version of Event Espresso 4 which is available with a support license purchase on EventEspresso.com. Template Settings allow you to configure some of the appearance options for both the Event List and Event Details pages.', |
|
| 2388 | + 'event_espresso' |
|
| 2389 | + ) . '</strong>'; |
|
| 2390 | + $this->display_admin_caf_preview_page('template_settings_tab'); |
|
| 2391 | + } |
|
| 2392 | + |
|
| 2393 | + |
|
| 2394 | + /** Event Category Stuff **/ |
|
| 2395 | + /** |
|
| 2396 | + * set the _category property with the category object for the loaded page. |
|
| 2397 | + * |
|
| 2398 | + * @access private |
|
| 2399 | + * @return void |
|
| 2400 | + */ |
|
| 2401 | + private function _set_category_object() |
|
| 2402 | + { |
|
| 2403 | + if (isset($this->_category->id) && ! empty($this->_category->id)) { |
|
| 2404 | + return; |
|
| 2405 | + } //already have the category object so get out. |
|
| 2406 | + // set default category object |
|
| 2407 | + $this->_set_empty_category_object(); |
|
| 2408 | + // only set if we've got an id |
|
| 2409 | + if (! isset($this->_req_data['EVT_CAT_ID'])) { |
|
| 2410 | + return; |
|
| 2411 | + } |
|
| 2412 | + $category_id = absint($this->_req_data['EVT_CAT_ID']); |
|
| 2413 | + $term = get_term($category_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
| 2414 | + if (! empty($term)) { |
|
| 2415 | + $this->_category->category_name = $term->name; |
|
| 2416 | + $this->_category->category_identifier = $term->slug; |
|
| 2417 | + $this->_category->category_desc = $term->description; |
|
| 2418 | + $this->_category->id = $term->term_id; |
|
| 2419 | + $this->_category->parent = $term->parent; |
|
| 2420 | + } |
|
| 2421 | + } |
|
| 2422 | + |
|
| 2423 | + |
|
| 2424 | + /** |
|
| 2425 | + * Clears out category properties. |
|
| 2426 | + */ |
|
| 2427 | + private function _set_empty_category_object() |
|
| 2428 | + { |
|
| 2429 | + $this->_category = new stdClass(); |
|
| 2430 | + $this->_category->category_name = $this->_category->category_identifier = $this->_category->category_desc = ''; |
|
| 2431 | + $this->_category->id = $this->_category->parent = 0; |
|
| 2432 | + } |
|
| 2433 | + |
|
| 2434 | + |
|
| 2435 | + /** |
|
| 2436 | + * @throws EE_Error |
|
| 2437 | + */ |
|
| 2438 | + protected function _category_list_table() |
|
| 2439 | + { |
|
| 2440 | + do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
|
| 2441 | + $this->_search_btn_label = esc_html__('Categories', 'event_espresso'); |
|
| 2442 | + $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
| 2443 | + 'add_category', |
|
| 2444 | + 'add_category', |
|
| 2445 | + array(), |
|
| 2446 | + 'add-new-h2' |
|
| 2447 | + ); |
|
| 2448 | + $this->display_admin_list_table_page_with_sidebar(); |
|
| 2449 | + } |
|
| 2450 | + |
|
| 2451 | + |
|
| 2452 | + /** |
|
| 2453 | + * Output category details view. |
|
| 2454 | + */ |
|
| 2455 | + protected function _category_details($view) |
|
| 2456 | + { |
|
| 2457 | + // load formatter helper |
|
| 2458 | + // load field generator helper |
|
| 2459 | + $route = $view == 'edit' ? 'update_category' : 'insert_category'; |
|
| 2460 | + $this->_set_add_edit_form_tags($route); |
|
| 2461 | + $this->_set_category_object(); |
|
| 2462 | + $id = ! empty($this->_category->id) ? $this->_category->id : ''; |
|
| 2463 | + $delete_action = 'delete_category'; |
|
| 2464 | + // custom redirect |
|
| 2465 | + $redirect = EE_Admin_Page::add_query_args_and_nonce( |
|
| 2466 | + array('action' => 'category_list'), |
|
| 2467 | + $this->_admin_base_url |
|
| 2468 | + ); |
|
| 2469 | + $this->_set_publish_post_box_vars('EVT_CAT_ID', $id, $delete_action, $redirect); |
|
| 2470 | + // take care of contents |
|
| 2471 | + $this->_template_args['admin_page_content'] = $this->_category_details_content(); |
|
| 2472 | + $this->display_admin_page_with_sidebar(); |
|
| 2473 | + } |
|
| 2474 | + |
|
| 2475 | + |
|
| 2476 | + /** |
|
| 2477 | + * Output category details content. |
|
| 2478 | + */ |
|
| 2479 | + protected function _category_details_content() |
|
| 2480 | + { |
|
| 2481 | + $editor_args['category_desc'] = array( |
|
| 2482 | + 'type' => 'wp_editor', |
|
| 2483 | + 'value' => EEH_Formatter::admin_format_content($this->_category->category_desc), |
|
| 2484 | + 'class' => 'my_editor_custom', |
|
| 2485 | + 'wpeditor_args' => array('media_buttons' => false), |
|
| 2486 | + ); |
|
| 2487 | + $_wp_editor = $this->_generate_admin_form_fields($editor_args, 'array'); |
|
| 2488 | + $all_terms = get_terms( |
|
| 2489 | + array(EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY), |
|
| 2490 | + array('hide_empty' => 0, 'exclude' => array($this->_category->id)) |
|
| 2491 | + ); |
|
| 2492 | + // setup category select for term parents. |
|
| 2493 | + $category_select_values[] = array( |
|
| 2494 | + 'text' => esc_html__('No Parent', 'event_espresso'), |
|
| 2495 | + 'id' => 0, |
|
| 2496 | + ); |
|
| 2497 | + foreach ($all_terms as $term) { |
|
| 2498 | + $category_select_values[] = array( |
|
| 2499 | + 'text' => $term->name, |
|
| 2500 | + 'id' => $term->term_id, |
|
| 2501 | + ); |
|
| 2502 | + } |
|
| 2503 | + $category_select = EEH_Form_Fields::select_input( |
|
| 2504 | + 'category_parent', |
|
| 2505 | + $category_select_values, |
|
| 2506 | + $this->_category->parent |
|
| 2507 | + ); |
|
| 2508 | + $template_args = array( |
|
| 2509 | + 'category' => $this->_category, |
|
| 2510 | + 'category_select' => $category_select, |
|
| 2511 | + 'unique_id_info_help_link' => $this->_get_help_tab_link('unique_id_info'), |
|
| 2512 | + 'category_desc_editor' => $_wp_editor['category_desc']['field'], |
|
| 2513 | + 'disable' => '', |
|
| 2514 | + 'disabled_message' => false, |
|
| 2515 | + ); |
|
| 2516 | + $template = EVENTS_TEMPLATE_PATH . 'event_category_details.template.php'; |
|
| 2517 | + return EEH_Template::display_template($template, $template_args, true); |
|
| 2518 | + } |
|
| 2519 | + |
|
| 2520 | + |
|
| 2521 | + /** |
|
| 2522 | + * Handles deleting categories. |
|
| 2523 | + */ |
|
| 2524 | + protected function _delete_categories() |
|
| 2525 | + { |
|
| 2526 | + $cat_ids = isset($this->_req_data['EVT_CAT_ID']) ? (array) $this->_req_data['EVT_CAT_ID'] |
|
| 2527 | + : (array) $this->_req_data['category_id']; |
|
| 2528 | + foreach ($cat_ids as $cat_id) { |
|
| 2529 | + $this->_delete_category($cat_id); |
|
| 2530 | + } |
|
| 2531 | + // doesn't matter what page we're coming from... we're going to the same place after delete. |
|
| 2532 | + $query_args = array( |
|
| 2533 | + 'action' => 'category_list', |
|
| 2534 | + ); |
|
| 2535 | + $this->_redirect_after_action(0, '', '', $query_args); |
|
| 2536 | + } |
|
| 2537 | + |
|
| 2538 | + |
|
| 2539 | + /** |
|
| 2540 | + * Handles deleting specific category. |
|
| 2541 | + * |
|
| 2542 | + * @param int $cat_id |
|
| 2543 | + */ |
|
| 2544 | + protected function _delete_category($cat_id) |
|
| 2545 | + { |
|
| 2546 | + $cat_id = absint($cat_id); |
|
| 2547 | + wp_delete_term($cat_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
| 2548 | + } |
|
| 2549 | + |
|
| 2550 | + |
|
| 2551 | + /** |
|
| 2552 | + * Handles triggering the update or insertion of a new category. |
|
| 2553 | + * |
|
| 2554 | + * @param bool $new_category true means we're triggering the insert of a new category. |
|
| 2555 | + */ |
|
| 2556 | + protected function _insert_or_update_category($new_category) |
|
| 2557 | + { |
|
| 2558 | + $cat_id = $new_category ? $this->_insert_category() : $this->_insert_category(true); |
|
| 2559 | + $success = 0; // we already have a success message so lets not send another. |
|
| 2560 | + if ($cat_id) { |
|
| 2561 | + $query_args = array( |
|
| 2562 | + 'action' => 'edit_category', |
|
| 2563 | + 'EVT_CAT_ID' => $cat_id, |
|
| 2564 | + ); |
|
| 2565 | + } else { |
|
| 2566 | + $query_args = array('action' => 'add_category'); |
|
| 2567 | + } |
|
| 2568 | + $this->_redirect_after_action($success, '', '', $query_args, true); |
|
| 2569 | + } |
|
| 2570 | + |
|
| 2571 | + |
|
| 2572 | + /** |
|
| 2573 | + * Inserts or updates category |
|
| 2574 | + * |
|
| 2575 | + * @param bool $update (true indicates we're updating a category). |
|
| 2576 | + * @return bool|mixed|string |
|
| 2577 | + */ |
|
| 2578 | + private function _insert_category($update = false) |
|
| 2579 | + { |
|
| 2580 | + $cat_id = $update ? $this->_req_data['EVT_CAT_ID'] : ''; |
|
| 2581 | + $category_name = isset($this->_req_data['category_name']) ? $this->_req_data['category_name'] : ''; |
|
| 2582 | + $category_desc = isset($this->_req_data['category_desc']) ? $this->_req_data['category_desc'] : ''; |
|
| 2583 | + $category_parent = isset($this->_req_data['category_parent']) ? $this->_req_data['category_parent'] : 0; |
|
| 2584 | + if (empty($category_name)) { |
|
| 2585 | + $msg = esc_html__('You must add a name for the category.', 'event_espresso'); |
|
| 2586 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2587 | + return false; |
|
| 2588 | + } |
|
| 2589 | + $term_args = array( |
|
| 2590 | + 'name' => $category_name, |
|
| 2591 | + 'description' => $category_desc, |
|
| 2592 | + 'parent' => $category_parent, |
|
| 2593 | + ); |
|
| 2594 | + // was the category_identifier input disabled? |
|
| 2595 | + if (isset($this->_req_data['category_identifier'])) { |
|
| 2596 | + $term_args['slug'] = $this->_req_data['category_identifier']; |
|
| 2597 | + } |
|
| 2598 | + $insert_ids = $update |
|
| 2599 | + ? wp_update_term($cat_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args) |
|
| 2600 | + : wp_insert_term($category_name, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args); |
|
| 2601 | + if (! is_array($insert_ids)) { |
|
| 2602 | + $msg = esc_html__( |
|
| 2603 | + 'An error occurred and the category has not been saved to the database.', |
|
| 2604 | + 'event_espresso' |
|
| 2605 | + ); |
|
| 2606 | + EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
|
| 2607 | + } else { |
|
| 2608 | + $cat_id = $insert_ids['term_id']; |
|
| 2609 | + $msg = sprintf(esc_html__('The category %s was successfully saved', 'event_espresso'), $category_name); |
|
| 2610 | + EE_Error::add_success($msg); |
|
| 2611 | + } |
|
| 2612 | + return $cat_id; |
|
| 2613 | + } |
|
| 2614 | + |
|
| 2615 | + |
|
| 2616 | + /** |
|
| 2617 | + * Gets categories or count of categories matching the arguments in the request. |
|
| 2618 | + * |
|
| 2619 | + * @param int $per_page |
|
| 2620 | + * @param int $current_page |
|
| 2621 | + * @param bool $count |
|
| 2622 | + * @return EE_Base_Class[]|EE_Term_Taxonomy[]|int |
|
| 2623 | + */ |
|
| 2624 | + public function get_categories($per_page = 10, $current_page = 1, $count = false) |
|
| 2625 | + { |
|
| 2626 | + // testing term stuff |
|
| 2627 | + $orderby = isset($this->_req_data['orderby']) ? $this->_req_data['orderby'] : 'Term.term_id'; |
|
| 2628 | + $order = isset($this->_req_data['order']) ? $this->_req_data['order'] : 'DESC'; |
|
| 2629 | + $limit = ($current_page - 1) * $per_page; |
|
| 2630 | + $where = array('taxonomy' => EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
|
| 2631 | + if (isset($this->_req_data['s'])) { |
|
| 2632 | + $sstr = '%' . $this->_req_data['s'] . '%'; |
|
| 2633 | + $where['OR'] = array( |
|
| 2634 | + 'Term.name' => array('LIKE', $sstr), |
|
| 2635 | + 'description' => array('LIKE', $sstr), |
|
| 2636 | + ); |
|
| 2637 | + } |
|
| 2638 | + $query_params = array( |
|
| 2639 | + $where, |
|
| 2640 | + 'order_by' => array($orderby => $order), |
|
| 2641 | + 'limit' => $limit . ',' . $per_page, |
|
| 2642 | + 'force_join' => array('Term'), |
|
| 2643 | + ); |
|
| 2644 | + $categories = $count |
|
| 2645 | + ? EEM_Term_Taxonomy::instance()->count($query_params, 'term_id') |
|
| 2646 | + : EEM_Term_Taxonomy::instance()->get_all($query_params); |
|
| 2647 | + return $categories; |
|
| 2648 | + } |
|
| 2649 | + |
|
| 2650 | + /* end category stuff */ |
|
| 2651 | + /**************/ |
|
| 2652 | + |
|
| 2653 | + |
|
| 2654 | + /** |
|
| 2655 | + * Callback for the `ee_save_timezone_setting` ajax action. |
|
| 2656 | + * |
|
| 2657 | + * @throws EE_Error |
|
| 2658 | + */ |
|
| 2659 | + public function save_timezonestring_setting() |
|
| 2660 | + { |
|
| 2661 | + $timezone_string = isset($this->_req_data['timezone_selected']) |
|
| 2662 | + ? $this->_req_data['timezone_selected'] |
|
| 2663 | + : ''; |
|
| 2664 | + if (empty($timezone_string) || ! EEH_DTT_Helper::validate_timezone($timezone_string, false)) { |
|
| 2665 | + EE_Error::add_error( |
|
| 2666 | + esc_html__('An invalid timezone string submitted.', 'event_espresso'), |
|
| 2667 | + __FILE__, |
|
| 2668 | + __FUNCTION__, |
|
| 2669 | + __LINE__ |
|
| 2670 | + ); |
|
| 2671 | + $this->_template_args['error'] = true; |
|
| 2672 | + $this->_return_json(); |
|
| 2673 | + } |
|
| 2674 | + |
|
| 2675 | + update_option('timezone_string', $timezone_string); |
|
| 2676 | + EE_Error::add_success( |
|
| 2677 | + esc_html__('Your timezone string was updated.', 'event_espresso') |
|
| 2678 | + ); |
|
| 2679 | + $this->_template_args['success'] = true; |
|
| 2680 | + $this->_return_json(true, array('action' => 'create_new')); |
|
| 2681 | + } |
|
| 2682 | 2682 | } |
@@ -548,11 +548,11 @@ discard block |
||
| 548 | 548 | { |
| 549 | 549 | wp_register_style( |
| 550 | 550 | 'events-admin-css', |
| 551 | - EVENTS_ASSETS_URL . 'events-admin-page.css', |
|
| 551 | + EVENTS_ASSETS_URL.'events-admin-page.css', |
|
| 552 | 552 | array(), |
| 553 | 553 | EVENT_ESPRESSO_VERSION |
| 554 | 554 | ); |
| 555 | - wp_register_style('ee-cat-admin', EVENTS_ASSETS_URL . 'ee-cat-admin.css', array(), EVENT_ESPRESSO_VERSION); |
|
| 555 | + wp_register_style('ee-cat-admin', EVENTS_ASSETS_URL.'ee-cat-admin.css', array(), EVENT_ESPRESSO_VERSION); |
|
| 556 | 556 | wp_enqueue_style('events-admin-css'); |
| 557 | 557 | wp_enqueue_style('ee-cat-admin'); |
| 558 | 558 | // todo note: we also need to load_scripts_styles per view (i.e. default/view_report/event_details |
@@ -560,7 +560,7 @@ discard block |
||
| 560 | 560 | // scripts |
| 561 | 561 | wp_register_script( |
| 562 | 562 | 'event_editor_js', |
| 563 | - EVENTS_ASSETS_URL . 'event_editor.js', |
|
| 563 | + EVENTS_ASSETS_URL.'event_editor.js', |
|
| 564 | 564 | array('ee_admin_js', 'jquery-ui-slider', 'jquery-ui-timepicker-addon'), |
| 565 | 565 | EVENT_ESPRESSO_VERSION, |
| 566 | 566 | true |
@@ -586,7 +586,7 @@ discard block |
||
| 586 | 586 | wp_enqueue_style('espresso-ui-theme'); |
| 587 | 587 | wp_register_style( |
| 588 | 588 | 'event-editor-css', |
| 589 | - EVENTS_ASSETS_URL . 'event-editor.css', |
|
| 589 | + EVENTS_ASSETS_URL.'event-editor.css', |
|
| 590 | 590 | array('ee-admin-css'), |
| 591 | 591 | EVENT_ESPRESSO_VERSION |
| 592 | 592 | ); |
@@ -594,7 +594,7 @@ discard block |
||
| 594 | 594 | // scripts |
| 595 | 595 | wp_register_script( |
| 596 | 596 | 'event-datetime-metabox', |
| 597 | - EVENTS_ASSETS_URL . 'event-datetime-metabox.js', |
|
| 597 | + EVENTS_ASSETS_URL.'event-datetime-metabox.js', |
|
| 598 | 598 | array('event_editor_js', 'ee-datepicker'), |
| 599 | 599 | EVENT_ESPRESSO_VERSION |
| 600 | 600 | ); |
@@ -663,7 +663,7 @@ discard block |
||
| 663 | 663 | public function verify_event_edit($event = null, $req_type = '') |
| 664 | 664 | { |
| 665 | 665 | // don't need to do this when processing |
| 666 | - if (! empty($req_type)) { |
|
| 666 | + if ( ! empty($req_type)) { |
|
| 667 | 667 | return; |
| 668 | 668 | } |
| 669 | 669 | // no event? |
@@ -672,7 +672,7 @@ discard block |
||
| 672 | 672 | $event = $this->_cpt_model_obj; |
| 673 | 673 | } |
| 674 | 674 | // STILL no event? |
| 675 | - if (! $event instanceof EE_Event) { |
|
| 675 | + if ( ! $event instanceof EE_Event) { |
|
| 676 | 676 | return; |
| 677 | 677 | } |
| 678 | 678 | $orig_status = $event->status(); |
@@ -710,7 +710,7 @@ discard block |
||
| 710 | 710 | ); |
| 711 | 711 | } |
| 712 | 712 | // now we need to determine if the event has any tickets on sale. If not then we dont' show the error |
| 713 | - if (! $event->tickets_on_sale()) { |
|
| 713 | + if ( ! $event->tickets_on_sale()) { |
|
| 714 | 714 | return; |
| 715 | 715 | } |
| 716 | 716 | // made it here so show warning |
@@ -755,7 +755,7 @@ discard block |
||
| 755 | 755 | { |
| 756 | 756 | $has_timezone_string = get_option('timezone_string'); |
| 757 | 757 | // only nag them about setting their timezone if it's their first event, and they haven't already done it |
| 758 | - if (! $has_timezone_string && ! EEM_Event::instance()->exists(array())) { |
|
| 758 | + if ( ! $has_timezone_string && ! EEM_Event::instance()->exists(array())) { |
|
| 759 | 759 | EE_Error::add_attention( |
| 760 | 760 | sprintf( |
| 761 | 761 | __( |
@@ -839,31 +839,31 @@ discard block |
||
| 839 | 839 | $items = apply_filters('FHEE__Events_Admin_Page___event_legend_items__items', $items); |
| 840 | 840 | $statuses = array( |
| 841 | 841 | 'sold_out_status' => array( |
| 842 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::sold_out, |
|
| 842 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::sold_out, |
|
| 843 | 843 | 'desc' => EEH_Template::pretty_status(EE_Datetime::sold_out, false, 'sentence'), |
| 844 | 844 | ), |
| 845 | 845 | 'active_status' => array( |
| 846 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::active, |
|
| 846 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::active, |
|
| 847 | 847 | 'desc' => EEH_Template::pretty_status(EE_Datetime::active, false, 'sentence'), |
| 848 | 848 | ), |
| 849 | 849 | 'upcoming_status' => array( |
| 850 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::upcoming, |
|
| 850 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::upcoming, |
|
| 851 | 851 | 'desc' => EEH_Template::pretty_status(EE_Datetime::upcoming, false, 'sentence'), |
| 852 | 852 | ), |
| 853 | 853 | 'postponed_status' => array( |
| 854 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::postponed, |
|
| 854 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::postponed, |
|
| 855 | 855 | 'desc' => EEH_Template::pretty_status(EE_Datetime::postponed, false, 'sentence'), |
| 856 | 856 | ), |
| 857 | 857 | 'cancelled_status' => array( |
| 858 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::cancelled, |
|
| 858 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::cancelled, |
|
| 859 | 859 | 'desc' => EEH_Template::pretty_status(EE_Datetime::cancelled, false, 'sentence'), |
| 860 | 860 | ), |
| 861 | 861 | 'expired_status' => array( |
| 862 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::expired, |
|
| 862 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::expired, |
|
| 863 | 863 | 'desc' => EEH_Template::pretty_status(EE_Datetime::expired, false, 'sentence'), |
| 864 | 864 | ), |
| 865 | 865 | 'inactive_status' => array( |
| 866 | - 'class' => 'ee-status-legend ee-status-legend-' . EE_Datetime::inactive, |
|
| 866 | + 'class' => 'ee-status-legend ee-status-legend-'.EE_Datetime::inactive, |
|
| 867 | 867 | 'desc' => EEH_Template::pretty_status(EE_Datetime::inactive, false, 'sentence'), |
| 868 | 868 | ), |
| 869 | 869 | ); |
@@ -877,7 +877,7 @@ discard block |
||
| 877 | 877 | */ |
| 878 | 878 | private function _event_model() |
| 879 | 879 | { |
| 880 | - if (! $this->_event_model instanceof EEM_Event) { |
|
| 880 | + if ( ! $this->_event_model instanceof EEM_Event) { |
|
| 881 | 881 | $this->_event_model = EE_Registry::instance()->load_model('Event'); |
| 882 | 882 | } |
| 883 | 883 | return $this->_event_model; |
@@ -897,7 +897,7 @@ discard block |
||
| 897 | 897 | public function extra_permalink_field_buttons($return, $id, $new_title, $new_slug) |
| 898 | 898 | { |
| 899 | 899 | // make sure this is only when editing |
| 900 | - if (! empty($id)) { |
|
| 900 | + if ( ! empty($id)) { |
|
| 901 | 901 | $post = get_post($id); |
| 902 | 902 | $return .= '<a class="button button-small" onclick="prompt(\'Shortcode:\', jQuery(\'#shortcode\').val()); return false;" href="#" tabindex="-1">' |
| 903 | 903 | . esc_html__('Shortcode', 'event_espresso') |
@@ -931,7 +931,7 @@ discard block |
||
| 931 | 931 | 'button' |
| 932 | 932 | ); |
| 933 | 933 | $this->_template_args['after_list_table']['legend'] = $this->_display_legend($this->_event_legend_items()); |
| 934 | - $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
| 934 | + $this->_admin_page_title .= ' '.$this->get_action_link_or_button( |
|
| 935 | 935 | 'create_new', |
| 936 | 936 | 'add', |
| 937 | 937 | array(), |
@@ -1071,7 +1071,7 @@ discard block |
||
| 1071 | 1071 | */ |
| 1072 | 1072 | protected function _default_venue_update(\EE_Event $evtobj, $data) |
| 1073 | 1073 | { |
| 1074 | - require_once(EE_MODELS . 'EEM_Venue.model.php'); |
|
| 1074 | + require_once(EE_MODELS.'EEM_Venue.model.php'); |
|
| 1075 | 1075 | $venue_model = EE_Registry::instance()->load_model('Venue'); |
| 1076 | 1076 | $rows_affected = null; |
| 1077 | 1077 | $venue_id = ! empty($data['venue_id']) ? $data['venue_id'] : null; |
@@ -1102,7 +1102,7 @@ discard block |
||
| 1102 | 1102 | 'status' => 'publish', |
| 1103 | 1103 | ); |
| 1104 | 1104 | // if we've got the venue_id then we're just updating the existing venue so let's do that and then get out. |
| 1105 | - if (! empty($venue_id)) { |
|
| 1105 | + if ( ! empty($venue_id)) { |
|
| 1106 | 1106 | $update_where = array($venue_model->primary_key_name() => $venue_id); |
| 1107 | 1107 | $rows_affected = $venue_model->update($venue_array, array($update_where)); |
| 1108 | 1108 | // we've gotta make sure that the venue is always attached to a revision.. add_relation_to should take care of making sure that the relation is already present. |
@@ -1144,7 +1144,7 @@ discard block |
||
| 1144 | 1144 | 'DTT_order' => $row, |
| 1145 | 1145 | ); |
| 1146 | 1146 | // if we have an id then let's get existing object first and then set the new values. Otherwise we instantiate a new object for save. |
| 1147 | - if (! empty($dtt['DTT_ID'])) { |
|
| 1147 | + if ( ! empty($dtt['DTT_ID'])) { |
|
| 1148 | 1148 | $DTM = EE_Registry::instance() |
| 1149 | 1149 | ->load_model('Datetime', array($evtobj->get_timezone())) |
| 1150 | 1150 | ->get_one_by_ID($dtt['DTT_ID']); |
@@ -1154,7 +1154,7 @@ discard block |
||
| 1154 | 1154 | $DTM->set($field, $value); |
| 1155 | 1155 | } |
| 1156 | 1156 | // make sure the $dtt_id here is saved just in case after the add_relation_to() the autosave replaces it. We need to do this so we dont' TRASH the parent DTT. |
| 1157 | - $saved_dtts[ $DTM->ID() ] = $DTM; |
|
| 1157 | + $saved_dtts[$DTM->ID()] = $DTM; |
|
| 1158 | 1158 | } else { |
| 1159 | 1159 | $DTM = EE_Registry::instance()->load_class( |
| 1160 | 1160 | 'Datetime', |
@@ -1187,14 +1187,14 @@ discard block |
||
| 1187 | 1187 | foreach ($data['edit_tickets'] as $row => $tkt) { |
| 1188 | 1188 | $incoming_date_formats = array('Y-m-d', 'h:i a'); |
| 1189 | 1189 | $update_prices = false; |
| 1190 | - $ticket_price = isset($data['edit_prices'][ $row ][1]['PRC_amount']) |
|
| 1191 | - ? $data['edit_prices'][ $row ][1]['PRC_amount'] : 0; |
|
| 1190 | + $ticket_price = isset($data['edit_prices'][$row][1]['PRC_amount']) |
|
| 1191 | + ? $data['edit_prices'][$row][1]['PRC_amount'] : 0; |
|
| 1192 | 1192 | // trim inputs to ensure any excess whitespace is removed. |
| 1193 | 1193 | $tkt = array_map('trim', $tkt); |
| 1194 | 1194 | if (empty($tkt['TKT_start_date'])) { |
| 1195 | 1195 | // let's use now in the set timezone. |
| 1196 | 1196 | $now = new DateTime('now', new DateTimeZone($evtobj->get_timezone())); |
| 1197 | - $tkt['TKT_start_date'] = $now->format($incoming_date_formats[0] . ' ' . $incoming_date_formats[1]); |
|
| 1197 | + $tkt['TKT_start_date'] = $now->format($incoming_date_formats[0].' '.$incoming_date_formats[1]); |
|
| 1198 | 1198 | } |
| 1199 | 1199 | if (empty($tkt['TKT_end_date'])) { |
| 1200 | 1200 | // use the start date of the first datetime |
@@ -1229,7 +1229,7 @@ discard block |
||
| 1229 | 1229 | // if we have a TKT_ID then we need to get that existing TKT_obj and update it |
| 1230 | 1230 | // we actually do our saves a head of doing any add_relations to because its entirely possible that this ticket didn't removed or added to any datetime in the session but DID have it's items modified. |
| 1231 | 1231 | // keep in mind that if the TKT has been sold (and we have changed pricing information), then we won't be updating the tkt but instead a new tkt will be created and the old one archived. |
| 1232 | - if (! empty($tkt['TKT_ID'])) { |
|
| 1232 | + if ( ! empty($tkt['TKT_ID'])) { |
|
| 1233 | 1233 | $TKT = EE_Registry::instance() |
| 1234 | 1234 | ->load_model('Ticket', array($evtobj->get_timezone())) |
| 1235 | 1235 | ->get_one_by_ID($tkt['TKT_ID']); |
@@ -1264,7 +1264,7 @@ discard block |
||
| 1264 | 1264 | $TKT->set('TKT_deleted', 1); |
| 1265 | 1265 | $TKT->save(); |
| 1266 | 1266 | // make sure this ticket is still recorded in our saved_tkts so we don't run it through the regular trash routine. |
| 1267 | - $saved_tickets[ $TKT->ID() ] = $TKT; |
|
| 1267 | + $saved_tickets[$TKT->ID()] = $TKT; |
|
| 1268 | 1268 | // create new ticket that's a copy of the existing except a new id of course (and not archived) AND has the new TKT_price associated with it. |
| 1269 | 1269 | $TKT = clone $TKT; |
| 1270 | 1270 | $TKT->set('TKT_ID', 0); |
@@ -1309,9 +1309,9 @@ discard block |
||
| 1309 | 1309 | } |
| 1310 | 1310 | // initially let's add the ticket to the dtt |
| 1311 | 1311 | $saved_dtt->_add_relation_to($TKT, 'Ticket'); |
| 1312 | - $saved_tickets[ $TKT->ID() ] = $TKT; |
|
| 1312 | + $saved_tickets[$TKT->ID()] = $TKT; |
|
| 1313 | 1313 | // add prices to ticket |
| 1314 | - $this->_add_prices_to_ticket($data['edit_prices'][ $row ], $TKT, $update_prices); |
|
| 1314 | + $this->_add_prices_to_ticket($data['edit_prices'][$row], $TKT, $update_prices); |
|
| 1315 | 1315 | } |
| 1316 | 1316 | // however now we need to handle permanently deleting tickets via the ui. Keep in mind that the ui does not allow deleting/archiving tickets that have ticket sold. However, it does allow for deleting tickets that have no tickets sold, in which case we want to get rid of permanently because there is no need to save in db. |
| 1317 | 1317 | $old_tickets = isset($old_tickets[0]) && $old_tickets[0] == '' ? array() : $old_tickets; |
@@ -1468,7 +1468,7 @@ discard block |
||
| 1468 | 1468 | $publish_box_extra_args['event_editor_overview_add'] = ob_get_clean(); |
| 1469 | 1469 | // load template |
| 1470 | 1470 | EEH_Template::display_template( |
| 1471 | - EVENTS_TEMPLATE_PATH . 'event_publish_box_extras.template.php', |
|
| 1471 | + EVENTS_TEMPLATE_PATH.'event_publish_box_extras.template.php', |
|
| 1472 | 1472 | $publish_box_extra_args |
| 1473 | 1473 | ); |
| 1474 | 1474 | } |
@@ -1560,7 +1560,7 @@ discard block |
||
| 1560 | 1560 | 'default_where_conditions' => 'none', |
| 1561 | 1561 | ) |
| 1562 | 1562 | ); |
| 1563 | - if (! empty($related_tickets)) { |
|
| 1563 | + if ( ! empty($related_tickets)) { |
|
| 1564 | 1564 | $template_args['total_ticket_rows'] = count($related_tickets); |
| 1565 | 1565 | $row = 0; |
| 1566 | 1566 | foreach ($related_tickets as $ticket) { |
@@ -1594,7 +1594,7 @@ discard block |
||
| 1594 | 1594 | ); |
| 1595 | 1595 | $template = apply_filters( |
| 1596 | 1596 | 'FHEE__Events_Admin_Page__ticket_metabox__template', |
| 1597 | - EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_main.template.php' |
|
| 1597 | + EVENTS_TEMPLATE_PATH.'event_tickets_metabox_main.template.php' |
|
| 1598 | 1598 | ); |
| 1599 | 1599 | EEH_Template::display_template($template, $template_args); |
| 1600 | 1600 | } |
@@ -1612,7 +1612,7 @@ discard block |
||
| 1612 | 1612 | private function _get_ticket_row($ticket, $skeleton = false, $row = 0) |
| 1613 | 1613 | { |
| 1614 | 1614 | $template_args = array( |
| 1615 | - 'tkt_status_class' => ' tkt-status-' . $ticket->ticket_status(), |
|
| 1615 | + 'tkt_status_class' => ' tkt-status-'.$ticket->ticket_status(), |
|
| 1616 | 1616 | 'tkt_archive_class' => $ticket->ticket_status() === EE_Ticket::archived && ! $skeleton ? ' tkt-archived' |
| 1617 | 1617 | : '', |
| 1618 | 1618 | 'ticketrow' => $skeleton ? 'TICKETNUM' : $row, |
@@ -1624,10 +1624,10 @@ discard block |
||
| 1624 | 1624 | 'TKT_qty' => $ticket->get_pretty('TKT_qty', 'input'), |
| 1625 | 1625 | 'edit_ticketrow_name' => $skeleton ? 'TICKETNAMEATTR' : 'edit_tickets', |
| 1626 | 1626 | 'TKT_sold' => $skeleton ? 0 : $ticket->get('TKT_sold'), |
| 1627 | - 'trash_icon' => ($skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted'))) |
|
| 1628 | - && (! empty($ticket) && $ticket->get('TKT_sold') === 0) |
|
| 1627 | + 'trash_icon' => ($skeleton || ( ! empty($ticket) && ! $ticket->get('TKT_deleted'))) |
|
| 1628 | + && ( ! empty($ticket) && $ticket->get('TKT_sold') === 0) |
|
| 1629 | 1629 | ? 'trash-icon dashicons dashicons-post-trash clickable' : 'ee-lock-icon', |
| 1630 | - 'disabled' => $skeleton || (! empty($ticket) && ! $ticket->get('TKT_deleted')) ? '' |
|
| 1630 | + 'disabled' => $skeleton || ( ! empty($ticket) && ! $ticket->get('TKT_deleted')) ? '' |
|
| 1631 | 1631 | : ' disabled=disabled', |
| 1632 | 1632 | ); |
| 1633 | 1633 | $price = $ticket->ID() !== 0 |
@@ -1654,7 +1654,7 @@ discard block |
||
| 1654 | 1654 | array('order_by' => array('DTT_EVT_start' => 'ASC')) |
| 1655 | 1655 | ) |
| 1656 | 1656 | : null; |
| 1657 | - if (! empty($earliest_dtt)) { |
|
| 1657 | + if ( ! empty($earliest_dtt)) { |
|
| 1658 | 1658 | $template_args['TKT_end_date'] = $earliest_dtt->get_datetime('DTT_EVT_start', 'Y-m-d', 'h:i a'); |
| 1659 | 1659 | } else { |
| 1660 | 1660 | $template_args['TKT_end_date'] = date( |
@@ -1666,7 +1666,7 @@ discard block |
||
| 1666 | 1666 | $template_args = array_merge($template_args, $price_args); |
| 1667 | 1667 | $template = apply_filters( |
| 1668 | 1668 | 'FHEE__Events_Admin_Page__get_ticket_row__template', |
| 1669 | - EVENTS_TEMPLATE_PATH . 'event_tickets_metabox_ticket_row.template.php', |
|
| 1669 | + EVENTS_TEMPLATE_PATH.'event_tickets_metabox_ticket_row.template.php', |
|
| 1670 | 1670 | $ticket |
| 1671 | 1671 | ); |
| 1672 | 1672 | return EEH_Template::display_template($template, $template_args, true); |
@@ -1720,7 +1720,7 @@ discard block |
||
| 1720 | 1720 | $default_reg_status_values |
| 1721 | 1721 | ); |
| 1722 | 1722 | EEH_Template::display_template( |
| 1723 | - EVENTS_TEMPLATE_PATH . 'event_registration_options.template.php', |
|
| 1723 | + EVENTS_TEMPLATE_PATH.'event_registration_options.template.php', |
|
| 1724 | 1724 | $template_args |
| 1725 | 1725 | ); |
| 1726 | 1726 | } |
@@ -1742,7 +1742,7 @@ discard block |
||
| 1742 | 1742 | { |
| 1743 | 1743 | $EEME = $this->_event_model(); |
| 1744 | 1744 | $offset = ($current_page - 1) * $per_page; |
| 1745 | - $limit = $count ? null : $offset . ',' . $per_page; |
|
| 1745 | + $limit = $count ? null : $offset.','.$per_page; |
|
| 1746 | 1746 | $orderby = isset($this->_req_data['orderby']) ? $this->_req_data['orderby'] : 'EVT_ID'; |
| 1747 | 1747 | $order = isset($this->_req_data['order']) ? $this->_req_data['order'] : "DESC"; |
| 1748 | 1748 | if (isset($this->_req_data['month_range'])) { |
@@ -1771,7 +1771,7 @@ discard block |
||
| 1771 | 1771 | // categories? |
| 1772 | 1772 | $category = isset($this->_req_data['EVT_CAT']) && $this->_req_data['EVT_CAT'] > 0 |
| 1773 | 1773 | ? $this->_req_data['EVT_CAT'] : null; |
| 1774 | - if (! empty($category)) { |
|
| 1774 | + if ( ! empty($category)) { |
|
| 1775 | 1775 | $where['Term_Taxonomy.taxonomy'] = EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY; |
| 1776 | 1776 | $where['Term_Taxonomy.term_id'] = $category; |
| 1777 | 1777 | } |
@@ -1779,7 +1779,7 @@ discard block |
||
| 1779 | 1779 | $start_formats = EEM_Datetime::instance()->get_formats_for('DTT_EVT_start'); |
| 1780 | 1780 | if (isset($this->_req_data['month_range']) && $this->_req_data['month_range'] != '') { |
| 1781 | 1781 | $DateTime = new DateTime( |
| 1782 | - $year_r . '-' . $month_r . '-01 00:00:00', |
|
| 1782 | + $year_r.'-'.$month_r.'-01 00:00:00', |
|
| 1783 | 1783 | new DateTimeZone(EEM_Datetime::instance()->get_timezone()) |
| 1784 | 1784 | ); |
| 1785 | 1785 | $start = $DateTime->format(implode(' ', $start_formats)); |
@@ -1805,11 +1805,11 @@ discard block |
||
| 1805 | 1805 | ->format(implode(' ', $start_formats)); |
| 1806 | 1806 | $where['Datetime.DTT_EVT_start'] = array('BETWEEN', array($start, $end)); |
| 1807 | 1807 | } |
| 1808 | - if (! EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events')) { |
|
| 1808 | + if ( ! EE_Registry::instance()->CAP->current_user_can('ee_read_others_events', 'get_events')) { |
|
| 1809 | 1809 | $where['EVT_wp_user'] = get_current_user_id(); |
| 1810 | 1810 | } else { |
| 1811 | - if (! isset($where['status'])) { |
|
| 1812 | - if (! EE_Registry::instance()->CAP->current_user_can('ee_read_private_events', 'get_events')) { |
|
| 1811 | + if ( ! isset($where['status'])) { |
|
| 1812 | + if ( ! EE_Registry::instance()->CAP->current_user_can('ee_read_private_events', 'get_events')) { |
|
| 1813 | 1813 | $where['OR'] = array( |
| 1814 | 1814 | 'status*restrict_private' => array('!=', 'private'), |
| 1815 | 1815 | 'AND' => array( |
@@ -1829,7 +1829,7 @@ discard block |
||
| 1829 | 1829 | } |
| 1830 | 1830 | // search query handling |
| 1831 | 1831 | if (isset($this->_req_data['s'])) { |
| 1832 | - $search_string = '%' . $this->_req_data['s'] . '%'; |
|
| 1832 | + $search_string = '%'.$this->_req_data['s'].'%'; |
|
| 1833 | 1833 | $where['OR'] = array( |
| 1834 | 1834 | 'EVT_name' => array('LIKE', $search_string), |
| 1835 | 1835 | 'EVT_desc' => array('LIKE', $search_string), |
@@ -1918,7 +1918,7 @@ discard block |
||
| 1918 | 1918 | // clean status |
| 1919 | 1919 | $event_status = sanitize_key($event_status); |
| 1920 | 1920 | // grab status |
| 1921 | - if (! empty($event_status)) { |
|
| 1921 | + if ( ! empty($event_status)) { |
|
| 1922 | 1922 | $success = $this->_change_event_status($EVT_ID, $event_status); |
| 1923 | 1923 | } else { |
| 1924 | 1924 | $success = false; |
@@ -1955,7 +1955,7 @@ discard block |
||
| 1955 | 1955 | // clean status |
| 1956 | 1956 | $event_status = sanitize_key($event_status); |
| 1957 | 1957 | // grab status |
| 1958 | - if (! empty($event_status)) { |
|
| 1958 | + if ( ! empty($event_status)) { |
|
| 1959 | 1959 | $success = true; |
| 1960 | 1960 | // determine the event id and set to array. |
| 1961 | 1961 | $EVT_IDs = isset($this->_req_data['EVT_IDs']) ? (array) $this->_req_data['EVT_IDs'] : array(); |
@@ -2002,7 +2002,7 @@ discard block |
||
| 2002 | 2002 | private function _change_event_status($EVT_ID = 0, $event_status = '') |
| 2003 | 2003 | { |
| 2004 | 2004 | // grab event id |
| 2005 | - if (! $EVT_ID) { |
|
| 2005 | + if ( ! $EVT_ID) { |
|
| 2006 | 2006 | $msg = esc_html__( |
| 2007 | 2007 | 'An error occurred. No Event ID or an invalid Event ID was received.', |
| 2008 | 2008 | 'event_espresso' |
@@ -2068,8 +2068,8 @@ discard block |
||
| 2068 | 2068 | // get list of events with no prices |
| 2069 | 2069 | $espresso_no_ticket_prices = get_option('ee_no_ticket_prices', array()); |
| 2070 | 2070 | // remove this event from the list of events with no prices |
| 2071 | - if (isset($espresso_no_ticket_prices[ $EVT_ID ])) { |
|
| 2072 | - unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
| 2071 | + if (isset($espresso_no_ticket_prices[$EVT_ID])) { |
|
| 2072 | + unset($espresso_no_ticket_prices[$EVT_ID]); |
|
| 2073 | 2073 | } |
| 2074 | 2074 | update_option('ee_no_ticket_prices', $espresso_no_ticket_prices); |
| 2075 | 2075 | } else { |
@@ -2111,7 +2111,7 @@ discard block |
||
| 2111 | 2111 | $results = $this->_permanently_delete_event($EVT_ID); |
| 2112 | 2112 | $success = $results !== false ? $success : false; |
| 2113 | 2113 | // remove this event from the list of events with no prices |
| 2114 | - unset($espresso_no_ticket_prices[ $EVT_ID ]); |
|
| 2114 | + unset($espresso_no_ticket_prices[$EVT_ID]); |
|
| 2115 | 2115 | } else { |
| 2116 | 2116 | $success = false; |
| 2117 | 2117 | $msg = esc_html__( |
@@ -2138,7 +2138,7 @@ discard block |
||
| 2138 | 2138 | private function _permanently_delete_event($EVT_ID = 0) |
| 2139 | 2139 | { |
| 2140 | 2140 | // grab event id |
| 2141 | - if (! $EVT_ID) { |
|
| 2141 | + if ( ! $EVT_ID) { |
|
| 2142 | 2142 | $msg = esc_html__( |
| 2143 | 2143 | 'An error occurred. No Event ID or an invalid Event ID was received.', |
| 2144 | 2144 | 'event_espresso' |
@@ -2146,12 +2146,12 @@ discard block |
||
| 2146 | 2146 | EE_Error::add_error($msg, __FILE__, __FUNCTION__, __LINE__); |
| 2147 | 2147 | return false; |
| 2148 | 2148 | } |
| 2149 | - if (! $this->_cpt_model_obj instanceof EE_Event |
|
| 2149 | + if ( ! $this->_cpt_model_obj instanceof EE_Event |
|
| 2150 | 2150 | || $this->_cpt_model_obj->ID() !== $EVT_ID |
| 2151 | 2151 | ) { |
| 2152 | 2152 | $this->_cpt_model_obj = EEM_Event::instance()->get_one_by_ID($EVT_ID); |
| 2153 | 2153 | } |
| 2154 | - if (! $this->_cpt_model_obj instanceof EE_Event) { |
|
| 2154 | + if ( ! $this->_cpt_model_obj instanceof EE_Event) { |
|
| 2155 | 2155 | return false; |
| 2156 | 2156 | } |
| 2157 | 2157 | // need to delete related tickets and prices first. |
@@ -2173,7 +2173,7 @@ discard block |
||
| 2173 | 2173 | } |
| 2174 | 2174 | // any attached question groups? |
| 2175 | 2175 | $question_groups = $this->_cpt_model_obj->get_many_related('Question_Group'); |
| 2176 | - if (! empty($question_groups)) { |
|
| 2176 | + if ( ! empty($question_groups)) { |
|
| 2177 | 2177 | foreach ($question_groups as $question_group) { |
| 2178 | 2178 | $this->_cpt_model_obj->_remove_relation_to($question_group, 'Question_Group'); |
| 2179 | 2179 | } |
@@ -2386,7 +2386,7 @@ discard block |
||
| 2386 | 2386 | . esc_html__( |
| 2387 | 2387 | 'Template Settings is a feature that is only available in the premium version of Event Espresso 4 which is available with a support license purchase on EventEspresso.com. Template Settings allow you to configure some of the appearance options for both the Event List and Event Details pages.', |
| 2388 | 2388 | 'event_espresso' |
| 2389 | - ) . '</strong>'; |
|
| 2389 | + ).'</strong>'; |
|
| 2390 | 2390 | $this->display_admin_caf_preview_page('template_settings_tab'); |
| 2391 | 2391 | } |
| 2392 | 2392 | |
@@ -2406,12 +2406,12 @@ discard block |
||
| 2406 | 2406 | // set default category object |
| 2407 | 2407 | $this->_set_empty_category_object(); |
| 2408 | 2408 | // only set if we've got an id |
| 2409 | - if (! isset($this->_req_data['EVT_CAT_ID'])) { |
|
| 2409 | + if ( ! isset($this->_req_data['EVT_CAT_ID'])) { |
|
| 2410 | 2410 | return; |
| 2411 | 2411 | } |
| 2412 | 2412 | $category_id = absint($this->_req_data['EVT_CAT_ID']); |
| 2413 | 2413 | $term = get_term($category_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
| 2414 | - if (! empty($term)) { |
|
| 2414 | + if ( ! empty($term)) { |
|
| 2415 | 2415 | $this->_category->category_name = $term->name; |
| 2416 | 2416 | $this->_category->category_identifier = $term->slug; |
| 2417 | 2417 | $this->_category->category_desc = $term->description; |
@@ -2439,7 +2439,7 @@ discard block |
||
| 2439 | 2439 | { |
| 2440 | 2440 | do_action('AHEE_log', __FILE__, __FUNCTION__, ''); |
| 2441 | 2441 | $this->_search_btn_label = esc_html__('Categories', 'event_espresso'); |
| 2442 | - $this->_admin_page_title .= ' ' . $this->get_action_link_or_button( |
|
| 2442 | + $this->_admin_page_title .= ' '.$this->get_action_link_or_button( |
|
| 2443 | 2443 | 'add_category', |
| 2444 | 2444 | 'add_category', |
| 2445 | 2445 | array(), |
@@ -2513,7 +2513,7 @@ discard block |
||
| 2513 | 2513 | 'disable' => '', |
| 2514 | 2514 | 'disabled_message' => false, |
| 2515 | 2515 | ); |
| 2516 | - $template = EVENTS_TEMPLATE_PATH . 'event_category_details.template.php'; |
|
| 2516 | + $template = EVENTS_TEMPLATE_PATH.'event_category_details.template.php'; |
|
| 2517 | 2517 | return EEH_Template::display_template($template, $template_args, true); |
| 2518 | 2518 | } |
| 2519 | 2519 | |
@@ -2598,7 +2598,7 @@ discard block |
||
| 2598 | 2598 | $insert_ids = $update |
| 2599 | 2599 | ? wp_update_term($cat_id, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args) |
| 2600 | 2600 | : wp_insert_term($category_name, EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY, $term_args); |
| 2601 | - if (! is_array($insert_ids)) { |
|
| 2601 | + if ( ! is_array($insert_ids)) { |
|
| 2602 | 2602 | $msg = esc_html__( |
| 2603 | 2603 | 'An error occurred and the category has not been saved to the database.', |
| 2604 | 2604 | 'event_espresso' |
@@ -2629,7 +2629,7 @@ discard block |
||
| 2629 | 2629 | $limit = ($current_page - 1) * $per_page; |
| 2630 | 2630 | $where = array('taxonomy' => EEM_CPT_Base::EVENT_CATEGORY_TAXONOMY); |
| 2631 | 2631 | if (isset($this->_req_data['s'])) { |
| 2632 | - $sstr = '%' . $this->_req_data['s'] . '%'; |
|
| 2632 | + $sstr = '%'.$this->_req_data['s'].'%'; |
|
| 2633 | 2633 | $where['OR'] = array( |
| 2634 | 2634 | 'Term.name' => array('LIKE', $sstr), |
| 2635 | 2635 | 'description' => array('LIKE', $sstr), |
@@ -2638,7 +2638,7 @@ discard block |
||
| 2638 | 2638 | $query_params = array( |
| 2639 | 2639 | $where, |
| 2640 | 2640 | 'order_by' => array($orderby => $order), |
| 2641 | - 'limit' => $limit . ',' . $per_page, |
|
| 2641 | + 'limit' => $limit.','.$per_page, |
|
| 2642 | 2642 | 'force_join' => array('Term'), |
| 2643 | 2643 | ); |
| 2644 | 2644 | $categories = $count |