Completed
Branch BUG/pantheon-session-fatal (924226)
by
unknown
55:28 queued 42:17
created
core/db_classes/EE_Line_Item.class.php 3 patches
Doc Comments   +1 added lines, -1 removed lines patch added patch discarded remove patch
@@ -656,7 +656,7 @@
 block discarded – undo
656 656
     /**
657 657
      * Gets the event that's related to the ticket, if this line item represents a ticket.
658 658
      *
659
-     * @return EE_Event|null
659
+     * @return EE_Base_Class|null
660 660
      * @throws EE_Error
661 661
      */
662 662
     public function ticket_event()
Please login to merge, or discard this patch.
Indentation   +1426 added lines, -1426 removed lines patch added patch discarded remove patch
@@ -14,1430 +14,1430 @@
 block discarded – undo
14 14
 class EE_Line_Item extends EE_Base_Class implements EEI_Line_Item
15 15
 {
16 16
 
17
-    /**
18
-     * for children line items (currently not a normal relation)
19
-     *
20
-     * @type EE_Line_Item[]
21
-     */
22
-    protected $_children = array();
23
-
24
-    /**
25
-     * for the parent line item
26
-     *
27
-     * @var EE_Line_Item
28
-     */
29
-    protected $_parent;
30
-
31
-
32
-    /**
33
-     *
34
-     * @param array  $props_n_values          incoming values
35
-     * @param string $timezone                incoming timezone (if not set the timezone set for the website will be
36
-     *                                        used.)
37
-     * @param array  $date_formats            incoming date_formats in an array where the first value is the
38
-     *                                        date_format and the second value is the time format
39
-     * @return EE_Line_Item
40
-     * @throws EE_Error
41
-     */
42
-    public static function new_instance($props_n_values = array(), $timezone = null, $date_formats = array())
43
-    {
44
-        $has_object = parent::_check_for_object($props_n_values, __CLASS__, $timezone, $date_formats);
45
-        return $has_object
46
-            ? $has_object
47
-            : new self($props_n_values, false, $timezone);
48
-    }
49
-
50
-
51
-    /**
52
-     * @param array  $props_n_values  incoming values from the database
53
-     * @param string $timezone        incoming timezone as set by the model.  If not set the timezone for
54
-     *                                the website will be used.
55
-     * @return EE_Line_Item
56
-     * @throws EE_Error
57
-     */
58
-    public static function new_instance_from_db($props_n_values = array(), $timezone = null)
59
-    {
60
-        return new self($props_n_values, true, $timezone);
61
-    }
62
-
63
-
64
-    /**
65
-     * Adds some defaults if they're not specified
66
-     *
67
-     * @param array  $fieldValues
68
-     * @param bool   $bydb
69
-     * @param string $timezone
70
-     * @throws EE_Error
71
-     */
72
-    protected function __construct($fieldValues = array(), $bydb = false, $timezone = '')
73
-    {
74
-        parent::__construct($fieldValues, $bydb, $timezone);
75
-        if (! $this->get('LIN_code')) {
76
-            $this->set_code($this->generate_code());
77
-        }
78
-    }
79
-
80
-
81
-    /**
82
-     * Gets ID
83
-     *
84
-     * @return int
85
-     * @throws EE_Error
86
-     */
87
-    public function ID()
88
-    {
89
-        return $this->get('LIN_ID');
90
-    }
91
-
92
-
93
-    /**
94
-     * Gets TXN_ID
95
-     *
96
-     * @return int
97
-     * @throws EE_Error
98
-     */
99
-    public function TXN_ID()
100
-    {
101
-        return $this->get('TXN_ID');
102
-    }
103
-
104
-
105
-    /**
106
-     * Sets TXN_ID
107
-     *
108
-     * @param int $TXN_ID
109
-     * @throws EE_Error
110
-     */
111
-    public function set_TXN_ID($TXN_ID)
112
-    {
113
-        $this->set('TXN_ID', $TXN_ID);
114
-    }
115
-
116
-
117
-    /**
118
-     * Gets name
119
-     *
120
-     * @return string
121
-     * @throws EE_Error
122
-     */
123
-    public function name()
124
-    {
125
-        $name = $this->get('LIN_name');
126
-        if (! $name) {
127
-            $name = ucwords(str_replace('-', ' ', $this->type()));
128
-        }
129
-        return $name;
130
-    }
131
-
132
-
133
-    /**
134
-     * Sets name
135
-     *
136
-     * @param string $name
137
-     * @throws EE_Error
138
-     */
139
-    public function set_name($name)
140
-    {
141
-        $this->set('LIN_name', $name);
142
-    }
143
-
144
-
145
-    /**
146
-     * Gets desc
147
-     *
148
-     * @return string
149
-     * @throws EE_Error
150
-     */
151
-    public function desc()
152
-    {
153
-        return $this->get('LIN_desc');
154
-    }
155
-
156
-
157
-    /**
158
-     * Sets desc
159
-     *
160
-     * @param string $desc
161
-     * @throws EE_Error
162
-     */
163
-    public function set_desc($desc)
164
-    {
165
-        $this->set('LIN_desc', $desc);
166
-    }
167
-
168
-
169
-    /**
170
-     * Gets quantity
171
-     *
172
-     * @return int
173
-     * @throws EE_Error
174
-     */
175
-    public function quantity()
176
-    {
177
-        return $this->get('LIN_quantity');
178
-    }
179
-
180
-
181
-    /**
182
-     * Sets quantity
183
-     *
184
-     * @param int $quantity
185
-     * @throws EE_Error
186
-     */
187
-    public function set_quantity($quantity)
188
-    {
189
-        $this->set('LIN_quantity', max($quantity, 0));
190
-    }
191
-
192
-
193
-    /**
194
-     * Gets item_id
195
-     *
196
-     * @return string
197
-     * @throws EE_Error
198
-     */
199
-    public function OBJ_ID()
200
-    {
201
-        return $this->get('OBJ_ID');
202
-    }
203
-
204
-
205
-    /**
206
-     * Sets item_id
207
-     *
208
-     * @param string $item_id
209
-     * @throws EE_Error
210
-     */
211
-    public function set_OBJ_ID($item_id)
212
-    {
213
-        $this->set('OBJ_ID', $item_id);
214
-    }
215
-
216
-
217
-    /**
218
-     * Gets item_type
219
-     *
220
-     * @return string
221
-     * @throws EE_Error
222
-     */
223
-    public function OBJ_type()
224
-    {
225
-        return $this->get('OBJ_type');
226
-    }
227
-
228
-
229
-    /**
230
-     * Gets item_type
231
-     *
232
-     * @return string
233
-     * @throws EE_Error
234
-     */
235
-    public function OBJ_type_i18n()
236
-    {
237
-        $obj_type = $this->OBJ_type();
238
-        switch ($obj_type) {
239
-            case 'Event':
240
-                $obj_type = __('Event', 'event_espresso');
241
-                break;
242
-            case 'Price':
243
-                $obj_type = __('Price', 'event_espresso');
244
-                break;
245
-            case 'Promotion':
246
-                $obj_type = __('Promotion', 'event_espresso');
247
-                break;
248
-            case 'Ticket':
249
-                $obj_type = __('Ticket', 'event_espresso');
250
-                break;
251
-            case 'Transaction':
252
-                $obj_type = __('Transaction', 'event_espresso');
253
-                break;
254
-        }
255
-        return apply_filters('FHEE__EE_Line_Item__OBJ_type_i18n', $obj_type, $this);
256
-    }
257
-
258
-
259
-    /**
260
-     * Sets item_type
261
-     *
262
-     * @param string $OBJ_type
263
-     * @throws EE_Error
264
-     */
265
-    public function set_OBJ_type($OBJ_type)
266
-    {
267
-        $this->set('OBJ_type', $OBJ_type);
268
-    }
269
-
270
-
271
-    /**
272
-     * Gets unit_price
273
-     *
274
-     * @return float
275
-     * @throws EE_Error
276
-     */
277
-    public function unit_price()
278
-    {
279
-        return $this->get('LIN_unit_price');
280
-    }
281
-
282
-
283
-    /**
284
-     * Sets unit_price
285
-     *
286
-     * @param float $unit_price
287
-     * @throws EE_Error
288
-     */
289
-    public function set_unit_price($unit_price)
290
-    {
291
-        $this->set('LIN_unit_price', $unit_price);
292
-    }
293
-
294
-
295
-    /**
296
-     * Checks if this item is a percentage modifier or not
297
-     *
298
-     * @return boolean
299
-     * @throws EE_Error
300
-     */
301
-    public function is_percent()
302
-    {
303
-        if ($this->is_tax_sub_total()) {
304
-            // tax subtotals HAVE a percent on them, that percentage only applies
305
-            // to taxable items, so its' an exception. Treat it like a flat line item
306
-            return false;
307
-        }
308
-        $unit_price = abs($this->get('LIN_unit_price'));
309
-        $percent = abs($this->get('LIN_percent'));
310
-        if ($unit_price < .001 && $percent) {
311
-            return true;
312
-        }
313
-        if ($unit_price >= .001 && ! $percent) {
314
-            return false;
315
-        }
316
-        if ($unit_price >= .001 && $percent) {
317
-            throw new EE_Error(
318
-                sprintf(
319
-                    esc_html__('A Line Item can not have a unit price of (%s) AND a percent (%s)!', 'event_espresso'),
320
-                    $unit_price,
321
-                    $percent
322
-                )
323
-            );
324
-        }
325
-        // if they're both 0, assume its not a percent item
326
-        return false;
327
-    }
328
-
329
-
330
-    /**
331
-     * Gets percent (between 100-.001)
332
-     *
333
-     * @return float
334
-     * @throws EE_Error
335
-     */
336
-    public function percent()
337
-    {
338
-        return $this->get('LIN_percent');
339
-    }
340
-
341
-
342
-    /**
343
-     * Sets percent (between 100-0.01)
344
-     *
345
-     * @param float $percent
346
-     * @throws EE_Error
347
-     */
348
-    public function set_percent($percent)
349
-    {
350
-        $this->set('LIN_percent', $percent);
351
-    }
352
-
353
-
354
-    /**
355
-     * Gets total
356
-     *
357
-     * @return float
358
-     * @throws EE_Error
359
-     */
360
-    public function total()
361
-    {
362
-        return $this->get('LIN_total');
363
-    }
364
-
365
-
366
-    /**
367
-     * Sets total
368
-     *
369
-     * @param float $total
370
-     * @throws EE_Error
371
-     */
372
-    public function set_total($total)
373
-    {
374
-        $this->set('LIN_total', $total);
375
-    }
376
-
377
-
378
-    /**
379
-     * Gets order
380
-     *
381
-     * @return int
382
-     * @throws EE_Error
383
-     */
384
-    public function order()
385
-    {
386
-        return $this->get('LIN_order');
387
-    }
388
-
389
-
390
-    /**
391
-     * Sets order
392
-     *
393
-     * @param int $order
394
-     * @throws EE_Error
395
-     */
396
-    public function set_order($order)
397
-    {
398
-        $this->set('LIN_order', $order);
399
-    }
400
-
401
-
402
-    /**
403
-     * Gets parent
404
-     *
405
-     * @return int
406
-     * @throws EE_Error
407
-     */
408
-    public function parent_ID()
409
-    {
410
-        return $this->get('LIN_parent');
411
-    }
412
-
413
-
414
-    /**
415
-     * Sets parent
416
-     *
417
-     * @param int $parent
418
-     * @throws EE_Error
419
-     */
420
-    public function set_parent_ID($parent)
421
-    {
422
-        $this->set('LIN_parent', $parent);
423
-    }
424
-
425
-
426
-    /**
427
-     * Gets type
428
-     *
429
-     * @return string
430
-     * @throws EE_Error
431
-     */
432
-    public function type()
433
-    {
434
-        return $this->get('LIN_type');
435
-    }
436
-
437
-
438
-    /**
439
-     * Sets type
440
-     *
441
-     * @param string $type
442
-     * @throws EE_Error
443
-     */
444
-    public function set_type($type)
445
-    {
446
-        $this->set('LIN_type', $type);
447
-    }
448
-
449
-
450
-    /**
451
-     * Gets the line item of which this item is a composite. Eg, if this is a subtotal, the parent might be a total\
452
-     * If this line item is saved to the DB, fetches the parent from the DB. However, if this line item isn't in the DB
453
-     * it uses its cached reference to its parent line item (which would have been set by `EE_Line_Item::set_parent()`
454
-     * or indirectly by `EE_Line_item::add_child_line_item()`)
455
-     *
456
-     * @return EE_Base_Class|EE_Line_Item
457
-     * @throws EE_Error
458
-     */
459
-    public function parent()
460
-    {
461
-        return $this->ID()
462
-            ? $this->get_model()->get_one_by_ID($this->parent_ID())
463
-            : $this->_parent;
464
-    }
465
-
466
-
467
-    /**
468
-     * Gets ALL the children of this line item (ie, all the parts that contribute towards this total).
469
-     *
470
-     * @return EE_Base_Class[]|EE_Line_Item[]
471
-     * @throws EE_Error
472
-     */
473
-    public function children()
474
-    {
475
-        if ($this->ID()) {
476
-            return $this->get_model()->get_all(
477
-                array(
478
-                    array('LIN_parent' => $this->ID()),
479
-                    'order_by' => array('LIN_order' => 'ASC'),
480
-                )
481
-            );
482
-        }
483
-        if (! is_array($this->_children)) {
484
-            $this->_children = array();
485
-        }
486
-        return $this->_children;
487
-    }
488
-
489
-
490
-    /**
491
-     * Gets code
492
-     *
493
-     * @return string
494
-     * @throws EE_Error
495
-     */
496
-    public function code()
497
-    {
498
-        return $this->get('LIN_code');
499
-    }
500
-
501
-
502
-    /**
503
-     * Sets code
504
-     *
505
-     * @param string $code
506
-     * @throws EE_Error
507
-     */
508
-    public function set_code($code)
509
-    {
510
-        $this->set('LIN_code', $code);
511
-    }
512
-
513
-
514
-    /**
515
-     * Gets is_taxable
516
-     *
517
-     * @return boolean
518
-     * @throws EE_Error
519
-     */
520
-    public function is_taxable()
521
-    {
522
-        return $this->get('LIN_is_taxable');
523
-    }
524
-
525
-
526
-    /**
527
-     * Sets is_taxable
528
-     *
529
-     * @param boolean $is_taxable
530
-     * @throws EE_Error
531
-     */
532
-    public function set_is_taxable($is_taxable)
533
-    {
534
-        $this->set('LIN_is_taxable', $is_taxable);
535
-    }
536
-
537
-
538
-    /**
539
-     * Gets the object that this model-joins-to.
540
-     * returns one of the model objects that the field OBJ_ID can point to... see the 'OBJ_ID' field on
541
-     * EEM_Promotion_Object
542
-     *
543
-     *        Eg, if this line item join model object is for a ticket, this will return the EE_Ticket object
544
-     *
545
-     * @return EE_Base_Class | NULL
546
-     * @throws EE_Error
547
-     */
548
-    public function get_object()
549
-    {
550
-        $model_name_of_related_obj = $this->OBJ_type();
551
-        return $this->get_model()->has_relation($model_name_of_related_obj)
552
-            ? $this->get_first_related($model_name_of_related_obj)
553
-            : null;
554
-    }
555
-
556
-
557
-    /**
558
-     * Like EE_Line_Item::get_object(), but can only ever actually return an EE_Ticket.
559
-     * (IE, if this line item is for a price or something else, will return NULL)
560
-     *
561
-     * @param array $query_params
562
-     * @return EE_Base_Class|EE_Ticket
563
-     * @throws EE_Error
564
-     */
565
-    public function ticket($query_params = array())
566
-    {
567
-        // we're going to assume that when this method is called we always want to receive the attached ticket EVEN if that ticket is archived.  This can be overridden via the incoming $query_params argument
568
-        $remove_defaults = array('default_where_conditions' => 'none');
569
-        $query_params = array_merge($remove_defaults, $query_params);
570
-        return $this->get_first_related('Ticket', $query_params);
571
-    }
572
-
573
-
574
-    /**
575
-     * Gets the EE_Datetime that's related to the ticket, IF this is for a ticket
576
-     *
577
-     * @return EE_Datetime | NULL
578
-     * @throws EE_Error
579
-     */
580
-    public function get_ticket_datetime()
581
-    {
582
-        if ($this->OBJ_type() === 'Ticket') {
583
-            $ticket = $this->ticket();
584
-            if ($ticket instanceof EE_Ticket) {
585
-                $datetime = $ticket->first_datetime();
586
-                if ($datetime instanceof EE_Datetime) {
587
-                    return $datetime;
588
-                }
589
-            }
590
-        }
591
-        return null;
592
-    }
593
-
594
-
595
-    /**
596
-     * Gets the event's name that's related to the ticket, if this is for
597
-     * a ticket
598
-     *
599
-     * @return string
600
-     * @throws EE_Error
601
-     */
602
-    public function ticket_event_name()
603
-    {
604
-        $event_name = esc_html__('Unknown', 'event_espresso');
605
-        $event = $this->ticket_event();
606
-        if ($event instanceof EE_Event) {
607
-            $event_name = $event->name();
608
-        }
609
-        return $event_name;
610
-    }
611
-
612
-
613
-    /**
614
-     * Gets the event that's related to the ticket, if this line item represents a ticket.
615
-     *
616
-     * @return EE_Event|null
617
-     * @throws EE_Error
618
-     */
619
-    public function ticket_event()
620
-    {
621
-        $event = null;
622
-        $ticket = $this->ticket();
623
-        if ($ticket instanceof EE_Ticket) {
624
-            $datetime = $ticket->first_datetime();
625
-            if ($datetime instanceof EE_Datetime) {
626
-                $event = $datetime->event();
627
-            }
628
-        }
629
-        return $event;
630
-    }
631
-
632
-
633
-    /**
634
-     * Gets the first datetime for this lien item, assuming it's for a ticket
635
-     *
636
-     * @param string $date_format
637
-     * @param string $time_format
638
-     * @return string
639
-     * @throws EE_Error
640
-     */
641
-    public function ticket_datetime_start($date_format = '', $time_format = '')
642
-    {
643
-        $first_datetime_string = esc_html__('Unknown', 'event_espresso');
644
-        $datetime = $this->get_ticket_datetime();
645
-        if ($datetime) {
646
-            $first_datetime_string = $datetime->start_date_and_time($date_format, $time_format);
647
-        }
648
-        return $first_datetime_string;
649
-    }
650
-
651
-
652
-    /**
653
-     * Adds the line item as a child to this line item. If there is another child line
654
-     * item with the same LIN_code, it is overwritten by this new one
655
-     *
656
-     * @param EEI_Line_Item $line_item
657
-     * @param bool          $set_order
658
-     * @return bool success
659
-     * @throws EE_Error
660
-     */
661
-    public function add_child_line_item(EEI_Line_Item $line_item, $set_order = true)
662
-    {
663
-        // should we calculate the LIN_order for this line item ?
664
-        if ($set_order || $line_item->order() === null) {
665
-            $line_item->set_order(count($this->children()));
666
-        }
667
-        if ($this->ID()) {
668
-            // check for any duplicate line items (with the same code), if so, this replaces it
669
-            $line_item_with_same_code = $this->get_child_line_item($line_item->code());
670
-            if ($line_item_with_same_code instanceof EE_Line_Item && $line_item_with_same_code !== $line_item) {
671
-                $this->delete_child_line_item($line_item_with_same_code->code());
672
-            }
673
-            $line_item->set_parent_ID($this->ID());
674
-            if ($this->TXN_ID()) {
675
-                $line_item->set_TXN_ID($this->TXN_ID());
676
-            }
677
-            return $line_item->save();
678
-        }
679
-        $this->_children[ $line_item->code() ] = $line_item;
680
-        if ($line_item->parent() !== $this) {
681
-            $line_item->set_parent($this);
682
-        }
683
-        return true;
684
-    }
685
-
686
-
687
-    /**
688
-     * Similar to EE_Base_Class::_add_relation_to, except this isn't a normal relation.
689
-     * If this line item is saved to the DB, this is just a wrapper for set_parent_ID() and save()
690
-     * However, if this line item is NOT saved to the DB, this just caches the parent on
691
-     * the EE_Line_Item::_parent property.
692
-     *
693
-     * @param EE_Line_Item $line_item
694
-     * @throws EE_Error
695
-     */
696
-    public function set_parent($line_item)
697
-    {
698
-        if ($this->ID()) {
699
-            if (! $line_item->ID()) {
700
-                $line_item->save();
701
-            }
702
-            $this->set_parent_ID($line_item->ID());
703
-            $this->save();
704
-        } else {
705
-            $this->_parent = $line_item;
706
-            $this->set_parent_ID($line_item->ID());
707
-        }
708
-    }
709
-
710
-
711
-    /**
712
-     * Gets the child line item as specified by its code. Because this returns an object (by reference)
713
-     * you can modify this child line item and the parent (this object) can know about them
714
-     * because it also has a reference to that line item
715
-     *
716
-     * @param string $code
717
-     * @return EE_Base_Class|EE_Line_Item|EE_Soft_Delete_Base_Class|NULL
718
-     * @throws EE_Error
719
-     */
720
-    public function get_child_line_item($code)
721
-    {
722
-        if ($this->ID()) {
723
-            return $this->get_model()->get_one(
724
-                array(array('LIN_parent' => $this->ID(), 'LIN_code' => $code))
725
-            );
726
-        }
727
-        return isset($this->_children[ $code ])
728
-            ? $this->_children[ $code ]
729
-            : null;
730
-    }
731
-
732
-
733
-    /**
734
-     * Returns how many items are deleted (or, if this item has not been saved ot the DB yet, just how many it HAD
735
-     * cached on it)
736
-     *
737
-     * @return int
738
-     * @throws EE_Error
739
-     */
740
-    public function delete_children_line_items()
741
-    {
742
-        if ($this->ID()) {
743
-            return $this->get_model()->delete(array(array('LIN_parent' => $this->ID())));
744
-        }
745
-        $count = count($this->_children);
746
-        $this->_children = array();
747
-        return $count;
748
-    }
749
-
750
-
751
-    /**
752
-     * If this line item has been saved to the DB, deletes its child with LIN_code == $code. If this line
753
-     * HAS NOT been saved to the DB, removes the child line item with index $code.
754
-     * Also searches through the child's children for a matching line item. However, once a line item has been found
755
-     * and deleted, stops searching (so if there are line items with duplicate codes, only the first one found will be
756
-     * deleted)
757
-     *
758
-     * @param string $code
759
-     * @param bool   $stop_search_once_found
760
-     * @return int count of items deleted (or simply removed from the line item's cache, if not has not been saved to
761
-     *             the DB yet)
762
-     * @throws EE_Error
763
-     */
764
-    public function delete_child_line_item($code, $stop_search_once_found = true)
765
-    {
766
-        if ($this->ID()) {
767
-            $items_deleted = 0;
768
-            if ($this->code() === $code) {
769
-                $items_deleted += EEH_Line_Item::delete_all_child_items($this);
770
-                $items_deleted += (int) $this->delete();
771
-                if ($stop_search_once_found) {
772
-                    return $items_deleted;
773
-                }
774
-            }
775
-            foreach ($this->children() as $child_line_item) {
776
-                $items_deleted += $child_line_item->delete_child_line_item($code, $stop_search_once_found);
777
-            }
778
-            return $items_deleted;
779
-        }
780
-        if (isset($this->_children[ $code ])) {
781
-            unset($this->_children[ $code ]);
782
-            return 1;
783
-        }
784
-        return 0;
785
-    }
786
-
787
-
788
-    /**
789
-     * If this line item is in the database, is of the type subtotal, and
790
-     * has no children, why do we have it? It should be deleted so this function
791
-     * does that
792
-     *
793
-     * @return boolean
794
-     * @throws EE_Error
795
-     */
796
-    public function delete_if_childless_subtotal()
797
-    {
798
-        if ($this->ID() && $this->type() === EEM_Line_Item::type_sub_total && ! $this->children()) {
799
-            return $this->delete();
800
-        }
801
-        return false;
802
-    }
803
-
804
-
805
-    /**
806
-     * Creates a code and returns a string. doesn't assign the code to this model object
807
-     *
808
-     * @return string
809
-     * @throws EE_Error
810
-     */
811
-    public function generate_code()
812
-    {
813
-        // each line item in the cart requires a unique identifier
814
-        return md5($this->get('OBJ_type') . $this->get('OBJ_ID') . microtime());
815
-    }
816
-
817
-
818
-    /**
819
-     * @return bool
820
-     * @throws EE_Error
821
-     */
822
-    public function is_tax()
823
-    {
824
-        return $this->type() === EEM_Line_Item::type_tax;
825
-    }
826
-
827
-
828
-    /**
829
-     * @return bool
830
-     * @throws EE_Error
831
-     */
832
-    public function is_tax_sub_total()
833
-    {
834
-        return $this->type() === EEM_Line_Item::type_tax_sub_total;
835
-    }
836
-
837
-
838
-    /**
839
-     * @return bool
840
-     * @throws EE_Error
841
-     */
842
-    public function is_line_item()
843
-    {
844
-        return $this->type() === EEM_Line_Item::type_line_item;
845
-    }
846
-
847
-
848
-    /**
849
-     * @return bool
850
-     * @throws EE_Error
851
-     */
852
-    public function is_sub_line_item()
853
-    {
854
-        return $this->type() === EEM_Line_Item::type_sub_line_item;
855
-    }
856
-
857
-
858
-    /**
859
-     * @return bool
860
-     * @throws EE_Error
861
-     */
862
-    public function is_sub_total()
863
-    {
864
-        return $this->type() === EEM_Line_Item::type_sub_total;
865
-    }
866
-
867
-
868
-    /**
869
-     * Whether or not this line item is a cancellation line item
870
-     *
871
-     * @return boolean
872
-     * @throws EE_Error
873
-     */
874
-    public function is_cancellation()
875
-    {
876
-        return EEM_Line_Item::type_cancellation === $this->type();
877
-    }
878
-
879
-
880
-    /**
881
-     * @return bool
882
-     * @throws EE_Error
883
-     */
884
-    public function is_total()
885
-    {
886
-        return $this->type() === EEM_Line_Item::type_total;
887
-    }
888
-
889
-
890
-    /**
891
-     * @return bool
892
-     * @throws EE_Error
893
-     */
894
-    public function is_cancelled()
895
-    {
896
-        return $this->type() === EEM_Line_Item::type_cancellation;
897
-    }
898
-
899
-
900
-    /**
901
-     * @return string like '2, 004.00', formatted according to the localized currency
902
-     * @throws EE_Error
903
-     */
904
-    public function unit_price_no_code()
905
-    {
906
-        return $this->get_pretty('LIN_unit_price', 'no_currency_code');
907
-    }
908
-
909
-
910
-    /**
911
-     * @return string like '2, 004.00', formatted according to the localized currency
912
-     * @throws EE_Error
913
-     */
914
-    public function total_no_code()
915
-    {
916
-        return $this->get_pretty('LIN_total', 'no_currency_code');
917
-    }
918
-
919
-
920
-    /**
921
-     * Gets the final total on this item, taking taxes into account.
922
-     * Has the side-effect of setting the sub-total as it was just calculated.
923
-     * If this is used on a grand-total line item, also updates the transaction's
924
-     * TXN_total (provided this line item is allowed to persist, otherwise we don't
925
-     * want to change a persistable transaction with info from a non-persistent line item)
926
-     *
927
-     * @return float
928
-     * @throws EE_Error
929
-     * @throws InvalidArgumentException
930
-     * @throws InvalidInterfaceException
931
-     * @throws InvalidDataTypeException
932
-     */
933
-    public function recalculate_total_including_taxes()
934
-    {
935
-        $pre_tax_total = $this->recalculate_pre_tax_total();
936
-        $tax_total = $this->recalculate_taxes_and_tax_total();
937
-        $total = $pre_tax_total + $tax_total;
938
-        // no negative totals plz
939
-        $total = max($total, 0);
940
-        $this->set_total($total);
941
-        // only update the related transaction's total
942
-        // if we intend to save this line item and its a grand total
943
-        if ($this->allow_persist() && $this->type() === EEM_Line_Item::type_total
944
-            && $this->transaction()
945
-               instanceof
946
-               EE_Transaction
947
-        ) {
948
-            $this->transaction()->set_total($total);
949
-            if ($this->transaction()->ID()) {
950
-                $this->transaction()->save();
951
-            }
952
-        }
953
-        $this->maybe_save();
954
-        return $total;
955
-    }
956
-
957
-
958
-    /**
959
-     * Recursively goes through all the children and recalculates sub-totals EXCEPT for
960
-     * tax-sub-totals (they're a an odd beast). Updates the 'total' on each line item according to either its
961
-     * unit price * quantity or the total of all its children EXCEPT when we're only calculating the taxable total and
962
-     * when this is called on the grand total
963
-     *
964
-     * @return float
965
-     * @throws InvalidArgumentException
966
-     * @throws InvalidInterfaceException
967
-     * @throws InvalidDataTypeException
968
-     * @throws EE_Error
969
-     */
970
-    public function recalculate_pre_tax_total()
971
-    {
972
-        $total = 0;
973
-        $my_children = $this->children();
974
-        $has_children = ! empty($my_children);
975
-        if ($has_children && $this->is_line_item()) {
976
-            $total = $this->_recalculate_pretax_total_for_line_item($total, $my_children);
977
-        } elseif (! $has_children && ($this->is_sub_line_item() || $this->is_line_item())) {
978
-            $total = $this->unit_price() * $this->quantity();
979
-        } elseif ($this->is_sub_total() || $this->is_total()) {
980
-            $total = $this->_recalculate_pretax_total_for_subtotal($total, $my_children);
981
-        } elseif ($this->is_tax_sub_total() || $this->is_tax() || $this->is_cancelled()) {
982
-            // completely ignore tax totals, tax sub-totals, and cancelled line items, when calculating the pre-tax-total
983
-            return 0;
984
-        }
985
-        // ensure all non-line items and non-sub-line-items have a quantity of 1 (except for Events)
986
-        if (! $this->is_line_item() && ! $this->is_sub_line_item() && ! $this->is_cancellation()
987
-        ) {
988
-            if ($this->OBJ_type() !== 'Event') {
989
-                $this->set_quantity(1);
990
-            }
991
-            if (! $this->is_percent()) {
992
-                $this->set_unit_price($total);
993
-            }
994
-        }
995
-        // we don't want to bother saving grand totals, because that needs to factor in taxes anyways
996
-        // so it ought to be
997
-        if (! $this->is_total()) {
998
-            $this->set_total($total);
999
-            // if not a percent line item, make sure we keep the unit price in sync
1000
-            if ($has_children
1001
-                && $this->is_line_item()
1002
-                && ! $this->is_percent()
1003
-            ) {
1004
-                if ($this->quantity() === 0) {
1005
-                    $new_unit_price = 0;
1006
-                } else {
1007
-                    $new_unit_price = $this->total() / $this->quantity();
1008
-                }
1009
-                $this->set_unit_price($new_unit_price);
1010
-            }
1011
-            $this->maybe_save();
1012
-        }
1013
-        return $total;
1014
-    }
1015
-
1016
-
1017
-    /**
1018
-     * Calculates the pretax total when this line item is a subtotal or total line item.
1019
-     * Basically does a sum-then-round approach (ie, any percent line item that are children
1020
-     * will calculate their total based on the un-rounded total we're working with so far, and
1021
-     * THEN round the result; instead of rounding as we go like with sub-line-items)
1022
-     *
1023
-     * @param float          $calculated_total_so_far
1024
-     * @param EE_Line_Item[] $my_children
1025
-     * @return float
1026
-     * @throws InvalidArgumentException
1027
-     * @throws InvalidInterfaceException
1028
-     * @throws InvalidDataTypeException
1029
-     * @throws EE_Error
1030
-     */
1031
-    protected function _recalculate_pretax_total_for_subtotal($calculated_total_so_far, $my_children = null)
1032
-    {
1033
-        if ($my_children === null) {
1034
-            $my_children = $this->children();
1035
-        }
1036
-        $subtotal_quantity = 0;
1037
-        // get the total of all its children
1038
-        foreach ($my_children as $child_line_item) {
1039
-            if ($child_line_item instanceof EE_Line_Item && ! $child_line_item->is_cancellation()) {
1040
-                // percentage line items are based on total so far
1041
-                if ($child_line_item->is_percent()) {
1042
-                    // round as we go so that the line items add up ok
1043
-                    $percent_total = round(
1044
-                        $calculated_total_so_far * $child_line_item->percent() / 100,
1045
-                        EE_Registry::instance()->CFG->currency->dec_plc
1046
-                    );
1047
-                    $child_line_item->set_total($percent_total);
1048
-                    // so far all percent line items should have a quantity of 1
1049
-                    // (ie, no double percent discounts. Although that might be requested someday)
1050
-                    $child_line_item->set_quantity(1);
1051
-                    $child_line_item->maybe_save();
1052
-                    $calculated_total_so_far += $percent_total;
1053
-                } else {
1054
-                    // verify flat sub-line-item quantities match their parent
1055
-                    if ($child_line_item->is_sub_line_item()) {
1056
-                        $child_line_item->set_quantity($this->quantity());
1057
-                    }
1058
-                    $calculated_total_so_far += $child_line_item->recalculate_pre_tax_total();
1059
-                    $subtotal_quantity += $child_line_item->quantity();
1060
-                }
1061
-            }
1062
-        }
1063
-        if ($this->is_sub_total()) {
1064
-            // no negative totals plz
1065
-            $calculated_total_so_far = max($calculated_total_so_far, 0);
1066
-            $subtotal_quantity = $subtotal_quantity > 0 ? 1 : 0;
1067
-            $this->set_quantity($subtotal_quantity);
1068
-            $this->maybe_save();
1069
-        }
1070
-        return $calculated_total_so_far;
1071
-    }
1072
-
1073
-
1074
-    /**
1075
-     * Calculates the pretax total for a normal line item, in a round-then-sum approach
1076
-     * (where each sub-line-item is applied to the base price for the line item
1077
-     * and the result is immediately rounded, rather than summing all the sub-line-items
1078
-     * then rounding, like we do when recalculating pretax totals on totals and subtotals).
1079
-     *
1080
-     * @param float          $calculated_total_so_far
1081
-     * @param EE_Line_Item[] $my_children
1082
-     * @return float
1083
-     * @throws InvalidArgumentException
1084
-     * @throws InvalidInterfaceException
1085
-     * @throws InvalidDataTypeException
1086
-     * @throws EE_Error
1087
-     */
1088
-    protected function _recalculate_pretax_total_for_line_item($calculated_total_so_far, $my_children = null)
1089
-    {
1090
-        if ($my_children === null) {
1091
-            $my_children = $this->children();
1092
-        }
1093
-        // we need to keep track of the running total for a single item,
1094
-        // because we need to round as we go
1095
-        $unit_price_for_total = 0;
1096
-        $quantity_for_total = 1;
1097
-        // get the total of all its children
1098
-        foreach ($my_children as $child_line_item) {
1099
-            if ($child_line_item instanceof EE_Line_Item && ! $child_line_item->is_cancellation()) {
1100
-                if ($child_line_item->is_percent()) {
1101
-                    // it should be the unit-price-so-far multiplied by teh percent multiplied by the quantity
1102
-                    // not total multiplied by percent, because that ignores rounding along-the-way
1103
-                    $percent_unit_price = round(
1104
-                        $unit_price_for_total * $child_line_item->percent() / 100,
1105
-                        EE_Registry::instance()->CFG->currency->dec_plc
1106
-                    );
1107
-                    $percent_total = $percent_unit_price * $quantity_for_total;
1108
-                    $child_line_item->set_total($percent_total);
1109
-                    // so far all percent line items should have a quantity of 1
1110
-                    // (ie, no double percent discounts. Although that might be requested someday)
1111
-                    $child_line_item->set_quantity(1);
1112
-                    $child_line_item->maybe_save();
1113
-                    $calculated_total_so_far += $percent_total;
1114
-                    $unit_price_for_total += $percent_unit_price;
1115
-                } else {
1116
-                    // verify flat sub-line-item quantities match their parent
1117
-                    if ($child_line_item->is_sub_line_item()) {
1118
-                        $child_line_item->set_quantity($this->quantity());
1119
-                    }
1120
-                    $quantity_for_total = $child_line_item->quantity();
1121
-                    $calculated_total_so_far += $child_line_item->recalculate_pre_tax_total();
1122
-                    $unit_price_for_total += $child_line_item->unit_price();
1123
-                }
1124
-            }
1125
-        }
1126
-        return $calculated_total_so_far;
1127
-    }
1128
-
1129
-
1130
-    /**
1131
-     * Recalculates the total on each individual tax (based on a recalculation of the pre-tax total), sets
1132
-     * the totals on each tax calculated, and returns the final tax total. Re-saves tax line items
1133
-     * and tax sub-total if already in the DB
1134
-     *
1135
-     * @return float
1136
-     * @throws EE_Error
1137
-     */
1138
-    public function recalculate_taxes_and_tax_total()
1139
-    {
1140
-        // get all taxes
1141
-        $taxes = $this->tax_descendants();
1142
-        // calculate the pretax total
1143
-        $taxable_total = $this->taxable_total();
1144
-        $tax_total = 0;
1145
-        foreach ($taxes as $tax) {
1146
-            $total_on_this_tax = $taxable_total * $tax->percent() / 100;
1147
-            // remember the total on this line item
1148
-            $tax->set_total($total_on_this_tax);
1149
-            $tax->maybe_save();
1150
-            $tax_total += $tax->total();
1151
-        }
1152
-        $this->_recalculate_tax_sub_total();
1153
-        return $tax_total;
1154
-    }
1155
-
1156
-
1157
-    /**
1158
-     * Simply forces all the tax-sub-totals to recalculate. Assumes the taxes have been calculated
1159
-     *
1160
-     * @return void
1161
-     * @throws EE_Error
1162
-     */
1163
-    private function _recalculate_tax_sub_total()
1164
-    {
1165
-        if ($this->is_tax_sub_total()) {
1166
-            $total = 0;
1167
-            $total_percent = 0;
1168
-            // simply loop through all its children (which should be taxes) and sum their total
1169
-            foreach ($this->children() as $child_tax) {
1170
-                if ($child_tax instanceof EE_Line_Item) {
1171
-                    $total += $child_tax->total();
1172
-                    $total_percent += $child_tax->percent();
1173
-                }
1174
-            }
1175
-            $this->set_total($total);
1176
-            $this->set_percent($total_percent);
1177
-            $this->maybe_save();
1178
-        } elseif ($this->is_total()) {
1179
-            foreach ($this->children() as $maybe_tax_subtotal) {
1180
-                if ($maybe_tax_subtotal instanceof EE_Line_Item) {
1181
-                    $maybe_tax_subtotal->_recalculate_tax_sub_total();
1182
-                }
1183
-            }
1184
-        }
1185
-    }
1186
-
1187
-
1188
-    /**
1189
-     * Gets the total tax on this line item. Assumes taxes have already been calculated using
1190
-     * recalculate_taxes_and_total
1191
-     *
1192
-     * @return float
1193
-     * @throws EE_Error
1194
-     */
1195
-    public function get_total_tax()
1196
-    {
1197
-        $this->_recalculate_tax_sub_total();
1198
-        $total = 0;
1199
-        foreach ($this->tax_descendants() as $tax_line_item) {
1200
-            if ($tax_line_item instanceof EE_Line_Item) {
1201
-                $total += $tax_line_item->total();
1202
-            }
1203
-        }
1204
-        return $total;
1205
-    }
1206
-
1207
-
1208
-    /**
1209
-     * Gets the total for all the items purchased only
1210
-     *
1211
-     * @return float
1212
-     * @throws EE_Error
1213
-     */
1214
-    public function get_items_total()
1215
-    {
1216
-        // by default, let's make sure we're consistent with the existing line item
1217
-        if ($this->is_total()) {
1218
-            $pretax_subtotal_li = EEH_Line_Item::get_pre_tax_subtotal($this);
1219
-            if ($pretax_subtotal_li instanceof EE_Line_Item) {
1220
-                return $pretax_subtotal_li->total();
1221
-            }
1222
-        }
1223
-        $total = 0;
1224
-        foreach ($this->get_items() as $item) {
1225
-            if ($item instanceof EE_Line_Item) {
1226
-                $total += $item->total();
1227
-            }
1228
-        }
1229
-        return $total;
1230
-    }
1231
-
1232
-
1233
-    /**
1234
-     * Gets all the descendants (ie, children or children of children etc) that
1235
-     * are of the type 'tax'
1236
-     *
1237
-     * @return EE_Line_Item[]
1238
-     */
1239
-    public function tax_descendants()
1240
-    {
1241
-        return EEH_Line_Item::get_tax_descendants($this);
1242
-    }
1243
-
1244
-
1245
-    /**
1246
-     * Gets all the real items purchased which are children of this item
1247
-     *
1248
-     * @return EE_Line_Item[]
1249
-     */
1250
-    public function get_items()
1251
-    {
1252
-        return EEH_Line_Item::get_line_item_descendants($this);
1253
-    }
1254
-
1255
-
1256
-    /**
1257
-     * Returns the amount taxable among this line item's children (or if it has no children,
1258
-     * how much of it is taxable). Does not recalculate totals or subtotals.
1259
-     * If the taxable total is negative, (eg, if none of the tickets were taxable,
1260
-     * but there is a "Taxable" discount), returns 0.
1261
-     *
1262
-     * @return float
1263
-     * @throws EE_Error
1264
-     */
1265
-    public function taxable_total()
1266
-    {
1267
-        $total = 0;
1268
-        if ($this->children()) {
1269
-            foreach ($this->children() as $child_line_item) {
1270
-                if ($child_line_item->type() === EEM_Line_Item::type_line_item && $child_line_item->is_taxable()) {
1271
-                    // if it's a percent item, only take into account the percent
1272
-                    // that's taxable too (the taxable total so far)
1273
-                    if ($child_line_item->is_percent()) {
1274
-                        $total += ($total * $child_line_item->percent() / 100);
1275
-                    } else {
1276
-                        $total += $child_line_item->total();
1277
-                    }
1278
-                } elseif ($child_line_item->type() === EEM_Line_Item::type_sub_total) {
1279
-                    $total += $child_line_item->taxable_total();
1280
-                }
1281
-            }
1282
-        }
1283
-        return max($total, 0);
1284
-    }
1285
-
1286
-
1287
-    /**
1288
-     * Gets the transaction for this line item
1289
-     *
1290
-     * @return EE_Base_Class|EE_Transaction
1291
-     * @throws EE_Error
1292
-     */
1293
-    public function transaction()
1294
-    {
1295
-        return $this->get_first_related('Transaction');
1296
-    }
1297
-
1298
-
1299
-    /**
1300
-     * Saves this line item to the DB, and recursively saves its descendants.
1301
-     * Because there currently is no proper parent-child relation on the model,
1302
-     * save_this_and_cached() will NOT save the descendants.
1303
-     * Also sets the transaction on this line item and all its descendants before saving
1304
-     *
1305
-     * @param int $txn_id if none is provided, assumes $this->TXN_ID()
1306
-     * @return int count of items saved
1307
-     * @throws EE_Error
1308
-     */
1309
-    public function save_this_and_descendants_to_txn($txn_id = null)
1310
-    {
1311
-        $count = 0;
1312
-        if (! $txn_id) {
1313
-            $txn_id = $this->TXN_ID();
1314
-        }
1315
-        $this->set_TXN_ID($txn_id);
1316
-        $children = $this->children();
1317
-        $count += $this->save()
1318
-            ? 1
1319
-            : 0;
1320
-        foreach ($children as $child_line_item) {
1321
-            if ($child_line_item instanceof EE_Line_Item) {
1322
-                $child_line_item->set_parent_ID($this->ID());
1323
-                $count += $child_line_item->save_this_and_descendants_to_txn($txn_id);
1324
-            }
1325
-        }
1326
-        return $count;
1327
-    }
1328
-
1329
-
1330
-    /**
1331
-     * Saves this line item to the DB, and recursively saves its descendants.
1332
-     *
1333
-     * @return int count of items saved
1334
-     * @throws EE_Error
1335
-     */
1336
-    public function save_this_and_descendants()
1337
-    {
1338
-        $count = 0;
1339
-        $children = $this->children();
1340
-        $count += $this->save()
1341
-            ? 1
1342
-            : 0;
1343
-        foreach ($children as $child_line_item) {
1344
-            if ($child_line_item instanceof EE_Line_Item) {
1345
-                $child_line_item->set_parent_ID($this->ID());
1346
-                $count += $child_line_item->save_this_and_descendants();
1347
-            }
1348
-        }
1349
-        return $count;
1350
-    }
1351
-
1352
-
1353
-    /**
1354
-     * returns the cancellation line item if this item was cancelled
1355
-     *
1356
-     * @return EE_Line_Item[]
1357
-     * @throws InvalidArgumentException
1358
-     * @throws InvalidInterfaceException
1359
-     * @throws InvalidDataTypeException
1360
-     * @throws ReflectionException
1361
-     * @throws EE_Error
1362
-     */
1363
-    public function get_cancellations()
1364
-    {
1365
-        EE_Registry::instance()->load_helper('Line_Item');
1366
-        return EEH_Line_Item::get_descendants_of_type($this, EEM_Line_Item::type_cancellation);
1367
-    }
1368
-
1369
-
1370
-    /**
1371
-     * If this item has an ID, then this saves it again to update the db
1372
-     *
1373
-     * @return int count of items saved
1374
-     * @throws EE_Error
1375
-     */
1376
-    public function maybe_save()
1377
-    {
1378
-        if ($this->ID()) {
1379
-            return $this->save();
1380
-        }
1381
-        return false;
1382
-    }
1383
-
1384
-
1385
-    /**
1386
-     * clears the cached children and parent from the line item
1387
-     *
1388
-     * @return void
1389
-     */
1390
-    public function clear_related_line_item_cache()
1391
-    {
1392
-        $this->_children = array();
1393
-        $this->_parent = null;
1394
-    }
1395
-
1396
-
1397
-    /**
1398
-     * @param bool $raw
1399
-     * @return int
1400
-     * @throws EE_Error
1401
-     */
1402
-    public function timestamp($raw = false)
1403
-    {
1404
-        return $raw
1405
-            ? $this->get_raw('LIN_timestamp')
1406
-            : $this->get('LIN_timestamp');
1407
-    }
1408
-
1409
-
1410
-
1411
-
1412
-    /************************* DEPRECATED *************************/
1413
-    /**
1414
-     * @deprecated 4.6.0
1415
-     * @param string $type one of the constants on EEM_Line_Item
1416
-     * @return EE_Line_Item[]
1417
-     */
1418
-    protected function _get_descendants_of_type($type)
1419
-    {
1420
-        EE_Error::doing_it_wrong(
1421
-            'EE_Line_Item::_get_descendants_of_type()',
1422
-            __('Method replaced with EEH_Line_Item::get_descendants_of_type()', 'event_espresso'),
1423
-            '4.6.0'
1424
-        );
1425
-        return EEH_Line_Item::get_descendants_of_type($this, $type);
1426
-    }
1427
-
1428
-
1429
-    /**
1430
-     * @deprecated 4.6.0
1431
-     * @param string $type like one of the EEM_Line_Item::type_*
1432
-     * @return EE_Line_Item
1433
-     */
1434
-    public function get_nearest_descendant_of_type($type)
1435
-    {
1436
-        EE_Error::doing_it_wrong(
1437
-            'EE_Line_Item::get_nearest_descendant_of_type()',
1438
-            __('Method replaced with EEH_Line_Item::get_nearest_descendant_of_type()', 'event_espresso'),
1439
-            '4.6.0'
1440
-        );
1441
-        return EEH_Line_Item::get_nearest_descendant_of_type($this, $type);
1442
-    }
17
+	/**
18
+	 * for children line items (currently not a normal relation)
19
+	 *
20
+	 * @type EE_Line_Item[]
21
+	 */
22
+	protected $_children = array();
23
+
24
+	/**
25
+	 * for the parent line item
26
+	 *
27
+	 * @var EE_Line_Item
28
+	 */
29
+	protected $_parent;
30
+
31
+
32
+	/**
33
+	 *
34
+	 * @param array  $props_n_values          incoming values
35
+	 * @param string $timezone                incoming timezone (if not set the timezone set for the website will be
36
+	 *                                        used.)
37
+	 * @param array  $date_formats            incoming date_formats in an array where the first value is the
38
+	 *                                        date_format and the second value is the time format
39
+	 * @return EE_Line_Item
40
+	 * @throws EE_Error
41
+	 */
42
+	public static function new_instance($props_n_values = array(), $timezone = null, $date_formats = array())
43
+	{
44
+		$has_object = parent::_check_for_object($props_n_values, __CLASS__, $timezone, $date_formats);
45
+		return $has_object
46
+			? $has_object
47
+			: new self($props_n_values, false, $timezone);
48
+	}
49
+
50
+
51
+	/**
52
+	 * @param array  $props_n_values  incoming values from the database
53
+	 * @param string $timezone        incoming timezone as set by the model.  If not set the timezone for
54
+	 *                                the website will be used.
55
+	 * @return EE_Line_Item
56
+	 * @throws EE_Error
57
+	 */
58
+	public static function new_instance_from_db($props_n_values = array(), $timezone = null)
59
+	{
60
+		return new self($props_n_values, true, $timezone);
61
+	}
62
+
63
+
64
+	/**
65
+	 * Adds some defaults if they're not specified
66
+	 *
67
+	 * @param array  $fieldValues
68
+	 * @param bool   $bydb
69
+	 * @param string $timezone
70
+	 * @throws EE_Error
71
+	 */
72
+	protected function __construct($fieldValues = array(), $bydb = false, $timezone = '')
73
+	{
74
+		parent::__construct($fieldValues, $bydb, $timezone);
75
+		if (! $this->get('LIN_code')) {
76
+			$this->set_code($this->generate_code());
77
+		}
78
+	}
79
+
80
+
81
+	/**
82
+	 * Gets ID
83
+	 *
84
+	 * @return int
85
+	 * @throws EE_Error
86
+	 */
87
+	public function ID()
88
+	{
89
+		return $this->get('LIN_ID');
90
+	}
91
+
92
+
93
+	/**
94
+	 * Gets TXN_ID
95
+	 *
96
+	 * @return int
97
+	 * @throws EE_Error
98
+	 */
99
+	public function TXN_ID()
100
+	{
101
+		return $this->get('TXN_ID');
102
+	}
103
+
104
+
105
+	/**
106
+	 * Sets TXN_ID
107
+	 *
108
+	 * @param int $TXN_ID
109
+	 * @throws EE_Error
110
+	 */
111
+	public function set_TXN_ID($TXN_ID)
112
+	{
113
+		$this->set('TXN_ID', $TXN_ID);
114
+	}
115
+
116
+
117
+	/**
118
+	 * Gets name
119
+	 *
120
+	 * @return string
121
+	 * @throws EE_Error
122
+	 */
123
+	public function name()
124
+	{
125
+		$name = $this->get('LIN_name');
126
+		if (! $name) {
127
+			$name = ucwords(str_replace('-', ' ', $this->type()));
128
+		}
129
+		return $name;
130
+	}
131
+
132
+
133
+	/**
134
+	 * Sets name
135
+	 *
136
+	 * @param string $name
137
+	 * @throws EE_Error
138
+	 */
139
+	public function set_name($name)
140
+	{
141
+		$this->set('LIN_name', $name);
142
+	}
143
+
144
+
145
+	/**
146
+	 * Gets desc
147
+	 *
148
+	 * @return string
149
+	 * @throws EE_Error
150
+	 */
151
+	public function desc()
152
+	{
153
+		return $this->get('LIN_desc');
154
+	}
155
+
156
+
157
+	/**
158
+	 * Sets desc
159
+	 *
160
+	 * @param string $desc
161
+	 * @throws EE_Error
162
+	 */
163
+	public function set_desc($desc)
164
+	{
165
+		$this->set('LIN_desc', $desc);
166
+	}
167
+
168
+
169
+	/**
170
+	 * Gets quantity
171
+	 *
172
+	 * @return int
173
+	 * @throws EE_Error
174
+	 */
175
+	public function quantity()
176
+	{
177
+		return $this->get('LIN_quantity');
178
+	}
179
+
180
+
181
+	/**
182
+	 * Sets quantity
183
+	 *
184
+	 * @param int $quantity
185
+	 * @throws EE_Error
186
+	 */
187
+	public function set_quantity($quantity)
188
+	{
189
+		$this->set('LIN_quantity', max($quantity, 0));
190
+	}
191
+
192
+
193
+	/**
194
+	 * Gets item_id
195
+	 *
196
+	 * @return string
197
+	 * @throws EE_Error
198
+	 */
199
+	public function OBJ_ID()
200
+	{
201
+		return $this->get('OBJ_ID');
202
+	}
203
+
204
+
205
+	/**
206
+	 * Sets item_id
207
+	 *
208
+	 * @param string $item_id
209
+	 * @throws EE_Error
210
+	 */
211
+	public function set_OBJ_ID($item_id)
212
+	{
213
+		$this->set('OBJ_ID', $item_id);
214
+	}
215
+
216
+
217
+	/**
218
+	 * Gets item_type
219
+	 *
220
+	 * @return string
221
+	 * @throws EE_Error
222
+	 */
223
+	public function OBJ_type()
224
+	{
225
+		return $this->get('OBJ_type');
226
+	}
227
+
228
+
229
+	/**
230
+	 * Gets item_type
231
+	 *
232
+	 * @return string
233
+	 * @throws EE_Error
234
+	 */
235
+	public function OBJ_type_i18n()
236
+	{
237
+		$obj_type = $this->OBJ_type();
238
+		switch ($obj_type) {
239
+			case 'Event':
240
+				$obj_type = __('Event', 'event_espresso');
241
+				break;
242
+			case 'Price':
243
+				$obj_type = __('Price', 'event_espresso');
244
+				break;
245
+			case 'Promotion':
246
+				$obj_type = __('Promotion', 'event_espresso');
247
+				break;
248
+			case 'Ticket':
249
+				$obj_type = __('Ticket', 'event_espresso');
250
+				break;
251
+			case 'Transaction':
252
+				$obj_type = __('Transaction', 'event_espresso');
253
+				break;
254
+		}
255
+		return apply_filters('FHEE__EE_Line_Item__OBJ_type_i18n', $obj_type, $this);
256
+	}
257
+
258
+
259
+	/**
260
+	 * Sets item_type
261
+	 *
262
+	 * @param string $OBJ_type
263
+	 * @throws EE_Error
264
+	 */
265
+	public function set_OBJ_type($OBJ_type)
266
+	{
267
+		$this->set('OBJ_type', $OBJ_type);
268
+	}
269
+
270
+
271
+	/**
272
+	 * Gets unit_price
273
+	 *
274
+	 * @return float
275
+	 * @throws EE_Error
276
+	 */
277
+	public function unit_price()
278
+	{
279
+		return $this->get('LIN_unit_price');
280
+	}
281
+
282
+
283
+	/**
284
+	 * Sets unit_price
285
+	 *
286
+	 * @param float $unit_price
287
+	 * @throws EE_Error
288
+	 */
289
+	public function set_unit_price($unit_price)
290
+	{
291
+		$this->set('LIN_unit_price', $unit_price);
292
+	}
293
+
294
+
295
+	/**
296
+	 * Checks if this item is a percentage modifier or not
297
+	 *
298
+	 * @return boolean
299
+	 * @throws EE_Error
300
+	 */
301
+	public function is_percent()
302
+	{
303
+		if ($this->is_tax_sub_total()) {
304
+			// tax subtotals HAVE a percent on them, that percentage only applies
305
+			// to taxable items, so its' an exception. Treat it like a flat line item
306
+			return false;
307
+		}
308
+		$unit_price = abs($this->get('LIN_unit_price'));
309
+		$percent = abs($this->get('LIN_percent'));
310
+		if ($unit_price < .001 && $percent) {
311
+			return true;
312
+		}
313
+		if ($unit_price >= .001 && ! $percent) {
314
+			return false;
315
+		}
316
+		if ($unit_price >= .001 && $percent) {
317
+			throw new EE_Error(
318
+				sprintf(
319
+					esc_html__('A Line Item can not have a unit price of (%s) AND a percent (%s)!', 'event_espresso'),
320
+					$unit_price,
321
+					$percent
322
+				)
323
+			);
324
+		}
325
+		// if they're both 0, assume its not a percent item
326
+		return false;
327
+	}
328
+
329
+
330
+	/**
331
+	 * Gets percent (between 100-.001)
332
+	 *
333
+	 * @return float
334
+	 * @throws EE_Error
335
+	 */
336
+	public function percent()
337
+	{
338
+		return $this->get('LIN_percent');
339
+	}
340
+
341
+
342
+	/**
343
+	 * Sets percent (between 100-0.01)
344
+	 *
345
+	 * @param float $percent
346
+	 * @throws EE_Error
347
+	 */
348
+	public function set_percent($percent)
349
+	{
350
+		$this->set('LIN_percent', $percent);
351
+	}
352
+
353
+
354
+	/**
355
+	 * Gets total
356
+	 *
357
+	 * @return float
358
+	 * @throws EE_Error
359
+	 */
360
+	public function total()
361
+	{
362
+		return $this->get('LIN_total');
363
+	}
364
+
365
+
366
+	/**
367
+	 * Sets total
368
+	 *
369
+	 * @param float $total
370
+	 * @throws EE_Error
371
+	 */
372
+	public function set_total($total)
373
+	{
374
+		$this->set('LIN_total', $total);
375
+	}
376
+
377
+
378
+	/**
379
+	 * Gets order
380
+	 *
381
+	 * @return int
382
+	 * @throws EE_Error
383
+	 */
384
+	public function order()
385
+	{
386
+		return $this->get('LIN_order');
387
+	}
388
+
389
+
390
+	/**
391
+	 * Sets order
392
+	 *
393
+	 * @param int $order
394
+	 * @throws EE_Error
395
+	 */
396
+	public function set_order($order)
397
+	{
398
+		$this->set('LIN_order', $order);
399
+	}
400
+
401
+
402
+	/**
403
+	 * Gets parent
404
+	 *
405
+	 * @return int
406
+	 * @throws EE_Error
407
+	 */
408
+	public function parent_ID()
409
+	{
410
+		return $this->get('LIN_parent');
411
+	}
412
+
413
+
414
+	/**
415
+	 * Sets parent
416
+	 *
417
+	 * @param int $parent
418
+	 * @throws EE_Error
419
+	 */
420
+	public function set_parent_ID($parent)
421
+	{
422
+		$this->set('LIN_parent', $parent);
423
+	}
424
+
425
+
426
+	/**
427
+	 * Gets type
428
+	 *
429
+	 * @return string
430
+	 * @throws EE_Error
431
+	 */
432
+	public function type()
433
+	{
434
+		return $this->get('LIN_type');
435
+	}
436
+
437
+
438
+	/**
439
+	 * Sets type
440
+	 *
441
+	 * @param string $type
442
+	 * @throws EE_Error
443
+	 */
444
+	public function set_type($type)
445
+	{
446
+		$this->set('LIN_type', $type);
447
+	}
448
+
449
+
450
+	/**
451
+	 * Gets the line item of which this item is a composite. Eg, if this is a subtotal, the parent might be a total\
452
+	 * If this line item is saved to the DB, fetches the parent from the DB. However, if this line item isn't in the DB
453
+	 * it uses its cached reference to its parent line item (which would have been set by `EE_Line_Item::set_parent()`
454
+	 * or indirectly by `EE_Line_item::add_child_line_item()`)
455
+	 *
456
+	 * @return EE_Base_Class|EE_Line_Item
457
+	 * @throws EE_Error
458
+	 */
459
+	public function parent()
460
+	{
461
+		return $this->ID()
462
+			? $this->get_model()->get_one_by_ID($this->parent_ID())
463
+			: $this->_parent;
464
+	}
465
+
466
+
467
+	/**
468
+	 * Gets ALL the children of this line item (ie, all the parts that contribute towards this total).
469
+	 *
470
+	 * @return EE_Base_Class[]|EE_Line_Item[]
471
+	 * @throws EE_Error
472
+	 */
473
+	public function children()
474
+	{
475
+		if ($this->ID()) {
476
+			return $this->get_model()->get_all(
477
+				array(
478
+					array('LIN_parent' => $this->ID()),
479
+					'order_by' => array('LIN_order' => 'ASC'),
480
+				)
481
+			);
482
+		}
483
+		if (! is_array($this->_children)) {
484
+			$this->_children = array();
485
+		}
486
+		return $this->_children;
487
+	}
488
+
489
+
490
+	/**
491
+	 * Gets code
492
+	 *
493
+	 * @return string
494
+	 * @throws EE_Error
495
+	 */
496
+	public function code()
497
+	{
498
+		return $this->get('LIN_code');
499
+	}
500
+
501
+
502
+	/**
503
+	 * Sets code
504
+	 *
505
+	 * @param string $code
506
+	 * @throws EE_Error
507
+	 */
508
+	public function set_code($code)
509
+	{
510
+		$this->set('LIN_code', $code);
511
+	}
512
+
513
+
514
+	/**
515
+	 * Gets is_taxable
516
+	 *
517
+	 * @return boolean
518
+	 * @throws EE_Error
519
+	 */
520
+	public function is_taxable()
521
+	{
522
+		return $this->get('LIN_is_taxable');
523
+	}
524
+
525
+
526
+	/**
527
+	 * Sets is_taxable
528
+	 *
529
+	 * @param boolean $is_taxable
530
+	 * @throws EE_Error
531
+	 */
532
+	public function set_is_taxable($is_taxable)
533
+	{
534
+		$this->set('LIN_is_taxable', $is_taxable);
535
+	}
536
+
537
+
538
+	/**
539
+	 * Gets the object that this model-joins-to.
540
+	 * returns one of the model objects that the field OBJ_ID can point to... see the 'OBJ_ID' field on
541
+	 * EEM_Promotion_Object
542
+	 *
543
+	 *        Eg, if this line item join model object is for a ticket, this will return the EE_Ticket object
544
+	 *
545
+	 * @return EE_Base_Class | NULL
546
+	 * @throws EE_Error
547
+	 */
548
+	public function get_object()
549
+	{
550
+		$model_name_of_related_obj = $this->OBJ_type();
551
+		return $this->get_model()->has_relation($model_name_of_related_obj)
552
+			? $this->get_first_related($model_name_of_related_obj)
553
+			: null;
554
+	}
555
+
556
+
557
+	/**
558
+	 * Like EE_Line_Item::get_object(), but can only ever actually return an EE_Ticket.
559
+	 * (IE, if this line item is for a price or something else, will return NULL)
560
+	 *
561
+	 * @param array $query_params
562
+	 * @return EE_Base_Class|EE_Ticket
563
+	 * @throws EE_Error
564
+	 */
565
+	public function ticket($query_params = array())
566
+	{
567
+		// we're going to assume that when this method is called we always want to receive the attached ticket EVEN if that ticket is archived.  This can be overridden via the incoming $query_params argument
568
+		$remove_defaults = array('default_where_conditions' => 'none');
569
+		$query_params = array_merge($remove_defaults, $query_params);
570
+		return $this->get_first_related('Ticket', $query_params);
571
+	}
572
+
573
+
574
+	/**
575
+	 * Gets the EE_Datetime that's related to the ticket, IF this is for a ticket
576
+	 *
577
+	 * @return EE_Datetime | NULL
578
+	 * @throws EE_Error
579
+	 */
580
+	public function get_ticket_datetime()
581
+	{
582
+		if ($this->OBJ_type() === 'Ticket') {
583
+			$ticket = $this->ticket();
584
+			if ($ticket instanceof EE_Ticket) {
585
+				$datetime = $ticket->first_datetime();
586
+				if ($datetime instanceof EE_Datetime) {
587
+					return $datetime;
588
+				}
589
+			}
590
+		}
591
+		return null;
592
+	}
593
+
594
+
595
+	/**
596
+	 * Gets the event's name that's related to the ticket, if this is for
597
+	 * a ticket
598
+	 *
599
+	 * @return string
600
+	 * @throws EE_Error
601
+	 */
602
+	public function ticket_event_name()
603
+	{
604
+		$event_name = esc_html__('Unknown', 'event_espresso');
605
+		$event = $this->ticket_event();
606
+		if ($event instanceof EE_Event) {
607
+			$event_name = $event->name();
608
+		}
609
+		return $event_name;
610
+	}
611
+
612
+
613
+	/**
614
+	 * Gets the event that's related to the ticket, if this line item represents a ticket.
615
+	 *
616
+	 * @return EE_Event|null
617
+	 * @throws EE_Error
618
+	 */
619
+	public function ticket_event()
620
+	{
621
+		$event = null;
622
+		$ticket = $this->ticket();
623
+		if ($ticket instanceof EE_Ticket) {
624
+			$datetime = $ticket->first_datetime();
625
+			if ($datetime instanceof EE_Datetime) {
626
+				$event = $datetime->event();
627
+			}
628
+		}
629
+		return $event;
630
+	}
631
+
632
+
633
+	/**
634
+	 * Gets the first datetime for this lien item, assuming it's for a ticket
635
+	 *
636
+	 * @param string $date_format
637
+	 * @param string $time_format
638
+	 * @return string
639
+	 * @throws EE_Error
640
+	 */
641
+	public function ticket_datetime_start($date_format = '', $time_format = '')
642
+	{
643
+		$first_datetime_string = esc_html__('Unknown', 'event_espresso');
644
+		$datetime = $this->get_ticket_datetime();
645
+		if ($datetime) {
646
+			$first_datetime_string = $datetime->start_date_and_time($date_format, $time_format);
647
+		}
648
+		return $first_datetime_string;
649
+	}
650
+
651
+
652
+	/**
653
+	 * Adds the line item as a child to this line item. If there is another child line
654
+	 * item with the same LIN_code, it is overwritten by this new one
655
+	 *
656
+	 * @param EEI_Line_Item $line_item
657
+	 * @param bool          $set_order
658
+	 * @return bool success
659
+	 * @throws EE_Error
660
+	 */
661
+	public function add_child_line_item(EEI_Line_Item $line_item, $set_order = true)
662
+	{
663
+		// should we calculate the LIN_order for this line item ?
664
+		if ($set_order || $line_item->order() === null) {
665
+			$line_item->set_order(count($this->children()));
666
+		}
667
+		if ($this->ID()) {
668
+			// check for any duplicate line items (with the same code), if so, this replaces it
669
+			$line_item_with_same_code = $this->get_child_line_item($line_item->code());
670
+			if ($line_item_with_same_code instanceof EE_Line_Item && $line_item_with_same_code !== $line_item) {
671
+				$this->delete_child_line_item($line_item_with_same_code->code());
672
+			}
673
+			$line_item->set_parent_ID($this->ID());
674
+			if ($this->TXN_ID()) {
675
+				$line_item->set_TXN_ID($this->TXN_ID());
676
+			}
677
+			return $line_item->save();
678
+		}
679
+		$this->_children[ $line_item->code() ] = $line_item;
680
+		if ($line_item->parent() !== $this) {
681
+			$line_item->set_parent($this);
682
+		}
683
+		return true;
684
+	}
685
+
686
+
687
+	/**
688
+	 * Similar to EE_Base_Class::_add_relation_to, except this isn't a normal relation.
689
+	 * If this line item is saved to the DB, this is just a wrapper for set_parent_ID() and save()
690
+	 * However, if this line item is NOT saved to the DB, this just caches the parent on
691
+	 * the EE_Line_Item::_parent property.
692
+	 *
693
+	 * @param EE_Line_Item $line_item
694
+	 * @throws EE_Error
695
+	 */
696
+	public function set_parent($line_item)
697
+	{
698
+		if ($this->ID()) {
699
+			if (! $line_item->ID()) {
700
+				$line_item->save();
701
+			}
702
+			$this->set_parent_ID($line_item->ID());
703
+			$this->save();
704
+		} else {
705
+			$this->_parent = $line_item;
706
+			$this->set_parent_ID($line_item->ID());
707
+		}
708
+	}
709
+
710
+
711
+	/**
712
+	 * Gets the child line item as specified by its code. Because this returns an object (by reference)
713
+	 * you can modify this child line item and the parent (this object) can know about them
714
+	 * because it also has a reference to that line item
715
+	 *
716
+	 * @param string $code
717
+	 * @return EE_Base_Class|EE_Line_Item|EE_Soft_Delete_Base_Class|NULL
718
+	 * @throws EE_Error
719
+	 */
720
+	public function get_child_line_item($code)
721
+	{
722
+		if ($this->ID()) {
723
+			return $this->get_model()->get_one(
724
+				array(array('LIN_parent' => $this->ID(), 'LIN_code' => $code))
725
+			);
726
+		}
727
+		return isset($this->_children[ $code ])
728
+			? $this->_children[ $code ]
729
+			: null;
730
+	}
731
+
732
+
733
+	/**
734
+	 * Returns how many items are deleted (or, if this item has not been saved ot the DB yet, just how many it HAD
735
+	 * cached on it)
736
+	 *
737
+	 * @return int
738
+	 * @throws EE_Error
739
+	 */
740
+	public function delete_children_line_items()
741
+	{
742
+		if ($this->ID()) {
743
+			return $this->get_model()->delete(array(array('LIN_parent' => $this->ID())));
744
+		}
745
+		$count = count($this->_children);
746
+		$this->_children = array();
747
+		return $count;
748
+	}
749
+
750
+
751
+	/**
752
+	 * If this line item has been saved to the DB, deletes its child with LIN_code == $code. If this line
753
+	 * HAS NOT been saved to the DB, removes the child line item with index $code.
754
+	 * Also searches through the child's children for a matching line item. However, once a line item has been found
755
+	 * and deleted, stops searching (so if there are line items with duplicate codes, only the first one found will be
756
+	 * deleted)
757
+	 *
758
+	 * @param string $code
759
+	 * @param bool   $stop_search_once_found
760
+	 * @return int count of items deleted (or simply removed from the line item's cache, if not has not been saved to
761
+	 *             the DB yet)
762
+	 * @throws EE_Error
763
+	 */
764
+	public function delete_child_line_item($code, $stop_search_once_found = true)
765
+	{
766
+		if ($this->ID()) {
767
+			$items_deleted = 0;
768
+			if ($this->code() === $code) {
769
+				$items_deleted += EEH_Line_Item::delete_all_child_items($this);
770
+				$items_deleted += (int) $this->delete();
771
+				if ($stop_search_once_found) {
772
+					return $items_deleted;
773
+				}
774
+			}
775
+			foreach ($this->children() as $child_line_item) {
776
+				$items_deleted += $child_line_item->delete_child_line_item($code, $stop_search_once_found);
777
+			}
778
+			return $items_deleted;
779
+		}
780
+		if (isset($this->_children[ $code ])) {
781
+			unset($this->_children[ $code ]);
782
+			return 1;
783
+		}
784
+		return 0;
785
+	}
786
+
787
+
788
+	/**
789
+	 * If this line item is in the database, is of the type subtotal, and
790
+	 * has no children, why do we have it? It should be deleted so this function
791
+	 * does that
792
+	 *
793
+	 * @return boolean
794
+	 * @throws EE_Error
795
+	 */
796
+	public function delete_if_childless_subtotal()
797
+	{
798
+		if ($this->ID() && $this->type() === EEM_Line_Item::type_sub_total && ! $this->children()) {
799
+			return $this->delete();
800
+		}
801
+		return false;
802
+	}
803
+
804
+
805
+	/**
806
+	 * Creates a code and returns a string. doesn't assign the code to this model object
807
+	 *
808
+	 * @return string
809
+	 * @throws EE_Error
810
+	 */
811
+	public function generate_code()
812
+	{
813
+		// each line item in the cart requires a unique identifier
814
+		return md5($this->get('OBJ_type') . $this->get('OBJ_ID') . microtime());
815
+	}
816
+
817
+
818
+	/**
819
+	 * @return bool
820
+	 * @throws EE_Error
821
+	 */
822
+	public function is_tax()
823
+	{
824
+		return $this->type() === EEM_Line_Item::type_tax;
825
+	}
826
+
827
+
828
+	/**
829
+	 * @return bool
830
+	 * @throws EE_Error
831
+	 */
832
+	public function is_tax_sub_total()
833
+	{
834
+		return $this->type() === EEM_Line_Item::type_tax_sub_total;
835
+	}
836
+
837
+
838
+	/**
839
+	 * @return bool
840
+	 * @throws EE_Error
841
+	 */
842
+	public function is_line_item()
843
+	{
844
+		return $this->type() === EEM_Line_Item::type_line_item;
845
+	}
846
+
847
+
848
+	/**
849
+	 * @return bool
850
+	 * @throws EE_Error
851
+	 */
852
+	public function is_sub_line_item()
853
+	{
854
+		return $this->type() === EEM_Line_Item::type_sub_line_item;
855
+	}
856
+
857
+
858
+	/**
859
+	 * @return bool
860
+	 * @throws EE_Error
861
+	 */
862
+	public function is_sub_total()
863
+	{
864
+		return $this->type() === EEM_Line_Item::type_sub_total;
865
+	}
866
+
867
+
868
+	/**
869
+	 * Whether or not this line item is a cancellation line item
870
+	 *
871
+	 * @return boolean
872
+	 * @throws EE_Error
873
+	 */
874
+	public function is_cancellation()
875
+	{
876
+		return EEM_Line_Item::type_cancellation === $this->type();
877
+	}
878
+
879
+
880
+	/**
881
+	 * @return bool
882
+	 * @throws EE_Error
883
+	 */
884
+	public function is_total()
885
+	{
886
+		return $this->type() === EEM_Line_Item::type_total;
887
+	}
888
+
889
+
890
+	/**
891
+	 * @return bool
892
+	 * @throws EE_Error
893
+	 */
894
+	public function is_cancelled()
895
+	{
896
+		return $this->type() === EEM_Line_Item::type_cancellation;
897
+	}
898
+
899
+
900
+	/**
901
+	 * @return string like '2, 004.00', formatted according to the localized currency
902
+	 * @throws EE_Error
903
+	 */
904
+	public function unit_price_no_code()
905
+	{
906
+		return $this->get_pretty('LIN_unit_price', 'no_currency_code');
907
+	}
908
+
909
+
910
+	/**
911
+	 * @return string like '2, 004.00', formatted according to the localized currency
912
+	 * @throws EE_Error
913
+	 */
914
+	public function total_no_code()
915
+	{
916
+		return $this->get_pretty('LIN_total', 'no_currency_code');
917
+	}
918
+
919
+
920
+	/**
921
+	 * Gets the final total on this item, taking taxes into account.
922
+	 * Has the side-effect of setting the sub-total as it was just calculated.
923
+	 * If this is used on a grand-total line item, also updates the transaction's
924
+	 * TXN_total (provided this line item is allowed to persist, otherwise we don't
925
+	 * want to change a persistable transaction with info from a non-persistent line item)
926
+	 *
927
+	 * @return float
928
+	 * @throws EE_Error
929
+	 * @throws InvalidArgumentException
930
+	 * @throws InvalidInterfaceException
931
+	 * @throws InvalidDataTypeException
932
+	 */
933
+	public function recalculate_total_including_taxes()
934
+	{
935
+		$pre_tax_total = $this->recalculate_pre_tax_total();
936
+		$tax_total = $this->recalculate_taxes_and_tax_total();
937
+		$total = $pre_tax_total + $tax_total;
938
+		// no negative totals plz
939
+		$total = max($total, 0);
940
+		$this->set_total($total);
941
+		// only update the related transaction's total
942
+		// if we intend to save this line item and its a grand total
943
+		if ($this->allow_persist() && $this->type() === EEM_Line_Item::type_total
944
+			&& $this->transaction()
945
+			   instanceof
946
+			   EE_Transaction
947
+		) {
948
+			$this->transaction()->set_total($total);
949
+			if ($this->transaction()->ID()) {
950
+				$this->transaction()->save();
951
+			}
952
+		}
953
+		$this->maybe_save();
954
+		return $total;
955
+	}
956
+
957
+
958
+	/**
959
+	 * Recursively goes through all the children and recalculates sub-totals EXCEPT for
960
+	 * tax-sub-totals (they're a an odd beast). Updates the 'total' on each line item according to either its
961
+	 * unit price * quantity or the total of all its children EXCEPT when we're only calculating the taxable total and
962
+	 * when this is called on the grand total
963
+	 *
964
+	 * @return float
965
+	 * @throws InvalidArgumentException
966
+	 * @throws InvalidInterfaceException
967
+	 * @throws InvalidDataTypeException
968
+	 * @throws EE_Error
969
+	 */
970
+	public function recalculate_pre_tax_total()
971
+	{
972
+		$total = 0;
973
+		$my_children = $this->children();
974
+		$has_children = ! empty($my_children);
975
+		if ($has_children && $this->is_line_item()) {
976
+			$total = $this->_recalculate_pretax_total_for_line_item($total, $my_children);
977
+		} elseif (! $has_children && ($this->is_sub_line_item() || $this->is_line_item())) {
978
+			$total = $this->unit_price() * $this->quantity();
979
+		} elseif ($this->is_sub_total() || $this->is_total()) {
980
+			$total = $this->_recalculate_pretax_total_for_subtotal($total, $my_children);
981
+		} elseif ($this->is_tax_sub_total() || $this->is_tax() || $this->is_cancelled()) {
982
+			// completely ignore tax totals, tax sub-totals, and cancelled line items, when calculating the pre-tax-total
983
+			return 0;
984
+		}
985
+		// ensure all non-line items and non-sub-line-items have a quantity of 1 (except for Events)
986
+		if (! $this->is_line_item() && ! $this->is_sub_line_item() && ! $this->is_cancellation()
987
+		) {
988
+			if ($this->OBJ_type() !== 'Event') {
989
+				$this->set_quantity(1);
990
+			}
991
+			if (! $this->is_percent()) {
992
+				$this->set_unit_price($total);
993
+			}
994
+		}
995
+		// we don't want to bother saving grand totals, because that needs to factor in taxes anyways
996
+		// so it ought to be
997
+		if (! $this->is_total()) {
998
+			$this->set_total($total);
999
+			// if not a percent line item, make sure we keep the unit price in sync
1000
+			if ($has_children
1001
+				&& $this->is_line_item()
1002
+				&& ! $this->is_percent()
1003
+			) {
1004
+				if ($this->quantity() === 0) {
1005
+					$new_unit_price = 0;
1006
+				} else {
1007
+					$new_unit_price = $this->total() / $this->quantity();
1008
+				}
1009
+				$this->set_unit_price($new_unit_price);
1010
+			}
1011
+			$this->maybe_save();
1012
+		}
1013
+		return $total;
1014
+	}
1015
+
1016
+
1017
+	/**
1018
+	 * Calculates the pretax total when this line item is a subtotal or total line item.
1019
+	 * Basically does a sum-then-round approach (ie, any percent line item that are children
1020
+	 * will calculate their total based on the un-rounded total we're working with so far, and
1021
+	 * THEN round the result; instead of rounding as we go like with sub-line-items)
1022
+	 *
1023
+	 * @param float          $calculated_total_so_far
1024
+	 * @param EE_Line_Item[] $my_children
1025
+	 * @return float
1026
+	 * @throws InvalidArgumentException
1027
+	 * @throws InvalidInterfaceException
1028
+	 * @throws InvalidDataTypeException
1029
+	 * @throws EE_Error
1030
+	 */
1031
+	protected function _recalculate_pretax_total_for_subtotal($calculated_total_so_far, $my_children = null)
1032
+	{
1033
+		if ($my_children === null) {
1034
+			$my_children = $this->children();
1035
+		}
1036
+		$subtotal_quantity = 0;
1037
+		// get the total of all its children
1038
+		foreach ($my_children as $child_line_item) {
1039
+			if ($child_line_item instanceof EE_Line_Item && ! $child_line_item->is_cancellation()) {
1040
+				// percentage line items are based on total so far
1041
+				if ($child_line_item->is_percent()) {
1042
+					// round as we go so that the line items add up ok
1043
+					$percent_total = round(
1044
+						$calculated_total_so_far * $child_line_item->percent() / 100,
1045
+						EE_Registry::instance()->CFG->currency->dec_plc
1046
+					);
1047
+					$child_line_item->set_total($percent_total);
1048
+					// so far all percent line items should have a quantity of 1
1049
+					// (ie, no double percent discounts. Although that might be requested someday)
1050
+					$child_line_item->set_quantity(1);
1051
+					$child_line_item->maybe_save();
1052
+					$calculated_total_so_far += $percent_total;
1053
+				} else {
1054
+					// verify flat sub-line-item quantities match their parent
1055
+					if ($child_line_item->is_sub_line_item()) {
1056
+						$child_line_item->set_quantity($this->quantity());
1057
+					}
1058
+					$calculated_total_so_far += $child_line_item->recalculate_pre_tax_total();
1059
+					$subtotal_quantity += $child_line_item->quantity();
1060
+				}
1061
+			}
1062
+		}
1063
+		if ($this->is_sub_total()) {
1064
+			// no negative totals plz
1065
+			$calculated_total_so_far = max($calculated_total_so_far, 0);
1066
+			$subtotal_quantity = $subtotal_quantity > 0 ? 1 : 0;
1067
+			$this->set_quantity($subtotal_quantity);
1068
+			$this->maybe_save();
1069
+		}
1070
+		return $calculated_total_so_far;
1071
+	}
1072
+
1073
+
1074
+	/**
1075
+	 * Calculates the pretax total for a normal line item, in a round-then-sum approach
1076
+	 * (where each sub-line-item is applied to the base price for the line item
1077
+	 * and the result is immediately rounded, rather than summing all the sub-line-items
1078
+	 * then rounding, like we do when recalculating pretax totals on totals and subtotals).
1079
+	 *
1080
+	 * @param float          $calculated_total_so_far
1081
+	 * @param EE_Line_Item[] $my_children
1082
+	 * @return float
1083
+	 * @throws InvalidArgumentException
1084
+	 * @throws InvalidInterfaceException
1085
+	 * @throws InvalidDataTypeException
1086
+	 * @throws EE_Error
1087
+	 */
1088
+	protected function _recalculate_pretax_total_for_line_item($calculated_total_so_far, $my_children = null)
1089
+	{
1090
+		if ($my_children === null) {
1091
+			$my_children = $this->children();
1092
+		}
1093
+		// we need to keep track of the running total for a single item,
1094
+		// because we need to round as we go
1095
+		$unit_price_for_total = 0;
1096
+		$quantity_for_total = 1;
1097
+		// get the total of all its children
1098
+		foreach ($my_children as $child_line_item) {
1099
+			if ($child_line_item instanceof EE_Line_Item && ! $child_line_item->is_cancellation()) {
1100
+				if ($child_line_item->is_percent()) {
1101
+					// it should be the unit-price-so-far multiplied by teh percent multiplied by the quantity
1102
+					// not total multiplied by percent, because that ignores rounding along-the-way
1103
+					$percent_unit_price = round(
1104
+						$unit_price_for_total * $child_line_item->percent() / 100,
1105
+						EE_Registry::instance()->CFG->currency->dec_plc
1106
+					);
1107
+					$percent_total = $percent_unit_price * $quantity_for_total;
1108
+					$child_line_item->set_total($percent_total);
1109
+					// so far all percent line items should have a quantity of 1
1110
+					// (ie, no double percent discounts. Although that might be requested someday)
1111
+					$child_line_item->set_quantity(1);
1112
+					$child_line_item->maybe_save();
1113
+					$calculated_total_so_far += $percent_total;
1114
+					$unit_price_for_total += $percent_unit_price;
1115
+				} else {
1116
+					// verify flat sub-line-item quantities match their parent
1117
+					if ($child_line_item->is_sub_line_item()) {
1118
+						$child_line_item->set_quantity($this->quantity());
1119
+					}
1120
+					$quantity_for_total = $child_line_item->quantity();
1121
+					$calculated_total_so_far += $child_line_item->recalculate_pre_tax_total();
1122
+					$unit_price_for_total += $child_line_item->unit_price();
1123
+				}
1124
+			}
1125
+		}
1126
+		return $calculated_total_so_far;
1127
+	}
1128
+
1129
+
1130
+	/**
1131
+	 * Recalculates the total on each individual tax (based on a recalculation of the pre-tax total), sets
1132
+	 * the totals on each tax calculated, and returns the final tax total. Re-saves tax line items
1133
+	 * and tax sub-total if already in the DB
1134
+	 *
1135
+	 * @return float
1136
+	 * @throws EE_Error
1137
+	 */
1138
+	public function recalculate_taxes_and_tax_total()
1139
+	{
1140
+		// get all taxes
1141
+		$taxes = $this->tax_descendants();
1142
+		// calculate the pretax total
1143
+		$taxable_total = $this->taxable_total();
1144
+		$tax_total = 0;
1145
+		foreach ($taxes as $tax) {
1146
+			$total_on_this_tax = $taxable_total * $tax->percent() / 100;
1147
+			// remember the total on this line item
1148
+			$tax->set_total($total_on_this_tax);
1149
+			$tax->maybe_save();
1150
+			$tax_total += $tax->total();
1151
+		}
1152
+		$this->_recalculate_tax_sub_total();
1153
+		return $tax_total;
1154
+	}
1155
+
1156
+
1157
+	/**
1158
+	 * Simply forces all the tax-sub-totals to recalculate. Assumes the taxes have been calculated
1159
+	 *
1160
+	 * @return void
1161
+	 * @throws EE_Error
1162
+	 */
1163
+	private function _recalculate_tax_sub_total()
1164
+	{
1165
+		if ($this->is_tax_sub_total()) {
1166
+			$total = 0;
1167
+			$total_percent = 0;
1168
+			// simply loop through all its children (which should be taxes) and sum their total
1169
+			foreach ($this->children() as $child_tax) {
1170
+				if ($child_tax instanceof EE_Line_Item) {
1171
+					$total += $child_tax->total();
1172
+					$total_percent += $child_tax->percent();
1173
+				}
1174
+			}
1175
+			$this->set_total($total);
1176
+			$this->set_percent($total_percent);
1177
+			$this->maybe_save();
1178
+		} elseif ($this->is_total()) {
1179
+			foreach ($this->children() as $maybe_tax_subtotal) {
1180
+				if ($maybe_tax_subtotal instanceof EE_Line_Item) {
1181
+					$maybe_tax_subtotal->_recalculate_tax_sub_total();
1182
+				}
1183
+			}
1184
+		}
1185
+	}
1186
+
1187
+
1188
+	/**
1189
+	 * Gets the total tax on this line item. Assumes taxes have already been calculated using
1190
+	 * recalculate_taxes_and_total
1191
+	 *
1192
+	 * @return float
1193
+	 * @throws EE_Error
1194
+	 */
1195
+	public function get_total_tax()
1196
+	{
1197
+		$this->_recalculate_tax_sub_total();
1198
+		$total = 0;
1199
+		foreach ($this->tax_descendants() as $tax_line_item) {
1200
+			if ($tax_line_item instanceof EE_Line_Item) {
1201
+				$total += $tax_line_item->total();
1202
+			}
1203
+		}
1204
+		return $total;
1205
+	}
1206
+
1207
+
1208
+	/**
1209
+	 * Gets the total for all the items purchased only
1210
+	 *
1211
+	 * @return float
1212
+	 * @throws EE_Error
1213
+	 */
1214
+	public function get_items_total()
1215
+	{
1216
+		// by default, let's make sure we're consistent with the existing line item
1217
+		if ($this->is_total()) {
1218
+			$pretax_subtotal_li = EEH_Line_Item::get_pre_tax_subtotal($this);
1219
+			if ($pretax_subtotal_li instanceof EE_Line_Item) {
1220
+				return $pretax_subtotal_li->total();
1221
+			}
1222
+		}
1223
+		$total = 0;
1224
+		foreach ($this->get_items() as $item) {
1225
+			if ($item instanceof EE_Line_Item) {
1226
+				$total += $item->total();
1227
+			}
1228
+		}
1229
+		return $total;
1230
+	}
1231
+
1232
+
1233
+	/**
1234
+	 * Gets all the descendants (ie, children or children of children etc) that
1235
+	 * are of the type 'tax'
1236
+	 *
1237
+	 * @return EE_Line_Item[]
1238
+	 */
1239
+	public function tax_descendants()
1240
+	{
1241
+		return EEH_Line_Item::get_tax_descendants($this);
1242
+	}
1243
+
1244
+
1245
+	/**
1246
+	 * Gets all the real items purchased which are children of this item
1247
+	 *
1248
+	 * @return EE_Line_Item[]
1249
+	 */
1250
+	public function get_items()
1251
+	{
1252
+		return EEH_Line_Item::get_line_item_descendants($this);
1253
+	}
1254
+
1255
+
1256
+	/**
1257
+	 * Returns the amount taxable among this line item's children (or if it has no children,
1258
+	 * how much of it is taxable). Does not recalculate totals or subtotals.
1259
+	 * If the taxable total is negative, (eg, if none of the tickets were taxable,
1260
+	 * but there is a "Taxable" discount), returns 0.
1261
+	 *
1262
+	 * @return float
1263
+	 * @throws EE_Error
1264
+	 */
1265
+	public function taxable_total()
1266
+	{
1267
+		$total = 0;
1268
+		if ($this->children()) {
1269
+			foreach ($this->children() as $child_line_item) {
1270
+				if ($child_line_item->type() === EEM_Line_Item::type_line_item && $child_line_item->is_taxable()) {
1271
+					// if it's a percent item, only take into account the percent
1272
+					// that's taxable too (the taxable total so far)
1273
+					if ($child_line_item->is_percent()) {
1274
+						$total += ($total * $child_line_item->percent() / 100);
1275
+					} else {
1276
+						$total += $child_line_item->total();
1277
+					}
1278
+				} elseif ($child_line_item->type() === EEM_Line_Item::type_sub_total) {
1279
+					$total += $child_line_item->taxable_total();
1280
+				}
1281
+			}
1282
+		}
1283
+		return max($total, 0);
1284
+	}
1285
+
1286
+
1287
+	/**
1288
+	 * Gets the transaction for this line item
1289
+	 *
1290
+	 * @return EE_Base_Class|EE_Transaction
1291
+	 * @throws EE_Error
1292
+	 */
1293
+	public function transaction()
1294
+	{
1295
+		return $this->get_first_related('Transaction');
1296
+	}
1297
+
1298
+
1299
+	/**
1300
+	 * Saves this line item to the DB, and recursively saves its descendants.
1301
+	 * Because there currently is no proper parent-child relation on the model,
1302
+	 * save_this_and_cached() will NOT save the descendants.
1303
+	 * Also sets the transaction on this line item and all its descendants before saving
1304
+	 *
1305
+	 * @param int $txn_id if none is provided, assumes $this->TXN_ID()
1306
+	 * @return int count of items saved
1307
+	 * @throws EE_Error
1308
+	 */
1309
+	public function save_this_and_descendants_to_txn($txn_id = null)
1310
+	{
1311
+		$count = 0;
1312
+		if (! $txn_id) {
1313
+			$txn_id = $this->TXN_ID();
1314
+		}
1315
+		$this->set_TXN_ID($txn_id);
1316
+		$children = $this->children();
1317
+		$count += $this->save()
1318
+			? 1
1319
+			: 0;
1320
+		foreach ($children as $child_line_item) {
1321
+			if ($child_line_item instanceof EE_Line_Item) {
1322
+				$child_line_item->set_parent_ID($this->ID());
1323
+				$count += $child_line_item->save_this_and_descendants_to_txn($txn_id);
1324
+			}
1325
+		}
1326
+		return $count;
1327
+	}
1328
+
1329
+
1330
+	/**
1331
+	 * Saves this line item to the DB, and recursively saves its descendants.
1332
+	 *
1333
+	 * @return int count of items saved
1334
+	 * @throws EE_Error
1335
+	 */
1336
+	public function save_this_and_descendants()
1337
+	{
1338
+		$count = 0;
1339
+		$children = $this->children();
1340
+		$count += $this->save()
1341
+			? 1
1342
+			: 0;
1343
+		foreach ($children as $child_line_item) {
1344
+			if ($child_line_item instanceof EE_Line_Item) {
1345
+				$child_line_item->set_parent_ID($this->ID());
1346
+				$count += $child_line_item->save_this_and_descendants();
1347
+			}
1348
+		}
1349
+		return $count;
1350
+	}
1351
+
1352
+
1353
+	/**
1354
+	 * returns the cancellation line item if this item was cancelled
1355
+	 *
1356
+	 * @return EE_Line_Item[]
1357
+	 * @throws InvalidArgumentException
1358
+	 * @throws InvalidInterfaceException
1359
+	 * @throws InvalidDataTypeException
1360
+	 * @throws ReflectionException
1361
+	 * @throws EE_Error
1362
+	 */
1363
+	public function get_cancellations()
1364
+	{
1365
+		EE_Registry::instance()->load_helper('Line_Item');
1366
+		return EEH_Line_Item::get_descendants_of_type($this, EEM_Line_Item::type_cancellation);
1367
+	}
1368
+
1369
+
1370
+	/**
1371
+	 * If this item has an ID, then this saves it again to update the db
1372
+	 *
1373
+	 * @return int count of items saved
1374
+	 * @throws EE_Error
1375
+	 */
1376
+	public function maybe_save()
1377
+	{
1378
+		if ($this->ID()) {
1379
+			return $this->save();
1380
+		}
1381
+		return false;
1382
+	}
1383
+
1384
+
1385
+	/**
1386
+	 * clears the cached children and parent from the line item
1387
+	 *
1388
+	 * @return void
1389
+	 */
1390
+	public function clear_related_line_item_cache()
1391
+	{
1392
+		$this->_children = array();
1393
+		$this->_parent = null;
1394
+	}
1395
+
1396
+
1397
+	/**
1398
+	 * @param bool $raw
1399
+	 * @return int
1400
+	 * @throws EE_Error
1401
+	 */
1402
+	public function timestamp($raw = false)
1403
+	{
1404
+		return $raw
1405
+			? $this->get_raw('LIN_timestamp')
1406
+			: $this->get('LIN_timestamp');
1407
+	}
1408
+
1409
+
1410
+
1411
+
1412
+	/************************* DEPRECATED *************************/
1413
+	/**
1414
+	 * @deprecated 4.6.0
1415
+	 * @param string $type one of the constants on EEM_Line_Item
1416
+	 * @return EE_Line_Item[]
1417
+	 */
1418
+	protected function _get_descendants_of_type($type)
1419
+	{
1420
+		EE_Error::doing_it_wrong(
1421
+			'EE_Line_Item::_get_descendants_of_type()',
1422
+			__('Method replaced with EEH_Line_Item::get_descendants_of_type()', 'event_espresso'),
1423
+			'4.6.0'
1424
+		);
1425
+		return EEH_Line_Item::get_descendants_of_type($this, $type);
1426
+	}
1427
+
1428
+
1429
+	/**
1430
+	 * @deprecated 4.6.0
1431
+	 * @param string $type like one of the EEM_Line_Item::type_*
1432
+	 * @return EE_Line_Item
1433
+	 */
1434
+	public function get_nearest_descendant_of_type($type)
1435
+	{
1436
+		EE_Error::doing_it_wrong(
1437
+			'EE_Line_Item::get_nearest_descendant_of_type()',
1438
+			__('Method replaced with EEH_Line_Item::get_nearest_descendant_of_type()', 'event_espresso'),
1439
+			'4.6.0'
1440
+		);
1441
+		return EEH_Line_Item::get_nearest_descendant_of_type($this, $type);
1442
+	}
1443 1443
 }
Please login to merge, or discard this patch.
Spacing   +15 added lines, -15 removed lines patch added patch discarded remove patch
@@ -72,7 +72,7 @@  discard block
 block discarded – undo
72 72
     protected function __construct($fieldValues = array(), $bydb = false, $timezone = '')
73 73
     {
74 74
         parent::__construct($fieldValues, $bydb, $timezone);
75
-        if (! $this->get('LIN_code')) {
75
+        if ( ! $this->get('LIN_code')) {
76 76
             $this->set_code($this->generate_code());
77 77
         }
78 78
     }
@@ -123,7 +123,7 @@  discard block
 block discarded – undo
123 123
     public function name()
124 124
     {
125 125
         $name = $this->get('LIN_name');
126
-        if (! $name) {
126
+        if ( ! $name) {
127 127
             $name = ucwords(str_replace('-', ' ', $this->type()));
128 128
         }
129 129
         return $name;
@@ -480,7 +480,7 @@  discard block
 block discarded – undo
480 480
                 )
481 481
             );
482 482
         }
483
-        if (! is_array($this->_children)) {
483
+        if ( ! is_array($this->_children)) {
484 484
             $this->_children = array();
485 485
         }
486 486
         return $this->_children;
@@ -676,7 +676,7 @@  discard block
 block discarded – undo
676 676
             }
677 677
             return $line_item->save();
678 678
         }
679
-        $this->_children[ $line_item->code() ] = $line_item;
679
+        $this->_children[$line_item->code()] = $line_item;
680 680
         if ($line_item->parent() !== $this) {
681 681
             $line_item->set_parent($this);
682 682
         }
@@ -696,7 +696,7 @@  discard block
 block discarded – undo
696 696
     public function set_parent($line_item)
697 697
     {
698 698
         if ($this->ID()) {
699
-            if (! $line_item->ID()) {
699
+            if ( ! $line_item->ID()) {
700 700
                 $line_item->save();
701 701
             }
702 702
             $this->set_parent_ID($line_item->ID());
@@ -724,8 +724,8 @@  discard block
 block discarded – undo
724 724
                 array(array('LIN_parent' => $this->ID(), 'LIN_code' => $code))
725 725
             );
726 726
         }
727
-        return isset($this->_children[ $code ])
728
-            ? $this->_children[ $code ]
727
+        return isset($this->_children[$code])
728
+            ? $this->_children[$code]
729 729
             : null;
730 730
     }
731 731
 
@@ -777,8 +777,8 @@  discard block
 block discarded – undo
777 777
             }
778 778
             return $items_deleted;
779 779
         }
780
-        if (isset($this->_children[ $code ])) {
781
-            unset($this->_children[ $code ]);
780
+        if (isset($this->_children[$code])) {
781
+            unset($this->_children[$code]);
782 782
             return 1;
783 783
         }
784 784
         return 0;
@@ -811,7 +811,7 @@  discard block
 block discarded – undo
811 811
     public function generate_code()
812 812
     {
813 813
         // each line item in the cart requires a unique identifier
814
-        return md5($this->get('OBJ_type') . $this->get('OBJ_ID') . microtime());
814
+        return md5($this->get('OBJ_type').$this->get('OBJ_ID').microtime());
815 815
     }
816 816
 
817 817
 
@@ -974,7 +974,7 @@  discard block
 block discarded – undo
974 974
         $has_children = ! empty($my_children);
975 975
         if ($has_children && $this->is_line_item()) {
976 976
             $total = $this->_recalculate_pretax_total_for_line_item($total, $my_children);
977
-        } elseif (! $has_children && ($this->is_sub_line_item() || $this->is_line_item())) {
977
+        } elseif ( ! $has_children && ($this->is_sub_line_item() || $this->is_line_item())) {
978 978
             $total = $this->unit_price() * $this->quantity();
979 979
         } elseif ($this->is_sub_total() || $this->is_total()) {
980 980
             $total = $this->_recalculate_pretax_total_for_subtotal($total, $my_children);
@@ -983,18 +983,18 @@  discard block
 block discarded – undo
983 983
             return 0;
984 984
         }
985 985
         // ensure all non-line items and non-sub-line-items have a quantity of 1 (except for Events)
986
-        if (! $this->is_line_item() && ! $this->is_sub_line_item() && ! $this->is_cancellation()
986
+        if ( ! $this->is_line_item() && ! $this->is_sub_line_item() && ! $this->is_cancellation()
987 987
         ) {
988 988
             if ($this->OBJ_type() !== 'Event') {
989 989
                 $this->set_quantity(1);
990 990
             }
991
-            if (! $this->is_percent()) {
991
+            if ( ! $this->is_percent()) {
992 992
                 $this->set_unit_price($total);
993 993
             }
994 994
         }
995 995
         // we don't want to bother saving grand totals, because that needs to factor in taxes anyways
996 996
         // so it ought to be
997
-        if (! $this->is_total()) {
997
+        if ( ! $this->is_total()) {
998 998
             $this->set_total($total);
999 999
             // if not a percent line item, make sure we keep the unit price in sync
1000 1000
             if ($has_children
@@ -1309,7 +1309,7 @@  discard block
 block discarded – undo
1309 1309
     public function save_this_and_descendants_to_txn($txn_id = null)
1310 1310
     {
1311 1311
         $count = 0;
1312
-        if (! $txn_id) {
1312
+        if ( ! $txn_id) {
1313 1313
             $txn_id = $this->TXN_ID();
1314 1314
         }
1315 1315
         $this->set_TXN_ID($txn_id);
Please login to merge, or discard this patch.
core/libraries/form_sections/base/EE_Form_Section_Proper.form.php 3 patches
Doc Comments   +1 added lines, -2 removed lines patch added patch discarded remove patch
@@ -459,7 +459,7 @@  discard block
 block discarded – undo
459 459
      *                                                      with construction finalize being called later
460 460
      *                                                      (realizing that the subsections' html names
461 461
      *                                                      might not be set yet, etc.)
462
-     * @return EE_Form_Section_Base
462
+     * @return EE_Form_Section_Validatable|null
463 463
      * @throws EE_Error
464 464
      */
465 465
     public function get_subsection($name, $require_construction_to_be_finalized = true)
@@ -1289,7 +1289,6 @@  discard block
 block discarded – undo
1289 1289
     /**
1290 1290
      * Sets the submission error message (aka validation error message for this form section and all sub-sections)
1291 1291
      * @param string                           $form_submission_error_message
1292
-     * @param EE_Form_Section_Validatable $form_section unused
1293 1292
      * @throws EE_Error
1294 1293
      */
1295 1294
     public function set_submission_error_message(
Please login to merge, or discard this patch.
Spacing   +38 added lines, -38 removed lines patch added patch discarded remove patch
@@ -111,8 +111,8 @@  discard block
 block discarded – undo
111 111
             // AND we are going to make sure they're in that specified order
112 112
             $reordered_subsections = array();
113 113
             foreach ($options_array['include'] as $input_name) {
114
-                if (isset($this->_subsections[ $input_name ])) {
115
-                    $reordered_subsections[ $input_name ] = $this->_subsections[ $input_name ];
114
+                if (isset($this->_subsections[$input_name])) {
115
+                    $reordered_subsections[$input_name] = $this->_subsections[$input_name];
116 116
                 }
117 117
             }
118 118
             $this->_subsections = $reordered_subsections;
@@ -124,7 +124,7 @@  discard block
 block discarded – undo
124 124
         if (isset($options_array['layout_strategy'])) {
125 125
             $this->_layout_strategy = $options_array['layout_strategy'];
126 126
         }
127
-        if (! $this->_layout_strategy) {
127
+        if ( ! $this->_layout_strategy) {
128 128
             $this->_layout_strategy = is_admin() ? new EE_Admin_Two_Column_Layout() : new EE_Two_Column_Layout();
129 129
         }
130 130
         $this->_layout_strategy->_construct_finalize($this);
@@ -313,7 +313,7 @@  discard block
 block discarded – undo
313 313
         if ($validate) {
314 314
             $this->_validate();
315 315
             // if it's invalid, we're going to want to re-display so remember what they submitted
316
-            if (! $this->is_valid()) {
316
+            if ( ! $this->is_valid()) {
317 317
                 $this->store_submitted_form_data_in_session();
318 318
             }
319 319
         }
@@ -426,11 +426,11 @@  discard block
 block discarded – undo
426 426
     public function populate_defaults($default_data)
427 427
     {
428 428
         foreach ($this->subsections(false) as $subsection_name => $subsection) {
429
-            if (isset($default_data[ $subsection_name ])) {
429
+            if (isset($default_data[$subsection_name])) {
430 430
                 if ($subsection instanceof EE_Form_Input_Base) {
431
-                    $subsection->set_default($default_data[ $subsection_name ]);
431
+                    $subsection->set_default($default_data[$subsection_name]);
432 432
                 } elseif ($subsection instanceof EE_Form_Section_Proper) {
433
-                    $subsection->populate_defaults($default_data[ $subsection_name ]);
433
+                    $subsection->populate_defaults($default_data[$subsection_name]);
434 434
                 }
435 435
             }
436 436
         }
@@ -445,7 +445,7 @@  discard block
 block discarded – undo
445 445
      */
446 446
     public function subsection_exists($name)
447 447
     {
448
-        return isset($this->_subsections[ $name ]) ? true : false;
448
+        return isset($this->_subsections[$name]) ? true : false;
449 449
     }
450 450
 
451 451
 
@@ -467,7 +467,7 @@  discard block
 block discarded – undo
467 467
         if ($require_construction_to_be_finalized) {
468 468
             $this->ensure_construct_finalized_called();
469 469
         }
470
-        return $this->subsection_exists($name) ? $this->_subsections[ $name ] : null;
470
+        return $this->subsection_exists($name) ? $this->_subsections[$name] : null;
471 471
     }
472 472
 
473 473
 
@@ -482,7 +482,7 @@  discard block
 block discarded – undo
482 482
         $validatable_subsections = array();
483 483
         foreach ($this->subsections() as $name => $obj) {
484 484
             if ($obj instanceof EE_Form_Section_Validatable) {
485
-                $validatable_subsections[ $name ] = $obj;
485
+                $validatable_subsections[$name] = $obj;
486 486
             }
487 487
         }
488 488
         return $validatable_subsections;
@@ -509,7 +509,7 @@  discard block
 block discarded – undo
509 509
             $name,
510 510
             $require_construction_to_be_finalized
511 511
         );
512
-        if (! $subsection instanceof EE_Form_Input_Base) {
512
+        if ( ! $subsection instanceof EE_Form_Input_Base) {
513 513
             throw new EE_Error(
514 514
                 sprintf(
515 515
                     esc_html__(
@@ -546,7 +546,7 @@  discard block
 block discarded – undo
546 546
             $name,
547 547
             $require_construction_to_be_finalized
548 548
         );
549
-        if (! $subsection instanceof EE_Form_Section_Proper) {
549
+        if ( ! $subsection instanceof EE_Form_Section_Proper) {
550 550
             throw new EE_Error(
551 551
                 sprintf(
552 552
                     esc_html__(
@@ -586,7 +586,7 @@  discard block
 block discarded – undo
586 586
     public function is_valid()
587 587
     {
588 588
         if ($this->is_valid === null) {
589
-            if (! $this->has_received_submission()) {
589
+            if ( ! $this->has_received_submission()) {
590 590
                 throw new EE_Error(
591 591
                     sprintf(
592 592
                         esc_html__(
@@ -596,14 +596,14 @@  discard block
 block discarded – undo
596 596
                     )
597 597
                 );
598 598
             }
599
-            if (! parent::is_valid()) {
599
+            if ( ! parent::is_valid()) {
600 600
                 $this->is_valid = false;
601 601
             } else {
602 602
                 // ok so no general errors to this entire form section.
603 603
                 // so let's check the subsections, but only set errors if that hasn't been done yet
604 604
                 $this->is_valid = true;
605 605
                 foreach ($this->get_validatable_subsections() as $subsection) {
606
-                    if (! $subsection->is_valid()) {
606
+                    if ( ! $subsection->is_valid()) {
607 607
                         $this->is_valid = false;
608 608
                     }
609 609
                 }
@@ -620,7 +620,7 @@  discard block
 block discarded – undo
620 620
      */
621 621
     protected function _set_default_name_if_empty()
622 622
     {
623
-        if (! $this->_name) {
623
+        if ( ! $this->_name) {
624 624
             $classname    = get_class($this);
625 625
             $default_name = str_replace('EE_', '', $classname);
626 626
             $this->_name  = $default_name;
@@ -710,7 +710,7 @@  discard block
 block discarded – undo
710 710
     {
711 711
         wp_register_script(
712 712
             'ee_form_section_validation',
713
-            EE_GLOBAL_ASSETS_URL . 'scripts' . DS . 'form_section_validation.js',
713
+            EE_GLOBAL_ASSETS_URL.'scripts'.DS.'form_section_validation.js',
714 714
             array('jquery-validate', 'jquery-ui-datepicker', 'jquery-validate-extra-methods'),
715 715
             EVENT_ESPRESSO_VERSION,
716 716
             true
@@ -754,13 +754,13 @@  discard block
 block discarded – undo
754 754
         // we only want to localize vars ONCE for the entire form,
755 755
         // so if the form section doesn't have a parent, then it must be the top dog
756 756
         if ($return_for_subsection || ! $this->parent_section()) {
757
-            EE_Form_Section_Proper::$_js_localization['form_data'][ $this->html_id() ] = array(
757
+            EE_Form_Section_Proper::$_js_localization['form_data'][$this->html_id()] = array(
758 758
                 'form_section_id'  => $this->html_id(true),
759 759
                 'validation_rules' => $this->get_jquery_validation_rules(),
760 760
                 'other_data'       => $this->get_other_js_data(),
761 761
                 'errors'           => $this->subsection_validation_errors_by_html_name(),
762 762
             );
763
-            EE_Form_Section_Proper::$_scripts_localized                                = true;
763
+            EE_Form_Section_Proper::$_scripts_localized = true;
764 764
         }
765 765
     }
766 766
 
@@ -795,7 +795,7 @@  discard block
 block discarded – undo
795 795
         $inputs = array();
796 796
         foreach ($this->subsections() as $subsection) {
797 797
             if ($subsection instanceof EE_Form_Input_Base) {
798
-                $inputs[ $subsection->html_name() ] = $subsection;
798
+                $inputs[$subsection->html_name()] = $subsection;
799 799
             } elseif ($subsection instanceof EE_Form_Section_Proper) {
800 800
                 $inputs += $subsection->inputs_in_subsections();
801 801
             }
@@ -818,7 +818,7 @@  discard block
 block discarded – undo
818 818
         $errors = array();
819 819
         foreach ($inputs as $form_input) {
820 820
             if ($form_input instanceof EE_Form_Input_Base && $form_input->get_validation_errors()) {
821
-                $errors[ $form_input->html_name() ] = $form_input->get_validation_error_string();
821
+                $errors[$form_input->html_name()] = $form_input->get_validation_error_string();
822 822
             }
823 823
         }
824 824
         return $errors;
@@ -841,7 +841,7 @@  discard block
 block discarded – undo
841 841
         $email_validation_level = isset(EE_Registry::instance()->CFG->registration->email_validation_level)
842 842
             ? EE_Registry::instance()->CFG->registration->email_validation_level
843 843
             : 'wp_default';
844
-        EE_Form_Section_Proper::$_js_localization['email_validation_level']   = $email_validation_level;
844
+        EE_Form_Section_Proper::$_js_localization['email_validation_level'] = $email_validation_level;
845 845
         wp_enqueue_script('ee_form_section_validation');
846 846
         wp_localize_script(
847 847
             'ee_form_section_validation',
@@ -858,7 +858,7 @@  discard block
 block discarded – undo
858 858
      */
859 859
     public function ensure_scripts_localized()
860 860
     {
861
-        if (! EE_Form_Section_Proper::$_scripts_localized) {
861
+        if ( ! EE_Form_Section_Proper::$_scripts_localized) {
862 862
             $this->_enqueue_and_localize_form_js();
863 863
         }
864 864
     }
@@ -954,8 +954,8 @@  discard block
 block discarded – undo
954 954
         // reset the cache of whether this form is valid or not- we're re-validating it now
955 955
         $this->is_valid = null;
956 956
         foreach ($this->get_validatable_subsections() as $subsection_name => $subsection) {
957
-            if (method_exists($this, '_validate_' . $subsection_name)) {
958
-                call_user_func_array(array($this, '_validate_' . $subsection_name), array($subsection));
957
+            if (method_exists($this, '_validate_'.$subsection_name)) {
958
+                call_user_func_array(array($this, '_validate_'.$subsection_name), array($subsection));
959 959
             }
960 960
             $subsection->_validate();
961 961
         }
@@ -973,9 +973,9 @@  discard block
 block discarded – undo
973 973
         $inputs = array();
974 974
         foreach ($this->subsections() as $subsection_name => $subsection) {
975 975
             if ($subsection instanceof EE_Form_Section_Proper) {
976
-                $inputs[ $subsection_name ] = $subsection->valid_data();
976
+                $inputs[$subsection_name] = $subsection->valid_data();
977 977
             } elseif ($subsection instanceof EE_Form_Input_Base) {
978
-                $inputs[ $subsection_name ] = $subsection->normalized_value();
978
+                $inputs[$subsection_name] = $subsection->normalized_value();
979 979
             }
980 980
         }
981 981
         return $inputs;
@@ -993,7 +993,7 @@  discard block
 block discarded – undo
993 993
         $inputs = array();
994 994
         foreach ($this->subsections() as $subsection_name => $subsection) {
995 995
             if ($subsection instanceof EE_Form_Input_Base) {
996
-                $inputs[ $subsection_name ] = $subsection;
996
+                $inputs[$subsection_name] = $subsection;
997 997
             }
998 998
         }
999 999
         return $inputs;
@@ -1011,7 +1011,7 @@  discard block
 block discarded – undo
1011 1011
         $form_sections = array();
1012 1012
         foreach ($this->subsections() as $name => $obj) {
1013 1013
             if ($obj instanceof EE_Form_Section_Proper) {
1014
-                $form_sections[ $name ] = $obj;
1014
+                $form_sections[$name] = $obj;
1015 1015
             }
1016 1016
         }
1017 1017
         return $form_sections;
@@ -1118,7 +1118,7 @@  discard block
 block discarded – undo
1118 1118
         $input_values = array();
1119 1119
         foreach ($this->subsections() as $subsection_name => $subsection) {
1120 1120
             if ($subsection instanceof EE_Form_Input_Base) {
1121
-                $input_values[ $subsection_name ] = $pretty
1121
+                $input_values[$subsection_name] = $pretty
1122 1122
                     ? $subsection->pretty_value()
1123 1123
                     : $subsection->normalized_value();
1124 1124
             } elseif ($subsection instanceof EE_Form_Section_Proper && $include_subform_inputs) {
@@ -1130,7 +1130,7 @@  discard block
 block discarded – undo
1130 1130
                 if ($flatten) {
1131 1131
                     $input_values = array_merge($input_values, $subform_input_values);
1132 1132
                 } else {
1133
-                    $input_values[ $subsection_name ] = $subform_input_values;
1133
+                    $input_values[$subsection_name] = $subform_input_values;
1134 1134
                 }
1135 1135
             }
1136 1136
         }
@@ -1158,7 +1158,7 @@  discard block
 block discarded – undo
1158 1158
             if ($subsection instanceof EE_Form_Input_Base) {
1159 1159
                 // is this input part of an array of inputs?
1160 1160
                 if (strpos($subsection->html_name(), '[') !== false) {
1161
-                    $full_input_name  = EEH_Array::convert_array_values_to_keys(
1161
+                    $full_input_name = EEH_Array::convert_array_values_to_keys(
1162 1162
                         explode(
1163 1163
                             '[',
1164 1164
                             str_replace(']', '', $subsection->html_name())
@@ -1167,7 +1167,7 @@  discard block
 block discarded – undo
1167 1167
                     );
1168 1168
                     $submitted_values = array_replace_recursive($submitted_values, $full_input_name);
1169 1169
                 } else {
1170
-                    $submitted_values[ $subsection->html_name() ] = $subsection->raw_value();
1170
+                    $submitted_values[$subsection->html_name()] = $subsection->raw_value();
1171 1171
                 }
1172 1172
             } elseif ($subsection instanceof EE_Form_Section_Proper && $include_subforms) {
1173 1173
                 $subform_input_values = $subsection->submitted_values($include_subforms);
@@ -1202,7 +1202,7 @@  discard block
 block discarded – undo
1202 1202
     public function exclude(array $inputs_to_exclude = array())
1203 1203
     {
1204 1204
         foreach ($inputs_to_exclude as $input_to_exclude_name) {
1205
-            unset($this->_subsections[ $input_to_exclude_name ]);
1205
+            unset($this->_subsections[$input_to_exclude_name]);
1206 1206
         }
1207 1207
     }
1208 1208
 
@@ -1245,7 +1245,7 @@  discard block
 block discarded – undo
1245 1245
     public function add_subsections($new_subsections, $subsection_name_to_target = null, $add_before = true)
1246 1246
     {
1247 1247
         foreach ($new_subsections as $subsection_name => $subsection) {
1248
-            if (! $subsection instanceof EE_Form_Section_Base) {
1248
+            if ( ! $subsection instanceof EE_Form_Section_Base) {
1249 1249
                 EE_Error::add_error(
1250 1250
                     sprintf(
1251 1251
                         esc_html__(
@@ -1257,7 +1257,7 @@  discard block
 block discarded – undo
1257 1257
                         $this->name()
1258 1258
                     )
1259 1259
                 );
1260
-                unset($new_subsections[ $subsection_name ]);
1260
+                unset($new_subsections[$subsection_name]);
1261 1261
             }
1262 1262
         }
1263 1263
         $this->_subsections = EEH_Array::insert_into_array(
@@ -1372,7 +1372,7 @@  discard block
 block discarded – undo
1372 1372
     public function html_name_prefix()
1373 1373
     {
1374 1374
         if ($this->parent_section() instanceof EE_Form_Section_Proper) {
1375
-            return $this->parent_section()->html_name_prefix() . '[' . $this->name() . ']';
1375
+            return $this->parent_section()->html_name_prefix().'['.$this->name().']';
1376 1376
         }
1377 1377
         return $this->name();
1378 1378
     }
@@ -1412,7 +1412,7 @@  discard block
 block discarded – undo
1412 1412
      */
1413 1413
     public function ensure_construct_finalized_called()
1414 1414
     {
1415
-        if (! $this->_construction_finalized) {
1415
+        if ( ! $this->_construction_finalized) {
1416 1416
             $this->_construct_finalize($this->_parent_section, $this->_name);
1417 1417
         }
1418 1418
     }
Please login to merge, or discard this patch.
Indentation   +1524 added lines, -1524 removed lines patch added patch discarded remove patch
@@ -14,1528 +14,1528 @@
 block discarded – undo
14 14
 class EE_Form_Section_Proper extends EE_Form_Section_Validatable
15 15
 {
16 16
 
17
-    const SUBMITTED_FORM_DATA_SSN_KEY = 'submitted_form_data';
18
-
19
-    /**
20
-     * Subsections
21
-     *
22
-     * @var EE_Form_Section_Validatable[]
23
-     */
24
-    protected $_subsections = array();
25
-
26
-    /**
27
-     * Strategy for laying out the form
28
-     *
29
-     * @var EE_Form_Section_Layout_Base
30
-     */
31
-    protected $_layout_strategy;
32
-
33
-    /**
34
-     * Whether or not this form has received and validated a form submission yet
35
-     *
36
-     * @var boolean
37
-     */
38
-    protected $_received_submission = false;
39
-
40
-    /**
41
-     * message displayed to users upon successful form submission
42
-     *
43
-     * @var string
44
-     */
45
-    protected $_form_submission_success_message = '';
46
-
47
-    /**
48
-     * message displayed to users upon unsuccessful form submission
49
-     *
50
-     * @var string
51
-     */
52
-    protected $_form_submission_error_message = '';
53
-
54
-    /**
55
-     * @var array like $_REQUEST
56
-     */
57
-    protected $cached_request_data;
58
-
59
-    /**
60
-     * Stores whether this form (and its sub-sections) were found to be valid or not.
61
-     * Starts off as null, but once the form is validated, it set to either true or false
62
-     * @var boolean|null
63
-     */
64
-    protected $is_valid;
65
-
66
-    /**
67
-     * Stores all the data that will localized for form validation
68
-     *
69
-     * @var array
70
-     */
71
-    static protected $_js_localization = array();
72
-
73
-    /**
74
-     * whether or not the form's localized validation JS vars have been set
75
-     *
76
-     * @type boolean
77
-     */
78
-    static protected $_scripts_localized = false;
79
-
80
-
81
-    /**
82
-     * when constructing a proper form section, calls _construct_finalize on children
83
-     * so that they know who their parent is, and what name they've been given.
84
-     *
85
-     * @param array[] $options_array   {
86
-     * @type          $subsections     EE_Form_Section_Validatable[] where keys are the section's name
87
-     * @type          $include         string[] numerically-indexed where values are section names to be included,
88
-     *                                 and in that order. This is handy if you want
89
-     *                                 the subsections to be ordered differently than the default, and if you override
90
-     *                                 which fields are shown
91
-     * @type          $exclude         string[] values are subsections to be excluded. This is handy if you want
92
-     *                                 to remove certain default subsections (note: if you specify BOTH 'include' AND
93
-     *                                 'exclude', the inclusions will be applied first, and the exclusions will exclude
94
-     *                                 items from that list of inclusions)
95
-     * @type          $layout_strategy EE_Form_Section_Layout_Base strategy for laying out the form
96
-     *                                 } @see EE_Form_Section_Validatable::__construct()
97
-     * @throws EE_Error
98
-     */
99
-    public function __construct($options_array = array())
100
-    {
101
-        $options_array = (array) apply_filters(
102
-            'FHEE__EE_Form_Section_Proper___construct__options_array',
103
-            $options_array,
104
-            $this
105
-        );
106
-        // call parent first, as it may be setting the name
107
-        parent::__construct($options_array);
108
-        // if they've included subsections in the constructor, add them now
109
-        if (isset($options_array['include'])) {
110
-            // we are going to make sure we ONLY have those subsections to include
111
-            // AND we are going to make sure they're in that specified order
112
-            $reordered_subsections = array();
113
-            foreach ($options_array['include'] as $input_name) {
114
-                if (isset($this->_subsections[ $input_name ])) {
115
-                    $reordered_subsections[ $input_name ] = $this->_subsections[ $input_name ];
116
-                }
117
-            }
118
-            $this->_subsections = $reordered_subsections;
119
-        }
120
-        if (isset($options_array['exclude'])) {
121
-            $exclude            = $options_array['exclude'];
122
-            $this->_subsections = array_diff_key($this->_subsections, array_flip($exclude));
123
-        }
124
-        if (isset($options_array['layout_strategy'])) {
125
-            $this->_layout_strategy = $options_array['layout_strategy'];
126
-        }
127
-        if (! $this->_layout_strategy) {
128
-            $this->_layout_strategy = is_admin() ? new EE_Admin_Two_Column_Layout() : new EE_Two_Column_Layout();
129
-        }
130
-        $this->_layout_strategy->_construct_finalize($this);
131
-        // ok so we are definitely going to want the forms JS,
132
-        // so enqueue it or remember to enqueue it during wp_enqueue_scripts
133
-        if (did_action('wp_enqueue_scripts') || did_action('admin_enqueue_scripts')) {
134
-            // ok so they've constructed this object after when they should have.
135
-            // just enqueue the generic form scripts and initialize the form immediately in the JS
136
-            EE_Form_Section_Proper::wp_enqueue_scripts(true);
137
-        } else {
138
-            add_action('wp_enqueue_scripts', array('EE_Form_Section_Proper', 'wp_enqueue_scripts'));
139
-            add_action('admin_enqueue_scripts', array('EE_Form_Section_Proper', 'wp_enqueue_scripts'));
140
-        }
141
-        add_action('wp_footer', array($this, 'ensure_scripts_localized'), 1);
142
-        /**
143
-         * Gives other plugins a chance to hook in before construct finalize is called.
144
-         * The form probably doesn't yet have a parent form section.
145
-         * Since 4.9.32, when this action was introduced, this is the best place to add a subsection onto a form,
146
-         * assuming you don't care what the form section's name, HTML ID, or HTML name etc are.
147
-         * Also see AHEE__EE_Form_Section_Proper___construct_finalize__end
148
-         *
149
-         * @since 4.9.32
150
-         * @param EE_Form_Section_Proper $this          before __construct is done, but all of its logic,
151
-         *                                              except maybe calling _construct_finalize has been done
152
-         * @param array                  $options_array options passed into the constructor
153
-         */
154
-        do_action(
155
-            'AHEE__EE_Form_Input_Base___construct__before_construct_finalize_called',
156
-            $this,
157
-            $options_array
158
-        );
159
-        if (isset($options_array['name'])) {
160
-            $this->_construct_finalize(null, $options_array['name']);
161
-        }
162
-    }
163
-
164
-
165
-    /**
166
-     * Finishes construction given the parent form section and this form section's name
167
-     *
168
-     * @param EE_Form_Section_Proper $parent_form_section
169
-     * @param string                 $name
170
-     * @throws EE_Error
171
-     */
172
-    public function _construct_finalize($parent_form_section, $name)
173
-    {
174
-        parent::_construct_finalize($parent_form_section, $name);
175
-        $this->_set_default_name_if_empty();
176
-        $this->_set_default_html_id_if_empty();
177
-        foreach ($this->_subsections as $subsection_name => $subsection) {
178
-            if ($subsection instanceof EE_Form_Section_Base) {
179
-                $subsection->_construct_finalize($this, $subsection_name);
180
-            } else {
181
-                throw new EE_Error(
182
-                    sprintf(
183
-                        esc_html__(
184
-                            'Subsection "%s" is not an instanceof EE_Form_Section_Base on form "%s". It is a "%s"',
185
-                            'event_espresso'
186
-                        ),
187
-                        $subsection_name,
188
-                        get_class($this),
189
-                        $subsection ? get_class($subsection) : esc_html__('NULL', 'event_espresso')
190
-                    )
191
-                );
192
-            }
193
-        }
194
-        /**
195
-         * Action performed just after form has been given a name (and HTML ID etc) and is fully constructed.
196
-         * If you have code that should modify the form and needs it and its subsections to have a name, HTML ID
197
-         * (or other attributes derived from the name like the HTML label id, etc), this is where it should be done.
198
-         * This might only happen just before displaying the form, or just before it receives form submission data.
199
-         * If you need to modify the form or its subsections before _construct_finalize is called on it (and we've
200
-         * ensured it has a name, HTML IDs, etc
201
-         *
202
-         * @param EE_Form_Section_Proper      $this
203
-         * @param EE_Form_Section_Proper|null $parent_form_section
204
-         * @param string                      $name
205
-         */
206
-        do_action(
207
-            'AHEE__EE_Form_Section_Proper___construct_finalize__end',
208
-            $this,
209
-            $parent_form_section,
210
-            $name
211
-        );
212
-    }
213
-
214
-
215
-    /**
216
-     * Gets the layout strategy for this form section
217
-     *
218
-     * @return EE_Form_Section_Layout_Base
219
-     */
220
-    public function get_layout_strategy()
221
-    {
222
-        return $this->_layout_strategy;
223
-    }
224
-
225
-
226
-    /**
227
-     * Gets the HTML for a single input for this form section according
228
-     * to the layout strategy
229
-     *
230
-     * @param EE_Form_Input_Base $input
231
-     * @return string
232
-     */
233
-    public function get_html_for_input($input)
234
-    {
235
-        return $this->_layout_strategy->layout_input($input);
236
-    }
237
-
238
-
239
-    /**
240
-     * was_submitted - checks if form inputs are present in request data
241
-     * Basically an alias for form_data_present_in() (which is used by both
242
-     * proper form sections and form inputs)
243
-     *
244
-     * @param null $form_data
245
-     * @return boolean
246
-     * @throws EE_Error
247
-     */
248
-    public function was_submitted($form_data = null)
249
-    {
250
-        return $this->form_data_present_in($form_data);
251
-    }
252
-
253
-    /**
254
-     * Gets the cached request data; but if there is none, or $req_data was set with
255
-     * something different, refresh the cache, and then return it
256
-     * @param null $req_data
257
-     * @return array
258
-     */
259
-    protected function getCachedRequest($req_data = null)
260
-    {
261
-        if ($this->cached_request_data === null
262
-            || (
263
-                $req_data !== null &&
264
-                $req_data !== $this->cached_request_data
265
-            )
266
-        ) {
267
-            $req_data = apply_filters(
268
-                'FHEE__EE_Form_Section_Proper__receive_form_submission__req_data',
269
-                $req_data,
270
-                $this
271
-            );
272
-            if ($req_data === null) {
273
-                $req_data = array_merge($_GET, $_POST);
274
-            }
275
-            $req_data = apply_filters(
276
-                'FHEE__EE_Form_Section_Proper__receive_form_submission__request_data',
277
-                $req_data,
278
-                $this
279
-            );
280
-            $this->cached_request_data = (array) $req_data;
281
-        }
282
-        return $this->cached_request_data;
283
-    }
284
-
285
-
286
-    /**
287
-     * After the form section is initially created, call this to sanitize the data in the submission
288
-     * which relates to this form section, validate it, and set it as properties on the form.
289
-     *
290
-     * @param array|null $req_data should usually be $_POST (the default).
291
-     *                             However, you CAN supply a different array.
292
-     *                             Consider using set_defaults() instead however.
293
-     *                             (If you rendered the form in the page using echo $form_x->get_html()
294
-     *                             the inputs will have the correct name in the request data for this function
295
-     *                             to find them and populate the form with them.
296
-     *                             If you have a flat form (with only input subsections),
297
-     *                             you can supply a flat array where keys
298
-     *                             are the form input names and values are their values)
299
-     * @param boolean    $validate whether or not to perform validation on this data. Default is,
300
-     *                             of course, to validate that data, and set errors on the invalid values.
301
-     *                             But if the data has already been validated
302
-     *                             (eg you validated the data then stored it in the DB)
303
-     *                             you may want to skip this step.
304
-     * @throws InvalidArgumentException
305
-     * @throws InvalidInterfaceException
306
-     * @throws InvalidDataTypeException
307
-     * @throws EE_Error
308
-     */
309
-    public function receive_form_submission($req_data = null, $validate = true)
310
-    {
311
-        $req_data = $this->getCachedRequest($req_data);
312
-        $this->_normalize($req_data);
313
-        if ($validate) {
314
-            $this->_validate();
315
-            // if it's invalid, we're going to want to re-display so remember what they submitted
316
-            if (! $this->is_valid()) {
317
-                $this->store_submitted_form_data_in_session();
318
-            }
319
-        }
320
-        if ($this->submission_error_message() === '' && ! $this->is_valid()) {
321
-            $this->set_submission_error_message();
322
-        }
323
-        do_action(
324
-            'AHEE__EE_Form_Section_Proper__receive_form_submission__end',
325
-            $req_data,
326
-            $this,
327
-            $validate
328
-        );
329
-    }
330
-
331
-
332
-    /**
333
-     * caches the originally submitted input values in the session
334
-     * so that they can be used to repopulate the form if it failed validation
335
-     *
336
-     * @return boolean whether or not the data was successfully stored in the session
337
-     * @throws InvalidArgumentException
338
-     * @throws InvalidInterfaceException
339
-     * @throws InvalidDataTypeException
340
-     * @throws EE_Error
341
-     */
342
-    protected function store_submitted_form_data_in_session()
343
-    {
344
-        return EE_Registry::instance()->SSN->set_session_data(
345
-            array(
346
-                EE_Form_Section_Proper::SUBMITTED_FORM_DATA_SSN_KEY => $this->submitted_values(true),
347
-            )
348
-        );
349
-    }
350
-
351
-
352
-    /**
353
-     * retrieves the originally submitted input values in the session
354
-     * so that they can be used to repopulate the form if it failed validation
355
-     *
356
-     * @return array
357
-     * @throws InvalidArgumentException
358
-     * @throws InvalidInterfaceException
359
-     * @throws InvalidDataTypeException
360
-     */
361
-    protected function get_submitted_form_data_from_session()
362
-    {
363
-        $session = EE_Registry::instance()->SSN;
364
-        if ($session instanceof EE_Session) {
365
-            return $session->get_session_data(
366
-                EE_Form_Section_Proper::SUBMITTED_FORM_DATA_SSN_KEY
367
-            );
368
-        }
369
-        return array();
370
-    }
371
-
372
-
373
-    /**
374
-     * flushed the originally submitted input values from the session
375
-     *
376
-     * @return boolean whether or not the data was successfully removed from the session
377
-     * @throws InvalidArgumentException
378
-     * @throws InvalidInterfaceException
379
-     * @throws InvalidDataTypeException
380
-     */
381
-    public static function flush_submitted_form_data_from_session()
382
-    {
383
-        return EE_Registry::instance()->SSN->reset_data(
384
-            array(EE_Form_Section_Proper::SUBMITTED_FORM_DATA_SSN_KEY)
385
-        );
386
-    }
387
-
388
-
389
-    /**
390
-     * Populates this form and its subsections with data from the session.
391
-     * (Wrapper for EE_Form_Section_Proper::receive_form_submission, so it shows
392
-     * validation errors when displaying too)
393
-     * Returns true if the form was populated from the session, false otherwise
394
-     *
395
-     * @return boolean
396
-     * @throws InvalidArgumentException
397
-     * @throws InvalidInterfaceException
398
-     * @throws InvalidDataTypeException
399
-     * @throws EE_Error
400
-     */
401
-    public function populate_from_session()
402
-    {
403
-        $form_data_in_session = $this->get_submitted_form_data_from_session();
404
-        if (empty($form_data_in_session)) {
405
-            return false;
406
-        }
407
-        $this->receive_form_submission($form_data_in_session);
408
-        add_action('shutdown', array('EE_Form_Section_Proper', 'flush_submitted_form_data_from_session'));
409
-        if ($this->form_data_present_in($form_data_in_session)) {
410
-            return true;
411
-        }
412
-        return false;
413
-    }
414
-
415
-
416
-    /**
417
-     * Populates the default data for the form, given an array where keys are
418
-     * the input names, and values are their values (preferably normalized to be their
419
-     * proper PHP types, not all strings... although that should be ok too).
420
-     * Proper subsections are sub-arrays, the key being the subsection's name, and
421
-     * the value being an array formatted in teh same way
422
-     *
423
-     * @param array $default_data
424
-     * @throws EE_Error
425
-     */
426
-    public function populate_defaults($default_data)
427
-    {
428
-        foreach ($this->subsections(false) as $subsection_name => $subsection) {
429
-            if (isset($default_data[ $subsection_name ])) {
430
-                if ($subsection instanceof EE_Form_Input_Base) {
431
-                    $subsection->set_default($default_data[ $subsection_name ]);
432
-                } elseif ($subsection instanceof EE_Form_Section_Proper) {
433
-                    $subsection->populate_defaults($default_data[ $subsection_name ]);
434
-                }
435
-            }
436
-        }
437
-    }
438
-
439
-
440
-    /**
441
-     * returns true if subsection exists
442
-     *
443
-     * @param string $name
444
-     * @return boolean
445
-     */
446
-    public function subsection_exists($name)
447
-    {
448
-        return isset($this->_subsections[ $name ]) ? true : false;
449
-    }
450
-
451
-
452
-    /**
453
-     * Gets the subsection specified by its name
454
-     *
455
-     * @param string  $name
456
-     * @param boolean $require_construction_to_be_finalized most client code should leave this as TRUE
457
-     *                                                      so that the inputs will be properly configured.
458
-     *                                                      However, some client code may be ok
459
-     *                                                      with construction finalize being called later
460
-     *                                                      (realizing that the subsections' html names
461
-     *                                                      might not be set yet, etc.)
462
-     * @return EE_Form_Section_Base
463
-     * @throws EE_Error
464
-     */
465
-    public function get_subsection($name, $require_construction_to_be_finalized = true)
466
-    {
467
-        if ($require_construction_to_be_finalized) {
468
-            $this->ensure_construct_finalized_called();
469
-        }
470
-        return $this->subsection_exists($name) ? $this->_subsections[ $name ] : null;
471
-    }
472
-
473
-
474
-    /**
475
-     * Gets all the validatable subsections of this form section
476
-     *
477
-     * @return EE_Form_Section_Validatable[]
478
-     * @throws EE_Error
479
-     */
480
-    public function get_validatable_subsections()
481
-    {
482
-        $validatable_subsections = array();
483
-        foreach ($this->subsections() as $name => $obj) {
484
-            if ($obj instanceof EE_Form_Section_Validatable) {
485
-                $validatable_subsections[ $name ] = $obj;
486
-            }
487
-        }
488
-        return $validatable_subsections;
489
-    }
490
-
491
-
492
-    /**
493
-     * Gets an input by the given name. If not found, or if its not an EE_FOrm_Input_Base child,
494
-     * throw an EE_Error.
495
-     *
496
-     * @param string  $name
497
-     * @param boolean $require_construction_to_be_finalized most client code should
498
-     *                                                      leave this as TRUE so that the inputs will be properly
499
-     *                                                      configured. However, some client code may be ok with
500
-     *                                                      construction finalize being called later
501
-     *                                                      (realizing that the subsections' html names might not be
502
-     *                                                      set yet, etc.)
503
-     * @return EE_Form_Input_Base
504
-     * @throws EE_Error
505
-     */
506
-    public function get_input($name, $require_construction_to_be_finalized = true)
507
-    {
508
-        $subsection = $this->get_subsection(
509
-            $name,
510
-            $require_construction_to_be_finalized
511
-        );
512
-        if (! $subsection instanceof EE_Form_Input_Base) {
513
-            throw new EE_Error(
514
-                sprintf(
515
-                    esc_html__(
516
-                        "Subsection '%s' is not an instanceof EE_Form_Input_Base on form '%s'. It is a '%s'",
517
-                        'event_espresso'
518
-                    ),
519
-                    $name,
520
-                    get_class($this),
521
-                    $subsection ? get_class($subsection) : esc_html__('NULL', 'event_espresso')
522
-                )
523
-            );
524
-        }
525
-        return $subsection;
526
-    }
527
-
528
-
529
-    /**
530
-     * Like get_input(), gets the proper subsection of the form given the name,
531
-     * otherwise throws an EE_Error
532
-     *
533
-     * @param string  $name
534
-     * @param boolean $require_construction_to_be_finalized most client code should
535
-     *                                                      leave this as TRUE so that the inputs will be properly
536
-     *                                                      configured. However, some client code may be ok with
537
-     *                                                      construction finalize being called later
538
-     *                                                      (realizing that the subsections' html names might not be
539
-     *                                                      set yet, etc.)
540
-     * @return EE_Form_Section_Proper
541
-     * @throws EE_Error
542
-     */
543
-    public function get_proper_subsection($name, $require_construction_to_be_finalized = true)
544
-    {
545
-        $subsection = $this->get_subsection(
546
-            $name,
547
-            $require_construction_to_be_finalized
548
-        );
549
-        if (! $subsection instanceof EE_Form_Section_Proper) {
550
-            throw new EE_Error(
551
-                sprintf(
552
-                    esc_html__(
553
-                        "Subsection '%'s is not an instanceof EE_Form_Section_Proper on form '%s'",
554
-                        'event_espresso'
555
-                    ),
556
-                    $name,
557
-                    get_class($this)
558
-                )
559
-            );
560
-        }
561
-        return $subsection;
562
-    }
563
-
564
-
565
-    /**
566
-     * Gets the value of the specified input. Should be called after receive_form_submission()
567
-     * or populate_defaults() on the form, where the normalized value on the input is set.
568
-     *
569
-     * @param string $name
570
-     * @return mixed depending on the input's type and its normalization strategy
571
-     * @throws EE_Error
572
-     */
573
-    public function get_input_value($name)
574
-    {
575
-        $input = $this->get_input($name);
576
-        return $input->normalized_value();
577
-    }
578
-
579
-
580
-    /**
581
-     * Checks if this form section itself is valid, and then checks its subsections
582
-     *
583
-     * @throws EE_Error
584
-     * @return boolean
585
-     */
586
-    public function is_valid()
587
-    {
588
-        if ($this->is_valid === null) {
589
-            if (! $this->has_received_submission()) {
590
-                throw new EE_Error(
591
-                    sprintf(
592
-                        esc_html__(
593
-                            'You cannot check if a form is valid before receiving the form submission using receive_form_submission',
594
-                            'event_espresso'
595
-                        )
596
-                    )
597
-                );
598
-            }
599
-            if (! parent::is_valid()) {
600
-                $this->is_valid = false;
601
-            } else {
602
-                // ok so no general errors to this entire form section.
603
-                // so let's check the subsections, but only set errors if that hasn't been done yet
604
-                $this->is_valid = true;
605
-                foreach ($this->get_validatable_subsections() as $subsection) {
606
-                    if (! $subsection->is_valid()) {
607
-                        $this->is_valid = false;
608
-                    }
609
-                }
610
-            }
611
-        }
612
-        return $this->is_valid;
613
-    }
614
-
615
-
616
-    /**
617
-     * gets the default name of this form section if none is specified
618
-     *
619
-     * @return void
620
-     */
621
-    protected function _set_default_name_if_empty()
622
-    {
623
-        if (! $this->_name) {
624
-            $classname    = get_class($this);
625
-            $default_name = str_replace('EE_', '', $classname);
626
-            $this->_name  = $default_name;
627
-        }
628
-    }
629
-
630
-
631
-    /**
632
-     * Returns the HTML for the form, except for the form opening and closing tags
633
-     * (as the form section doesn't know where you necessarily want to send the information to),
634
-     * and except for a submit button. Enqueues JS and CSS; if called early enough we will
635
-     * try to enqueue them in the header, otherwise they'll be enqueued in the footer.
636
-     * Not doing_it_wrong because theoretically this CAN be used properly,
637
-     * provided its used during "wp_enqueue_scripts", or it doesn't need to enqueue
638
-     * any CSS.
639
-     *
640
-     * @throws InvalidArgumentException
641
-     * @throws InvalidInterfaceException
642
-     * @throws InvalidDataTypeException
643
-     * @throws EE_Error
644
-     */
645
-    public function get_html_and_js()
646
-    {
647
-        $this->enqueue_js();
648
-        return $this->get_html();
649
-    }
650
-
651
-
652
-    /**
653
-     * returns HTML for displaying this form section. recursively calls display_section() on all subsections
654
-     *
655
-     * @param bool $display_previously_submitted_data
656
-     * @return string
657
-     * @throws InvalidArgumentException
658
-     * @throws InvalidInterfaceException
659
-     * @throws InvalidDataTypeException
660
-     * @throws EE_Error
661
-     * @throws EE_Error
662
-     * @throws EE_Error
663
-     */
664
-    public function get_html($display_previously_submitted_data = true)
665
-    {
666
-        $this->ensure_construct_finalized_called();
667
-        if ($display_previously_submitted_data) {
668
-            $this->populate_from_session();
669
-        }
670
-        return $this->_form_html_filter
671
-            ? $this->_form_html_filter->filterHtml($this->_layout_strategy->layout_form(), $this)
672
-            : $this->_layout_strategy->layout_form();
673
-    }
674
-
675
-
676
-    /**
677
-     * enqueues JS and CSS for the form.
678
-     * It is preferred to call this before wp_enqueue_scripts so the
679
-     * scripts and styles can be put in the header, but if called later
680
-     * they will be put in the footer (which is OK for JS, but in HTML4 CSS should
681
-     * only be in the header; but in HTML5 its ok in the body.
682
-     * See http://stackoverflow.com/questions/4957446/load-external-css-file-in-body-tag.
683
-     * So if your form enqueues CSS, it's preferred to call this before wp_enqueue_scripts.)
684
-     *
685
-     * @return void
686
-     * @throws EE_Error
687
-     */
688
-    public function enqueue_js()
689
-    {
690
-        $this->_enqueue_and_localize_form_js();
691
-        foreach ($this->subsections() as $subsection) {
692
-            $subsection->enqueue_js();
693
-        }
694
-    }
695
-
696
-
697
-    /**
698
-     * adds a filter so that jquery validate gets enqueued in EE_System::wp_enqueue_scripts().
699
-     * This must be done BEFORE wp_enqueue_scripts() gets called, which is on
700
-     * the wp_enqueue_scripts hook.
701
-     * However, registering the form js and localizing it can happen when we
702
-     * actually output the form (which is preferred, seeing how teh form's fields
703
-     * could change until it's actually outputted)
704
-     *
705
-     * @param boolean $init_form_validation_automatically whether or not we want the form validation
706
-     *                                                    to be triggered automatically or not
707
-     * @return void
708
-     */
709
-    public static function wp_enqueue_scripts($init_form_validation_automatically = true)
710
-    {
711
-        wp_register_script(
712
-            'ee_form_section_validation',
713
-            EE_GLOBAL_ASSETS_URL . 'scripts' . DS . 'form_section_validation.js',
714
-            array('jquery-validate', 'jquery-ui-datepicker', 'jquery-validate-extra-methods'),
715
-            EVENT_ESPRESSO_VERSION,
716
-            true
717
-        );
718
-        wp_localize_script(
719
-            'ee_form_section_validation',
720
-            'ee_form_section_validation_init',
721
-            array('init' => $init_form_validation_automatically ? '1' : '0')
722
-        );
723
-    }
724
-
725
-
726
-    /**
727
-     * gets the variables used by form_section_validation.js.
728
-     * This needs to be called AFTER we've called $this->_enqueue_jquery_validate_script,
729
-     * but before the wordpress hook wp_loaded
730
-     *
731
-     * @throws EE_Error
732
-     */
733
-    public function _enqueue_and_localize_form_js()
734
-    {
735
-        $this->ensure_construct_finalized_called();
736
-        // actually, we don't want to localize just yet. There may be other forms on the page.
737
-        // so we need to add our form section data to a static variable accessible by all form sections
738
-        // and localize it just before the footer
739
-        $this->localize_validation_rules();
740
-        add_action('wp_footer', array('EE_Form_Section_Proper', 'localize_script_for_all_forms'), 2);
741
-        add_action('admin_footer', array('EE_Form_Section_Proper', 'localize_script_for_all_forms'));
742
-    }
743
-
744
-
745
-    /**
746
-     * add our form section data to a static variable accessible by all form sections
747
-     *
748
-     * @param bool $return_for_subsection
749
-     * @return void
750
-     * @throws EE_Error
751
-     */
752
-    public function localize_validation_rules($return_for_subsection = false)
753
-    {
754
-        // we only want to localize vars ONCE for the entire form,
755
-        // so if the form section doesn't have a parent, then it must be the top dog
756
-        if ($return_for_subsection || ! $this->parent_section()) {
757
-            EE_Form_Section_Proper::$_js_localization['form_data'][ $this->html_id() ] = array(
758
-                'form_section_id'  => $this->html_id(true),
759
-                'validation_rules' => $this->get_jquery_validation_rules(),
760
-                'other_data'       => $this->get_other_js_data(),
761
-                'errors'           => $this->subsection_validation_errors_by_html_name(),
762
-            );
763
-            EE_Form_Section_Proper::$_scripts_localized                                = true;
764
-        }
765
-    }
766
-
767
-
768
-    /**
769
-     * Gets an array of extra data that will be useful for client-side javascript.
770
-     * This is primarily data added by inputs and forms in addition to any
771
-     * scripts they might enqueue
772
-     *
773
-     * @param array $form_other_js_data
774
-     * @return array
775
-     * @throws EE_Error
776
-     */
777
-    public function get_other_js_data($form_other_js_data = array())
778
-    {
779
-        foreach ($this->subsections() as $subsection) {
780
-            $form_other_js_data = $subsection->get_other_js_data($form_other_js_data);
781
-        }
782
-        return $form_other_js_data;
783
-    }
784
-
785
-
786
-    /**
787
-     * Gets a flat array of inputs for this form section and its subsections.
788
-     * Keys are their form names, and values are the inputs themselves
789
-     *
790
-     * @return EE_Form_Input_Base
791
-     * @throws EE_Error
792
-     */
793
-    public function inputs_in_subsections()
794
-    {
795
-        $inputs = array();
796
-        foreach ($this->subsections() as $subsection) {
797
-            if ($subsection instanceof EE_Form_Input_Base) {
798
-                $inputs[ $subsection->html_name() ] = $subsection;
799
-            } elseif ($subsection instanceof EE_Form_Section_Proper) {
800
-                $inputs += $subsection->inputs_in_subsections();
801
-            }
802
-        }
803
-        return $inputs;
804
-    }
805
-
806
-
807
-    /**
808
-     * Gets a flat array of all the validation errors.
809
-     * Keys are html names (because those should be unique)
810
-     * and values are a string of all their validation errors
811
-     *
812
-     * @return string[]
813
-     * @throws EE_Error
814
-     */
815
-    public function subsection_validation_errors_by_html_name()
816
-    {
817
-        $inputs = $this->inputs();
818
-        $errors = array();
819
-        foreach ($inputs as $form_input) {
820
-            if ($form_input instanceof EE_Form_Input_Base && $form_input->get_validation_errors()) {
821
-                $errors[ $form_input->html_name() ] = $form_input->get_validation_error_string();
822
-            }
823
-        }
824
-        return $errors;
825
-    }
826
-
827
-
828
-    /**
829
-     * passes all the form data required by the JS to the JS, and enqueues the few required JS files.
830
-     * Should be setup by each form during the _enqueues_and_localize_form_js
831
-     *
832
-     * @throws InvalidArgumentException
833
-     * @throws InvalidInterfaceException
834
-     * @throws InvalidDataTypeException
835
-     */
836
-    public static function localize_script_for_all_forms()
837
-    {
838
-        // allow inputs and stuff to hook in their JS and stuff here
839
-        do_action('AHEE__EE_Form_Section_Proper__localize_script_for_all_forms__begin');
840
-        EE_Form_Section_Proper::$_js_localization['localized_error_messages'] = EE_Form_Section_Proper::_get_localized_error_messages();
841
-        $email_validation_level = isset(EE_Registry::instance()->CFG->registration->email_validation_level)
842
-            ? EE_Registry::instance()->CFG->registration->email_validation_level
843
-            : 'wp_default';
844
-        EE_Form_Section_Proper::$_js_localization['email_validation_level']   = $email_validation_level;
845
-        wp_enqueue_script('ee_form_section_validation');
846
-        wp_localize_script(
847
-            'ee_form_section_validation',
848
-            'ee_form_section_vars',
849
-            EE_Form_Section_Proper::$_js_localization
850
-        );
851
-    }
852
-
853
-
854
-    /**
855
-     * ensure_scripts_localized
856
-     *
857
-     * @throws EE_Error
858
-     */
859
-    public function ensure_scripts_localized()
860
-    {
861
-        if (! EE_Form_Section_Proper::$_scripts_localized) {
862
-            $this->_enqueue_and_localize_form_js();
863
-        }
864
-    }
865
-
866
-
867
-    /**
868
-     * Gets the hard-coded validation error messages to be used in the JS. The convention
869
-     * is that the key here should be the same as the custom validation rule put in the JS file
870
-     *
871
-     * @return array keys are custom validation rules, and values are internationalized strings
872
-     */
873
-    private static function _get_localized_error_messages()
874
-    {
875
-        return array(
876
-            'validUrl' => esc_html__('This is not a valid absolute URL. Eg, http://domain.com/monkey.jpg', 'event_espresso'),
877
-            'regex'    => esc_html__('Please check your input', 'event_espresso'),
878
-        );
879
-    }
880
-
881
-
882
-    /**
883
-     * @return array
884
-     */
885
-    public static function js_localization()
886
-    {
887
-        return self::$_js_localization;
888
-    }
889
-
890
-
891
-    /**
892
-     * @return void
893
-     */
894
-    public static function reset_js_localization()
895
-    {
896
-        self::$_js_localization = array();
897
-    }
898
-
899
-
900
-    /**
901
-     * Gets the JS to put inside the jquery validation rules for subsection of this form section.
902
-     * See parent function for more...
903
-     *
904
-     * @return array
905
-     * @throws EE_Error
906
-     */
907
-    public function get_jquery_validation_rules()
908
-    {
909
-        $jquery_validation_rules = array();
910
-        foreach ($this->get_validatable_subsections() as $subsection) {
911
-            $jquery_validation_rules = array_merge(
912
-                $jquery_validation_rules,
913
-                $subsection->get_jquery_validation_rules()
914
-            );
915
-        }
916
-        return $jquery_validation_rules;
917
-    }
918
-
919
-
920
-    /**
921
-     * Sanitizes all the data and sets the sanitized value of each field
922
-     *
923
-     * @param array $req_data like $_POST
924
-     * @return void
925
-     * @throws EE_Error
926
-     */
927
-    protected function _normalize($req_data)
928
-    {
929
-        $this->_received_submission = true;
930
-        $this->_validation_errors   = array();
931
-        foreach ($this->get_validatable_subsections() as $subsection) {
932
-            try {
933
-                $subsection->_normalize($req_data);
934
-            } catch (EE_Validation_Error $e) {
935
-                $subsection->add_validation_error($e);
936
-            }
937
-        }
938
-    }
939
-
940
-
941
-    /**
942
-     * Performs validation on this form section and its subsections.
943
-     * For each subsection,
944
-     * calls _validate_{subsection_name} on THIS form (if the function exists)
945
-     * and passes it the subsection, then calls _validate on that subsection.
946
-     * If you need to perform validation on the form as a whole (considering multiple)
947
-     * you would be best to override this _validate method,
948
-     * calling parent::_validate() first.
949
-     *
950
-     * @throws EE_Error
951
-     */
952
-    protected function _validate()
953
-    {
954
-        // reset the cache of whether this form is valid or not- we're re-validating it now
955
-        $this->is_valid = null;
956
-        foreach ($this->get_validatable_subsections() as $subsection_name => $subsection) {
957
-            if (method_exists($this, '_validate_' . $subsection_name)) {
958
-                call_user_func_array(array($this, '_validate_' . $subsection_name), array($subsection));
959
-            }
960
-            $subsection->_validate();
961
-        }
962
-    }
963
-
964
-
965
-    /**
966
-     * Gets all the validated inputs for the form section
967
-     *
968
-     * @return array
969
-     * @throws EE_Error
970
-     */
971
-    public function valid_data()
972
-    {
973
-        $inputs = array();
974
-        foreach ($this->subsections() as $subsection_name => $subsection) {
975
-            if ($subsection instanceof EE_Form_Section_Proper) {
976
-                $inputs[ $subsection_name ] = $subsection->valid_data();
977
-            } elseif ($subsection instanceof EE_Form_Input_Base) {
978
-                $inputs[ $subsection_name ] = $subsection->normalized_value();
979
-            }
980
-        }
981
-        return $inputs;
982
-    }
983
-
984
-
985
-    /**
986
-     * Gets all the inputs on this form section
987
-     *
988
-     * @return EE_Form_Input_Base[]
989
-     * @throws EE_Error
990
-     */
991
-    public function inputs()
992
-    {
993
-        $inputs = array();
994
-        foreach ($this->subsections() as $subsection_name => $subsection) {
995
-            if ($subsection instanceof EE_Form_Input_Base) {
996
-                $inputs[ $subsection_name ] = $subsection;
997
-            }
998
-        }
999
-        return $inputs;
1000
-    }
1001
-
1002
-
1003
-    /**
1004
-     * Gets all the subsections which are a proper form
1005
-     *
1006
-     * @return EE_Form_Section_Proper[]
1007
-     * @throws EE_Error
1008
-     */
1009
-    public function subforms()
1010
-    {
1011
-        $form_sections = array();
1012
-        foreach ($this->subsections() as $name => $obj) {
1013
-            if ($obj instanceof EE_Form_Section_Proper) {
1014
-                $form_sections[ $name ] = $obj;
1015
-            }
1016
-        }
1017
-        return $form_sections;
1018
-    }
1019
-
1020
-
1021
-    /**
1022
-     * Gets all the subsections (inputs, proper subsections, or html-only sections).
1023
-     * Consider using inputs() or subforms()
1024
-     * if you only want form inputs or proper form sections.
1025
-     *
1026
-     * @param boolean $require_construction_to_be_finalized most client code should
1027
-     *                                                      leave this as TRUE so that the inputs will be properly
1028
-     *                                                      configured. However, some client code may be ok with
1029
-     *                                                      construction finalize being called later
1030
-     *                                                      (realizing that the subsections' html names might not be
1031
-     *                                                      set yet, etc.)
1032
-     * @return EE_Form_Section_Proper[]
1033
-     * @throws EE_Error
1034
-     */
1035
-    public function subsections($require_construction_to_be_finalized = true)
1036
-    {
1037
-        if ($require_construction_to_be_finalized) {
1038
-            $this->ensure_construct_finalized_called();
1039
-        }
1040
-        return $this->_subsections;
1041
-    }
1042
-
1043
-
1044
-    /**
1045
-     * Returns whether this form has any subforms or inputs
1046
-     * @return bool
1047
-     */
1048
-    public function hasSubsections()
1049
-    {
1050
-        return ! empty($this->_subsections);
1051
-    }
1052
-
1053
-
1054
-    /**
1055
-     * Returns a simple array where keys are input names, and values are their normalized
1056
-     * values. (Similar to calling get_input_value on inputs)
1057
-     *
1058
-     * @param boolean $include_subform_inputs Whether to include inputs from subforms,
1059
-     *                                        or just this forms' direct children inputs
1060
-     * @param boolean $flatten                Whether to force the results into 1-dimensional array,
1061
-     *                                        or allow multidimensional array
1062
-     * @return array if $flatten is TRUE it will always be a 1-dimensional array
1063
-     *                                        with array keys being input names
1064
-     *                                        (regardless of whether they are from a subsection or not),
1065
-     *                                        and if $flatten is FALSE it can be a multidimensional array
1066
-     *                                        where keys are always subsection names and values are either
1067
-     *                                        the input's normalized value, or an array like the top-level array
1068
-     * @throws EE_Error
1069
-     */
1070
-    public function input_values($include_subform_inputs = false, $flatten = false)
1071
-    {
1072
-        return $this->_input_values(false, $include_subform_inputs, $flatten);
1073
-    }
1074
-
1075
-
1076
-    /**
1077
-     * Similar to EE_Form_Section_Proper::input_values(), except this returns the 'display_value'
1078
-     * of each input. On some inputs (especially radio boxes or checkboxes), the value stored
1079
-     * is not necessarily the value we want to display to users. This creates an array
1080
-     * where keys are the input names, and values are their display values
1081
-     *
1082
-     * @param boolean $include_subform_inputs Whether to include inputs from subforms,
1083
-     *                                        or just this forms' direct children inputs
1084
-     * @param boolean $flatten                Whether to force the results into 1-dimensional array,
1085
-     *                                        or allow multidimensional array
1086
-     * @return array if $flatten is TRUE it will always be a 1-dimensional array
1087
-     *                                        with array keys being input names
1088
-     *                                        (regardless of whether they are from a subsection or not),
1089
-     *                                        and if $flatten is FALSE it can be a multidimensional array
1090
-     *                                        where keys are always subsection names and values are either
1091
-     *                                        the input's normalized value, or an array like the top-level array
1092
-     * @throws EE_Error
1093
-     */
1094
-    public function input_pretty_values($include_subform_inputs = false, $flatten = false)
1095
-    {
1096
-        return $this->_input_values(true, $include_subform_inputs, $flatten);
1097
-    }
1098
-
1099
-
1100
-    /**
1101
-     * Gets the input values from the form
1102
-     *
1103
-     * @param boolean $pretty                 Whether to retrieve the pretty value,
1104
-     *                                        or just the normalized value
1105
-     * @param boolean $include_subform_inputs Whether to include inputs from subforms,
1106
-     *                                        or just this forms' direct children inputs
1107
-     * @param boolean $flatten                Whether to force the results into 1-dimensional array,
1108
-     *                                        or allow multidimensional array
1109
-     * @return array if $flatten is TRUE it will always be a 1-dimensional array with array keys being
1110
-     *                                        input names (regardless of whether they are from a subsection or not),
1111
-     *                                        and if $flatten is FALSE it can be a multidimensional array
1112
-     *                                        where keys are always subsection names and values are either
1113
-     *                                        the input's normalized value, or an array like the top-level array
1114
-     * @throws EE_Error
1115
-     */
1116
-    public function _input_values($pretty = false, $include_subform_inputs = false, $flatten = false)
1117
-    {
1118
-        $input_values = array();
1119
-        foreach ($this->subsections() as $subsection_name => $subsection) {
1120
-            if ($subsection instanceof EE_Form_Input_Base) {
1121
-                $input_values[ $subsection_name ] = $pretty
1122
-                    ? $subsection->pretty_value()
1123
-                    : $subsection->normalized_value();
1124
-            } elseif ($subsection instanceof EE_Form_Section_Proper && $include_subform_inputs) {
1125
-                $subform_input_values = $subsection->_input_values(
1126
-                    $pretty,
1127
-                    $include_subform_inputs,
1128
-                    $flatten
1129
-                );
1130
-                if ($flatten) {
1131
-                    $input_values = array_merge($input_values, $subform_input_values);
1132
-                } else {
1133
-                    $input_values[ $subsection_name ] = $subform_input_values;
1134
-                }
1135
-            }
1136
-        }
1137
-        return $input_values;
1138
-    }
1139
-
1140
-
1141
-    /**
1142
-     * Gets the originally submitted input values from the form
1143
-     *
1144
-     * @param boolean $include_subforms  Whether to include inputs from subforms,
1145
-     *                                   or just this forms' direct children inputs
1146
-     * @return array                     if $flatten is TRUE it will always be a 1-dimensional array
1147
-     *                                   with array keys being input names
1148
-     *                                   (regardless of whether they are from a subsection or not),
1149
-     *                                   and if $flatten is FALSE it can be a multidimensional array
1150
-     *                                   where keys are always subsection names and values are either
1151
-     *                                   the input's normalized value, or an array like the top-level array
1152
-     * @throws EE_Error
1153
-     */
1154
-    public function submitted_values($include_subforms = false)
1155
-    {
1156
-        $submitted_values = array();
1157
-        foreach ($this->subsections() as $subsection) {
1158
-            if ($subsection instanceof EE_Form_Input_Base) {
1159
-                // is this input part of an array of inputs?
1160
-                if (strpos($subsection->html_name(), '[') !== false) {
1161
-                    $full_input_name  = EEH_Array::convert_array_values_to_keys(
1162
-                        explode(
1163
-                            '[',
1164
-                            str_replace(']', '', $subsection->html_name())
1165
-                        ),
1166
-                        $subsection->raw_value()
1167
-                    );
1168
-                    $submitted_values = array_replace_recursive($submitted_values, $full_input_name);
1169
-                } else {
1170
-                    $submitted_values[ $subsection->html_name() ] = $subsection->raw_value();
1171
-                }
1172
-            } elseif ($subsection instanceof EE_Form_Section_Proper && $include_subforms) {
1173
-                $subform_input_values = $subsection->submitted_values($include_subforms);
1174
-                $submitted_values     = array_replace_recursive($submitted_values, $subform_input_values);
1175
-            }
1176
-        }
1177
-        return $submitted_values;
1178
-    }
1179
-
1180
-
1181
-    /**
1182
-     * Indicates whether or not this form has received a submission yet
1183
-     * (ie, had receive_form_submission called on it yet)
1184
-     *
1185
-     * @return boolean
1186
-     * @throws EE_Error
1187
-     */
1188
-    public function has_received_submission()
1189
-    {
1190
-        $this->ensure_construct_finalized_called();
1191
-        return $this->_received_submission;
1192
-    }
1193
-
1194
-
1195
-    /**
1196
-     * Equivalent to passing 'exclude' in the constructor's options array.
1197
-     * Removes the listed inputs from the form
1198
-     *
1199
-     * @param array $inputs_to_exclude values are the input names
1200
-     * @return void
1201
-     */
1202
-    public function exclude(array $inputs_to_exclude = array())
1203
-    {
1204
-        foreach ($inputs_to_exclude as $input_to_exclude_name) {
1205
-            unset($this->_subsections[ $input_to_exclude_name ]);
1206
-        }
1207
-    }
1208
-
1209
-
1210
-    /**
1211
-     * Changes these inputs' display strategy to be EE_Hidden_Display_Strategy.
1212
-     * @param array $inputs_to_hide
1213
-     * @throws EE_Error
1214
-     */
1215
-    public function hide(array $inputs_to_hide = array())
1216
-    {
1217
-        foreach ($inputs_to_hide as $input_to_hide) {
1218
-            $input = $this->get_input($input_to_hide);
1219
-            $input->set_display_strategy(new EE_Hidden_Display_Strategy());
1220
-        }
1221
-    }
1222
-
1223
-
1224
-    /**
1225
-     * add_subsections
1226
-     * Adds the listed subsections to the form section.
1227
-     * If $subsection_name_to_target is provided,
1228
-     * then new subsections are added before or after that subsection,
1229
-     * otherwise to the start or end of the entire subsections array.
1230
-     *
1231
-     * @param EE_Form_Section_Base[] $new_subsections           array of new form subsections
1232
-     *                                                          where keys are their names
1233
-     * @param string                 $subsection_name_to_target an existing for section that $new_subsections
1234
-     *                                                          should be added before or after
1235
-     *                                                          IF $subsection_name_to_target is null,
1236
-     *                                                          then $new_subsections will be added to
1237
-     *                                                          the beginning or end of the entire subsections array
1238
-     * @param boolean                $add_before                whether to add $new_subsections, before or after
1239
-     *                                                          $subsection_name_to_target,
1240
-     *                                                          or if $subsection_name_to_target is null,
1241
-     *                                                          before or after entire subsections array
1242
-     * @return void
1243
-     * @throws EE_Error
1244
-     */
1245
-    public function add_subsections($new_subsections, $subsection_name_to_target = null, $add_before = true)
1246
-    {
1247
-        foreach ($new_subsections as $subsection_name => $subsection) {
1248
-            if (! $subsection instanceof EE_Form_Section_Base) {
1249
-                EE_Error::add_error(
1250
-                    sprintf(
1251
-                        esc_html__(
1252
-                            "Trying to add a %s as a subsection (it was named '%s') to the form section '%s'. It was removed.",
1253
-                            'event_espresso'
1254
-                        ),
1255
-                        get_class($subsection),
1256
-                        $subsection_name,
1257
-                        $this->name()
1258
-                    )
1259
-                );
1260
-                unset($new_subsections[ $subsection_name ]);
1261
-            }
1262
-        }
1263
-        $this->_subsections = EEH_Array::insert_into_array(
1264
-            $this->_subsections,
1265
-            $new_subsections,
1266
-            $subsection_name_to_target,
1267
-            $add_before
1268
-        );
1269
-        if ($this->_construction_finalized) {
1270
-            foreach ($this->_subsections as $name => $subsection) {
1271
-                $subsection->_construct_finalize($this, $name);
1272
-            }
1273
-        }
1274
-    }
1275
-
1276
-
1277
-    /**
1278
-     * @param string $subsection_name
1279
-     * @param bool   $recursive
1280
-     * @return bool
1281
-     */
1282
-    public function has_subsection($subsection_name, $recursive = false)
1283
-    {
1284
-        foreach ($this->_subsections as $name => $subsection) {
1285
-            if ($name === $subsection_name
1286
-                || (
1287
-                    $recursive
1288
-                    && $subsection instanceof EE_Form_Section_Proper
1289
-                    && $subsection->has_subsection($subsection_name, $recursive)
1290
-                )
1291
-            ) {
1292
-                return true;
1293
-            }
1294
-        }
1295
-        return false;
1296
-    }
1297
-
1298
-
1299
-
1300
-    /**
1301
-     * Just gets all validatable subsections to clean their sensitive data
1302
-     *
1303
-     * @throws EE_Error
1304
-     */
1305
-    public function clean_sensitive_data()
1306
-    {
1307
-        foreach ($this->get_validatable_subsections() as $subsection) {
1308
-            $subsection->clean_sensitive_data();
1309
-        }
1310
-    }
1311
-
1312
-
1313
-    /**
1314
-     * Sets the submission error message (aka validation error message for this form section and all sub-sections)
1315
-     * @param string                           $form_submission_error_message
1316
-     * @param EE_Form_Section_Validatable $form_section unused
1317
-     * @throws EE_Error
1318
-     */
1319
-    public function set_submission_error_message(
1320
-        $form_submission_error_message = ''
1321
-    ) {
1322
-        $this->_form_submission_error_message = ! empty($form_submission_error_message)
1323
-            ? $form_submission_error_message
1324
-            : $this->getAllValidationErrorsString();
1325
-    }
1326
-
1327
-
1328
-    /**
1329
-     * Returns the cached error message. A default value is set for this during _validate(),
1330
-     * (called during receive_form_submission) but it can be explicitly set using
1331
-     * set_submission_error_message
1332
-     *
1333
-     * @return string
1334
-     */
1335
-    public function submission_error_message()
1336
-    {
1337
-        return $this->_form_submission_error_message;
1338
-    }
1339
-
1340
-
1341
-    /**
1342
-     * Sets a message to display if the data submitted to the form was valid.
1343
-     * @param string $form_submission_success_message
1344
-     */
1345
-    public function set_submission_success_message($form_submission_success_message = '')
1346
-    {
1347
-        $this->_form_submission_success_message = ! empty($form_submission_success_message)
1348
-            ? $form_submission_success_message
1349
-            : esc_html__('Form submitted successfully', 'event_espresso');
1350
-    }
1351
-
1352
-
1353
-    /**
1354
-     * Gets a message appropriate for display when the form is correctly submitted
1355
-     * @return string
1356
-     */
1357
-    public function submission_success_message()
1358
-    {
1359
-        return $this->_form_submission_success_message;
1360
-    }
1361
-
1362
-
1363
-    /**
1364
-     * Returns the prefix that should be used on child of this form section for
1365
-     * their html names. If this form section itself has a parent, prepends ITS
1366
-     * prefix onto this form section's prefix. Used primarily by
1367
-     * EE_Form_Input_Base::_set_default_html_name_if_empty
1368
-     *
1369
-     * @return string
1370
-     * @throws EE_Error
1371
-     */
1372
-    public function html_name_prefix()
1373
-    {
1374
-        if ($this->parent_section() instanceof EE_Form_Section_Proper) {
1375
-            return $this->parent_section()->html_name_prefix() . '[' . $this->name() . ']';
1376
-        }
1377
-        return $this->name();
1378
-    }
1379
-
1380
-
1381
-    /**
1382
-     * Gets the name, but first checks _construct_finalize has been called. If not,
1383
-     * calls it (assumes there is no parent and that we want the name to be whatever
1384
-     * was set, which is probably nothing, or the classname)
1385
-     *
1386
-     * @return string
1387
-     * @throws EE_Error
1388
-     */
1389
-    public function name()
1390
-    {
1391
-        $this->ensure_construct_finalized_called();
1392
-        return parent::name();
1393
-    }
1394
-
1395
-
1396
-    /**
1397
-     * @return EE_Form_Section_Proper
1398
-     * @throws EE_Error
1399
-     */
1400
-    public function parent_section()
1401
-    {
1402
-        $this->ensure_construct_finalized_called();
1403
-        return parent::parent_section();
1404
-    }
1405
-
1406
-
1407
-    /**
1408
-     * make sure construction finalized was called, otherwise children might not be ready
1409
-     *
1410
-     * @return void
1411
-     * @throws EE_Error
1412
-     */
1413
-    public function ensure_construct_finalized_called()
1414
-    {
1415
-        if (! $this->_construction_finalized) {
1416
-            $this->_construct_finalize($this->_parent_section, $this->_name);
1417
-        }
1418
-    }
1419
-
1420
-
1421
-    /**
1422
-     * Checks if any of this form section's inputs, or any of its children's inputs,
1423
-     * are in teh form data. If any are found, returns true. Else false
1424
-     *
1425
-     * @param array $req_data
1426
-     * @return boolean
1427
-     * @throws EE_Error
1428
-     */
1429
-    public function form_data_present_in($req_data = null)
1430
-    {
1431
-        $req_data = $this->getCachedRequest($req_data);
1432
-        foreach ($this->subsections() as $subsection) {
1433
-            if ($subsection instanceof EE_Form_Input_Base) {
1434
-                if ($subsection->form_data_present_in($req_data)) {
1435
-                    return true;
1436
-                }
1437
-            } elseif ($subsection instanceof EE_Form_Section_Proper) {
1438
-                if ($subsection->form_data_present_in($req_data)) {
1439
-                    return true;
1440
-                }
1441
-            }
1442
-        }
1443
-        return false;
1444
-    }
1445
-
1446
-
1447
-    /**
1448
-     * Gets validation errors for this form section and subsections
1449
-     * Similar to EE_Form_Section_Validatable::get_validation_errors() except this
1450
-     * gets the validation errors for ALL subsection
1451
-     *
1452
-     * @return EE_Validation_Error[]
1453
-     * @throws EE_Error
1454
-     */
1455
-    public function get_validation_errors_accumulated()
1456
-    {
1457
-        $validation_errors = $this->get_validation_errors();
1458
-        foreach ($this->get_validatable_subsections() as $subsection) {
1459
-            if ($subsection instanceof EE_Form_Section_Proper) {
1460
-                $validation_errors_on_this_subsection = $subsection->get_validation_errors_accumulated();
1461
-            } else {
1462
-                $validation_errors_on_this_subsection = $subsection->get_validation_errors();
1463
-            }
1464
-            if ($validation_errors_on_this_subsection) {
1465
-                $validation_errors = array_merge($validation_errors, $validation_errors_on_this_subsection);
1466
-            }
1467
-        }
1468
-        return $validation_errors;
1469
-    }
1470
-
1471
-    /**
1472
-     * Fetch validation errors from children and grandchildren and puts them in a single string.
1473
-     * This traverses the form section tree to generate this, but you probably want to instead use
1474
-     * get_form_submission_error_message() which is usually this message cached (or a custom validation error message)
1475
-     *
1476
-     * @return string
1477
-     * @since 4.9.59.p
1478
-     */
1479
-    protected function getAllValidationErrorsString()
1480
-    {
1481
-        $submission_error_messages = array();
1482
-        // bad, bad, bad registrant
1483
-        foreach ($this->get_validation_errors_accumulated() as $validation_error) {
1484
-            if ($validation_error instanceof EE_Validation_Error) {
1485
-                $form_section = $validation_error->get_form_section();
1486
-                if ($form_section instanceof EE_Form_Input_Base) {
1487
-                    $label = $validation_error->get_form_section()->html_label_text();
1488
-                } elseif ($form_section instanceof EE_Form_Section_Validatable) {
1489
-                    $label = $validation_error->get_form_section()->name();
1490
-                } else {
1491
-                    $label = esc_html__('Unknown', 'event_espresso');
1492
-                }
1493
-                $submission_error_messages[] = sprintf(
1494
-                    __('%s : %s', 'event_espresso'),
1495
-                    $label,
1496
-                    $validation_error->getMessage()
1497
-                );
1498
-            }
1499
-        }
1500
-        return implode('<br', $submission_error_messages);
1501
-    }
1502
-
1503
-
1504
-    /**
1505
-     * This isn't just the name of an input, it's a path pointing to an input. The
1506
-     * path is similar to a folder path: slash (/) means to descend into a subsection,
1507
-     * dot-dot-slash (../) means to ascend into the parent section.
1508
-     * After a series of slashes and dot-dot-slashes, there should be the name of an input,
1509
-     * which will be returned.
1510
-     * Eg, if you want the related input to be conditional on a sibling input name 'foobar'
1511
-     * just use 'foobar'. If you want it to be conditional on an aunt/uncle input name
1512
-     * 'baz', use '../baz'. If you want it to be conditional on a cousin input,
1513
-     * the child of 'baz_section' named 'baz_child', use '../baz_section/baz_child'.
1514
-     * Etc
1515
-     *
1516
-     * @param string|false $form_section_path we accept false also because substr( '../', '../' ) = false
1517
-     * @return EE_Form_Section_Base
1518
-     * @throws EE_Error
1519
-     */
1520
-    public function find_section_from_path($form_section_path)
1521
-    {
1522
-        // check if we can find the input from purely going straight up the tree
1523
-        $input = parent::find_section_from_path($form_section_path);
1524
-        if ($input instanceof EE_Form_Section_Base) {
1525
-            return $input;
1526
-        }
1527
-        $next_slash_pos = strpos($form_section_path, '/');
1528
-        if ($next_slash_pos !== false) {
1529
-            $child_section_name = substr($form_section_path, 0, $next_slash_pos);
1530
-            $subpath            = substr($form_section_path, $next_slash_pos + 1);
1531
-        } else {
1532
-            $child_section_name = $form_section_path;
1533
-            $subpath            = '';
1534
-        }
1535
-        $child_section = $this->get_subsection($child_section_name);
1536
-        if ($child_section instanceof EE_Form_Section_Base) {
1537
-            return $child_section->find_section_from_path($subpath);
1538
-        }
1539
-        return null;
1540
-    }
17
+	const SUBMITTED_FORM_DATA_SSN_KEY = 'submitted_form_data';
18
+
19
+	/**
20
+	 * Subsections
21
+	 *
22
+	 * @var EE_Form_Section_Validatable[]
23
+	 */
24
+	protected $_subsections = array();
25
+
26
+	/**
27
+	 * Strategy for laying out the form
28
+	 *
29
+	 * @var EE_Form_Section_Layout_Base
30
+	 */
31
+	protected $_layout_strategy;
32
+
33
+	/**
34
+	 * Whether or not this form has received and validated a form submission yet
35
+	 *
36
+	 * @var boolean
37
+	 */
38
+	protected $_received_submission = false;
39
+
40
+	/**
41
+	 * message displayed to users upon successful form submission
42
+	 *
43
+	 * @var string
44
+	 */
45
+	protected $_form_submission_success_message = '';
46
+
47
+	/**
48
+	 * message displayed to users upon unsuccessful form submission
49
+	 *
50
+	 * @var string
51
+	 */
52
+	protected $_form_submission_error_message = '';
53
+
54
+	/**
55
+	 * @var array like $_REQUEST
56
+	 */
57
+	protected $cached_request_data;
58
+
59
+	/**
60
+	 * Stores whether this form (and its sub-sections) were found to be valid or not.
61
+	 * Starts off as null, but once the form is validated, it set to either true or false
62
+	 * @var boolean|null
63
+	 */
64
+	protected $is_valid;
65
+
66
+	/**
67
+	 * Stores all the data that will localized for form validation
68
+	 *
69
+	 * @var array
70
+	 */
71
+	static protected $_js_localization = array();
72
+
73
+	/**
74
+	 * whether or not the form's localized validation JS vars have been set
75
+	 *
76
+	 * @type boolean
77
+	 */
78
+	static protected $_scripts_localized = false;
79
+
80
+
81
+	/**
82
+	 * when constructing a proper form section, calls _construct_finalize on children
83
+	 * so that they know who their parent is, and what name they've been given.
84
+	 *
85
+	 * @param array[] $options_array   {
86
+	 * @type          $subsections     EE_Form_Section_Validatable[] where keys are the section's name
87
+	 * @type          $include         string[] numerically-indexed where values are section names to be included,
88
+	 *                                 and in that order. This is handy if you want
89
+	 *                                 the subsections to be ordered differently than the default, and if you override
90
+	 *                                 which fields are shown
91
+	 * @type          $exclude         string[] values are subsections to be excluded. This is handy if you want
92
+	 *                                 to remove certain default subsections (note: if you specify BOTH 'include' AND
93
+	 *                                 'exclude', the inclusions will be applied first, and the exclusions will exclude
94
+	 *                                 items from that list of inclusions)
95
+	 * @type          $layout_strategy EE_Form_Section_Layout_Base strategy for laying out the form
96
+	 *                                 } @see EE_Form_Section_Validatable::__construct()
97
+	 * @throws EE_Error
98
+	 */
99
+	public function __construct($options_array = array())
100
+	{
101
+		$options_array = (array) apply_filters(
102
+			'FHEE__EE_Form_Section_Proper___construct__options_array',
103
+			$options_array,
104
+			$this
105
+		);
106
+		// call parent first, as it may be setting the name
107
+		parent::__construct($options_array);
108
+		// if they've included subsections in the constructor, add them now
109
+		if (isset($options_array['include'])) {
110
+			// we are going to make sure we ONLY have those subsections to include
111
+			// AND we are going to make sure they're in that specified order
112
+			$reordered_subsections = array();
113
+			foreach ($options_array['include'] as $input_name) {
114
+				if (isset($this->_subsections[ $input_name ])) {
115
+					$reordered_subsections[ $input_name ] = $this->_subsections[ $input_name ];
116
+				}
117
+			}
118
+			$this->_subsections = $reordered_subsections;
119
+		}
120
+		if (isset($options_array['exclude'])) {
121
+			$exclude            = $options_array['exclude'];
122
+			$this->_subsections = array_diff_key($this->_subsections, array_flip($exclude));
123
+		}
124
+		if (isset($options_array['layout_strategy'])) {
125
+			$this->_layout_strategy = $options_array['layout_strategy'];
126
+		}
127
+		if (! $this->_layout_strategy) {
128
+			$this->_layout_strategy = is_admin() ? new EE_Admin_Two_Column_Layout() : new EE_Two_Column_Layout();
129
+		}
130
+		$this->_layout_strategy->_construct_finalize($this);
131
+		// ok so we are definitely going to want the forms JS,
132
+		// so enqueue it or remember to enqueue it during wp_enqueue_scripts
133
+		if (did_action('wp_enqueue_scripts') || did_action('admin_enqueue_scripts')) {
134
+			// ok so they've constructed this object after when they should have.
135
+			// just enqueue the generic form scripts and initialize the form immediately in the JS
136
+			EE_Form_Section_Proper::wp_enqueue_scripts(true);
137
+		} else {
138
+			add_action('wp_enqueue_scripts', array('EE_Form_Section_Proper', 'wp_enqueue_scripts'));
139
+			add_action('admin_enqueue_scripts', array('EE_Form_Section_Proper', 'wp_enqueue_scripts'));
140
+		}
141
+		add_action('wp_footer', array($this, 'ensure_scripts_localized'), 1);
142
+		/**
143
+		 * Gives other plugins a chance to hook in before construct finalize is called.
144
+		 * The form probably doesn't yet have a parent form section.
145
+		 * Since 4.9.32, when this action was introduced, this is the best place to add a subsection onto a form,
146
+		 * assuming you don't care what the form section's name, HTML ID, or HTML name etc are.
147
+		 * Also see AHEE__EE_Form_Section_Proper___construct_finalize__end
148
+		 *
149
+		 * @since 4.9.32
150
+		 * @param EE_Form_Section_Proper $this          before __construct is done, but all of its logic,
151
+		 *                                              except maybe calling _construct_finalize has been done
152
+		 * @param array                  $options_array options passed into the constructor
153
+		 */
154
+		do_action(
155
+			'AHEE__EE_Form_Input_Base___construct__before_construct_finalize_called',
156
+			$this,
157
+			$options_array
158
+		);
159
+		if (isset($options_array['name'])) {
160
+			$this->_construct_finalize(null, $options_array['name']);
161
+		}
162
+	}
163
+
164
+
165
+	/**
166
+	 * Finishes construction given the parent form section and this form section's name
167
+	 *
168
+	 * @param EE_Form_Section_Proper $parent_form_section
169
+	 * @param string                 $name
170
+	 * @throws EE_Error
171
+	 */
172
+	public function _construct_finalize($parent_form_section, $name)
173
+	{
174
+		parent::_construct_finalize($parent_form_section, $name);
175
+		$this->_set_default_name_if_empty();
176
+		$this->_set_default_html_id_if_empty();
177
+		foreach ($this->_subsections as $subsection_name => $subsection) {
178
+			if ($subsection instanceof EE_Form_Section_Base) {
179
+				$subsection->_construct_finalize($this, $subsection_name);
180
+			} else {
181
+				throw new EE_Error(
182
+					sprintf(
183
+						esc_html__(
184
+							'Subsection "%s" is not an instanceof EE_Form_Section_Base on form "%s". It is a "%s"',
185
+							'event_espresso'
186
+						),
187
+						$subsection_name,
188
+						get_class($this),
189
+						$subsection ? get_class($subsection) : esc_html__('NULL', 'event_espresso')
190
+					)
191
+				);
192
+			}
193
+		}
194
+		/**
195
+		 * Action performed just after form has been given a name (and HTML ID etc) and is fully constructed.
196
+		 * If you have code that should modify the form and needs it and its subsections to have a name, HTML ID
197
+		 * (or other attributes derived from the name like the HTML label id, etc), this is where it should be done.
198
+		 * This might only happen just before displaying the form, or just before it receives form submission data.
199
+		 * If you need to modify the form or its subsections before _construct_finalize is called on it (and we've
200
+		 * ensured it has a name, HTML IDs, etc
201
+		 *
202
+		 * @param EE_Form_Section_Proper      $this
203
+		 * @param EE_Form_Section_Proper|null $parent_form_section
204
+		 * @param string                      $name
205
+		 */
206
+		do_action(
207
+			'AHEE__EE_Form_Section_Proper___construct_finalize__end',
208
+			$this,
209
+			$parent_form_section,
210
+			$name
211
+		);
212
+	}
213
+
214
+
215
+	/**
216
+	 * Gets the layout strategy for this form section
217
+	 *
218
+	 * @return EE_Form_Section_Layout_Base
219
+	 */
220
+	public function get_layout_strategy()
221
+	{
222
+		return $this->_layout_strategy;
223
+	}
224
+
225
+
226
+	/**
227
+	 * Gets the HTML for a single input for this form section according
228
+	 * to the layout strategy
229
+	 *
230
+	 * @param EE_Form_Input_Base $input
231
+	 * @return string
232
+	 */
233
+	public function get_html_for_input($input)
234
+	{
235
+		return $this->_layout_strategy->layout_input($input);
236
+	}
237
+
238
+
239
+	/**
240
+	 * was_submitted - checks if form inputs are present in request data
241
+	 * Basically an alias for form_data_present_in() (which is used by both
242
+	 * proper form sections and form inputs)
243
+	 *
244
+	 * @param null $form_data
245
+	 * @return boolean
246
+	 * @throws EE_Error
247
+	 */
248
+	public function was_submitted($form_data = null)
249
+	{
250
+		return $this->form_data_present_in($form_data);
251
+	}
252
+
253
+	/**
254
+	 * Gets the cached request data; but if there is none, or $req_data was set with
255
+	 * something different, refresh the cache, and then return it
256
+	 * @param null $req_data
257
+	 * @return array
258
+	 */
259
+	protected function getCachedRequest($req_data = null)
260
+	{
261
+		if ($this->cached_request_data === null
262
+			|| (
263
+				$req_data !== null &&
264
+				$req_data !== $this->cached_request_data
265
+			)
266
+		) {
267
+			$req_data = apply_filters(
268
+				'FHEE__EE_Form_Section_Proper__receive_form_submission__req_data',
269
+				$req_data,
270
+				$this
271
+			);
272
+			if ($req_data === null) {
273
+				$req_data = array_merge($_GET, $_POST);
274
+			}
275
+			$req_data = apply_filters(
276
+				'FHEE__EE_Form_Section_Proper__receive_form_submission__request_data',
277
+				$req_data,
278
+				$this
279
+			);
280
+			$this->cached_request_data = (array) $req_data;
281
+		}
282
+		return $this->cached_request_data;
283
+	}
284
+
285
+
286
+	/**
287
+	 * After the form section is initially created, call this to sanitize the data in the submission
288
+	 * which relates to this form section, validate it, and set it as properties on the form.
289
+	 *
290
+	 * @param array|null $req_data should usually be $_POST (the default).
291
+	 *                             However, you CAN supply a different array.
292
+	 *                             Consider using set_defaults() instead however.
293
+	 *                             (If you rendered the form in the page using echo $form_x->get_html()
294
+	 *                             the inputs will have the correct name in the request data for this function
295
+	 *                             to find them and populate the form with them.
296
+	 *                             If you have a flat form (with only input subsections),
297
+	 *                             you can supply a flat array where keys
298
+	 *                             are the form input names and values are their values)
299
+	 * @param boolean    $validate whether or not to perform validation on this data. Default is,
300
+	 *                             of course, to validate that data, and set errors on the invalid values.
301
+	 *                             But if the data has already been validated
302
+	 *                             (eg you validated the data then stored it in the DB)
303
+	 *                             you may want to skip this step.
304
+	 * @throws InvalidArgumentException
305
+	 * @throws InvalidInterfaceException
306
+	 * @throws InvalidDataTypeException
307
+	 * @throws EE_Error
308
+	 */
309
+	public function receive_form_submission($req_data = null, $validate = true)
310
+	{
311
+		$req_data = $this->getCachedRequest($req_data);
312
+		$this->_normalize($req_data);
313
+		if ($validate) {
314
+			$this->_validate();
315
+			// if it's invalid, we're going to want to re-display so remember what they submitted
316
+			if (! $this->is_valid()) {
317
+				$this->store_submitted_form_data_in_session();
318
+			}
319
+		}
320
+		if ($this->submission_error_message() === '' && ! $this->is_valid()) {
321
+			$this->set_submission_error_message();
322
+		}
323
+		do_action(
324
+			'AHEE__EE_Form_Section_Proper__receive_form_submission__end',
325
+			$req_data,
326
+			$this,
327
+			$validate
328
+		);
329
+	}
330
+
331
+
332
+	/**
333
+	 * caches the originally submitted input values in the session
334
+	 * so that they can be used to repopulate the form if it failed validation
335
+	 *
336
+	 * @return boolean whether or not the data was successfully stored in the session
337
+	 * @throws InvalidArgumentException
338
+	 * @throws InvalidInterfaceException
339
+	 * @throws InvalidDataTypeException
340
+	 * @throws EE_Error
341
+	 */
342
+	protected function store_submitted_form_data_in_session()
343
+	{
344
+		return EE_Registry::instance()->SSN->set_session_data(
345
+			array(
346
+				EE_Form_Section_Proper::SUBMITTED_FORM_DATA_SSN_KEY => $this->submitted_values(true),
347
+			)
348
+		);
349
+	}
350
+
351
+
352
+	/**
353
+	 * retrieves the originally submitted input values in the session
354
+	 * so that they can be used to repopulate the form if it failed validation
355
+	 *
356
+	 * @return array
357
+	 * @throws InvalidArgumentException
358
+	 * @throws InvalidInterfaceException
359
+	 * @throws InvalidDataTypeException
360
+	 */
361
+	protected function get_submitted_form_data_from_session()
362
+	{
363
+		$session = EE_Registry::instance()->SSN;
364
+		if ($session instanceof EE_Session) {
365
+			return $session->get_session_data(
366
+				EE_Form_Section_Proper::SUBMITTED_FORM_DATA_SSN_KEY
367
+			);
368
+		}
369
+		return array();
370
+	}
371
+
372
+
373
+	/**
374
+	 * flushed the originally submitted input values from the session
375
+	 *
376
+	 * @return boolean whether or not the data was successfully removed from the session
377
+	 * @throws InvalidArgumentException
378
+	 * @throws InvalidInterfaceException
379
+	 * @throws InvalidDataTypeException
380
+	 */
381
+	public static function flush_submitted_form_data_from_session()
382
+	{
383
+		return EE_Registry::instance()->SSN->reset_data(
384
+			array(EE_Form_Section_Proper::SUBMITTED_FORM_DATA_SSN_KEY)
385
+		);
386
+	}
387
+
388
+
389
+	/**
390
+	 * Populates this form and its subsections with data from the session.
391
+	 * (Wrapper for EE_Form_Section_Proper::receive_form_submission, so it shows
392
+	 * validation errors when displaying too)
393
+	 * Returns true if the form was populated from the session, false otherwise
394
+	 *
395
+	 * @return boolean
396
+	 * @throws InvalidArgumentException
397
+	 * @throws InvalidInterfaceException
398
+	 * @throws InvalidDataTypeException
399
+	 * @throws EE_Error
400
+	 */
401
+	public function populate_from_session()
402
+	{
403
+		$form_data_in_session = $this->get_submitted_form_data_from_session();
404
+		if (empty($form_data_in_session)) {
405
+			return false;
406
+		}
407
+		$this->receive_form_submission($form_data_in_session);
408
+		add_action('shutdown', array('EE_Form_Section_Proper', 'flush_submitted_form_data_from_session'));
409
+		if ($this->form_data_present_in($form_data_in_session)) {
410
+			return true;
411
+		}
412
+		return false;
413
+	}
414
+
415
+
416
+	/**
417
+	 * Populates the default data for the form, given an array where keys are
418
+	 * the input names, and values are their values (preferably normalized to be their
419
+	 * proper PHP types, not all strings... although that should be ok too).
420
+	 * Proper subsections are sub-arrays, the key being the subsection's name, and
421
+	 * the value being an array formatted in teh same way
422
+	 *
423
+	 * @param array $default_data
424
+	 * @throws EE_Error
425
+	 */
426
+	public function populate_defaults($default_data)
427
+	{
428
+		foreach ($this->subsections(false) as $subsection_name => $subsection) {
429
+			if (isset($default_data[ $subsection_name ])) {
430
+				if ($subsection instanceof EE_Form_Input_Base) {
431
+					$subsection->set_default($default_data[ $subsection_name ]);
432
+				} elseif ($subsection instanceof EE_Form_Section_Proper) {
433
+					$subsection->populate_defaults($default_data[ $subsection_name ]);
434
+				}
435
+			}
436
+		}
437
+	}
438
+
439
+
440
+	/**
441
+	 * returns true if subsection exists
442
+	 *
443
+	 * @param string $name
444
+	 * @return boolean
445
+	 */
446
+	public function subsection_exists($name)
447
+	{
448
+		return isset($this->_subsections[ $name ]) ? true : false;
449
+	}
450
+
451
+
452
+	/**
453
+	 * Gets the subsection specified by its name
454
+	 *
455
+	 * @param string  $name
456
+	 * @param boolean $require_construction_to_be_finalized most client code should leave this as TRUE
457
+	 *                                                      so that the inputs will be properly configured.
458
+	 *                                                      However, some client code may be ok
459
+	 *                                                      with construction finalize being called later
460
+	 *                                                      (realizing that the subsections' html names
461
+	 *                                                      might not be set yet, etc.)
462
+	 * @return EE_Form_Section_Base
463
+	 * @throws EE_Error
464
+	 */
465
+	public function get_subsection($name, $require_construction_to_be_finalized = true)
466
+	{
467
+		if ($require_construction_to_be_finalized) {
468
+			$this->ensure_construct_finalized_called();
469
+		}
470
+		return $this->subsection_exists($name) ? $this->_subsections[ $name ] : null;
471
+	}
472
+
473
+
474
+	/**
475
+	 * Gets all the validatable subsections of this form section
476
+	 *
477
+	 * @return EE_Form_Section_Validatable[]
478
+	 * @throws EE_Error
479
+	 */
480
+	public function get_validatable_subsections()
481
+	{
482
+		$validatable_subsections = array();
483
+		foreach ($this->subsections() as $name => $obj) {
484
+			if ($obj instanceof EE_Form_Section_Validatable) {
485
+				$validatable_subsections[ $name ] = $obj;
486
+			}
487
+		}
488
+		return $validatable_subsections;
489
+	}
490
+
491
+
492
+	/**
493
+	 * Gets an input by the given name. If not found, or if its not an EE_FOrm_Input_Base child,
494
+	 * throw an EE_Error.
495
+	 *
496
+	 * @param string  $name
497
+	 * @param boolean $require_construction_to_be_finalized most client code should
498
+	 *                                                      leave this as TRUE so that the inputs will be properly
499
+	 *                                                      configured. However, some client code may be ok with
500
+	 *                                                      construction finalize being called later
501
+	 *                                                      (realizing that the subsections' html names might not be
502
+	 *                                                      set yet, etc.)
503
+	 * @return EE_Form_Input_Base
504
+	 * @throws EE_Error
505
+	 */
506
+	public function get_input($name, $require_construction_to_be_finalized = true)
507
+	{
508
+		$subsection = $this->get_subsection(
509
+			$name,
510
+			$require_construction_to_be_finalized
511
+		);
512
+		if (! $subsection instanceof EE_Form_Input_Base) {
513
+			throw new EE_Error(
514
+				sprintf(
515
+					esc_html__(
516
+						"Subsection '%s' is not an instanceof EE_Form_Input_Base on form '%s'. It is a '%s'",
517
+						'event_espresso'
518
+					),
519
+					$name,
520
+					get_class($this),
521
+					$subsection ? get_class($subsection) : esc_html__('NULL', 'event_espresso')
522
+				)
523
+			);
524
+		}
525
+		return $subsection;
526
+	}
527
+
528
+
529
+	/**
530
+	 * Like get_input(), gets the proper subsection of the form given the name,
531
+	 * otherwise throws an EE_Error
532
+	 *
533
+	 * @param string  $name
534
+	 * @param boolean $require_construction_to_be_finalized most client code should
535
+	 *                                                      leave this as TRUE so that the inputs will be properly
536
+	 *                                                      configured. However, some client code may be ok with
537
+	 *                                                      construction finalize being called later
538
+	 *                                                      (realizing that the subsections' html names might not be
539
+	 *                                                      set yet, etc.)
540
+	 * @return EE_Form_Section_Proper
541
+	 * @throws EE_Error
542
+	 */
543
+	public function get_proper_subsection($name, $require_construction_to_be_finalized = true)
544
+	{
545
+		$subsection = $this->get_subsection(
546
+			$name,
547
+			$require_construction_to_be_finalized
548
+		);
549
+		if (! $subsection instanceof EE_Form_Section_Proper) {
550
+			throw new EE_Error(
551
+				sprintf(
552
+					esc_html__(
553
+						"Subsection '%'s is not an instanceof EE_Form_Section_Proper on form '%s'",
554
+						'event_espresso'
555
+					),
556
+					$name,
557
+					get_class($this)
558
+				)
559
+			);
560
+		}
561
+		return $subsection;
562
+	}
563
+
564
+
565
+	/**
566
+	 * Gets the value of the specified input. Should be called after receive_form_submission()
567
+	 * or populate_defaults() on the form, where the normalized value on the input is set.
568
+	 *
569
+	 * @param string $name
570
+	 * @return mixed depending on the input's type and its normalization strategy
571
+	 * @throws EE_Error
572
+	 */
573
+	public function get_input_value($name)
574
+	{
575
+		$input = $this->get_input($name);
576
+		return $input->normalized_value();
577
+	}
578
+
579
+
580
+	/**
581
+	 * Checks if this form section itself is valid, and then checks its subsections
582
+	 *
583
+	 * @throws EE_Error
584
+	 * @return boolean
585
+	 */
586
+	public function is_valid()
587
+	{
588
+		if ($this->is_valid === null) {
589
+			if (! $this->has_received_submission()) {
590
+				throw new EE_Error(
591
+					sprintf(
592
+						esc_html__(
593
+							'You cannot check if a form is valid before receiving the form submission using receive_form_submission',
594
+							'event_espresso'
595
+						)
596
+					)
597
+				);
598
+			}
599
+			if (! parent::is_valid()) {
600
+				$this->is_valid = false;
601
+			} else {
602
+				// ok so no general errors to this entire form section.
603
+				// so let's check the subsections, but only set errors if that hasn't been done yet
604
+				$this->is_valid = true;
605
+				foreach ($this->get_validatable_subsections() as $subsection) {
606
+					if (! $subsection->is_valid()) {
607
+						$this->is_valid = false;
608
+					}
609
+				}
610
+			}
611
+		}
612
+		return $this->is_valid;
613
+	}
614
+
615
+
616
+	/**
617
+	 * gets the default name of this form section if none is specified
618
+	 *
619
+	 * @return void
620
+	 */
621
+	protected function _set_default_name_if_empty()
622
+	{
623
+		if (! $this->_name) {
624
+			$classname    = get_class($this);
625
+			$default_name = str_replace('EE_', '', $classname);
626
+			$this->_name  = $default_name;
627
+		}
628
+	}
629
+
630
+
631
+	/**
632
+	 * Returns the HTML for the form, except for the form opening and closing tags
633
+	 * (as the form section doesn't know where you necessarily want to send the information to),
634
+	 * and except for a submit button. Enqueues JS and CSS; if called early enough we will
635
+	 * try to enqueue them in the header, otherwise they'll be enqueued in the footer.
636
+	 * Not doing_it_wrong because theoretically this CAN be used properly,
637
+	 * provided its used during "wp_enqueue_scripts", or it doesn't need to enqueue
638
+	 * any CSS.
639
+	 *
640
+	 * @throws InvalidArgumentException
641
+	 * @throws InvalidInterfaceException
642
+	 * @throws InvalidDataTypeException
643
+	 * @throws EE_Error
644
+	 */
645
+	public function get_html_and_js()
646
+	{
647
+		$this->enqueue_js();
648
+		return $this->get_html();
649
+	}
650
+
651
+
652
+	/**
653
+	 * returns HTML for displaying this form section. recursively calls display_section() on all subsections
654
+	 *
655
+	 * @param bool $display_previously_submitted_data
656
+	 * @return string
657
+	 * @throws InvalidArgumentException
658
+	 * @throws InvalidInterfaceException
659
+	 * @throws InvalidDataTypeException
660
+	 * @throws EE_Error
661
+	 * @throws EE_Error
662
+	 * @throws EE_Error
663
+	 */
664
+	public function get_html($display_previously_submitted_data = true)
665
+	{
666
+		$this->ensure_construct_finalized_called();
667
+		if ($display_previously_submitted_data) {
668
+			$this->populate_from_session();
669
+		}
670
+		return $this->_form_html_filter
671
+			? $this->_form_html_filter->filterHtml($this->_layout_strategy->layout_form(), $this)
672
+			: $this->_layout_strategy->layout_form();
673
+	}
674
+
675
+
676
+	/**
677
+	 * enqueues JS and CSS for the form.
678
+	 * It is preferred to call this before wp_enqueue_scripts so the
679
+	 * scripts and styles can be put in the header, but if called later
680
+	 * they will be put in the footer (which is OK for JS, but in HTML4 CSS should
681
+	 * only be in the header; but in HTML5 its ok in the body.
682
+	 * See http://stackoverflow.com/questions/4957446/load-external-css-file-in-body-tag.
683
+	 * So if your form enqueues CSS, it's preferred to call this before wp_enqueue_scripts.)
684
+	 *
685
+	 * @return void
686
+	 * @throws EE_Error
687
+	 */
688
+	public function enqueue_js()
689
+	{
690
+		$this->_enqueue_and_localize_form_js();
691
+		foreach ($this->subsections() as $subsection) {
692
+			$subsection->enqueue_js();
693
+		}
694
+	}
695
+
696
+
697
+	/**
698
+	 * adds a filter so that jquery validate gets enqueued in EE_System::wp_enqueue_scripts().
699
+	 * This must be done BEFORE wp_enqueue_scripts() gets called, which is on
700
+	 * the wp_enqueue_scripts hook.
701
+	 * However, registering the form js and localizing it can happen when we
702
+	 * actually output the form (which is preferred, seeing how teh form's fields
703
+	 * could change until it's actually outputted)
704
+	 *
705
+	 * @param boolean $init_form_validation_automatically whether or not we want the form validation
706
+	 *                                                    to be triggered automatically or not
707
+	 * @return void
708
+	 */
709
+	public static function wp_enqueue_scripts($init_form_validation_automatically = true)
710
+	{
711
+		wp_register_script(
712
+			'ee_form_section_validation',
713
+			EE_GLOBAL_ASSETS_URL . 'scripts' . DS . 'form_section_validation.js',
714
+			array('jquery-validate', 'jquery-ui-datepicker', 'jquery-validate-extra-methods'),
715
+			EVENT_ESPRESSO_VERSION,
716
+			true
717
+		);
718
+		wp_localize_script(
719
+			'ee_form_section_validation',
720
+			'ee_form_section_validation_init',
721
+			array('init' => $init_form_validation_automatically ? '1' : '0')
722
+		);
723
+	}
724
+
725
+
726
+	/**
727
+	 * gets the variables used by form_section_validation.js.
728
+	 * This needs to be called AFTER we've called $this->_enqueue_jquery_validate_script,
729
+	 * but before the wordpress hook wp_loaded
730
+	 *
731
+	 * @throws EE_Error
732
+	 */
733
+	public function _enqueue_and_localize_form_js()
734
+	{
735
+		$this->ensure_construct_finalized_called();
736
+		// actually, we don't want to localize just yet. There may be other forms on the page.
737
+		// so we need to add our form section data to a static variable accessible by all form sections
738
+		// and localize it just before the footer
739
+		$this->localize_validation_rules();
740
+		add_action('wp_footer', array('EE_Form_Section_Proper', 'localize_script_for_all_forms'), 2);
741
+		add_action('admin_footer', array('EE_Form_Section_Proper', 'localize_script_for_all_forms'));
742
+	}
743
+
744
+
745
+	/**
746
+	 * add our form section data to a static variable accessible by all form sections
747
+	 *
748
+	 * @param bool $return_for_subsection
749
+	 * @return void
750
+	 * @throws EE_Error
751
+	 */
752
+	public function localize_validation_rules($return_for_subsection = false)
753
+	{
754
+		// we only want to localize vars ONCE for the entire form,
755
+		// so if the form section doesn't have a parent, then it must be the top dog
756
+		if ($return_for_subsection || ! $this->parent_section()) {
757
+			EE_Form_Section_Proper::$_js_localization['form_data'][ $this->html_id() ] = array(
758
+				'form_section_id'  => $this->html_id(true),
759
+				'validation_rules' => $this->get_jquery_validation_rules(),
760
+				'other_data'       => $this->get_other_js_data(),
761
+				'errors'           => $this->subsection_validation_errors_by_html_name(),
762
+			);
763
+			EE_Form_Section_Proper::$_scripts_localized                                = true;
764
+		}
765
+	}
766
+
767
+
768
+	/**
769
+	 * Gets an array of extra data that will be useful for client-side javascript.
770
+	 * This is primarily data added by inputs and forms in addition to any
771
+	 * scripts they might enqueue
772
+	 *
773
+	 * @param array $form_other_js_data
774
+	 * @return array
775
+	 * @throws EE_Error
776
+	 */
777
+	public function get_other_js_data($form_other_js_data = array())
778
+	{
779
+		foreach ($this->subsections() as $subsection) {
780
+			$form_other_js_data = $subsection->get_other_js_data($form_other_js_data);
781
+		}
782
+		return $form_other_js_data;
783
+	}
784
+
785
+
786
+	/**
787
+	 * Gets a flat array of inputs for this form section and its subsections.
788
+	 * Keys are their form names, and values are the inputs themselves
789
+	 *
790
+	 * @return EE_Form_Input_Base
791
+	 * @throws EE_Error
792
+	 */
793
+	public function inputs_in_subsections()
794
+	{
795
+		$inputs = array();
796
+		foreach ($this->subsections() as $subsection) {
797
+			if ($subsection instanceof EE_Form_Input_Base) {
798
+				$inputs[ $subsection->html_name() ] = $subsection;
799
+			} elseif ($subsection instanceof EE_Form_Section_Proper) {
800
+				$inputs += $subsection->inputs_in_subsections();
801
+			}
802
+		}
803
+		return $inputs;
804
+	}
805
+
806
+
807
+	/**
808
+	 * Gets a flat array of all the validation errors.
809
+	 * Keys are html names (because those should be unique)
810
+	 * and values are a string of all their validation errors
811
+	 *
812
+	 * @return string[]
813
+	 * @throws EE_Error
814
+	 */
815
+	public function subsection_validation_errors_by_html_name()
816
+	{
817
+		$inputs = $this->inputs();
818
+		$errors = array();
819
+		foreach ($inputs as $form_input) {
820
+			if ($form_input instanceof EE_Form_Input_Base && $form_input->get_validation_errors()) {
821
+				$errors[ $form_input->html_name() ] = $form_input->get_validation_error_string();
822
+			}
823
+		}
824
+		return $errors;
825
+	}
826
+
827
+
828
+	/**
829
+	 * passes all the form data required by the JS to the JS, and enqueues the few required JS files.
830
+	 * Should be setup by each form during the _enqueues_and_localize_form_js
831
+	 *
832
+	 * @throws InvalidArgumentException
833
+	 * @throws InvalidInterfaceException
834
+	 * @throws InvalidDataTypeException
835
+	 */
836
+	public static function localize_script_for_all_forms()
837
+	{
838
+		// allow inputs and stuff to hook in their JS and stuff here
839
+		do_action('AHEE__EE_Form_Section_Proper__localize_script_for_all_forms__begin');
840
+		EE_Form_Section_Proper::$_js_localization['localized_error_messages'] = EE_Form_Section_Proper::_get_localized_error_messages();
841
+		$email_validation_level = isset(EE_Registry::instance()->CFG->registration->email_validation_level)
842
+			? EE_Registry::instance()->CFG->registration->email_validation_level
843
+			: 'wp_default';
844
+		EE_Form_Section_Proper::$_js_localization['email_validation_level']   = $email_validation_level;
845
+		wp_enqueue_script('ee_form_section_validation');
846
+		wp_localize_script(
847
+			'ee_form_section_validation',
848
+			'ee_form_section_vars',
849
+			EE_Form_Section_Proper::$_js_localization
850
+		);
851
+	}
852
+
853
+
854
+	/**
855
+	 * ensure_scripts_localized
856
+	 *
857
+	 * @throws EE_Error
858
+	 */
859
+	public function ensure_scripts_localized()
860
+	{
861
+		if (! EE_Form_Section_Proper::$_scripts_localized) {
862
+			$this->_enqueue_and_localize_form_js();
863
+		}
864
+	}
865
+
866
+
867
+	/**
868
+	 * Gets the hard-coded validation error messages to be used in the JS. The convention
869
+	 * is that the key here should be the same as the custom validation rule put in the JS file
870
+	 *
871
+	 * @return array keys are custom validation rules, and values are internationalized strings
872
+	 */
873
+	private static function _get_localized_error_messages()
874
+	{
875
+		return array(
876
+			'validUrl' => esc_html__('This is not a valid absolute URL. Eg, http://domain.com/monkey.jpg', 'event_espresso'),
877
+			'regex'    => esc_html__('Please check your input', 'event_espresso'),
878
+		);
879
+	}
880
+
881
+
882
+	/**
883
+	 * @return array
884
+	 */
885
+	public static function js_localization()
886
+	{
887
+		return self::$_js_localization;
888
+	}
889
+
890
+
891
+	/**
892
+	 * @return void
893
+	 */
894
+	public static function reset_js_localization()
895
+	{
896
+		self::$_js_localization = array();
897
+	}
898
+
899
+
900
+	/**
901
+	 * Gets the JS to put inside the jquery validation rules for subsection of this form section.
902
+	 * See parent function for more...
903
+	 *
904
+	 * @return array
905
+	 * @throws EE_Error
906
+	 */
907
+	public function get_jquery_validation_rules()
908
+	{
909
+		$jquery_validation_rules = array();
910
+		foreach ($this->get_validatable_subsections() as $subsection) {
911
+			$jquery_validation_rules = array_merge(
912
+				$jquery_validation_rules,
913
+				$subsection->get_jquery_validation_rules()
914
+			);
915
+		}
916
+		return $jquery_validation_rules;
917
+	}
918
+
919
+
920
+	/**
921
+	 * Sanitizes all the data and sets the sanitized value of each field
922
+	 *
923
+	 * @param array $req_data like $_POST
924
+	 * @return void
925
+	 * @throws EE_Error
926
+	 */
927
+	protected function _normalize($req_data)
928
+	{
929
+		$this->_received_submission = true;
930
+		$this->_validation_errors   = array();
931
+		foreach ($this->get_validatable_subsections() as $subsection) {
932
+			try {
933
+				$subsection->_normalize($req_data);
934
+			} catch (EE_Validation_Error $e) {
935
+				$subsection->add_validation_error($e);
936
+			}
937
+		}
938
+	}
939
+
940
+
941
+	/**
942
+	 * Performs validation on this form section and its subsections.
943
+	 * For each subsection,
944
+	 * calls _validate_{subsection_name} on THIS form (if the function exists)
945
+	 * and passes it the subsection, then calls _validate on that subsection.
946
+	 * If you need to perform validation on the form as a whole (considering multiple)
947
+	 * you would be best to override this _validate method,
948
+	 * calling parent::_validate() first.
949
+	 *
950
+	 * @throws EE_Error
951
+	 */
952
+	protected function _validate()
953
+	{
954
+		// reset the cache of whether this form is valid or not- we're re-validating it now
955
+		$this->is_valid = null;
956
+		foreach ($this->get_validatable_subsections() as $subsection_name => $subsection) {
957
+			if (method_exists($this, '_validate_' . $subsection_name)) {
958
+				call_user_func_array(array($this, '_validate_' . $subsection_name), array($subsection));
959
+			}
960
+			$subsection->_validate();
961
+		}
962
+	}
963
+
964
+
965
+	/**
966
+	 * Gets all the validated inputs for the form section
967
+	 *
968
+	 * @return array
969
+	 * @throws EE_Error
970
+	 */
971
+	public function valid_data()
972
+	{
973
+		$inputs = array();
974
+		foreach ($this->subsections() as $subsection_name => $subsection) {
975
+			if ($subsection instanceof EE_Form_Section_Proper) {
976
+				$inputs[ $subsection_name ] = $subsection->valid_data();
977
+			} elseif ($subsection instanceof EE_Form_Input_Base) {
978
+				$inputs[ $subsection_name ] = $subsection->normalized_value();
979
+			}
980
+		}
981
+		return $inputs;
982
+	}
983
+
984
+
985
+	/**
986
+	 * Gets all the inputs on this form section
987
+	 *
988
+	 * @return EE_Form_Input_Base[]
989
+	 * @throws EE_Error
990
+	 */
991
+	public function inputs()
992
+	{
993
+		$inputs = array();
994
+		foreach ($this->subsections() as $subsection_name => $subsection) {
995
+			if ($subsection instanceof EE_Form_Input_Base) {
996
+				$inputs[ $subsection_name ] = $subsection;
997
+			}
998
+		}
999
+		return $inputs;
1000
+	}
1001
+
1002
+
1003
+	/**
1004
+	 * Gets all the subsections which are a proper form
1005
+	 *
1006
+	 * @return EE_Form_Section_Proper[]
1007
+	 * @throws EE_Error
1008
+	 */
1009
+	public function subforms()
1010
+	{
1011
+		$form_sections = array();
1012
+		foreach ($this->subsections() as $name => $obj) {
1013
+			if ($obj instanceof EE_Form_Section_Proper) {
1014
+				$form_sections[ $name ] = $obj;
1015
+			}
1016
+		}
1017
+		return $form_sections;
1018
+	}
1019
+
1020
+
1021
+	/**
1022
+	 * Gets all the subsections (inputs, proper subsections, or html-only sections).
1023
+	 * Consider using inputs() or subforms()
1024
+	 * if you only want form inputs or proper form sections.
1025
+	 *
1026
+	 * @param boolean $require_construction_to_be_finalized most client code should
1027
+	 *                                                      leave this as TRUE so that the inputs will be properly
1028
+	 *                                                      configured. However, some client code may be ok with
1029
+	 *                                                      construction finalize being called later
1030
+	 *                                                      (realizing that the subsections' html names might not be
1031
+	 *                                                      set yet, etc.)
1032
+	 * @return EE_Form_Section_Proper[]
1033
+	 * @throws EE_Error
1034
+	 */
1035
+	public function subsections($require_construction_to_be_finalized = true)
1036
+	{
1037
+		if ($require_construction_to_be_finalized) {
1038
+			$this->ensure_construct_finalized_called();
1039
+		}
1040
+		return $this->_subsections;
1041
+	}
1042
+
1043
+
1044
+	/**
1045
+	 * Returns whether this form has any subforms or inputs
1046
+	 * @return bool
1047
+	 */
1048
+	public function hasSubsections()
1049
+	{
1050
+		return ! empty($this->_subsections);
1051
+	}
1052
+
1053
+
1054
+	/**
1055
+	 * Returns a simple array where keys are input names, and values are their normalized
1056
+	 * values. (Similar to calling get_input_value on inputs)
1057
+	 *
1058
+	 * @param boolean $include_subform_inputs Whether to include inputs from subforms,
1059
+	 *                                        or just this forms' direct children inputs
1060
+	 * @param boolean $flatten                Whether to force the results into 1-dimensional array,
1061
+	 *                                        or allow multidimensional array
1062
+	 * @return array if $flatten is TRUE it will always be a 1-dimensional array
1063
+	 *                                        with array keys being input names
1064
+	 *                                        (regardless of whether they are from a subsection or not),
1065
+	 *                                        and if $flatten is FALSE it can be a multidimensional array
1066
+	 *                                        where keys are always subsection names and values are either
1067
+	 *                                        the input's normalized value, or an array like the top-level array
1068
+	 * @throws EE_Error
1069
+	 */
1070
+	public function input_values($include_subform_inputs = false, $flatten = false)
1071
+	{
1072
+		return $this->_input_values(false, $include_subform_inputs, $flatten);
1073
+	}
1074
+
1075
+
1076
+	/**
1077
+	 * Similar to EE_Form_Section_Proper::input_values(), except this returns the 'display_value'
1078
+	 * of each input. On some inputs (especially radio boxes or checkboxes), the value stored
1079
+	 * is not necessarily the value we want to display to users. This creates an array
1080
+	 * where keys are the input names, and values are their display values
1081
+	 *
1082
+	 * @param boolean $include_subform_inputs Whether to include inputs from subforms,
1083
+	 *                                        or just this forms' direct children inputs
1084
+	 * @param boolean $flatten                Whether to force the results into 1-dimensional array,
1085
+	 *                                        or allow multidimensional array
1086
+	 * @return array if $flatten is TRUE it will always be a 1-dimensional array
1087
+	 *                                        with array keys being input names
1088
+	 *                                        (regardless of whether they are from a subsection or not),
1089
+	 *                                        and if $flatten is FALSE it can be a multidimensional array
1090
+	 *                                        where keys are always subsection names and values are either
1091
+	 *                                        the input's normalized value, or an array like the top-level array
1092
+	 * @throws EE_Error
1093
+	 */
1094
+	public function input_pretty_values($include_subform_inputs = false, $flatten = false)
1095
+	{
1096
+		return $this->_input_values(true, $include_subform_inputs, $flatten);
1097
+	}
1098
+
1099
+
1100
+	/**
1101
+	 * Gets the input values from the form
1102
+	 *
1103
+	 * @param boolean $pretty                 Whether to retrieve the pretty value,
1104
+	 *                                        or just the normalized value
1105
+	 * @param boolean $include_subform_inputs Whether to include inputs from subforms,
1106
+	 *                                        or just this forms' direct children inputs
1107
+	 * @param boolean $flatten                Whether to force the results into 1-dimensional array,
1108
+	 *                                        or allow multidimensional array
1109
+	 * @return array if $flatten is TRUE it will always be a 1-dimensional array with array keys being
1110
+	 *                                        input names (regardless of whether they are from a subsection or not),
1111
+	 *                                        and if $flatten is FALSE it can be a multidimensional array
1112
+	 *                                        where keys are always subsection names and values are either
1113
+	 *                                        the input's normalized value, or an array like the top-level array
1114
+	 * @throws EE_Error
1115
+	 */
1116
+	public function _input_values($pretty = false, $include_subform_inputs = false, $flatten = false)
1117
+	{
1118
+		$input_values = array();
1119
+		foreach ($this->subsections() as $subsection_name => $subsection) {
1120
+			if ($subsection instanceof EE_Form_Input_Base) {
1121
+				$input_values[ $subsection_name ] = $pretty
1122
+					? $subsection->pretty_value()
1123
+					: $subsection->normalized_value();
1124
+			} elseif ($subsection instanceof EE_Form_Section_Proper && $include_subform_inputs) {
1125
+				$subform_input_values = $subsection->_input_values(
1126
+					$pretty,
1127
+					$include_subform_inputs,
1128
+					$flatten
1129
+				);
1130
+				if ($flatten) {
1131
+					$input_values = array_merge($input_values, $subform_input_values);
1132
+				} else {
1133
+					$input_values[ $subsection_name ] = $subform_input_values;
1134
+				}
1135
+			}
1136
+		}
1137
+		return $input_values;
1138
+	}
1139
+
1140
+
1141
+	/**
1142
+	 * Gets the originally submitted input values from the form
1143
+	 *
1144
+	 * @param boolean $include_subforms  Whether to include inputs from subforms,
1145
+	 *                                   or just this forms' direct children inputs
1146
+	 * @return array                     if $flatten is TRUE it will always be a 1-dimensional array
1147
+	 *                                   with array keys being input names
1148
+	 *                                   (regardless of whether they are from a subsection or not),
1149
+	 *                                   and if $flatten is FALSE it can be a multidimensional array
1150
+	 *                                   where keys are always subsection names and values are either
1151
+	 *                                   the input's normalized value, or an array like the top-level array
1152
+	 * @throws EE_Error
1153
+	 */
1154
+	public function submitted_values($include_subforms = false)
1155
+	{
1156
+		$submitted_values = array();
1157
+		foreach ($this->subsections() as $subsection) {
1158
+			if ($subsection instanceof EE_Form_Input_Base) {
1159
+				// is this input part of an array of inputs?
1160
+				if (strpos($subsection->html_name(), '[') !== false) {
1161
+					$full_input_name  = EEH_Array::convert_array_values_to_keys(
1162
+						explode(
1163
+							'[',
1164
+							str_replace(']', '', $subsection->html_name())
1165
+						),
1166
+						$subsection->raw_value()
1167
+					);
1168
+					$submitted_values = array_replace_recursive($submitted_values, $full_input_name);
1169
+				} else {
1170
+					$submitted_values[ $subsection->html_name() ] = $subsection->raw_value();
1171
+				}
1172
+			} elseif ($subsection instanceof EE_Form_Section_Proper && $include_subforms) {
1173
+				$subform_input_values = $subsection->submitted_values($include_subforms);
1174
+				$submitted_values     = array_replace_recursive($submitted_values, $subform_input_values);
1175
+			}
1176
+		}
1177
+		return $submitted_values;
1178
+	}
1179
+
1180
+
1181
+	/**
1182
+	 * Indicates whether or not this form has received a submission yet
1183
+	 * (ie, had receive_form_submission called on it yet)
1184
+	 *
1185
+	 * @return boolean
1186
+	 * @throws EE_Error
1187
+	 */
1188
+	public function has_received_submission()
1189
+	{
1190
+		$this->ensure_construct_finalized_called();
1191
+		return $this->_received_submission;
1192
+	}
1193
+
1194
+
1195
+	/**
1196
+	 * Equivalent to passing 'exclude' in the constructor's options array.
1197
+	 * Removes the listed inputs from the form
1198
+	 *
1199
+	 * @param array $inputs_to_exclude values are the input names
1200
+	 * @return void
1201
+	 */
1202
+	public function exclude(array $inputs_to_exclude = array())
1203
+	{
1204
+		foreach ($inputs_to_exclude as $input_to_exclude_name) {
1205
+			unset($this->_subsections[ $input_to_exclude_name ]);
1206
+		}
1207
+	}
1208
+
1209
+
1210
+	/**
1211
+	 * Changes these inputs' display strategy to be EE_Hidden_Display_Strategy.
1212
+	 * @param array $inputs_to_hide
1213
+	 * @throws EE_Error
1214
+	 */
1215
+	public function hide(array $inputs_to_hide = array())
1216
+	{
1217
+		foreach ($inputs_to_hide as $input_to_hide) {
1218
+			$input = $this->get_input($input_to_hide);
1219
+			$input->set_display_strategy(new EE_Hidden_Display_Strategy());
1220
+		}
1221
+	}
1222
+
1223
+
1224
+	/**
1225
+	 * add_subsections
1226
+	 * Adds the listed subsections to the form section.
1227
+	 * If $subsection_name_to_target is provided,
1228
+	 * then new subsections are added before or after that subsection,
1229
+	 * otherwise to the start or end of the entire subsections array.
1230
+	 *
1231
+	 * @param EE_Form_Section_Base[] $new_subsections           array of new form subsections
1232
+	 *                                                          where keys are their names
1233
+	 * @param string                 $subsection_name_to_target an existing for section that $new_subsections
1234
+	 *                                                          should be added before or after
1235
+	 *                                                          IF $subsection_name_to_target is null,
1236
+	 *                                                          then $new_subsections will be added to
1237
+	 *                                                          the beginning or end of the entire subsections array
1238
+	 * @param boolean                $add_before                whether to add $new_subsections, before or after
1239
+	 *                                                          $subsection_name_to_target,
1240
+	 *                                                          or if $subsection_name_to_target is null,
1241
+	 *                                                          before or after entire subsections array
1242
+	 * @return void
1243
+	 * @throws EE_Error
1244
+	 */
1245
+	public function add_subsections($new_subsections, $subsection_name_to_target = null, $add_before = true)
1246
+	{
1247
+		foreach ($new_subsections as $subsection_name => $subsection) {
1248
+			if (! $subsection instanceof EE_Form_Section_Base) {
1249
+				EE_Error::add_error(
1250
+					sprintf(
1251
+						esc_html__(
1252
+							"Trying to add a %s as a subsection (it was named '%s') to the form section '%s'. It was removed.",
1253
+							'event_espresso'
1254
+						),
1255
+						get_class($subsection),
1256
+						$subsection_name,
1257
+						$this->name()
1258
+					)
1259
+				);
1260
+				unset($new_subsections[ $subsection_name ]);
1261
+			}
1262
+		}
1263
+		$this->_subsections = EEH_Array::insert_into_array(
1264
+			$this->_subsections,
1265
+			$new_subsections,
1266
+			$subsection_name_to_target,
1267
+			$add_before
1268
+		);
1269
+		if ($this->_construction_finalized) {
1270
+			foreach ($this->_subsections as $name => $subsection) {
1271
+				$subsection->_construct_finalize($this, $name);
1272
+			}
1273
+		}
1274
+	}
1275
+
1276
+
1277
+	/**
1278
+	 * @param string $subsection_name
1279
+	 * @param bool   $recursive
1280
+	 * @return bool
1281
+	 */
1282
+	public function has_subsection($subsection_name, $recursive = false)
1283
+	{
1284
+		foreach ($this->_subsections as $name => $subsection) {
1285
+			if ($name === $subsection_name
1286
+				|| (
1287
+					$recursive
1288
+					&& $subsection instanceof EE_Form_Section_Proper
1289
+					&& $subsection->has_subsection($subsection_name, $recursive)
1290
+				)
1291
+			) {
1292
+				return true;
1293
+			}
1294
+		}
1295
+		return false;
1296
+	}
1297
+
1298
+
1299
+
1300
+	/**
1301
+	 * Just gets all validatable subsections to clean their sensitive data
1302
+	 *
1303
+	 * @throws EE_Error
1304
+	 */
1305
+	public function clean_sensitive_data()
1306
+	{
1307
+		foreach ($this->get_validatable_subsections() as $subsection) {
1308
+			$subsection->clean_sensitive_data();
1309
+		}
1310
+	}
1311
+
1312
+
1313
+	/**
1314
+	 * Sets the submission error message (aka validation error message for this form section and all sub-sections)
1315
+	 * @param string                           $form_submission_error_message
1316
+	 * @param EE_Form_Section_Validatable $form_section unused
1317
+	 * @throws EE_Error
1318
+	 */
1319
+	public function set_submission_error_message(
1320
+		$form_submission_error_message = ''
1321
+	) {
1322
+		$this->_form_submission_error_message = ! empty($form_submission_error_message)
1323
+			? $form_submission_error_message
1324
+			: $this->getAllValidationErrorsString();
1325
+	}
1326
+
1327
+
1328
+	/**
1329
+	 * Returns the cached error message. A default value is set for this during _validate(),
1330
+	 * (called during receive_form_submission) but it can be explicitly set using
1331
+	 * set_submission_error_message
1332
+	 *
1333
+	 * @return string
1334
+	 */
1335
+	public function submission_error_message()
1336
+	{
1337
+		return $this->_form_submission_error_message;
1338
+	}
1339
+
1340
+
1341
+	/**
1342
+	 * Sets a message to display if the data submitted to the form was valid.
1343
+	 * @param string $form_submission_success_message
1344
+	 */
1345
+	public function set_submission_success_message($form_submission_success_message = '')
1346
+	{
1347
+		$this->_form_submission_success_message = ! empty($form_submission_success_message)
1348
+			? $form_submission_success_message
1349
+			: esc_html__('Form submitted successfully', 'event_espresso');
1350
+	}
1351
+
1352
+
1353
+	/**
1354
+	 * Gets a message appropriate for display when the form is correctly submitted
1355
+	 * @return string
1356
+	 */
1357
+	public function submission_success_message()
1358
+	{
1359
+		return $this->_form_submission_success_message;
1360
+	}
1361
+
1362
+
1363
+	/**
1364
+	 * Returns the prefix that should be used on child of this form section for
1365
+	 * their html names. If this form section itself has a parent, prepends ITS
1366
+	 * prefix onto this form section's prefix. Used primarily by
1367
+	 * EE_Form_Input_Base::_set_default_html_name_if_empty
1368
+	 *
1369
+	 * @return string
1370
+	 * @throws EE_Error
1371
+	 */
1372
+	public function html_name_prefix()
1373
+	{
1374
+		if ($this->parent_section() instanceof EE_Form_Section_Proper) {
1375
+			return $this->parent_section()->html_name_prefix() . '[' . $this->name() . ']';
1376
+		}
1377
+		return $this->name();
1378
+	}
1379
+
1380
+
1381
+	/**
1382
+	 * Gets the name, but first checks _construct_finalize has been called. If not,
1383
+	 * calls it (assumes there is no parent and that we want the name to be whatever
1384
+	 * was set, which is probably nothing, or the classname)
1385
+	 *
1386
+	 * @return string
1387
+	 * @throws EE_Error
1388
+	 */
1389
+	public function name()
1390
+	{
1391
+		$this->ensure_construct_finalized_called();
1392
+		return parent::name();
1393
+	}
1394
+
1395
+
1396
+	/**
1397
+	 * @return EE_Form_Section_Proper
1398
+	 * @throws EE_Error
1399
+	 */
1400
+	public function parent_section()
1401
+	{
1402
+		$this->ensure_construct_finalized_called();
1403
+		return parent::parent_section();
1404
+	}
1405
+
1406
+
1407
+	/**
1408
+	 * make sure construction finalized was called, otherwise children might not be ready
1409
+	 *
1410
+	 * @return void
1411
+	 * @throws EE_Error
1412
+	 */
1413
+	public function ensure_construct_finalized_called()
1414
+	{
1415
+		if (! $this->_construction_finalized) {
1416
+			$this->_construct_finalize($this->_parent_section, $this->_name);
1417
+		}
1418
+	}
1419
+
1420
+
1421
+	/**
1422
+	 * Checks if any of this form section's inputs, or any of its children's inputs,
1423
+	 * are in teh form data. If any are found, returns true. Else false
1424
+	 *
1425
+	 * @param array $req_data
1426
+	 * @return boolean
1427
+	 * @throws EE_Error
1428
+	 */
1429
+	public function form_data_present_in($req_data = null)
1430
+	{
1431
+		$req_data = $this->getCachedRequest($req_data);
1432
+		foreach ($this->subsections() as $subsection) {
1433
+			if ($subsection instanceof EE_Form_Input_Base) {
1434
+				if ($subsection->form_data_present_in($req_data)) {
1435
+					return true;
1436
+				}
1437
+			} elseif ($subsection instanceof EE_Form_Section_Proper) {
1438
+				if ($subsection->form_data_present_in($req_data)) {
1439
+					return true;
1440
+				}
1441
+			}
1442
+		}
1443
+		return false;
1444
+	}
1445
+
1446
+
1447
+	/**
1448
+	 * Gets validation errors for this form section and subsections
1449
+	 * Similar to EE_Form_Section_Validatable::get_validation_errors() except this
1450
+	 * gets the validation errors for ALL subsection
1451
+	 *
1452
+	 * @return EE_Validation_Error[]
1453
+	 * @throws EE_Error
1454
+	 */
1455
+	public function get_validation_errors_accumulated()
1456
+	{
1457
+		$validation_errors = $this->get_validation_errors();
1458
+		foreach ($this->get_validatable_subsections() as $subsection) {
1459
+			if ($subsection instanceof EE_Form_Section_Proper) {
1460
+				$validation_errors_on_this_subsection = $subsection->get_validation_errors_accumulated();
1461
+			} else {
1462
+				$validation_errors_on_this_subsection = $subsection->get_validation_errors();
1463
+			}
1464
+			if ($validation_errors_on_this_subsection) {
1465
+				$validation_errors = array_merge($validation_errors, $validation_errors_on_this_subsection);
1466
+			}
1467
+		}
1468
+		return $validation_errors;
1469
+	}
1470
+
1471
+	/**
1472
+	 * Fetch validation errors from children and grandchildren and puts them in a single string.
1473
+	 * This traverses the form section tree to generate this, but you probably want to instead use
1474
+	 * get_form_submission_error_message() which is usually this message cached (or a custom validation error message)
1475
+	 *
1476
+	 * @return string
1477
+	 * @since 4.9.59.p
1478
+	 */
1479
+	protected function getAllValidationErrorsString()
1480
+	{
1481
+		$submission_error_messages = array();
1482
+		// bad, bad, bad registrant
1483
+		foreach ($this->get_validation_errors_accumulated() as $validation_error) {
1484
+			if ($validation_error instanceof EE_Validation_Error) {
1485
+				$form_section = $validation_error->get_form_section();
1486
+				if ($form_section instanceof EE_Form_Input_Base) {
1487
+					$label = $validation_error->get_form_section()->html_label_text();
1488
+				} elseif ($form_section instanceof EE_Form_Section_Validatable) {
1489
+					$label = $validation_error->get_form_section()->name();
1490
+				} else {
1491
+					$label = esc_html__('Unknown', 'event_espresso');
1492
+				}
1493
+				$submission_error_messages[] = sprintf(
1494
+					__('%s : %s', 'event_espresso'),
1495
+					$label,
1496
+					$validation_error->getMessage()
1497
+				);
1498
+			}
1499
+		}
1500
+		return implode('<br', $submission_error_messages);
1501
+	}
1502
+
1503
+
1504
+	/**
1505
+	 * This isn't just the name of an input, it's a path pointing to an input. The
1506
+	 * path is similar to a folder path: slash (/) means to descend into a subsection,
1507
+	 * dot-dot-slash (../) means to ascend into the parent section.
1508
+	 * After a series of slashes and dot-dot-slashes, there should be the name of an input,
1509
+	 * which will be returned.
1510
+	 * Eg, if you want the related input to be conditional on a sibling input name 'foobar'
1511
+	 * just use 'foobar'. If you want it to be conditional on an aunt/uncle input name
1512
+	 * 'baz', use '../baz'. If you want it to be conditional on a cousin input,
1513
+	 * the child of 'baz_section' named 'baz_child', use '../baz_section/baz_child'.
1514
+	 * Etc
1515
+	 *
1516
+	 * @param string|false $form_section_path we accept false also because substr( '../', '../' ) = false
1517
+	 * @return EE_Form_Section_Base
1518
+	 * @throws EE_Error
1519
+	 */
1520
+	public function find_section_from_path($form_section_path)
1521
+	{
1522
+		// check if we can find the input from purely going straight up the tree
1523
+		$input = parent::find_section_from_path($form_section_path);
1524
+		if ($input instanceof EE_Form_Section_Base) {
1525
+			return $input;
1526
+		}
1527
+		$next_slash_pos = strpos($form_section_path, '/');
1528
+		if ($next_slash_pos !== false) {
1529
+			$child_section_name = substr($form_section_path, 0, $next_slash_pos);
1530
+			$subpath            = substr($form_section_path, $next_slash_pos + 1);
1531
+		} else {
1532
+			$child_section_name = $form_section_path;
1533
+			$subpath            = '';
1534
+		}
1535
+		$child_section = $this->get_subsection($child_section_name);
1536
+		if ($child_section instanceof EE_Form_Section_Base) {
1537
+			return $child_section->find_section_from_path($subpath);
1538
+		}
1539
+		return null;
1540
+	}
1541 1541
 }
Please login to merge, or discard this patch.
modules/ticket_sales_monitor/EED_Ticket_Sales_Monitor.module.php 3 patches
Doc Comments   +3 added lines, -2 removed lines patch added patch discarded remove patch
@@ -326,7 +326,7 @@  discard block
 block discarded – undo
326 326
      *
327 327
      * @param    EE_Ticket $ticket
328 328
      * @param int          $quantity
329
-     * @return bool
329
+     * @return integer
330 330
      * @throws EE_Error
331 331
      */
332 332
     protected function _reserve_ticket(EE_Ticket $ticket, $quantity = 1)
@@ -343,7 +343,7 @@  discard block
 block discarded – undo
343 343
     /**
344 344
      * @param  EE_Ticket $ticket
345 345
      * @param  int       $quantity
346
-     * @return bool
346
+     * @return integer
347 347
      * @throws EE_Error
348 348
      */
349 349
     protected function _release_reserved_ticket(EE_Ticket $ticket, $quantity = 1)
@@ -969,6 +969,7 @@  discard block
 block discarded – undo
969 969
     /**
970 970
      * @param EE_Ticket[]    $tickets_with_reservations
971 971
      * @param EE_Line_Item[] $valid_reserved_ticket_line_items
972
+     * @param string $source
972 973
      * @return int
973 974
      * @throws UnexpectedEntityException
974 975
      * @throws DomainException
Please login to merge, or discard this patch.
Indentation   +1043 added lines, -1043 removed lines patch added patch discarded remove patch
@@ -20,1048 +20,1048 @@
 block discarded – undo
20 20
 class EED_Ticket_Sales_Monitor extends EED_Module
21 21
 {
22 22
 
23
-    const debug = false;
24
-
25
-    private static $nl = '';
26
-
27
-    /**
28
-     * an array of raw ticket data from EED_Ticket_Selector
29
-     *
30
-     * @var array $ticket_selections
31
-     */
32
-    protected $ticket_selections = array();
33
-
34
-    /**
35
-     * the raw ticket data from EED_Ticket_Selector is organized in rows
36
-     * according to how they are displayed in the actual Ticket_Selector
37
-     * this tracks the current row being processed
38
-     *
39
-     * @var int $current_row
40
-     */
41
-    protected $current_row = 0;
42
-
43
-    /**
44
-     * an array for tracking names of tickets that have sold out
45
-     *
46
-     * @var array $sold_out_tickets
47
-     */
48
-    protected $sold_out_tickets = array();
49
-
50
-    /**
51
-     * an array for tracking names of tickets that have had their quantities reduced
52
-     *
53
-     * @var array $decremented_tickets
54
-     */
55
-    protected $decremented_tickets = array();
56
-
57
-
58
-    /**
59
-     * set_hooks - for hooking into EE Core, other modules, etc
60
-     *
61
-     * @return    void
62
-     */
63
-    public static function set_hooks()
64
-    {
65
-        self::$nl = defined('EE_TESTS_DIR') ? "\n" : '<br />';
66
-        // release tickets for expired carts
67
-        add_action(
68
-            'EED_Ticket_Selector__process_ticket_selections__before',
69
-            array('EED_Ticket_Sales_Monitor', 'release_tickets_for_expired_carts'),
70
-            1
71
-        );
72
-        // check ticket reserves AFTER MER does it's check (hence priority 20)
73
-        add_filter(
74
-            'FHEE__EE_Ticket_Selector___add_ticket_to_cart__ticket_qty',
75
-            array('EED_Ticket_Sales_Monitor', 'validate_ticket_sale'),
76
-            20,
77
-            3
78
-        );
79
-        // add notices for sold out tickets
80
-        add_action(
81
-            'AHEE__EE_Ticket_Selector__process_ticket_selections__after_tickets_added_to_cart',
82
-            array('EED_Ticket_Sales_Monitor', 'post_notices'),
83
-            10
84
-        );
85
-
86
-        // handle tickets deleted from cart
87
-        add_action(
88
-            'FHEE__EED_Multi_Event_Registration__delete_ticket__ticket_removed_from_cart',
89
-            array('EED_Ticket_Sales_Monitor', 'ticket_removed_from_cart'),
90
-            10,
91
-            2
92
-        );
93
-        // handle emptied carts
94
-        add_action(
95
-            'AHEE__EE_Session__reset_cart__before_reset',
96
-            array('EED_Ticket_Sales_Monitor', 'session_cart_reset'),
97
-            10,
98
-            1
99
-        );
100
-        add_action(
101
-            'AHEE__EED_Multi_Event_Registration__empty_event_cart__before_delete_cart',
102
-            array('EED_Ticket_Sales_Monitor', 'session_cart_reset'),
103
-            10,
104
-            1
105
-        );
106
-        // handle cancelled registrations
107
-        add_action(
108
-            'AHEE__EE_Session__reset_checkout__before_reset',
109
-            array('EED_Ticket_Sales_Monitor', 'session_checkout_reset'),
110
-            10,
111
-            1
112
-        );
113
-        // cron tasks
114
-        add_action(
115
-            'AHEE__EE_Cron_Tasks__process_expired_transactions__abandoned_transaction',
116
-            array('EED_Ticket_Sales_Monitor', 'process_abandoned_transactions'),
117
-            10,
118
-            1
119
-        );
120
-        add_action(
121
-            'AHEE__EE_Cron_Tasks__process_expired_transactions__incomplete_transaction',
122
-            array('EED_Ticket_Sales_Monitor', 'process_abandoned_transactions'),
123
-            10,
124
-            1
125
-        );
126
-        add_action(
127
-            'AHEE__EE_Cron_Tasks__process_expired_transactions__failed_transaction',
128
-            array('EED_Ticket_Sales_Monitor', 'process_failed_transactions'),
129
-            10,
130
-            1
131
-        );
132
-    }
133
-
134
-
135
-    /**
136
-     * set_hooks_admin - for hooking into EE Admin Core, other modules, etc
137
-     *
138
-     * @return void
139
-     */
140
-    public static function set_hooks_admin()
141
-    {
142
-        EED_Ticket_Sales_Monitor::set_hooks();
143
-    }
144
-
145
-
146
-    /**
147
-     * @return EED_Ticket_Sales_Monitor|EED_Module
148
-     */
149
-    public static function instance()
150
-    {
151
-        return parent::get_instance(__CLASS__);
152
-    }
153
-
154
-
155
-    /**
156
-     * @param WP_Query $WP_Query
157
-     * @return    void
158
-     */
159
-    public function run($WP_Query)
160
-    {
161
-    }
162
-
163
-
164
-
165
-    /********************************** PRE_TICKET_SALES  **********************************/
166
-
167
-
168
-    /**
169
-     * Retrieves grand totals from the line items that have no TXN ID
170
-     * and timestamps less than the current time minus the session lifespan.
171
-     * These are carts that have been abandoned before the "registrant" even attempted to checkout.
172
-     * We're going to release the tickets for these line items before attempting to add more to the cart.
173
-     *
174
-     * @return void
175
-     * @throws DomainException
176
-     * @throws EE_Error
177
-     * @throws InvalidArgumentException
178
-     * @throws InvalidDataTypeException
179
-     * @throws InvalidInterfaceException
180
-     * @throws UnexpectedEntityException
181
-     */
182
-    public static function release_tickets_for_expired_carts()
183
-    {
184
-        if (self::debug) {
185
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '()';
186
-        }
187
-        do_action('AHEE__EED_Ticket_Sales_Monitor__release_tickets_for_expired_carts__begin');
188
-        $expired_ticket_IDs = array();
189
-        /** @var EventEspresso\core\domain\values\session\SessionLifespan $session_lifespan */
190
-        $session_lifespan = LoaderFactory::getLoader()->getShared(
191
-            'EventEspresso\core\domain\values\session\SessionLifespan'
192
-        );
193
-        $timestamp = $session_lifespan->expiration();
194
-        $expired_ticket_line_items = EEM_Line_Item::instance()->getTicketLineItemsForExpiredCarts($timestamp);
195
-        if (self::debug) {
196
-            echo self::$nl . ' . time(): ' . time();
197
-            echo self::$nl . ' . time() as date: ' . date('Y-m-d H:i a');
198
-            echo self::$nl . ' . session expiration: ' . $session_lifespan->expiration();
199
-            echo self::$nl . ' . session expiration as date: ' . date('Y-m-d H:i a', $session_lifespan->expiration());
200
-            echo self::$nl . ' . timestamp: ' . $timestamp;
201
-            echo self::$nl . ' . $expired_ticket_line_items: ' . count($expired_ticket_line_items);
202
-        }
203
-        if (! empty($expired_ticket_line_items)) {
204
-            foreach ($expired_ticket_line_items as $expired_ticket_line_item) {
205
-                if (! $expired_ticket_line_item instanceof EE_Line_Item) {
206
-                    continue;
207
-                }
208
-                $expired_ticket_IDs[ $expired_ticket_line_item->OBJ_ID() ] = $expired_ticket_line_item->OBJ_ID();
209
-                if (self::debug) {
210
-                    echo self::$nl . ' . $expired_ticket_line_item->OBJ_ID(): ' . $expired_ticket_line_item->OBJ_ID();
211
-                    echo self::$nl . ' . $expired_ticket_line_item->timestamp(): '
212
-                         . date(
213
-                             'Y-m-d h:i a',
214
-                             $expired_ticket_line_item->timestamp(true)
215
-                         );
216
-                }
217
-            }
218
-            if (! empty($expired_ticket_IDs)) {
219
-                EED_Ticket_Sales_Monitor::release_reservations_for_tickets(
220
-                    \EEM_Ticket::instance()->get_tickets_with_IDs($expired_ticket_IDs),
221
-                    array(),
222
-                    __FUNCTION__
223
-                );
224
-                // now  let's get rid of expired line items so that they can't interfere with tracking
225
-                EED_Ticket_Sales_Monitor::clear_expired_line_items_with_no_transaction($timestamp);
226
-            }
227
-        }
228
-        do_action(
229
-            'AHEE__EED_Ticket_Sales_Monitor__release_tickets_for_expired_carts__end',
230
-            $expired_ticket_IDs,
231
-            $expired_ticket_line_items
232
-        );
233
-    }
234
-
235
-
236
-
237
-    /********************************** VALIDATE_TICKET_SALE  **********************************/
238
-
239
-
240
-    /**
241
-     * callback for 'FHEE__EED_Ticket_Selector__process_ticket_selections__valid_post_data'
242
-     *
243
-     * @param int       $qty
244
-     * @param EE_Ticket $ticket
245
-     * @return bool
246
-     * @throws UnexpectedEntityException
247
-     * @throws EE_Error
248
-     */
249
-    public static function validate_ticket_sale($qty = 1, EE_Ticket $ticket)
250
-    {
251
-        $qty = absint($qty);
252
-        if ($qty > 0) {
253
-            $qty = EED_Ticket_Sales_Monitor::instance()->_validate_ticket_sale($ticket, $qty);
254
-        }
255
-        if (self::debug) {
256
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '()';
257
-            echo self::$nl . self::$nl . '<b> RETURNED QTY: ' . $qty . '</b>';
258
-        }
259
-        return $qty;
260
-    }
261
-
262
-
263
-    /**
264
-     * checks whether an individual ticket is available for purchase based on datetime, and ticket details
265
-     *
266
-     * @param   EE_Ticket $ticket
267
-     * @param int         $qty
268
-     * @return int
269
-     * @throws UnexpectedEntityException
270
-     * @throws EE_Error
271
-     */
272
-    protected function _validate_ticket_sale(EE_Ticket $ticket, $qty = 1)
273
-    {
274
-        if (self::debug) {
275
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
276
-        }
277
-        if (! $ticket instanceof EE_Ticket) {
278
-            return 0;
279
-        }
280
-        if (self::debug) {
281
-            echo self::$nl . '<b> . ticket->ID: ' . $ticket->ID() . '</b>';
282
-            echo self::$nl . ' . original ticket->reserved: ' . $ticket->reserved();
283
-        }
284
-        $ticket->refresh_from_db();
285
-        // first let's determine the ticket availability based on sales
286
-        $available = $ticket->qty('saleable');
287
-        if (self::debug) {
288
-            echo self::$nl . ' . . . ticket->qty: ' . $ticket->qty();
289
-            echo self::$nl . ' . . . ticket->sold: ' . $ticket->sold();
290
-            echo self::$nl . ' . . . ticket->reserved: ' . $ticket->reserved();
291
-            echo self::$nl . ' . . . ticket->qty(saleable): ' . $ticket->qty('saleable');
292
-            echo self::$nl . ' . . . available: ' . $available;
293
-        }
294
-        if ($available < 1) {
295
-            $this->_ticket_sold_out($ticket);
296
-            return 0;
297
-        }
298
-        if (self::debug) {
299
-            echo self::$nl . ' . . . qty: ' . $qty;
300
-        }
301
-        if ($available < $qty) {
302
-            $qty = $available;
303
-            if (self::debug) {
304
-                echo self::$nl . ' . . . QTY ADJUSTED: ' . $qty;
305
-            }
306
-            $this->_ticket_quantity_decremented($ticket);
307
-        }
308
-        $this->_reserve_ticket($ticket, $qty);
309
-        return $qty;
310
-    }
311
-
312
-
313
-    /**
314
-     * increments ticket reserved based on quantity passed
315
-     *
316
-     * @param    EE_Ticket $ticket
317
-     * @param int          $quantity
318
-     * @return bool
319
-     * @throws EE_Error
320
-     */
321
-    protected function _reserve_ticket(EE_Ticket $ticket, $quantity = 1)
322
-    {
323
-        if (self::debug) {
324
-            echo self::$nl . self::$nl . ' . . . INCREASE RESERVED: ' . $quantity;
325
-        }
326
-        $ticket->increase_reserved($quantity, 'TicketSalesMonitor:' . __LINE__);
327
-        return $ticket->save();
328
-    }
329
-
330
-
331
-    /**
332
-     * @param  EE_Ticket $ticket
333
-     * @param  int       $quantity
334
-     * @return bool
335
-     * @throws EE_Error
336
-     */
337
-    protected function _release_reserved_ticket(EE_Ticket $ticket, $quantity = 1)
338
-    {
339
-        if (self::debug) {
340
-            echo self::$nl . ' . . . ticket->ID: ' . $ticket->ID();
341
-            echo self::$nl . ' . . . ticket->reserved before: ' . $ticket->reserved();
342
-        }
343
-        $ticket->decrease_reserved($quantity, true, 'TicketSalesMonitor:' . __LINE__);
344
-        if (self::debug) {
345
-            echo self::$nl . ' . . . ticket->reserved after: ' . $ticket->reserved();
346
-        }
347
-        return $ticket->save() ? 1 : 0;
348
-    }
349
-
350
-
351
-    /**
352
-     * removes quantities within the ticket selector based on zero ticket availability
353
-     *
354
-     * @param    EE_Ticket $ticket
355
-     * @return    void
356
-     * @throws UnexpectedEntityException
357
-     * @throws EE_Error
358
-     */
359
-    protected function _ticket_sold_out(EE_Ticket $ticket)
360
-    {
361
-        if (self::debug) {
362
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
363
-            echo self::$nl . ' . . ticket->name: ' . $this->_get_ticket_and_event_name($ticket);
364
-        }
365
-        $this->sold_out_tickets[] = $this->_get_ticket_and_event_name($ticket);
366
-    }
367
-
368
-
369
-    /**
370
-     * adjusts quantities within the ticket selector based on decreased ticket availability
371
-     *
372
-     * @param    EE_Ticket $ticket
373
-     * @return void
374
-     * @throws UnexpectedEntityException
375
-     * @throws EE_Error
376
-     */
377
-    protected function _ticket_quantity_decremented(EE_Ticket $ticket)
378
-    {
379
-        if (self::debug) {
380
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
381
-            echo self::$nl . ' . . ticket->name: ' . $this->_get_ticket_and_event_name($ticket);
382
-        }
383
-        $this->decremented_tickets[] = $this->_get_ticket_and_event_name($ticket);
384
-    }
385
-
386
-
387
-    /**
388
-     * builds string out of ticket and event name
389
-     *
390
-     * @param    EE_Ticket $ticket
391
-     * @return string
392
-     * @throws UnexpectedEntityException
393
-     * @throws EE_Error
394
-     */
395
-    protected function _get_ticket_and_event_name(EE_Ticket $ticket)
396
-    {
397
-        $event = $ticket->get_related_event();
398
-        if ($event instanceof EE_Event) {
399
-            $ticket_name = sprintf(
400
-                _x('%1$s for %2$s', 'ticket name for event name', 'event_espresso'),
401
-                $ticket->name(),
402
-                $event->name()
403
-            );
404
-        } else {
405
-            $ticket_name = $ticket->name();
406
-        }
407
-        return $ticket_name;
408
-    }
409
-
410
-
411
-
412
-    /********************************** EVENT CART  **********************************/
413
-
414
-
415
-    /**
416
-     * releases or reserves ticket(s) based on quantity passed
417
-     *
418
-     * @param  EE_Line_Item $line_item
419
-     * @param  int          $quantity
420
-     * @return void
421
-     * @throws EE_Error
422
-     * @throws InvalidArgumentException
423
-     * @throws InvalidDataTypeException
424
-     * @throws InvalidInterfaceException
425
-     */
426
-    public static function ticket_quantity_updated(EE_Line_Item $line_item, $quantity = 1)
427
-    {
428
-        $ticket = EEM_Ticket::instance()->get_one_by_ID(absint($line_item->OBJ_ID()));
429
-        if ($ticket instanceof EE_Ticket) {
430
-            $ticket->add_extra_meta(
431
-                EE_Ticket::META_KEY_TICKET_RESERVATIONS,
432
-                __LINE__ . ') ' . __METHOD__ . '()'
433
-            );
434
-            if ($quantity > 0) {
435
-                EED_Ticket_Sales_Monitor::instance()->_reserve_ticket($ticket, $quantity);
436
-            } else {
437
-                EED_Ticket_Sales_Monitor::instance()->_release_reserved_ticket($ticket, $quantity);
438
-            }
439
-        }
440
-    }
441
-
442
-
443
-    /**
444
-     * releases reserved ticket(s) based on quantity passed
445
-     *
446
-     * @param  EE_Ticket $ticket
447
-     * @param  int       $quantity
448
-     * @return void
449
-     * @throws EE_Error
450
-     */
451
-    public static function ticket_removed_from_cart(EE_Ticket $ticket, $quantity = 1)
452
-    {
453
-        $ticket->add_extra_meta(
454
-            EE_Ticket::META_KEY_TICKET_RESERVATIONS,
455
-            __LINE__ . ') ' . __METHOD__ . '()'
456
-        );
457
-        EED_Ticket_Sales_Monitor::instance()->_release_reserved_ticket($ticket, $quantity);
458
-    }
459
-
460
-
461
-
462
-    /********************************** POST_NOTICES  **********************************/
463
-
464
-
465
-    /**
466
-     * @return void
467
-     * @throws EE_Error
468
-     * @throws InvalidArgumentException
469
-     * @throws ReflectionException
470
-     * @throws InvalidDataTypeException
471
-     * @throws InvalidInterfaceException
472
-     */
473
-    public static function post_notices()
474
-    {
475
-        EED_Ticket_Sales_Monitor::instance()->_post_notices();
476
-    }
477
-
478
-
479
-    /**
480
-     * @return void
481
-     * @throws EE_Error
482
-     * @throws InvalidArgumentException
483
-     * @throws ReflectionException
484
-     * @throws InvalidDataTypeException
485
-     * @throws InvalidInterfaceException
486
-     */
487
-    protected function _post_notices()
488
-    {
489
-        if (self::debug) {
490
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
491
-        }
492
-        $refresh_msg = '';
493
-        $none_added_msg = '';
494
-        if (defined('DOING_AJAX') && DOING_AJAX) {
495
-            $refresh_msg = __(
496
-                'Please refresh the page to view updated ticket quantities.',
497
-                'event_espresso'
498
-            );
499
-            $none_added_msg = __('No tickets were added for the event.', 'event_espresso');
500
-        }
501
-        if (! empty($this->sold_out_tickets)) {
502
-            EE_Error::add_attention(
503
-                sprintf(
504
-                    apply_filters(
505
-                        'FHEE__EED_Ticket_Sales_Monitor___post_notices__sold_out_tickets_notice',
506
-                        __(
507
-                            'We\'re sorry...%1$sThe following items have sold out since you first viewed this page, and can no longer be registered for:%1$s%1$s%2$s%1$s%1$sPlease note that availability can change at any time due to cancellations, so please check back again later if registration for this event(s) is important to you.%1$s%1$s%3$s%1$s%4$s%1$s',
508
-                            'event_espresso'
509
-                        )
510
-                    ),
511
-                    '<br />',
512
-                    implode('<br />', $this->sold_out_tickets),
513
-                    $none_added_msg,
514
-                    $refresh_msg
515
-                )
516
-            );
517
-            // alter code flow in the Ticket Selector for better UX
518
-            add_filter('FHEE__EED_Ticket_Selector__process_ticket_selections__tckts_slctd', '__return_true');
519
-            add_filter('FHEE__EED_Ticket_Selector__process_ticket_selections__success', '__return_false');
520
-            $this->sold_out_tickets = array();
521
-            // and reset the cart
522
-            EED_Ticket_Sales_Monitor::session_cart_reset(EE_Registry::instance()->SSN);
523
-        }
524
-        if (! empty($this->decremented_tickets)) {
525
-            EE_Error::add_attention(
526
-                sprintf(
527
-                    apply_filters(
528
-                        'FHEE__EED_Ticket_Sales_Monitor___ticket_quantity_decremented__notice',
529
-                        __(
530
-                            'We\'re sorry...%1$sDue to sales that have occurred since you first viewed the last page, the following items have had their quantities adjusted to match the current available amount:%1$s%1$s%2$s%1$s%1$sPlease note that availability can change at any time due to cancellations, so please check back again later if registration for this event(s) is important to you.%1$s%1$s%3$s%1$s%4$s%1$s',
531
-                            'event_espresso'
532
-                        )
533
-                    ),
534
-                    '<br />',
535
-                    implode('<br />', $this->decremented_tickets),
536
-                    $none_added_msg,
537
-                    $refresh_msg
538
-                )
539
-            );
540
-            $this->decremented_tickets = array();
541
-        }
542
-    }
543
-
544
-
545
-
546
-    /********************************** RELEASE_ALL_RESERVED_TICKETS_FOR_TRANSACTION  **********************************/
547
-
548
-
549
-    /**
550
-     * releases reserved tickets for all registrations of an EE_Transaction
551
-     * by default, will NOT release tickets for finalized transactions
552
-     *
553
-     * @param    EE_Transaction $transaction
554
-     * @return int
555
-     * @throws EE_Error
556
-     * @throws InvalidSessionDataException
557
-     */
558
-    protected function _release_all_reserved_tickets_for_transaction(EE_Transaction $transaction)
559
-    {
560
-        if (self::debug) {
561
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
562
-            echo self::$nl . ' . transaction->ID: ' . $transaction->ID();
563
-            echo self::$nl . ' . TXN status_ID: ' . $transaction->status_ID();
564
-        }
565
-        // check if 'finalize_registration' step has been completed...
566
-        $finalized = $transaction->reg_step_completed('finalize_registration');
567
-        if (self::debug) {
568
-            // DEBUG LOG
569
-            EEH_Debug_Tools::log(
570
-                __CLASS__,
571
-                __FUNCTION__,
572
-                __LINE__,
573
-                array('finalized' => $finalized),
574
-                false,
575
-                'EE_Transaction: ' . $transaction->ID()
576
-            );
577
-        }
578
-        // how many tickets were released
579
-        $count = 0;
580
-        if (self::debug) {
581
-            echo self::$nl . ' . . . TXN finalized: ' . $finalized;
582
-        }
583
-        $release_tickets_with_TXN_status = array(
584
-            EEM_Transaction::failed_status_code,
585
-            EEM_Transaction::abandoned_status_code,
586
-            EEM_Transaction::incomplete_status_code,
587
-        );
588
-        $events = array();
589
-        // if the session is getting cleared BEFORE the TXN has been finalized or the transaction is not completed
590
-        if (! $finalized || in_array($transaction->status_ID(), $release_tickets_with_TXN_status, true)) {
591
-            // cancel any reserved tickets for registrations that were not approved
592
-            $registrations = $transaction->registrations();
593
-            if (self::debug) {
594
-                echo self::$nl . ' . . . # registrations: ' . count($registrations);
595
-                $reg = reset($registrations);
596
-                $ticket = $reg->ticket();
597
-                if ($ticket instanceof EE_Ticket) {
598
-                    $ticket->add_extra_meta(
599
-                        EE_Ticket::META_KEY_TICKET_RESERVATIONS,
600
-                        __LINE__ . ') Release All Tickets TXN:' . $transaction->ID()
601
-                    );
602
-                }
603
-            }
604
-            if (! empty($registrations)) {
605
-                foreach ($registrations as $registration) {
606
-                    if ($registration instanceof EE_Registration
607
-                        && $this->_release_reserved_ticket_for_registration($registration, $transaction)
608
-                    ) {
609
-                        $count++;
610
-                        $events[ $registration->event_ID() ] = $registration->event();
611
-                    }
612
-                }
613
-            }
614
-        }
615
-        if ($events !== array()) {
616
-            foreach ($events as $event) {
617
-                /** @var EE_Event $event */
618
-                $event->perform_sold_out_status_check();
619
-            }
620
-        }
621
-        return $count;
622
-    }
623
-
624
-
625
-    /**
626
-     * releases reserved tickets for an EE_Registration
627
-     * by default, will NOT release tickets for APPROVED registrations
628
-     *
629
-     * @param EE_Registration $registration
630
-     * @param EE_Transaction  $transaction
631
-     * @return int
632
-     * @throws EE_Error
633
-     */
634
-    protected function _release_reserved_ticket_for_registration(
635
-        EE_Registration $registration,
636
-        EE_Transaction $transaction
637
-    ) {
638
-        $STS_ID = $transaction->status_ID();
639
-        if (self::debug) {
640
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
641
-            echo self::$nl . ' . . registration->ID: ' . $registration->ID();
642
-            echo self::$nl . ' . . registration->status_ID: ' . $registration->status_ID();
643
-            echo self::$nl . ' . . transaction->status_ID(): ' . $STS_ID;
644
-        }
645
-        if (// release Tickets for Failed Transactions and Abandoned Transactions
646
-            $STS_ID === EEM_Transaction::failed_status_code
647
-            || $STS_ID === EEM_Transaction::abandoned_status_code
648
-            || (
649
-                // also release Tickets for Incomplete Transactions, but ONLY if the Registrations are NOT Approved
650
-                $STS_ID === EEM_Transaction::incomplete_status_code
651
-                && $registration->status_ID() !== EEM_Registration::status_id_approved
652
-            )
653
-        ) {
654
-            if (self::debug) {
655
-                echo self::$nl . self::$nl . ' . . RELEASE RESERVED TICKET';
656
-                $rsrvd = $registration->get_extra_meta(EE_Registration::HAS_RESERVED_TICKET_KEY, true);
657
-                echo self::$nl . ' . . . registration HAS_RESERVED_TICKET_KEY: ';
658
-                var_dump($rsrvd);
659
-            }
660
-            $registration->release_reserved_ticket(true, 'TicketSalesMonitor:' . __LINE__);
661
-            return 1;
662
-        }
663
-        return 0;
664
-    }
665
-
666
-
667
-
668
-    /********************************** SESSION_CART_RESET  **********************************/
669
-
670
-
671
-    /**
672
-     * callback hooked into 'AHEE__EE_Session__reset_cart__before_reset'
673
-     *
674
-     * @param EE_Session $session
675
-     * @return void
676
-     * @throws EE_Error
677
-     * @throws InvalidArgumentException
678
-     * @throws ReflectionException
679
-     * @throws InvalidDataTypeException
680
-     * @throws InvalidInterfaceException
681
-     */
682
-    public static function session_cart_reset(EE_Session $session)
683
-    {
684
-        // don't release tickets if checkout was already reset
685
-        if (did_action('AHEE__EE_Session__reset_checkout__before_reset')) {
686
-            return;
687
-        }
688
-        if (self::debug) {
689
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
690
-        }
691
-        // first check of the session has a valid Checkout object
692
-        $checkout = $session->checkout();
693
-        if ($checkout instanceof EE_Checkout) {
694
-            // and use that to clear ticket reservations because it will update the associated registration meta data
695
-            EED_Ticket_Sales_Monitor::instance()->_session_checkout_reset($checkout);
696
-            return;
697
-        }
698
-        $cart = $session->cart();
699
-        if ($cart instanceof EE_Cart) {
700
-            if (self::debug) {
701
-                echo self::$nl . self::$nl . ' cart instance of EE_Cart: ';
702
-            }
703
-            EED_Ticket_Sales_Monitor::instance()->_session_cart_reset($cart, $session);
704
-        } else {
705
-            if (self::debug) {
706
-                echo self::$nl . self::$nl . ' invalid EE_Cart: ';
707
-                var_export($cart, true);
708
-            }
709
-        }
710
-    }
711
-
712
-
713
-    /**
714
-     * releases reserved tickets in the EE_Cart
715
-     *
716
-     * @param EE_Cart $cart
717
-     * @return void
718
-     * @throws EE_Error
719
-     * @throws InvalidArgumentException
720
-     * @throws ReflectionException
721
-     * @throws InvalidDataTypeException
722
-     * @throws InvalidInterfaceException
723
-     */
724
-    protected function _session_cart_reset(EE_Cart $cart, EE_Session $session)
725
-    {
726
-        if (self::debug) {
727
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
728
-        }
729
-        $ticket_line_items = $cart->get_tickets();
730
-        if (empty($ticket_line_items)) {
731
-            return;
732
-        }
733
-        if (self::debug) {
734
-            echo '<br /> . ticket_line_item count: ' . count($ticket_line_items);
735
-        }
736
-        foreach ($ticket_line_items as $ticket_line_item) {
737
-            if (self::debug) {
738
-                echo self::$nl . ' . ticket_line_item->ID(): ' . $ticket_line_item->ID();
739
-            }
740
-            if ($ticket_line_item instanceof EE_Line_Item && $ticket_line_item->OBJ_type() === 'Ticket') {
741
-                if (self::debug) {
742
-                    echo self::$nl . ' . . ticket_line_item->OBJ_ID(): ' . $ticket_line_item->OBJ_ID();
743
-                }
744
-                $ticket = EEM_Ticket::instance()->get_one_by_ID($ticket_line_item->OBJ_ID());
745
-                if ($ticket instanceof EE_Ticket) {
746
-                    if (self::debug) {
747
-                        echo self::$nl . ' . . ticket->ID(): ' . $ticket->ID();
748
-                        echo self::$nl . ' . . ticket_line_item->quantity(): ' . $ticket_line_item->quantity();
749
-                    }
750
-                    $ticket->add_extra_meta(
751
-                        EE_Ticket::META_KEY_TICKET_RESERVATIONS,
752
-                        __LINE__ . ') ' . __METHOD__ . '() SID = ' . $session->id()
753
-                    );
754
-                    $this->_release_reserved_ticket($ticket, $ticket_line_item->quantity());
755
-                }
756
-            }
757
-        }
758
-        if (self::debug) {
759
-            echo self::$nl . self::$nl . ' RESET COMPLETED ';
760
-        }
761
-    }
762
-
763
-
764
-
765
-    /********************************** SESSION_CHECKOUT_RESET  **********************************/
766
-
767
-
768
-    /**
769
-     * callback hooked into 'AHEE__EE_Session__reset_checkout__before_reset'
770
-     *
771
-     * @param EE_Session $session
772
-     * @return void
773
-     * @throws EE_Error
774
-     * @throws InvalidSessionDataException
775
-     */
776
-    public static function session_checkout_reset(EE_Session $session)
777
-    {
778
-        // don't release tickets if cart was already reset
779
-        if (did_action('AHEE__EE_Session__reset_cart__before_reset')) {
780
-            return;
781
-        }
782
-        $checkout = $session->checkout();
783
-        if ($checkout instanceof EE_Checkout) {
784
-            EED_Ticket_Sales_Monitor::instance()->_session_checkout_reset($checkout);
785
-        }
786
-    }
787
-
788
-
789
-    /**
790
-     * releases reserved tickets for the EE_Checkout->transaction
791
-     *
792
-     * @param EE_Checkout $checkout
793
-     * @return void
794
-     * @throws EE_Error
795
-     * @throws InvalidSessionDataException
796
-     */
797
-    protected function _session_checkout_reset(EE_Checkout $checkout)
798
-    {
799
-        if (self::debug) {
800
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
801
-        }
802
-        // we want to release the each registration's reserved tickets if the session was cleared, but not if this is a revisit
803
-        if ($checkout->revisit || ! $checkout->transaction instanceof EE_Transaction) {
804
-            return;
805
-        }
806
-        $this->_release_all_reserved_tickets_for_transaction($checkout->transaction);
807
-    }
808
-
809
-
810
-
811
-    /********************************** SESSION_EXPIRED_RESET  **********************************/
812
-
813
-
814
-    /**
815
-     * @param    EE_Session $session
816
-     * @return    void
817
-     */
818
-    public static function session_expired_reset(EE_Session $session)
819
-    {
820
-    }
821
-
822
-
823
-
824
-    /********************************** PROCESS_ABANDONED_TRANSACTIONS  **********************************/
825
-
826
-
827
-    /**
828
-     * releases reserved tickets for all registrations of an ABANDONED EE_Transaction
829
-     * by default, will NOT release tickets for free transactions, or any that have received a payment
830
-     *
831
-     * @param EE_Transaction $transaction
832
-     * @return void
833
-     * @throws EE_Error
834
-     * @throws InvalidSessionDataException
835
-     */
836
-    public static function process_abandoned_transactions(EE_Transaction $transaction)
837
-    {
838
-        // is this TXN free or has any money been paid towards this TXN? If so, then leave it alone
839
-        if ($transaction->is_free() || $transaction->paid() > 0) {
840
-            if (self::debug) {
841
-                // DEBUG LOG
842
-                EEH_Debug_Tools::log(
843
-                    __CLASS__,
844
-                    __FUNCTION__,
845
-                    __LINE__,
846
-                    array($transaction),
847
-                    false,
848
-                    'EE_Transaction: ' . $transaction->ID()
849
-                );
850
-            }
851
-            return;
852
-        }
853
-        // have their been any successful payments made ?
854
-        $payments = $transaction->payments();
855
-        foreach ($payments as $payment) {
856
-            if ($payment instanceof EE_Payment && $payment->status() === EEM_Payment::status_id_approved) {
857
-                if (self::debug) {
858
-                    // DEBUG LOG
859
-                    EEH_Debug_Tools::log(
860
-                        __CLASS__,
861
-                        __FUNCTION__,
862
-                        __LINE__,
863
-                        array($payment),
864
-                        false,
865
-                        'EE_Transaction: ' . $transaction->ID()
866
-                    );
867
-                }
868
-                return;
869
-            }
870
-        }
871
-        // since you haven't even attempted to pay for your ticket...
872
-        EED_Ticket_Sales_Monitor::instance()->_release_all_reserved_tickets_for_transaction($transaction);
873
-    }
874
-
875
-
876
-
877
-    /********************************** PROCESS_FAILED_TRANSACTIONS  **********************************/
878
-
879
-
880
-    /**
881
-     * releases reserved tickets for absolutely ALL registrations of a FAILED EE_Transaction
882
-     *
883
-     * @param EE_Transaction $transaction
884
-     * @return void
885
-     * @throws EE_Error
886
-     * @throws InvalidSessionDataException
887
-     */
888
-    public static function process_failed_transactions(EE_Transaction $transaction)
889
-    {
890
-        // since you haven't even attempted to pay for your ticket...
891
-        EED_Ticket_Sales_Monitor::instance()->_release_all_reserved_tickets_for_transaction($transaction);
892
-    }
893
-
894
-
895
-
896
-    /********************************** RESET RESERVATION COUNTS  *********************************/
897
-
898
-
899
-    /**
900
-     * Resets all ticket and datetime reserved counts to zero
901
-     * Tickets that are currently associated with a Transaction that is in progress
902
-     *
903
-     * @throws EE_Error
904
-     * @throws DomainException
905
-     * @throws InvalidDataTypeException
906
-     * @throws InvalidInterfaceException
907
-     * @throws InvalidArgumentException
908
-     * @throws UnexpectedEntityException
909
-     */
910
-    public static function reset_reservation_counts()
911
-    {
912
-        /** @var EE_Line_Item[] $valid_reserved_tickets */
913
-        $valid_reserved_tickets = array();
914
-        /** @var EE_Transaction[] $transactions_not_in_progress */
915
-        $transactions_not_in_progress = EEM_Transaction::instance()->get_transactions_not_in_progress();
916
-        foreach ($transactions_not_in_progress as $transaction) {
917
-            // if this TXN has been fully completed, then skip it
918
-            if ($transaction->reg_step_completed('finalize_registration')) {
919
-                continue;
920
-            }
921
-            $total_line_item = $transaction->total_line_item();
922
-            // $transaction_in_progress->line
923
-            if (! $total_line_item instanceof EE_Line_Item) {
924
-                throw new DomainException(
925
-                    esc_html__(
926
-                        'Transaction does not have a valid Total Line Item associated with it.',
927
-                        'event_espresso'
928
-                    )
929
-                );
930
-            }
931
-            $valid_reserved_tickets += EED_Ticket_Sales_Monitor::get_ticket_line_items_for_grand_total(
932
-                $total_line_item
933
-            );
934
-        }
935
-        $total_line_items = EEM_Line_Item::instance()->get_total_line_items_for_active_carts();
936
-        foreach ($total_line_items as $total_line_item) {
937
-            $valid_reserved_tickets += EED_Ticket_Sales_Monitor::get_ticket_line_items_for_grand_total(
938
-                $total_line_item
939
-            );
940
-        }
941
-        $tickets_with_reservations = EEM_Ticket::instance()->get_tickets_with_reservations();
942
-        return EED_Ticket_Sales_Monitor::release_reservations_for_tickets(
943
-            $tickets_with_reservations,
944
-            $valid_reserved_tickets,
945
-            __FUNCTION__
946
-        );
947
-    }
948
-
949
-
950
-    /**
951
-     * @param EE_Line_Item $total_line_item
952
-     * @return EE_Line_Item[]
953
-     */
954
-    private static function get_ticket_line_items_for_grand_total(EE_Line_Item $total_line_item)
955
-    {
956
-        /** @var EE_Line_Item[] $valid_reserved_tickets */
957
-        $valid_reserved_tickets = array();
958
-        $ticket_line_items = EEH_Line_Item::get_ticket_line_items($total_line_item);
959
-        foreach ($ticket_line_items as $ticket_line_item) {
960
-            if ($ticket_line_item instanceof EE_Line_Item) {
961
-                $valid_reserved_tickets[] = $ticket_line_item;
962
-            }
963
-        }
964
-        return $valid_reserved_tickets;
965
-    }
966
-
967
-
968
-    /**
969
-     * @param EE_Ticket[]    $tickets_with_reservations
970
-     * @param EE_Line_Item[] $valid_reserved_ticket_line_items
971
-     * @return int
972
-     * @throws UnexpectedEntityException
973
-     * @throws DomainException
974
-     * @throws EE_Error
975
-     */
976
-    private static function release_reservations_for_tickets(
977
-        array $tickets_with_reservations,
978
-        array $valid_reserved_ticket_line_items = array(),
979
-        $source
980
-    ) {
981
-        if (self::debug) {
982
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '()';
983
-        }
984
-        $total_tickets_released = 0;
985
-        $sold_out_events = array();
986
-        foreach ($tickets_with_reservations as $ticket_with_reservations) {
987
-            if (! $ticket_with_reservations instanceof EE_Ticket) {
988
-                continue;
989
-            }
990
-            $reserved_qty = $ticket_with_reservations->reserved();
991
-            if (self::debug) {
992
-                echo self::$nl . ' . $ticket_with_reservations->ID(): ' . $ticket_with_reservations->ID();
993
-                echo self::$nl . ' . $reserved_qty: ' . $reserved_qty;
994
-            }
995
-            foreach ($valid_reserved_ticket_line_items as $valid_reserved_ticket_line_item) {
996
-                if ($valid_reserved_ticket_line_item instanceof EE_Line_Item
997
-                    && $valid_reserved_ticket_line_item->OBJ_ID() === $ticket_with_reservations->ID()
998
-                ) {
999
-                    if (self::debug) {
1000
-                        echo self::$nl . ' . $valid_reserved_ticket_line_item->quantity(): '
1001
-                             . $valid_reserved_ticket_line_item->quantity();
1002
-                    }
1003
-                    $reserved_qty -= $valid_reserved_ticket_line_item->quantity();
1004
-                }
1005
-            }
1006
-            if ($reserved_qty > 0) {
1007
-                $ticket_with_reservations->add_extra_meta(
1008
-                    EE_Ticket::META_KEY_TICKET_RESERVATIONS,
1009
-                    __LINE__ . ') ' . $source . '()'
1010
-                );
1011
-                $ticket_with_reservations->decrease_reserved($reserved_qty, true, 'TicketSalesMonitor:' . __LINE__);
1012
-                $ticket_with_reservations->save();
1013
-                $total_tickets_released += $reserved_qty;
1014
-                $event = $ticket_with_reservations->get_related_event();
1015
-                // track sold out events
1016
-                if ($event instanceof EE_Event && $event->is_sold_out()) {
1017
-                    $sold_out_events[] = $event;
1018
-                }
1019
-            }
1020
-        }
1021
-        if (self::debug) {
1022
-            echo self::$nl . ' . $total_tickets_released: ' . $total_tickets_released;
1023
-        }
1024
-        // double check whether sold out events should remain sold out after releasing tickets
1025
-        if ($sold_out_events !== array()) {
1026
-            foreach ($sold_out_events as $sold_out_event) {
1027
-                /** @var EE_Event $sold_out_event */
1028
-                $sold_out_event->perform_sold_out_status_check();
1029
-            }
1030
-        }
1031
-        return $total_tickets_released;
1032
-    }
1033
-
1034
-
1035
-
1036
-    /********************************** SHUTDOWN  **********************************/
1037
-
1038
-
1039
-    /**
1040
-     * @param int $timestamp
1041
-     * @return false|int
1042
-     * @throws EE_Error
1043
-     * @throws InvalidArgumentException
1044
-     * @throws InvalidDataTypeException
1045
-     * @throws InvalidInterfaceException
1046
-     */
1047
-    public static function clear_expired_line_items_with_no_transaction($timestamp = 0)
1048
-    {
1049
-        /** @type WPDB $wpdb */
1050
-        global $wpdb;
1051
-        if (! absint($timestamp)) {
1052
-            /** @var EventEspresso\core\domain\values\session\SessionLifespan $session_lifespan */
1053
-            $session_lifespan = LoaderFactory::getLoader()->getShared(
1054
-                'EventEspresso\core\domain\values\session\SessionLifespan'
1055
-            );
1056
-            $timestamp = $session_lifespan->expiration();
1057
-        }
1058
-        return $wpdb->query(
1059
-            $wpdb->prepare(
1060
-                'DELETE FROM ' . EEM_Line_Item::instance()->table() . '
23
+	const debug = false;
24
+
25
+	private static $nl = '';
26
+
27
+	/**
28
+	 * an array of raw ticket data from EED_Ticket_Selector
29
+	 *
30
+	 * @var array $ticket_selections
31
+	 */
32
+	protected $ticket_selections = array();
33
+
34
+	/**
35
+	 * the raw ticket data from EED_Ticket_Selector is organized in rows
36
+	 * according to how they are displayed in the actual Ticket_Selector
37
+	 * this tracks the current row being processed
38
+	 *
39
+	 * @var int $current_row
40
+	 */
41
+	protected $current_row = 0;
42
+
43
+	/**
44
+	 * an array for tracking names of tickets that have sold out
45
+	 *
46
+	 * @var array $sold_out_tickets
47
+	 */
48
+	protected $sold_out_tickets = array();
49
+
50
+	/**
51
+	 * an array for tracking names of tickets that have had their quantities reduced
52
+	 *
53
+	 * @var array $decremented_tickets
54
+	 */
55
+	protected $decremented_tickets = array();
56
+
57
+
58
+	/**
59
+	 * set_hooks - for hooking into EE Core, other modules, etc
60
+	 *
61
+	 * @return    void
62
+	 */
63
+	public static function set_hooks()
64
+	{
65
+		self::$nl = defined('EE_TESTS_DIR') ? "\n" : '<br />';
66
+		// release tickets for expired carts
67
+		add_action(
68
+			'EED_Ticket_Selector__process_ticket_selections__before',
69
+			array('EED_Ticket_Sales_Monitor', 'release_tickets_for_expired_carts'),
70
+			1
71
+		);
72
+		// check ticket reserves AFTER MER does it's check (hence priority 20)
73
+		add_filter(
74
+			'FHEE__EE_Ticket_Selector___add_ticket_to_cart__ticket_qty',
75
+			array('EED_Ticket_Sales_Monitor', 'validate_ticket_sale'),
76
+			20,
77
+			3
78
+		);
79
+		// add notices for sold out tickets
80
+		add_action(
81
+			'AHEE__EE_Ticket_Selector__process_ticket_selections__after_tickets_added_to_cart',
82
+			array('EED_Ticket_Sales_Monitor', 'post_notices'),
83
+			10
84
+		);
85
+
86
+		// handle tickets deleted from cart
87
+		add_action(
88
+			'FHEE__EED_Multi_Event_Registration__delete_ticket__ticket_removed_from_cart',
89
+			array('EED_Ticket_Sales_Monitor', 'ticket_removed_from_cart'),
90
+			10,
91
+			2
92
+		);
93
+		// handle emptied carts
94
+		add_action(
95
+			'AHEE__EE_Session__reset_cart__before_reset',
96
+			array('EED_Ticket_Sales_Monitor', 'session_cart_reset'),
97
+			10,
98
+			1
99
+		);
100
+		add_action(
101
+			'AHEE__EED_Multi_Event_Registration__empty_event_cart__before_delete_cart',
102
+			array('EED_Ticket_Sales_Monitor', 'session_cart_reset'),
103
+			10,
104
+			1
105
+		);
106
+		// handle cancelled registrations
107
+		add_action(
108
+			'AHEE__EE_Session__reset_checkout__before_reset',
109
+			array('EED_Ticket_Sales_Monitor', 'session_checkout_reset'),
110
+			10,
111
+			1
112
+		);
113
+		// cron tasks
114
+		add_action(
115
+			'AHEE__EE_Cron_Tasks__process_expired_transactions__abandoned_transaction',
116
+			array('EED_Ticket_Sales_Monitor', 'process_abandoned_transactions'),
117
+			10,
118
+			1
119
+		);
120
+		add_action(
121
+			'AHEE__EE_Cron_Tasks__process_expired_transactions__incomplete_transaction',
122
+			array('EED_Ticket_Sales_Monitor', 'process_abandoned_transactions'),
123
+			10,
124
+			1
125
+		);
126
+		add_action(
127
+			'AHEE__EE_Cron_Tasks__process_expired_transactions__failed_transaction',
128
+			array('EED_Ticket_Sales_Monitor', 'process_failed_transactions'),
129
+			10,
130
+			1
131
+		);
132
+	}
133
+
134
+
135
+	/**
136
+	 * set_hooks_admin - for hooking into EE Admin Core, other modules, etc
137
+	 *
138
+	 * @return void
139
+	 */
140
+	public static function set_hooks_admin()
141
+	{
142
+		EED_Ticket_Sales_Monitor::set_hooks();
143
+	}
144
+
145
+
146
+	/**
147
+	 * @return EED_Ticket_Sales_Monitor|EED_Module
148
+	 */
149
+	public static function instance()
150
+	{
151
+		return parent::get_instance(__CLASS__);
152
+	}
153
+
154
+
155
+	/**
156
+	 * @param WP_Query $WP_Query
157
+	 * @return    void
158
+	 */
159
+	public function run($WP_Query)
160
+	{
161
+	}
162
+
163
+
164
+
165
+	/********************************** PRE_TICKET_SALES  **********************************/
166
+
167
+
168
+	/**
169
+	 * Retrieves grand totals from the line items that have no TXN ID
170
+	 * and timestamps less than the current time minus the session lifespan.
171
+	 * These are carts that have been abandoned before the "registrant" even attempted to checkout.
172
+	 * We're going to release the tickets for these line items before attempting to add more to the cart.
173
+	 *
174
+	 * @return void
175
+	 * @throws DomainException
176
+	 * @throws EE_Error
177
+	 * @throws InvalidArgumentException
178
+	 * @throws InvalidDataTypeException
179
+	 * @throws InvalidInterfaceException
180
+	 * @throws UnexpectedEntityException
181
+	 */
182
+	public static function release_tickets_for_expired_carts()
183
+	{
184
+		if (self::debug) {
185
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '()';
186
+		}
187
+		do_action('AHEE__EED_Ticket_Sales_Monitor__release_tickets_for_expired_carts__begin');
188
+		$expired_ticket_IDs = array();
189
+		/** @var EventEspresso\core\domain\values\session\SessionLifespan $session_lifespan */
190
+		$session_lifespan = LoaderFactory::getLoader()->getShared(
191
+			'EventEspresso\core\domain\values\session\SessionLifespan'
192
+		);
193
+		$timestamp = $session_lifespan->expiration();
194
+		$expired_ticket_line_items = EEM_Line_Item::instance()->getTicketLineItemsForExpiredCarts($timestamp);
195
+		if (self::debug) {
196
+			echo self::$nl . ' . time(): ' . time();
197
+			echo self::$nl . ' . time() as date: ' . date('Y-m-d H:i a');
198
+			echo self::$nl . ' . session expiration: ' . $session_lifespan->expiration();
199
+			echo self::$nl . ' . session expiration as date: ' . date('Y-m-d H:i a', $session_lifespan->expiration());
200
+			echo self::$nl . ' . timestamp: ' . $timestamp;
201
+			echo self::$nl . ' . $expired_ticket_line_items: ' . count($expired_ticket_line_items);
202
+		}
203
+		if (! empty($expired_ticket_line_items)) {
204
+			foreach ($expired_ticket_line_items as $expired_ticket_line_item) {
205
+				if (! $expired_ticket_line_item instanceof EE_Line_Item) {
206
+					continue;
207
+				}
208
+				$expired_ticket_IDs[ $expired_ticket_line_item->OBJ_ID() ] = $expired_ticket_line_item->OBJ_ID();
209
+				if (self::debug) {
210
+					echo self::$nl . ' . $expired_ticket_line_item->OBJ_ID(): ' . $expired_ticket_line_item->OBJ_ID();
211
+					echo self::$nl . ' . $expired_ticket_line_item->timestamp(): '
212
+						 . date(
213
+							 'Y-m-d h:i a',
214
+							 $expired_ticket_line_item->timestamp(true)
215
+						 );
216
+				}
217
+			}
218
+			if (! empty($expired_ticket_IDs)) {
219
+				EED_Ticket_Sales_Monitor::release_reservations_for_tickets(
220
+					\EEM_Ticket::instance()->get_tickets_with_IDs($expired_ticket_IDs),
221
+					array(),
222
+					__FUNCTION__
223
+				);
224
+				// now  let's get rid of expired line items so that they can't interfere with tracking
225
+				EED_Ticket_Sales_Monitor::clear_expired_line_items_with_no_transaction($timestamp);
226
+			}
227
+		}
228
+		do_action(
229
+			'AHEE__EED_Ticket_Sales_Monitor__release_tickets_for_expired_carts__end',
230
+			$expired_ticket_IDs,
231
+			$expired_ticket_line_items
232
+		);
233
+	}
234
+
235
+
236
+
237
+	/********************************** VALIDATE_TICKET_SALE  **********************************/
238
+
239
+
240
+	/**
241
+	 * callback for 'FHEE__EED_Ticket_Selector__process_ticket_selections__valid_post_data'
242
+	 *
243
+	 * @param int       $qty
244
+	 * @param EE_Ticket $ticket
245
+	 * @return bool
246
+	 * @throws UnexpectedEntityException
247
+	 * @throws EE_Error
248
+	 */
249
+	public static function validate_ticket_sale($qty = 1, EE_Ticket $ticket)
250
+	{
251
+		$qty = absint($qty);
252
+		if ($qty > 0) {
253
+			$qty = EED_Ticket_Sales_Monitor::instance()->_validate_ticket_sale($ticket, $qty);
254
+		}
255
+		if (self::debug) {
256
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '()';
257
+			echo self::$nl . self::$nl . '<b> RETURNED QTY: ' . $qty . '</b>';
258
+		}
259
+		return $qty;
260
+	}
261
+
262
+
263
+	/**
264
+	 * checks whether an individual ticket is available for purchase based on datetime, and ticket details
265
+	 *
266
+	 * @param   EE_Ticket $ticket
267
+	 * @param int         $qty
268
+	 * @return int
269
+	 * @throws UnexpectedEntityException
270
+	 * @throws EE_Error
271
+	 */
272
+	protected function _validate_ticket_sale(EE_Ticket $ticket, $qty = 1)
273
+	{
274
+		if (self::debug) {
275
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
276
+		}
277
+		if (! $ticket instanceof EE_Ticket) {
278
+			return 0;
279
+		}
280
+		if (self::debug) {
281
+			echo self::$nl . '<b> . ticket->ID: ' . $ticket->ID() . '</b>';
282
+			echo self::$nl . ' . original ticket->reserved: ' . $ticket->reserved();
283
+		}
284
+		$ticket->refresh_from_db();
285
+		// first let's determine the ticket availability based on sales
286
+		$available = $ticket->qty('saleable');
287
+		if (self::debug) {
288
+			echo self::$nl . ' . . . ticket->qty: ' . $ticket->qty();
289
+			echo self::$nl . ' . . . ticket->sold: ' . $ticket->sold();
290
+			echo self::$nl . ' . . . ticket->reserved: ' . $ticket->reserved();
291
+			echo self::$nl . ' . . . ticket->qty(saleable): ' . $ticket->qty('saleable');
292
+			echo self::$nl . ' . . . available: ' . $available;
293
+		}
294
+		if ($available < 1) {
295
+			$this->_ticket_sold_out($ticket);
296
+			return 0;
297
+		}
298
+		if (self::debug) {
299
+			echo self::$nl . ' . . . qty: ' . $qty;
300
+		}
301
+		if ($available < $qty) {
302
+			$qty = $available;
303
+			if (self::debug) {
304
+				echo self::$nl . ' . . . QTY ADJUSTED: ' . $qty;
305
+			}
306
+			$this->_ticket_quantity_decremented($ticket);
307
+		}
308
+		$this->_reserve_ticket($ticket, $qty);
309
+		return $qty;
310
+	}
311
+
312
+
313
+	/**
314
+	 * increments ticket reserved based on quantity passed
315
+	 *
316
+	 * @param    EE_Ticket $ticket
317
+	 * @param int          $quantity
318
+	 * @return bool
319
+	 * @throws EE_Error
320
+	 */
321
+	protected function _reserve_ticket(EE_Ticket $ticket, $quantity = 1)
322
+	{
323
+		if (self::debug) {
324
+			echo self::$nl . self::$nl . ' . . . INCREASE RESERVED: ' . $quantity;
325
+		}
326
+		$ticket->increase_reserved($quantity, 'TicketSalesMonitor:' . __LINE__);
327
+		return $ticket->save();
328
+	}
329
+
330
+
331
+	/**
332
+	 * @param  EE_Ticket $ticket
333
+	 * @param  int       $quantity
334
+	 * @return bool
335
+	 * @throws EE_Error
336
+	 */
337
+	protected function _release_reserved_ticket(EE_Ticket $ticket, $quantity = 1)
338
+	{
339
+		if (self::debug) {
340
+			echo self::$nl . ' . . . ticket->ID: ' . $ticket->ID();
341
+			echo self::$nl . ' . . . ticket->reserved before: ' . $ticket->reserved();
342
+		}
343
+		$ticket->decrease_reserved($quantity, true, 'TicketSalesMonitor:' . __LINE__);
344
+		if (self::debug) {
345
+			echo self::$nl . ' . . . ticket->reserved after: ' . $ticket->reserved();
346
+		}
347
+		return $ticket->save() ? 1 : 0;
348
+	}
349
+
350
+
351
+	/**
352
+	 * removes quantities within the ticket selector based on zero ticket availability
353
+	 *
354
+	 * @param    EE_Ticket $ticket
355
+	 * @return    void
356
+	 * @throws UnexpectedEntityException
357
+	 * @throws EE_Error
358
+	 */
359
+	protected function _ticket_sold_out(EE_Ticket $ticket)
360
+	{
361
+		if (self::debug) {
362
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
363
+			echo self::$nl . ' . . ticket->name: ' . $this->_get_ticket_and_event_name($ticket);
364
+		}
365
+		$this->sold_out_tickets[] = $this->_get_ticket_and_event_name($ticket);
366
+	}
367
+
368
+
369
+	/**
370
+	 * adjusts quantities within the ticket selector based on decreased ticket availability
371
+	 *
372
+	 * @param    EE_Ticket $ticket
373
+	 * @return void
374
+	 * @throws UnexpectedEntityException
375
+	 * @throws EE_Error
376
+	 */
377
+	protected function _ticket_quantity_decremented(EE_Ticket $ticket)
378
+	{
379
+		if (self::debug) {
380
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
381
+			echo self::$nl . ' . . ticket->name: ' . $this->_get_ticket_and_event_name($ticket);
382
+		}
383
+		$this->decremented_tickets[] = $this->_get_ticket_and_event_name($ticket);
384
+	}
385
+
386
+
387
+	/**
388
+	 * builds string out of ticket and event name
389
+	 *
390
+	 * @param    EE_Ticket $ticket
391
+	 * @return string
392
+	 * @throws UnexpectedEntityException
393
+	 * @throws EE_Error
394
+	 */
395
+	protected function _get_ticket_and_event_name(EE_Ticket $ticket)
396
+	{
397
+		$event = $ticket->get_related_event();
398
+		if ($event instanceof EE_Event) {
399
+			$ticket_name = sprintf(
400
+				_x('%1$s for %2$s', 'ticket name for event name', 'event_espresso'),
401
+				$ticket->name(),
402
+				$event->name()
403
+			);
404
+		} else {
405
+			$ticket_name = $ticket->name();
406
+		}
407
+		return $ticket_name;
408
+	}
409
+
410
+
411
+
412
+	/********************************** EVENT CART  **********************************/
413
+
414
+
415
+	/**
416
+	 * releases or reserves ticket(s) based on quantity passed
417
+	 *
418
+	 * @param  EE_Line_Item $line_item
419
+	 * @param  int          $quantity
420
+	 * @return void
421
+	 * @throws EE_Error
422
+	 * @throws InvalidArgumentException
423
+	 * @throws InvalidDataTypeException
424
+	 * @throws InvalidInterfaceException
425
+	 */
426
+	public static function ticket_quantity_updated(EE_Line_Item $line_item, $quantity = 1)
427
+	{
428
+		$ticket = EEM_Ticket::instance()->get_one_by_ID(absint($line_item->OBJ_ID()));
429
+		if ($ticket instanceof EE_Ticket) {
430
+			$ticket->add_extra_meta(
431
+				EE_Ticket::META_KEY_TICKET_RESERVATIONS,
432
+				__LINE__ . ') ' . __METHOD__ . '()'
433
+			);
434
+			if ($quantity > 0) {
435
+				EED_Ticket_Sales_Monitor::instance()->_reserve_ticket($ticket, $quantity);
436
+			} else {
437
+				EED_Ticket_Sales_Monitor::instance()->_release_reserved_ticket($ticket, $quantity);
438
+			}
439
+		}
440
+	}
441
+
442
+
443
+	/**
444
+	 * releases reserved ticket(s) based on quantity passed
445
+	 *
446
+	 * @param  EE_Ticket $ticket
447
+	 * @param  int       $quantity
448
+	 * @return void
449
+	 * @throws EE_Error
450
+	 */
451
+	public static function ticket_removed_from_cart(EE_Ticket $ticket, $quantity = 1)
452
+	{
453
+		$ticket->add_extra_meta(
454
+			EE_Ticket::META_KEY_TICKET_RESERVATIONS,
455
+			__LINE__ . ') ' . __METHOD__ . '()'
456
+		);
457
+		EED_Ticket_Sales_Monitor::instance()->_release_reserved_ticket($ticket, $quantity);
458
+	}
459
+
460
+
461
+
462
+	/********************************** POST_NOTICES  **********************************/
463
+
464
+
465
+	/**
466
+	 * @return void
467
+	 * @throws EE_Error
468
+	 * @throws InvalidArgumentException
469
+	 * @throws ReflectionException
470
+	 * @throws InvalidDataTypeException
471
+	 * @throws InvalidInterfaceException
472
+	 */
473
+	public static function post_notices()
474
+	{
475
+		EED_Ticket_Sales_Monitor::instance()->_post_notices();
476
+	}
477
+
478
+
479
+	/**
480
+	 * @return void
481
+	 * @throws EE_Error
482
+	 * @throws InvalidArgumentException
483
+	 * @throws ReflectionException
484
+	 * @throws InvalidDataTypeException
485
+	 * @throws InvalidInterfaceException
486
+	 */
487
+	protected function _post_notices()
488
+	{
489
+		if (self::debug) {
490
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
491
+		}
492
+		$refresh_msg = '';
493
+		$none_added_msg = '';
494
+		if (defined('DOING_AJAX') && DOING_AJAX) {
495
+			$refresh_msg = __(
496
+				'Please refresh the page to view updated ticket quantities.',
497
+				'event_espresso'
498
+			);
499
+			$none_added_msg = __('No tickets were added for the event.', 'event_espresso');
500
+		}
501
+		if (! empty($this->sold_out_tickets)) {
502
+			EE_Error::add_attention(
503
+				sprintf(
504
+					apply_filters(
505
+						'FHEE__EED_Ticket_Sales_Monitor___post_notices__sold_out_tickets_notice',
506
+						__(
507
+							'We\'re sorry...%1$sThe following items have sold out since you first viewed this page, and can no longer be registered for:%1$s%1$s%2$s%1$s%1$sPlease note that availability can change at any time due to cancellations, so please check back again later if registration for this event(s) is important to you.%1$s%1$s%3$s%1$s%4$s%1$s',
508
+							'event_espresso'
509
+						)
510
+					),
511
+					'<br />',
512
+					implode('<br />', $this->sold_out_tickets),
513
+					$none_added_msg,
514
+					$refresh_msg
515
+				)
516
+			);
517
+			// alter code flow in the Ticket Selector for better UX
518
+			add_filter('FHEE__EED_Ticket_Selector__process_ticket_selections__tckts_slctd', '__return_true');
519
+			add_filter('FHEE__EED_Ticket_Selector__process_ticket_selections__success', '__return_false');
520
+			$this->sold_out_tickets = array();
521
+			// and reset the cart
522
+			EED_Ticket_Sales_Monitor::session_cart_reset(EE_Registry::instance()->SSN);
523
+		}
524
+		if (! empty($this->decremented_tickets)) {
525
+			EE_Error::add_attention(
526
+				sprintf(
527
+					apply_filters(
528
+						'FHEE__EED_Ticket_Sales_Monitor___ticket_quantity_decremented__notice',
529
+						__(
530
+							'We\'re sorry...%1$sDue to sales that have occurred since you first viewed the last page, the following items have had their quantities adjusted to match the current available amount:%1$s%1$s%2$s%1$s%1$sPlease note that availability can change at any time due to cancellations, so please check back again later if registration for this event(s) is important to you.%1$s%1$s%3$s%1$s%4$s%1$s',
531
+							'event_espresso'
532
+						)
533
+					),
534
+					'<br />',
535
+					implode('<br />', $this->decremented_tickets),
536
+					$none_added_msg,
537
+					$refresh_msg
538
+				)
539
+			);
540
+			$this->decremented_tickets = array();
541
+		}
542
+	}
543
+
544
+
545
+
546
+	/********************************** RELEASE_ALL_RESERVED_TICKETS_FOR_TRANSACTION  **********************************/
547
+
548
+
549
+	/**
550
+	 * releases reserved tickets for all registrations of an EE_Transaction
551
+	 * by default, will NOT release tickets for finalized transactions
552
+	 *
553
+	 * @param    EE_Transaction $transaction
554
+	 * @return int
555
+	 * @throws EE_Error
556
+	 * @throws InvalidSessionDataException
557
+	 */
558
+	protected function _release_all_reserved_tickets_for_transaction(EE_Transaction $transaction)
559
+	{
560
+		if (self::debug) {
561
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
562
+			echo self::$nl . ' . transaction->ID: ' . $transaction->ID();
563
+			echo self::$nl . ' . TXN status_ID: ' . $transaction->status_ID();
564
+		}
565
+		// check if 'finalize_registration' step has been completed...
566
+		$finalized = $transaction->reg_step_completed('finalize_registration');
567
+		if (self::debug) {
568
+			// DEBUG LOG
569
+			EEH_Debug_Tools::log(
570
+				__CLASS__,
571
+				__FUNCTION__,
572
+				__LINE__,
573
+				array('finalized' => $finalized),
574
+				false,
575
+				'EE_Transaction: ' . $transaction->ID()
576
+			);
577
+		}
578
+		// how many tickets were released
579
+		$count = 0;
580
+		if (self::debug) {
581
+			echo self::$nl . ' . . . TXN finalized: ' . $finalized;
582
+		}
583
+		$release_tickets_with_TXN_status = array(
584
+			EEM_Transaction::failed_status_code,
585
+			EEM_Transaction::abandoned_status_code,
586
+			EEM_Transaction::incomplete_status_code,
587
+		);
588
+		$events = array();
589
+		// if the session is getting cleared BEFORE the TXN has been finalized or the transaction is not completed
590
+		if (! $finalized || in_array($transaction->status_ID(), $release_tickets_with_TXN_status, true)) {
591
+			// cancel any reserved tickets for registrations that were not approved
592
+			$registrations = $transaction->registrations();
593
+			if (self::debug) {
594
+				echo self::$nl . ' . . . # registrations: ' . count($registrations);
595
+				$reg = reset($registrations);
596
+				$ticket = $reg->ticket();
597
+				if ($ticket instanceof EE_Ticket) {
598
+					$ticket->add_extra_meta(
599
+						EE_Ticket::META_KEY_TICKET_RESERVATIONS,
600
+						__LINE__ . ') Release All Tickets TXN:' . $transaction->ID()
601
+					);
602
+				}
603
+			}
604
+			if (! empty($registrations)) {
605
+				foreach ($registrations as $registration) {
606
+					if ($registration instanceof EE_Registration
607
+						&& $this->_release_reserved_ticket_for_registration($registration, $transaction)
608
+					) {
609
+						$count++;
610
+						$events[ $registration->event_ID() ] = $registration->event();
611
+					}
612
+				}
613
+			}
614
+		}
615
+		if ($events !== array()) {
616
+			foreach ($events as $event) {
617
+				/** @var EE_Event $event */
618
+				$event->perform_sold_out_status_check();
619
+			}
620
+		}
621
+		return $count;
622
+	}
623
+
624
+
625
+	/**
626
+	 * releases reserved tickets for an EE_Registration
627
+	 * by default, will NOT release tickets for APPROVED registrations
628
+	 *
629
+	 * @param EE_Registration $registration
630
+	 * @param EE_Transaction  $transaction
631
+	 * @return int
632
+	 * @throws EE_Error
633
+	 */
634
+	protected function _release_reserved_ticket_for_registration(
635
+		EE_Registration $registration,
636
+		EE_Transaction $transaction
637
+	) {
638
+		$STS_ID = $transaction->status_ID();
639
+		if (self::debug) {
640
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
641
+			echo self::$nl . ' . . registration->ID: ' . $registration->ID();
642
+			echo self::$nl . ' . . registration->status_ID: ' . $registration->status_ID();
643
+			echo self::$nl . ' . . transaction->status_ID(): ' . $STS_ID;
644
+		}
645
+		if (// release Tickets for Failed Transactions and Abandoned Transactions
646
+			$STS_ID === EEM_Transaction::failed_status_code
647
+			|| $STS_ID === EEM_Transaction::abandoned_status_code
648
+			|| (
649
+				// also release Tickets for Incomplete Transactions, but ONLY if the Registrations are NOT Approved
650
+				$STS_ID === EEM_Transaction::incomplete_status_code
651
+				&& $registration->status_ID() !== EEM_Registration::status_id_approved
652
+			)
653
+		) {
654
+			if (self::debug) {
655
+				echo self::$nl . self::$nl . ' . . RELEASE RESERVED TICKET';
656
+				$rsrvd = $registration->get_extra_meta(EE_Registration::HAS_RESERVED_TICKET_KEY, true);
657
+				echo self::$nl . ' . . . registration HAS_RESERVED_TICKET_KEY: ';
658
+				var_dump($rsrvd);
659
+			}
660
+			$registration->release_reserved_ticket(true, 'TicketSalesMonitor:' . __LINE__);
661
+			return 1;
662
+		}
663
+		return 0;
664
+	}
665
+
666
+
667
+
668
+	/********************************** SESSION_CART_RESET  **********************************/
669
+
670
+
671
+	/**
672
+	 * callback hooked into 'AHEE__EE_Session__reset_cart__before_reset'
673
+	 *
674
+	 * @param EE_Session $session
675
+	 * @return void
676
+	 * @throws EE_Error
677
+	 * @throws InvalidArgumentException
678
+	 * @throws ReflectionException
679
+	 * @throws InvalidDataTypeException
680
+	 * @throws InvalidInterfaceException
681
+	 */
682
+	public static function session_cart_reset(EE_Session $session)
683
+	{
684
+		// don't release tickets if checkout was already reset
685
+		if (did_action('AHEE__EE_Session__reset_checkout__before_reset')) {
686
+			return;
687
+		}
688
+		if (self::debug) {
689
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
690
+		}
691
+		// first check of the session has a valid Checkout object
692
+		$checkout = $session->checkout();
693
+		if ($checkout instanceof EE_Checkout) {
694
+			// and use that to clear ticket reservations because it will update the associated registration meta data
695
+			EED_Ticket_Sales_Monitor::instance()->_session_checkout_reset($checkout);
696
+			return;
697
+		}
698
+		$cart = $session->cart();
699
+		if ($cart instanceof EE_Cart) {
700
+			if (self::debug) {
701
+				echo self::$nl . self::$nl . ' cart instance of EE_Cart: ';
702
+			}
703
+			EED_Ticket_Sales_Monitor::instance()->_session_cart_reset($cart, $session);
704
+		} else {
705
+			if (self::debug) {
706
+				echo self::$nl . self::$nl . ' invalid EE_Cart: ';
707
+				var_export($cart, true);
708
+			}
709
+		}
710
+	}
711
+
712
+
713
+	/**
714
+	 * releases reserved tickets in the EE_Cart
715
+	 *
716
+	 * @param EE_Cart $cart
717
+	 * @return void
718
+	 * @throws EE_Error
719
+	 * @throws InvalidArgumentException
720
+	 * @throws ReflectionException
721
+	 * @throws InvalidDataTypeException
722
+	 * @throws InvalidInterfaceException
723
+	 */
724
+	protected function _session_cart_reset(EE_Cart $cart, EE_Session $session)
725
+	{
726
+		if (self::debug) {
727
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
728
+		}
729
+		$ticket_line_items = $cart->get_tickets();
730
+		if (empty($ticket_line_items)) {
731
+			return;
732
+		}
733
+		if (self::debug) {
734
+			echo '<br /> . ticket_line_item count: ' . count($ticket_line_items);
735
+		}
736
+		foreach ($ticket_line_items as $ticket_line_item) {
737
+			if (self::debug) {
738
+				echo self::$nl . ' . ticket_line_item->ID(): ' . $ticket_line_item->ID();
739
+			}
740
+			if ($ticket_line_item instanceof EE_Line_Item && $ticket_line_item->OBJ_type() === 'Ticket') {
741
+				if (self::debug) {
742
+					echo self::$nl . ' . . ticket_line_item->OBJ_ID(): ' . $ticket_line_item->OBJ_ID();
743
+				}
744
+				$ticket = EEM_Ticket::instance()->get_one_by_ID($ticket_line_item->OBJ_ID());
745
+				if ($ticket instanceof EE_Ticket) {
746
+					if (self::debug) {
747
+						echo self::$nl . ' . . ticket->ID(): ' . $ticket->ID();
748
+						echo self::$nl . ' . . ticket_line_item->quantity(): ' . $ticket_line_item->quantity();
749
+					}
750
+					$ticket->add_extra_meta(
751
+						EE_Ticket::META_KEY_TICKET_RESERVATIONS,
752
+						__LINE__ . ') ' . __METHOD__ . '() SID = ' . $session->id()
753
+					);
754
+					$this->_release_reserved_ticket($ticket, $ticket_line_item->quantity());
755
+				}
756
+			}
757
+		}
758
+		if (self::debug) {
759
+			echo self::$nl . self::$nl . ' RESET COMPLETED ';
760
+		}
761
+	}
762
+
763
+
764
+
765
+	/********************************** SESSION_CHECKOUT_RESET  **********************************/
766
+
767
+
768
+	/**
769
+	 * callback hooked into 'AHEE__EE_Session__reset_checkout__before_reset'
770
+	 *
771
+	 * @param EE_Session $session
772
+	 * @return void
773
+	 * @throws EE_Error
774
+	 * @throws InvalidSessionDataException
775
+	 */
776
+	public static function session_checkout_reset(EE_Session $session)
777
+	{
778
+		// don't release tickets if cart was already reset
779
+		if (did_action('AHEE__EE_Session__reset_cart__before_reset')) {
780
+			return;
781
+		}
782
+		$checkout = $session->checkout();
783
+		if ($checkout instanceof EE_Checkout) {
784
+			EED_Ticket_Sales_Monitor::instance()->_session_checkout_reset($checkout);
785
+		}
786
+	}
787
+
788
+
789
+	/**
790
+	 * releases reserved tickets for the EE_Checkout->transaction
791
+	 *
792
+	 * @param EE_Checkout $checkout
793
+	 * @return void
794
+	 * @throws EE_Error
795
+	 * @throws InvalidSessionDataException
796
+	 */
797
+	protected function _session_checkout_reset(EE_Checkout $checkout)
798
+	{
799
+		if (self::debug) {
800
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
801
+		}
802
+		// we want to release the each registration's reserved tickets if the session was cleared, but not if this is a revisit
803
+		if ($checkout->revisit || ! $checkout->transaction instanceof EE_Transaction) {
804
+			return;
805
+		}
806
+		$this->_release_all_reserved_tickets_for_transaction($checkout->transaction);
807
+	}
808
+
809
+
810
+
811
+	/********************************** SESSION_EXPIRED_RESET  **********************************/
812
+
813
+
814
+	/**
815
+	 * @param    EE_Session $session
816
+	 * @return    void
817
+	 */
818
+	public static function session_expired_reset(EE_Session $session)
819
+	{
820
+	}
821
+
822
+
823
+
824
+	/********************************** PROCESS_ABANDONED_TRANSACTIONS  **********************************/
825
+
826
+
827
+	/**
828
+	 * releases reserved tickets for all registrations of an ABANDONED EE_Transaction
829
+	 * by default, will NOT release tickets for free transactions, or any that have received a payment
830
+	 *
831
+	 * @param EE_Transaction $transaction
832
+	 * @return void
833
+	 * @throws EE_Error
834
+	 * @throws InvalidSessionDataException
835
+	 */
836
+	public static function process_abandoned_transactions(EE_Transaction $transaction)
837
+	{
838
+		// is this TXN free or has any money been paid towards this TXN? If so, then leave it alone
839
+		if ($transaction->is_free() || $transaction->paid() > 0) {
840
+			if (self::debug) {
841
+				// DEBUG LOG
842
+				EEH_Debug_Tools::log(
843
+					__CLASS__,
844
+					__FUNCTION__,
845
+					__LINE__,
846
+					array($transaction),
847
+					false,
848
+					'EE_Transaction: ' . $transaction->ID()
849
+				);
850
+			}
851
+			return;
852
+		}
853
+		// have their been any successful payments made ?
854
+		$payments = $transaction->payments();
855
+		foreach ($payments as $payment) {
856
+			if ($payment instanceof EE_Payment && $payment->status() === EEM_Payment::status_id_approved) {
857
+				if (self::debug) {
858
+					// DEBUG LOG
859
+					EEH_Debug_Tools::log(
860
+						__CLASS__,
861
+						__FUNCTION__,
862
+						__LINE__,
863
+						array($payment),
864
+						false,
865
+						'EE_Transaction: ' . $transaction->ID()
866
+					);
867
+				}
868
+				return;
869
+			}
870
+		}
871
+		// since you haven't even attempted to pay for your ticket...
872
+		EED_Ticket_Sales_Monitor::instance()->_release_all_reserved_tickets_for_transaction($transaction);
873
+	}
874
+
875
+
876
+
877
+	/********************************** PROCESS_FAILED_TRANSACTIONS  **********************************/
878
+
879
+
880
+	/**
881
+	 * releases reserved tickets for absolutely ALL registrations of a FAILED EE_Transaction
882
+	 *
883
+	 * @param EE_Transaction $transaction
884
+	 * @return void
885
+	 * @throws EE_Error
886
+	 * @throws InvalidSessionDataException
887
+	 */
888
+	public static function process_failed_transactions(EE_Transaction $transaction)
889
+	{
890
+		// since you haven't even attempted to pay for your ticket...
891
+		EED_Ticket_Sales_Monitor::instance()->_release_all_reserved_tickets_for_transaction($transaction);
892
+	}
893
+
894
+
895
+
896
+	/********************************** RESET RESERVATION COUNTS  *********************************/
897
+
898
+
899
+	/**
900
+	 * Resets all ticket and datetime reserved counts to zero
901
+	 * Tickets that are currently associated with a Transaction that is in progress
902
+	 *
903
+	 * @throws EE_Error
904
+	 * @throws DomainException
905
+	 * @throws InvalidDataTypeException
906
+	 * @throws InvalidInterfaceException
907
+	 * @throws InvalidArgumentException
908
+	 * @throws UnexpectedEntityException
909
+	 */
910
+	public static function reset_reservation_counts()
911
+	{
912
+		/** @var EE_Line_Item[] $valid_reserved_tickets */
913
+		$valid_reserved_tickets = array();
914
+		/** @var EE_Transaction[] $transactions_not_in_progress */
915
+		$transactions_not_in_progress = EEM_Transaction::instance()->get_transactions_not_in_progress();
916
+		foreach ($transactions_not_in_progress as $transaction) {
917
+			// if this TXN has been fully completed, then skip it
918
+			if ($transaction->reg_step_completed('finalize_registration')) {
919
+				continue;
920
+			}
921
+			$total_line_item = $transaction->total_line_item();
922
+			// $transaction_in_progress->line
923
+			if (! $total_line_item instanceof EE_Line_Item) {
924
+				throw new DomainException(
925
+					esc_html__(
926
+						'Transaction does not have a valid Total Line Item associated with it.',
927
+						'event_espresso'
928
+					)
929
+				);
930
+			}
931
+			$valid_reserved_tickets += EED_Ticket_Sales_Monitor::get_ticket_line_items_for_grand_total(
932
+				$total_line_item
933
+			);
934
+		}
935
+		$total_line_items = EEM_Line_Item::instance()->get_total_line_items_for_active_carts();
936
+		foreach ($total_line_items as $total_line_item) {
937
+			$valid_reserved_tickets += EED_Ticket_Sales_Monitor::get_ticket_line_items_for_grand_total(
938
+				$total_line_item
939
+			);
940
+		}
941
+		$tickets_with_reservations = EEM_Ticket::instance()->get_tickets_with_reservations();
942
+		return EED_Ticket_Sales_Monitor::release_reservations_for_tickets(
943
+			$tickets_with_reservations,
944
+			$valid_reserved_tickets,
945
+			__FUNCTION__
946
+		);
947
+	}
948
+
949
+
950
+	/**
951
+	 * @param EE_Line_Item $total_line_item
952
+	 * @return EE_Line_Item[]
953
+	 */
954
+	private static function get_ticket_line_items_for_grand_total(EE_Line_Item $total_line_item)
955
+	{
956
+		/** @var EE_Line_Item[] $valid_reserved_tickets */
957
+		$valid_reserved_tickets = array();
958
+		$ticket_line_items = EEH_Line_Item::get_ticket_line_items($total_line_item);
959
+		foreach ($ticket_line_items as $ticket_line_item) {
960
+			if ($ticket_line_item instanceof EE_Line_Item) {
961
+				$valid_reserved_tickets[] = $ticket_line_item;
962
+			}
963
+		}
964
+		return $valid_reserved_tickets;
965
+	}
966
+
967
+
968
+	/**
969
+	 * @param EE_Ticket[]    $tickets_with_reservations
970
+	 * @param EE_Line_Item[] $valid_reserved_ticket_line_items
971
+	 * @return int
972
+	 * @throws UnexpectedEntityException
973
+	 * @throws DomainException
974
+	 * @throws EE_Error
975
+	 */
976
+	private static function release_reservations_for_tickets(
977
+		array $tickets_with_reservations,
978
+		array $valid_reserved_ticket_line_items = array(),
979
+		$source
980
+	) {
981
+		if (self::debug) {
982
+			echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '()';
983
+		}
984
+		$total_tickets_released = 0;
985
+		$sold_out_events = array();
986
+		foreach ($tickets_with_reservations as $ticket_with_reservations) {
987
+			if (! $ticket_with_reservations instanceof EE_Ticket) {
988
+				continue;
989
+			}
990
+			$reserved_qty = $ticket_with_reservations->reserved();
991
+			if (self::debug) {
992
+				echo self::$nl . ' . $ticket_with_reservations->ID(): ' . $ticket_with_reservations->ID();
993
+				echo self::$nl . ' . $reserved_qty: ' . $reserved_qty;
994
+			}
995
+			foreach ($valid_reserved_ticket_line_items as $valid_reserved_ticket_line_item) {
996
+				if ($valid_reserved_ticket_line_item instanceof EE_Line_Item
997
+					&& $valid_reserved_ticket_line_item->OBJ_ID() === $ticket_with_reservations->ID()
998
+				) {
999
+					if (self::debug) {
1000
+						echo self::$nl . ' . $valid_reserved_ticket_line_item->quantity(): '
1001
+							 . $valid_reserved_ticket_line_item->quantity();
1002
+					}
1003
+					$reserved_qty -= $valid_reserved_ticket_line_item->quantity();
1004
+				}
1005
+			}
1006
+			if ($reserved_qty > 0) {
1007
+				$ticket_with_reservations->add_extra_meta(
1008
+					EE_Ticket::META_KEY_TICKET_RESERVATIONS,
1009
+					__LINE__ . ') ' . $source . '()'
1010
+				);
1011
+				$ticket_with_reservations->decrease_reserved($reserved_qty, true, 'TicketSalesMonitor:' . __LINE__);
1012
+				$ticket_with_reservations->save();
1013
+				$total_tickets_released += $reserved_qty;
1014
+				$event = $ticket_with_reservations->get_related_event();
1015
+				// track sold out events
1016
+				if ($event instanceof EE_Event && $event->is_sold_out()) {
1017
+					$sold_out_events[] = $event;
1018
+				}
1019
+			}
1020
+		}
1021
+		if (self::debug) {
1022
+			echo self::$nl . ' . $total_tickets_released: ' . $total_tickets_released;
1023
+		}
1024
+		// double check whether sold out events should remain sold out after releasing tickets
1025
+		if ($sold_out_events !== array()) {
1026
+			foreach ($sold_out_events as $sold_out_event) {
1027
+				/** @var EE_Event $sold_out_event */
1028
+				$sold_out_event->perform_sold_out_status_check();
1029
+			}
1030
+		}
1031
+		return $total_tickets_released;
1032
+	}
1033
+
1034
+
1035
+
1036
+	/********************************** SHUTDOWN  **********************************/
1037
+
1038
+
1039
+	/**
1040
+	 * @param int $timestamp
1041
+	 * @return false|int
1042
+	 * @throws EE_Error
1043
+	 * @throws InvalidArgumentException
1044
+	 * @throws InvalidDataTypeException
1045
+	 * @throws InvalidInterfaceException
1046
+	 */
1047
+	public static function clear_expired_line_items_with_no_transaction($timestamp = 0)
1048
+	{
1049
+		/** @type WPDB $wpdb */
1050
+		global $wpdb;
1051
+		if (! absint($timestamp)) {
1052
+			/** @var EventEspresso\core\domain\values\session\SessionLifespan $session_lifespan */
1053
+			$session_lifespan = LoaderFactory::getLoader()->getShared(
1054
+				'EventEspresso\core\domain\values\session\SessionLifespan'
1055
+			);
1056
+			$timestamp = $session_lifespan->expiration();
1057
+		}
1058
+		return $wpdb->query(
1059
+			$wpdb->prepare(
1060
+				'DELETE FROM ' . EEM_Line_Item::instance()->table() . '
1061 1061
                 WHERE TXN_ID = 0 AND LIN_timestamp <= %s',
1062
-                // use GMT time because that's what LIN_timestamps are in
1063
-                date('Y-m-d H:i:s', $timestamp)
1064
-            )
1065
-        );
1066
-    }
1062
+				// use GMT time because that's what LIN_timestamps are in
1063
+				date('Y-m-d H:i:s', $timestamp)
1064
+			)
1065
+		);
1066
+	}
1067 1067
 }
Please login to merge, or discard this patch.
Spacing   +83 added lines, -83 removed lines patch added patch discarded remove patch
@@ -182,7 +182,7 @@  discard block
 block discarded – undo
182 182
     public static function release_tickets_for_expired_carts()
183 183
     {
184 184
         if (self::debug) {
185
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '()';
185
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'()';
186 186
         }
187 187
         do_action('AHEE__EED_Ticket_Sales_Monitor__release_tickets_for_expired_carts__begin');
188 188
         $expired_ticket_IDs = array();
@@ -193,29 +193,29 @@  discard block
 block discarded – undo
193 193
         $timestamp = $session_lifespan->expiration();
194 194
         $expired_ticket_line_items = EEM_Line_Item::instance()->getTicketLineItemsForExpiredCarts($timestamp);
195 195
         if (self::debug) {
196
-            echo self::$nl . ' . time(): ' . time();
197
-            echo self::$nl . ' . time() as date: ' . date('Y-m-d H:i a');
198
-            echo self::$nl . ' . session expiration: ' . $session_lifespan->expiration();
199
-            echo self::$nl . ' . session expiration as date: ' . date('Y-m-d H:i a', $session_lifespan->expiration());
200
-            echo self::$nl . ' . timestamp: ' . $timestamp;
201
-            echo self::$nl . ' . $expired_ticket_line_items: ' . count($expired_ticket_line_items);
196
+            echo self::$nl.' . time(): '.time();
197
+            echo self::$nl.' . time() as date: '.date('Y-m-d H:i a');
198
+            echo self::$nl.' . session expiration: '.$session_lifespan->expiration();
199
+            echo self::$nl.' . session expiration as date: '.date('Y-m-d H:i a', $session_lifespan->expiration());
200
+            echo self::$nl.' . timestamp: '.$timestamp;
201
+            echo self::$nl.' . $expired_ticket_line_items: '.count($expired_ticket_line_items);
202 202
         }
203
-        if (! empty($expired_ticket_line_items)) {
203
+        if ( ! empty($expired_ticket_line_items)) {
204 204
             foreach ($expired_ticket_line_items as $expired_ticket_line_item) {
205
-                if (! $expired_ticket_line_item instanceof EE_Line_Item) {
205
+                if ( ! $expired_ticket_line_item instanceof EE_Line_Item) {
206 206
                     continue;
207 207
                 }
208
-                $expired_ticket_IDs[ $expired_ticket_line_item->OBJ_ID() ] = $expired_ticket_line_item->OBJ_ID();
208
+                $expired_ticket_IDs[$expired_ticket_line_item->OBJ_ID()] = $expired_ticket_line_item->OBJ_ID();
209 209
                 if (self::debug) {
210
-                    echo self::$nl . ' . $expired_ticket_line_item->OBJ_ID(): ' . $expired_ticket_line_item->OBJ_ID();
211
-                    echo self::$nl . ' . $expired_ticket_line_item->timestamp(): '
210
+                    echo self::$nl.' . $expired_ticket_line_item->OBJ_ID(): '.$expired_ticket_line_item->OBJ_ID();
211
+                    echo self::$nl.' . $expired_ticket_line_item->timestamp(): '
212 212
                          . date(
213 213
                              'Y-m-d h:i a',
214 214
                              $expired_ticket_line_item->timestamp(true)
215 215
                          );
216 216
                 }
217 217
             }
218
-            if (! empty($expired_ticket_IDs)) {
218
+            if ( ! empty($expired_ticket_IDs)) {
219 219
                 EED_Ticket_Sales_Monitor::release_reservations_for_tickets(
220 220
                     \EEM_Ticket::instance()->get_tickets_with_IDs($expired_ticket_IDs),
221 221
                     array(),
@@ -253,8 +253,8 @@  discard block
 block discarded – undo
253 253
             $qty = EED_Ticket_Sales_Monitor::instance()->_validate_ticket_sale($ticket, $qty);
254 254
         }
255 255
         if (self::debug) {
256
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '()';
257
-            echo self::$nl . self::$nl . '<b> RETURNED QTY: ' . $qty . '</b>';
256
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'()';
257
+            echo self::$nl.self::$nl.'<b> RETURNED QTY: '.$qty.'</b>';
258 258
         }
259 259
         return $qty;
260 260
     }
@@ -272,36 +272,36 @@  discard block
 block discarded – undo
272 272
     protected function _validate_ticket_sale(EE_Ticket $ticket, $qty = 1)
273 273
     {
274 274
         if (self::debug) {
275
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
275
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'() ';
276 276
         }
277
-        if (! $ticket instanceof EE_Ticket) {
277
+        if ( ! $ticket instanceof EE_Ticket) {
278 278
             return 0;
279 279
         }
280 280
         if (self::debug) {
281
-            echo self::$nl . '<b> . ticket->ID: ' . $ticket->ID() . '</b>';
282
-            echo self::$nl . ' . original ticket->reserved: ' . $ticket->reserved();
281
+            echo self::$nl.'<b> . ticket->ID: '.$ticket->ID().'</b>';
282
+            echo self::$nl.' . original ticket->reserved: '.$ticket->reserved();
283 283
         }
284 284
         $ticket->refresh_from_db();
285 285
         // first let's determine the ticket availability based on sales
286 286
         $available = $ticket->qty('saleable');
287 287
         if (self::debug) {
288
-            echo self::$nl . ' . . . ticket->qty: ' . $ticket->qty();
289
-            echo self::$nl . ' . . . ticket->sold: ' . $ticket->sold();
290
-            echo self::$nl . ' . . . ticket->reserved: ' . $ticket->reserved();
291
-            echo self::$nl . ' . . . ticket->qty(saleable): ' . $ticket->qty('saleable');
292
-            echo self::$nl . ' . . . available: ' . $available;
288
+            echo self::$nl.' . . . ticket->qty: '.$ticket->qty();
289
+            echo self::$nl.' . . . ticket->sold: '.$ticket->sold();
290
+            echo self::$nl.' . . . ticket->reserved: '.$ticket->reserved();
291
+            echo self::$nl.' . . . ticket->qty(saleable): '.$ticket->qty('saleable');
292
+            echo self::$nl.' . . . available: '.$available;
293 293
         }
294 294
         if ($available < 1) {
295 295
             $this->_ticket_sold_out($ticket);
296 296
             return 0;
297 297
         }
298 298
         if (self::debug) {
299
-            echo self::$nl . ' . . . qty: ' . $qty;
299
+            echo self::$nl.' . . . qty: '.$qty;
300 300
         }
301 301
         if ($available < $qty) {
302 302
             $qty = $available;
303 303
             if (self::debug) {
304
-                echo self::$nl . ' . . . QTY ADJUSTED: ' . $qty;
304
+                echo self::$nl.' . . . QTY ADJUSTED: '.$qty;
305 305
             }
306 306
             $this->_ticket_quantity_decremented($ticket);
307 307
         }
@@ -321,9 +321,9 @@  discard block
 block discarded – undo
321 321
     protected function _reserve_ticket(EE_Ticket $ticket, $quantity = 1)
322 322
     {
323 323
         if (self::debug) {
324
-            echo self::$nl . self::$nl . ' . . . INCREASE RESERVED: ' . $quantity;
324
+            echo self::$nl.self::$nl.' . . . INCREASE RESERVED: '.$quantity;
325 325
         }
326
-        $ticket->increase_reserved($quantity, 'TicketSalesMonitor:' . __LINE__);
326
+        $ticket->increase_reserved($quantity, 'TicketSalesMonitor:'.__LINE__);
327 327
         return $ticket->save();
328 328
     }
329 329
 
@@ -337,12 +337,12 @@  discard block
 block discarded – undo
337 337
     protected function _release_reserved_ticket(EE_Ticket $ticket, $quantity = 1)
338 338
     {
339 339
         if (self::debug) {
340
-            echo self::$nl . ' . . . ticket->ID: ' . $ticket->ID();
341
-            echo self::$nl . ' . . . ticket->reserved before: ' . $ticket->reserved();
340
+            echo self::$nl.' . . . ticket->ID: '.$ticket->ID();
341
+            echo self::$nl.' . . . ticket->reserved before: '.$ticket->reserved();
342 342
         }
343
-        $ticket->decrease_reserved($quantity, true, 'TicketSalesMonitor:' . __LINE__);
343
+        $ticket->decrease_reserved($quantity, true, 'TicketSalesMonitor:'.__LINE__);
344 344
         if (self::debug) {
345
-            echo self::$nl . ' . . . ticket->reserved after: ' . $ticket->reserved();
345
+            echo self::$nl.' . . . ticket->reserved after: '.$ticket->reserved();
346 346
         }
347 347
         return $ticket->save() ? 1 : 0;
348 348
     }
@@ -359,8 +359,8 @@  discard block
 block discarded – undo
359 359
     protected function _ticket_sold_out(EE_Ticket $ticket)
360 360
     {
361 361
         if (self::debug) {
362
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
363
-            echo self::$nl . ' . . ticket->name: ' . $this->_get_ticket_and_event_name($ticket);
362
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'() ';
363
+            echo self::$nl.' . . ticket->name: '.$this->_get_ticket_and_event_name($ticket);
364 364
         }
365 365
         $this->sold_out_tickets[] = $this->_get_ticket_and_event_name($ticket);
366 366
     }
@@ -377,8 +377,8 @@  discard block
 block discarded – undo
377 377
     protected function _ticket_quantity_decremented(EE_Ticket $ticket)
378 378
     {
379 379
         if (self::debug) {
380
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
381
-            echo self::$nl . ' . . ticket->name: ' . $this->_get_ticket_and_event_name($ticket);
380
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'() ';
381
+            echo self::$nl.' . . ticket->name: '.$this->_get_ticket_and_event_name($ticket);
382 382
         }
383 383
         $this->decremented_tickets[] = $this->_get_ticket_and_event_name($ticket);
384 384
     }
@@ -429,7 +429,7 @@  discard block
 block discarded – undo
429 429
         if ($ticket instanceof EE_Ticket) {
430 430
             $ticket->add_extra_meta(
431 431
                 EE_Ticket::META_KEY_TICKET_RESERVATIONS,
432
-                __LINE__ . ') ' . __METHOD__ . '()'
432
+                __LINE__.') '.__METHOD__.'()'
433 433
             );
434 434
             if ($quantity > 0) {
435 435
                 EED_Ticket_Sales_Monitor::instance()->_reserve_ticket($ticket, $quantity);
@@ -452,7 +452,7 @@  discard block
 block discarded – undo
452 452
     {
453 453
         $ticket->add_extra_meta(
454 454
             EE_Ticket::META_KEY_TICKET_RESERVATIONS,
455
-            __LINE__ . ') ' . __METHOD__ . '()'
455
+            __LINE__.') '.__METHOD__.'()'
456 456
         );
457 457
         EED_Ticket_Sales_Monitor::instance()->_release_reserved_ticket($ticket, $quantity);
458 458
     }
@@ -487,7 +487,7 @@  discard block
 block discarded – undo
487 487
     protected function _post_notices()
488 488
     {
489 489
         if (self::debug) {
490
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
490
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'() ';
491 491
         }
492 492
         $refresh_msg = '';
493 493
         $none_added_msg = '';
@@ -498,7 +498,7 @@  discard block
 block discarded – undo
498 498
             );
499 499
             $none_added_msg = __('No tickets were added for the event.', 'event_espresso');
500 500
         }
501
-        if (! empty($this->sold_out_tickets)) {
501
+        if ( ! empty($this->sold_out_tickets)) {
502 502
             EE_Error::add_attention(
503 503
                 sprintf(
504 504
                     apply_filters(
@@ -521,7 +521,7 @@  discard block
 block discarded – undo
521 521
             // and reset the cart
522 522
             EED_Ticket_Sales_Monitor::session_cart_reset(EE_Registry::instance()->SSN);
523 523
         }
524
-        if (! empty($this->decremented_tickets)) {
524
+        if ( ! empty($this->decremented_tickets)) {
525 525
             EE_Error::add_attention(
526 526
                 sprintf(
527 527
                     apply_filters(
@@ -558,9 +558,9 @@  discard block
 block discarded – undo
558 558
     protected function _release_all_reserved_tickets_for_transaction(EE_Transaction $transaction)
559 559
     {
560 560
         if (self::debug) {
561
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
562
-            echo self::$nl . ' . transaction->ID: ' . $transaction->ID();
563
-            echo self::$nl . ' . TXN status_ID: ' . $transaction->status_ID();
561
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'() ';
562
+            echo self::$nl.' . transaction->ID: '.$transaction->ID();
563
+            echo self::$nl.' . TXN status_ID: '.$transaction->status_ID();
564 564
         }
565 565
         // check if 'finalize_registration' step has been completed...
566 566
         $finalized = $transaction->reg_step_completed('finalize_registration');
@@ -572,13 +572,13 @@  discard block
 block discarded – undo
572 572
                 __LINE__,
573 573
                 array('finalized' => $finalized),
574 574
                 false,
575
-                'EE_Transaction: ' . $transaction->ID()
575
+                'EE_Transaction: '.$transaction->ID()
576 576
             );
577 577
         }
578 578
         // how many tickets were released
579 579
         $count = 0;
580 580
         if (self::debug) {
581
-            echo self::$nl . ' . . . TXN finalized: ' . $finalized;
581
+            echo self::$nl.' . . . TXN finalized: '.$finalized;
582 582
         }
583 583
         $release_tickets_with_TXN_status = array(
584 584
             EEM_Transaction::failed_status_code,
@@ -587,27 +587,27 @@  discard block
 block discarded – undo
587 587
         );
588 588
         $events = array();
589 589
         // if the session is getting cleared BEFORE the TXN has been finalized or the transaction is not completed
590
-        if (! $finalized || in_array($transaction->status_ID(), $release_tickets_with_TXN_status, true)) {
590
+        if ( ! $finalized || in_array($transaction->status_ID(), $release_tickets_with_TXN_status, true)) {
591 591
             // cancel any reserved tickets for registrations that were not approved
592 592
             $registrations = $transaction->registrations();
593 593
             if (self::debug) {
594
-                echo self::$nl . ' . . . # registrations: ' . count($registrations);
594
+                echo self::$nl.' . . . # registrations: '.count($registrations);
595 595
                 $reg = reset($registrations);
596 596
                 $ticket = $reg->ticket();
597 597
                 if ($ticket instanceof EE_Ticket) {
598 598
                     $ticket->add_extra_meta(
599 599
                         EE_Ticket::META_KEY_TICKET_RESERVATIONS,
600
-                        __LINE__ . ') Release All Tickets TXN:' . $transaction->ID()
600
+                        __LINE__.') Release All Tickets TXN:'.$transaction->ID()
601 601
                     );
602 602
                 }
603 603
             }
604
-            if (! empty($registrations)) {
604
+            if ( ! empty($registrations)) {
605 605
                 foreach ($registrations as $registration) {
606 606
                     if ($registration instanceof EE_Registration
607 607
                         && $this->_release_reserved_ticket_for_registration($registration, $transaction)
608 608
                     ) {
609 609
                         $count++;
610
-                        $events[ $registration->event_ID() ] = $registration->event();
610
+                        $events[$registration->event_ID()] = $registration->event();
611 611
                     }
612 612
                 }
613 613
             }
@@ -637,10 +637,10 @@  discard block
 block discarded – undo
637 637
     ) {
638 638
         $STS_ID = $transaction->status_ID();
639 639
         if (self::debug) {
640
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
641
-            echo self::$nl . ' . . registration->ID: ' . $registration->ID();
642
-            echo self::$nl . ' . . registration->status_ID: ' . $registration->status_ID();
643
-            echo self::$nl . ' . . transaction->status_ID(): ' . $STS_ID;
640
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'() ';
641
+            echo self::$nl.' . . registration->ID: '.$registration->ID();
642
+            echo self::$nl.' . . registration->status_ID: '.$registration->status_ID();
643
+            echo self::$nl.' . . transaction->status_ID(): '.$STS_ID;
644 644
         }
645 645
         if (// release Tickets for Failed Transactions and Abandoned Transactions
646 646
             $STS_ID === EEM_Transaction::failed_status_code
@@ -652,12 +652,12 @@  discard block
 block discarded – undo
652 652
             )
653 653
         ) {
654 654
             if (self::debug) {
655
-                echo self::$nl . self::$nl . ' . . RELEASE RESERVED TICKET';
655
+                echo self::$nl.self::$nl.' . . RELEASE RESERVED TICKET';
656 656
                 $rsrvd = $registration->get_extra_meta(EE_Registration::HAS_RESERVED_TICKET_KEY, true);
657
-                echo self::$nl . ' . . . registration HAS_RESERVED_TICKET_KEY: ';
657
+                echo self::$nl.' . . . registration HAS_RESERVED_TICKET_KEY: ';
658 658
                 var_dump($rsrvd);
659 659
             }
660
-            $registration->release_reserved_ticket(true, 'TicketSalesMonitor:' . __LINE__);
660
+            $registration->release_reserved_ticket(true, 'TicketSalesMonitor:'.__LINE__);
661 661
             return 1;
662 662
         }
663 663
         return 0;
@@ -686,7 +686,7 @@  discard block
 block discarded – undo
686 686
             return;
687 687
         }
688 688
         if (self::debug) {
689
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
689
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'() ';
690 690
         }
691 691
         // first check of the session has a valid Checkout object
692 692
         $checkout = $session->checkout();
@@ -698,12 +698,12 @@  discard block
 block discarded – undo
698 698
         $cart = $session->cart();
699 699
         if ($cart instanceof EE_Cart) {
700 700
             if (self::debug) {
701
-                echo self::$nl . self::$nl . ' cart instance of EE_Cart: ';
701
+                echo self::$nl.self::$nl.' cart instance of EE_Cart: ';
702 702
             }
703 703
             EED_Ticket_Sales_Monitor::instance()->_session_cart_reset($cart, $session);
704 704
         } else {
705 705
             if (self::debug) {
706
-                echo self::$nl . self::$nl . ' invalid EE_Cart: ';
706
+                echo self::$nl.self::$nl.' invalid EE_Cart: ';
707 707
                 var_export($cart, true);
708 708
             }
709 709
         }
@@ -724,39 +724,39 @@  discard block
 block discarded – undo
724 724
     protected function _session_cart_reset(EE_Cart $cart, EE_Session $session)
725 725
     {
726 726
         if (self::debug) {
727
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
727
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'() ';
728 728
         }
729 729
         $ticket_line_items = $cart->get_tickets();
730 730
         if (empty($ticket_line_items)) {
731 731
             return;
732 732
         }
733 733
         if (self::debug) {
734
-            echo '<br /> . ticket_line_item count: ' . count($ticket_line_items);
734
+            echo '<br /> . ticket_line_item count: '.count($ticket_line_items);
735 735
         }
736 736
         foreach ($ticket_line_items as $ticket_line_item) {
737 737
             if (self::debug) {
738
-                echo self::$nl . ' . ticket_line_item->ID(): ' . $ticket_line_item->ID();
738
+                echo self::$nl.' . ticket_line_item->ID(): '.$ticket_line_item->ID();
739 739
             }
740 740
             if ($ticket_line_item instanceof EE_Line_Item && $ticket_line_item->OBJ_type() === 'Ticket') {
741 741
                 if (self::debug) {
742
-                    echo self::$nl . ' . . ticket_line_item->OBJ_ID(): ' . $ticket_line_item->OBJ_ID();
742
+                    echo self::$nl.' . . ticket_line_item->OBJ_ID(): '.$ticket_line_item->OBJ_ID();
743 743
                 }
744 744
                 $ticket = EEM_Ticket::instance()->get_one_by_ID($ticket_line_item->OBJ_ID());
745 745
                 if ($ticket instanceof EE_Ticket) {
746 746
                     if (self::debug) {
747
-                        echo self::$nl . ' . . ticket->ID(): ' . $ticket->ID();
748
-                        echo self::$nl . ' . . ticket_line_item->quantity(): ' . $ticket_line_item->quantity();
747
+                        echo self::$nl.' . . ticket->ID(): '.$ticket->ID();
748
+                        echo self::$nl.' . . ticket_line_item->quantity(): '.$ticket_line_item->quantity();
749 749
                     }
750 750
                     $ticket->add_extra_meta(
751 751
                         EE_Ticket::META_KEY_TICKET_RESERVATIONS,
752
-                        __LINE__ . ') ' . __METHOD__ . '() SID = ' . $session->id()
752
+                        __LINE__.') '.__METHOD__.'() SID = '.$session->id()
753 753
                     );
754 754
                     $this->_release_reserved_ticket($ticket, $ticket_line_item->quantity());
755 755
                 }
756 756
             }
757 757
         }
758 758
         if (self::debug) {
759
-            echo self::$nl . self::$nl . ' RESET COMPLETED ';
759
+            echo self::$nl.self::$nl.' RESET COMPLETED ';
760 760
         }
761 761
     }
762 762
 
@@ -797,7 +797,7 @@  discard block
 block discarded – undo
797 797
     protected function _session_checkout_reset(EE_Checkout $checkout)
798 798
     {
799 799
         if (self::debug) {
800
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '() ';
800
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'() ';
801 801
         }
802 802
         // we want to release the each registration's reserved tickets if the session was cleared, but not if this is a revisit
803 803
         if ($checkout->revisit || ! $checkout->transaction instanceof EE_Transaction) {
@@ -845,7 +845,7 @@  discard block
 block discarded – undo
845 845
                     __LINE__,
846 846
                     array($transaction),
847 847
                     false,
848
-                    'EE_Transaction: ' . $transaction->ID()
848
+                    'EE_Transaction: '.$transaction->ID()
849 849
                 );
850 850
             }
851 851
             return;
@@ -862,7 +862,7 @@  discard block
 block discarded – undo
862 862
                         __LINE__,
863 863
                         array($payment),
864 864
                         false,
865
-                        'EE_Transaction: ' . $transaction->ID()
865
+                        'EE_Transaction: '.$transaction->ID()
866 866
                     );
867 867
                 }
868 868
                 return;
@@ -920,7 +920,7 @@  discard block
 block discarded – undo
920 920
             }
921 921
             $total_line_item = $transaction->total_line_item();
922 922
             // $transaction_in_progress->line
923
-            if (! $total_line_item instanceof EE_Line_Item) {
923
+            if ( ! $total_line_item instanceof EE_Line_Item) {
924 924
                 throw new DomainException(
925 925
                     esc_html__(
926 926
                         'Transaction does not have a valid Total Line Item associated with it.',
@@ -979,25 +979,25 @@  discard block
 block discarded – undo
979 979
         $source
980 980
     ) {
981 981
         if (self::debug) {
982
-            echo self::$nl . self::$nl . __LINE__ . ') ' . __METHOD__ . '()';
982
+            echo self::$nl.self::$nl.__LINE__.') '.__METHOD__.'()';
983 983
         }
984 984
         $total_tickets_released = 0;
985 985
         $sold_out_events = array();
986 986
         foreach ($tickets_with_reservations as $ticket_with_reservations) {
987
-            if (! $ticket_with_reservations instanceof EE_Ticket) {
987
+            if ( ! $ticket_with_reservations instanceof EE_Ticket) {
988 988
                 continue;
989 989
             }
990 990
             $reserved_qty = $ticket_with_reservations->reserved();
991 991
             if (self::debug) {
992
-                echo self::$nl . ' . $ticket_with_reservations->ID(): ' . $ticket_with_reservations->ID();
993
-                echo self::$nl . ' . $reserved_qty: ' . $reserved_qty;
992
+                echo self::$nl.' . $ticket_with_reservations->ID(): '.$ticket_with_reservations->ID();
993
+                echo self::$nl.' . $reserved_qty: '.$reserved_qty;
994 994
             }
995 995
             foreach ($valid_reserved_ticket_line_items as $valid_reserved_ticket_line_item) {
996 996
                 if ($valid_reserved_ticket_line_item instanceof EE_Line_Item
997 997
                     && $valid_reserved_ticket_line_item->OBJ_ID() === $ticket_with_reservations->ID()
998 998
                 ) {
999 999
                     if (self::debug) {
1000
-                        echo self::$nl . ' . $valid_reserved_ticket_line_item->quantity(): '
1000
+                        echo self::$nl.' . $valid_reserved_ticket_line_item->quantity(): '
1001 1001
                              . $valid_reserved_ticket_line_item->quantity();
1002 1002
                     }
1003 1003
                     $reserved_qty -= $valid_reserved_ticket_line_item->quantity();
@@ -1006,9 +1006,9 @@  discard block
 block discarded – undo
1006 1006
             if ($reserved_qty > 0) {
1007 1007
                 $ticket_with_reservations->add_extra_meta(
1008 1008
                     EE_Ticket::META_KEY_TICKET_RESERVATIONS,
1009
-                    __LINE__ . ') ' . $source . '()'
1009
+                    __LINE__.') '.$source.'()'
1010 1010
                 );
1011
-                $ticket_with_reservations->decrease_reserved($reserved_qty, true, 'TicketSalesMonitor:' . __LINE__);
1011
+                $ticket_with_reservations->decrease_reserved($reserved_qty, true, 'TicketSalesMonitor:'.__LINE__);
1012 1012
                 $ticket_with_reservations->save();
1013 1013
                 $total_tickets_released += $reserved_qty;
1014 1014
                 $event = $ticket_with_reservations->get_related_event();
@@ -1019,7 +1019,7 @@  discard block
 block discarded – undo
1019 1019
             }
1020 1020
         }
1021 1021
         if (self::debug) {
1022
-            echo self::$nl . ' . $total_tickets_released: ' . $total_tickets_released;
1022
+            echo self::$nl.' . $total_tickets_released: '.$total_tickets_released;
1023 1023
         }
1024 1024
         // double check whether sold out events should remain sold out after releasing tickets
1025 1025
         if ($sold_out_events !== array()) {
@@ -1048,7 +1048,7 @@  discard block
 block discarded – undo
1048 1048
     {
1049 1049
         /** @type WPDB $wpdb */
1050 1050
         global $wpdb;
1051
-        if (! absint($timestamp)) {
1051
+        if ( ! absint($timestamp)) {
1052 1052
             /** @var EventEspresso\core\domain\values\session\SessionLifespan $session_lifespan */
1053 1053
             $session_lifespan = LoaderFactory::getLoader()->getShared(
1054 1054
                 'EventEspresso\core\domain\values\session\SessionLifespan'
@@ -1057,7 +1057,7 @@  discard block
 block discarded – undo
1057 1057
         }
1058 1058
         return $wpdb->query(
1059 1059
             $wpdb->prepare(
1060
-                'DELETE FROM ' . EEM_Line_Item::instance()->table() . '
1060
+                'DELETE FROM '.EEM_Line_Item::instance()->table().'
1061 1061
                 WHERE TXN_ID = 0 AND LIN_timestamp <= %s',
1062 1062
                 // use GMT time because that's what LIN_timestamps are in
1063 1063
                 date('Y-m-d H:i:s', $timestamp)
Please login to merge, or discard this patch.
core/EE_Session.core.php 3 patches
Doc Comments   +3 added lines, -3 removed lines patch added patch discarded remove patch
@@ -542,7 +542,7 @@  discard block
 block discarded – undo
542 542
     /**
543 543
      * @initiate session
544 544
      * @access   private
545
-     * @return TRUE on success, FALSE on fail
545
+     * @return boolean on success, FALSE on fail
546 546
      * @throws EE_Error
547 547
      * @throws InvalidArgumentException
548 548
      * @throws InvalidDataTypeException
@@ -778,7 +778,7 @@  discard block
 block discarded – undo
778 778
      * @update session data  prior to saving to the db
779 779
      * @access public
780 780
      * @param bool $new_session
781
-     * @return TRUE on success, FALSE on fail
781
+     * @return boolean on success, FALSE on fail
782 782
      * @throws EE_Error
783 783
      * @throws InvalidArgumentException
784 784
      * @throws InvalidDataTypeException
@@ -879,7 +879,7 @@  discard block
 block discarded – undo
879 879
      * _save_session_to_db
880 880
      *
881 881
      * @param bool $clear_session
882
-     * @return string
882
+     * @return boolean
883 883
      * @throws EE_Error
884 884
      * @throws InvalidArgumentException
885 885
      * @throws InvalidDataTypeException
Please login to merge, or discard this patch.
Indentation   +1305 added lines, -1305 removed lines patch added patch discarded remove patch
@@ -24,1303 +24,1303 @@  discard block
 block discarded – undo
24 24
 class EE_Session implements SessionIdentifierInterface
25 25
 {
26 26
 
27
-    const session_id_prefix = 'ee_ssn_';
28
-
29
-    const hash_check_prefix = 'ee_shc_';
30
-
31
-    const OPTION_NAME_SETTINGS = 'ee_session_settings';
32
-
33
-    const STATUS_CLOSED = 0;
34
-
35
-    const STATUS_OPEN = 1;
36
-
37
-    /**
38
-     * instance of the EE_Session object
39
-     *
40
-     * @var EE_Session
41
-     */
42
-    private static $_instance;
43
-
44
-    /**
45
-     * @var CacheStorageInterface $cache_storage
46
-     */
47
-    protected $cache_storage;
48
-
49
-    /**
50
-     * EE_Encryption object
51
-     *
52
-     * @var EE_Encryption
53
-     */
54
-    protected $encryption;
55
-
56
-    /**
57
-     * the session id
58
-     *
59
-     * @var string
60
-     */
61
-    private $_sid;
62
-
63
-    /**
64
-     * session id salt
65
-     *
66
-     * @var string
67
-     */
68
-    private $_sid_salt;
69
-
70
-    /**
71
-     * session data
72
-     *
73
-     * @var array
74
-     */
75
-    private $_session_data = array();
76
-
77
-    /**
78
-     * how long an EE session lasts
79
-     * default session lifespan of 1 hour (for not so instant IPNs)
80
-     *
81
-     * @var SessionLifespan $session_lifespan
82
-     */
83
-    private $session_lifespan;
84
-
85
-    /**
86
-     * session expiration time as Unix timestamp in GMT
87
-     *
88
-     * @var int
89
-     */
90
-    private $_expiration;
91
-
92
-    /**
93
-     * whether or not session has expired at some point
94
-     *
95
-     * @var boolean
96
-     */
97
-    private $_expired = false;
98
-
99
-    /**
100
-     * current time as Unix timestamp in GMT
101
-     *
102
-     * @var int
103
-     */
104
-    private $_time;
105
-
106
-    /**
107
-     * whether to encrypt session data
108
-     *
109
-     * @var bool
110
-     */
111
-    private $_use_encryption;
112
-
113
-    /**
114
-     * well... according to the server...
115
-     *
116
-     * @var null
117
-     */
118
-    private $_user_agent;
119
-
120
-    /**
121
-     * do you really trust the server ?
122
-     *
123
-     * @var null
124
-     */
125
-    private $_ip_address;
126
-
127
-    /**
128
-     * current WP user_id
129
-     *
130
-     * @var null
131
-     */
132
-    private $_wp_user_id;
133
-
134
-    /**
135
-     * array for defining default session vars
136
-     *
137
-     * @var array
138
-     */
139
-    private $_default_session_vars = array(
140
-        'id'            => null,
141
-        'user_id'       => null,
142
-        'ip_address'    => null,
143
-        'user_agent'    => null,
144
-        'init_access'   => null,
145
-        'last_access'   => null,
146
-        'expiration'    => null,
147
-        'pages_visited' => array(),
148
-    );
149
-
150
-    /**
151
-     * timestamp for when last garbage collection cycle was performed
152
-     *
153
-     * @var int $_last_gc
154
-     */
155
-    private $_last_gc;
156
-
157
-    /**
158
-     * @var RequestInterface $request
159
-     */
160
-    protected $request;
161
-
162
-    /**
163
-     * whether session is active or not
164
-     *
165
-     * @var int $status
166
-     */
167
-    private $status = EE_Session::STATUS_CLOSED;
168
-
169
-
170
-    /**
171
-     * @singleton method used to instantiate class object
172
-     * @param CacheStorageInterface $cache_storage
173
-     * @param SessionLifespan|null  $lifespan
174
-     * @param RequestInterface      $request
175
-     * @param EE_Encryption         $encryption
176
-     * @return EE_Session
177
-     * @throws InvalidArgumentException
178
-     * @throws InvalidDataTypeException
179
-     * @throws InvalidInterfaceException
180
-     */
181
-    public static function instance(
182
-        CacheStorageInterface $cache_storage = null,
183
-        SessionLifespan $lifespan = null,
184
-        RequestInterface $request = null,
185
-        EE_Encryption $encryption = null
186
-    ) {
187
-        // check if class object is instantiated
188
-        // session loading is turned ON by default, but prior to the init hook, can be turned back OFF via:
189
-        // add_filter( 'FHEE_load_EE_Session', '__return_false' );
190
-        if (! self::$_instance instanceof EE_Session && apply_filters('FHEE_load_EE_Session', true)) {
191
-            self::$_instance = new self(
192
-                $cache_storage,
193
-                $lifespan,
194
-                $request,
195
-                $encryption
196
-            );
197
-        }
198
-        return self::$_instance;
199
-    }
200
-
201
-
202
-    /**
203
-     * protected constructor to prevent direct creation
204
-     *
205
-     * @param CacheStorageInterface $cache_storage
206
-     * @param SessionLifespan       $lifespan
207
-     * @param RequestInterface      $request
208
-     * @param EE_Encryption         $encryption
209
-     * @throws InvalidArgumentException
210
-     * @throws InvalidDataTypeException
211
-     * @throws InvalidInterfaceException
212
-     */
213
-    protected function __construct(
214
-        CacheStorageInterface $cache_storage,
215
-        SessionLifespan $lifespan,
216
-        RequestInterface $request,
217
-        EE_Encryption $encryption = null
218
-    ) {
219
-        // session loading is turned ON by default,
220
-        // but prior to the 'AHEE__EE_System__core_loaded_and_ready' hook
221
-        // (which currently fires on the init hook at priority 9),
222
-        // can be turned back OFF via: add_filter( 'FHEE_load_EE_Session', '__return_false' );
223
-        if (! apply_filters('FHEE_load_EE_Session', true)) {
224
-            return;
225
-        }
226
-        $this->session_lifespan = $lifespan;
227
-        $this->request = $request;
228
-        if (! defined('ESPRESSO_SESSION')) {
229
-            define('ESPRESSO_SESSION', true);
230
-        }
231
-        // retrieve session options from db
232
-        $session_settings = (array) get_option(EE_Session::OPTION_NAME_SETTINGS, array());
233
-        if (! empty($session_settings)) {
234
-            // cycle though existing session options
235
-            foreach ($session_settings as $var_name => $session_setting) {
236
-                // set values for class properties
237
-                $var_name = '_' . $var_name;
238
-                $this->{$var_name} = $session_setting;
239
-            }
240
-        }
241
-        $this->cache_storage = $cache_storage;
242
-        // are we using encryption?
243
-        $this->_use_encryption = $encryption instanceof EE_Encryption
244
-                                 && EE_Registry::instance()->CFG->admin->encode_session_data();
245
-        // encrypt data via: $this->encryption->encrypt();
246
-        $this->encryption = $encryption;
247
-        // filter hook allows outside functions/classes/plugins to change default empty cart
248
-        $extra_default_session_vars = apply_filters('FHEE__EE_Session__construct__extra_default_session_vars', array());
249
-        array_merge($this->_default_session_vars, $extra_default_session_vars);
250
-        // apply default session vars
251
-        $this->_set_defaults();
252
-        add_action('AHEE__EE_System__initialize', array($this, 'open_session'));
253
-        // check request for 'clear_session' param
254
-        add_action('AHEE__EE_Request_Handler__construct__complete', array($this, 'wp_loaded'));
255
-        // once everything is all said and done,
256
-        add_action('shutdown', array($this, 'update'), 100);
257
-        add_action('shutdown', array($this, 'garbageCollection'), 1000);
258
-        $this->configure_garbage_collection_filters();
259
-    }
260
-
261
-
262
-    /**
263
-     * @return bool
264
-     * @throws InvalidArgumentException
265
-     * @throws InvalidDataTypeException
266
-     * @throws InvalidInterfaceException
267
-     */
268
-    public static function isLoadedAndActive()
269
-    {
270
-        return did_action('AHEE__EE_System__core_loaded_and_ready')
271
-               && EE_Session::instance() instanceof EE_Session
272
-               && EE_Session::instance()->isActive();
273
-    }
274
-
275
-
276
-    /**
277
-     * @return bool
278
-     */
279
-    public function isActive()
280
-    {
281
-        return $this->status === EE_Session::STATUS_OPEN;
282
-    }
283
-
284
-
285
-    /**
286
-     * @return void
287
-     * @throws EE_Error
288
-     * @throws InvalidArgumentException
289
-     * @throws InvalidDataTypeException
290
-     * @throws InvalidInterfaceException
291
-     * @throws InvalidSessionDataException
292
-     */
293
-    public function open_session()
294
-    {
295
-        // check for existing session and retrieve it from db
296
-        if (! $this->_espresso_session()) {
297
-            // or just start a new one
298
-            $this->_create_espresso_session();
299
-        }
300
-    }
301
-
302
-
303
-    /**
304
-     * @return bool
305
-     */
306
-    public function expired()
307
-    {
308
-        return $this->_expired;
309
-    }
310
-
311
-
312
-    /**
313
-     * @return void
314
-     */
315
-    public function reset_expired()
316
-    {
317
-        $this->_expired = false;
318
-    }
319
-
320
-
321
-    /**
322
-     * @return int
323
-     */
324
-    public function expiration()
325
-    {
326
-        return $this->_expiration;
327
-    }
328
-
329
-
330
-    /**
331
-     * @return int
332
-     */
333
-    public function extension()
334
-    {
335
-        return apply_filters('FHEE__EE_Session__extend_expiration__seconds_added', 10 * MINUTE_IN_SECONDS);
336
-    }
337
-
338
-
339
-    /**
340
-     * @param int $time number of seconds to add to session expiration
341
-     */
342
-    public function extend_expiration($time = 0)
343
-    {
344
-        $time = $time ? $time : $this->extension();
345
-        $this->_expiration += absint($time);
346
-    }
347
-
348
-
349
-    /**
350
-     * @return int
351
-     */
352
-    public function lifespan()
353
-    {
354
-        return $this->session_lifespan->inSeconds();
355
-    }
356
-
357
-
358
-    /**
359
-     * This just sets some defaults for the _session data property
360
-     *
361
-     * @access private
362
-     * @return void
363
-     */
364
-    private function _set_defaults()
365
-    {
366
-        // set some defaults
367
-        foreach ($this->_default_session_vars as $key => $default_var) {
368
-            if (is_array($default_var)) {
369
-                $this->_session_data[ $key ] = array();
370
-            } else {
371
-                $this->_session_data[ $key ] = '';
372
-            }
373
-        }
374
-    }
375
-
376
-
377
-    /**
378
-     * @retrieve  session data
379
-     * @access    public
380
-     * @return    string
381
-     */
382
-    public function id()
383
-    {
384
-        return $this->_sid;
385
-    }
386
-
387
-
388
-    /**
389
-     * @param \EE_Cart $cart
390
-     * @return bool
391
-     */
392
-    public function set_cart(EE_Cart $cart)
393
-    {
394
-        $this->_session_data['cart'] = $cart;
395
-        return true;
396
-    }
397
-
398
-
399
-    /**
400
-     * reset_cart
401
-     */
402
-    public function reset_cart()
403
-    {
404
-        do_action('AHEE__EE_Session__reset_cart__before_reset', $this);
405
-        $this->_session_data['cart'] = null;
406
-    }
407
-
408
-
409
-    /**
410
-     * @return \EE_Cart
411
-     */
412
-    public function cart()
413
-    {
414
-        return isset($this->_session_data['cart']) && $this->_session_data['cart'] instanceof EE_Cart
415
-            ? $this->_session_data['cart']
416
-            : null;
417
-    }
418
-
419
-
420
-    /**
421
-     * @param \EE_Checkout $checkout
422
-     * @return bool
423
-     */
424
-    public function set_checkout(EE_Checkout $checkout)
425
-    {
426
-        $this->_session_data['checkout'] = $checkout;
427
-        return true;
428
-    }
429
-
430
-
431
-    /**
432
-     * reset_checkout
433
-     */
434
-    public function reset_checkout()
435
-    {
436
-        do_action('AHEE__EE_Session__reset_checkout__before_reset', $this);
437
-        $this->_session_data['checkout'] = null;
438
-    }
439
-
440
-
441
-    /**
442
-     * @return \EE_Checkout
443
-     */
444
-    public function checkout()
445
-    {
446
-        return isset($this->_session_data['checkout']) && $this->_session_data['checkout'] instanceof EE_Checkout
447
-            ? $this->_session_data['checkout']
448
-            : null;
449
-    }
450
-
451
-
452
-    /**
453
-     * @param \EE_Transaction $transaction
454
-     * @return bool
455
-     * @throws EE_Error
456
-     */
457
-    public function set_transaction(EE_Transaction $transaction)
458
-    {
459
-        // first remove the session from the transaction before we save the transaction in the session
460
-        $transaction->set_txn_session_data(null);
461
-        $this->_session_data['transaction'] = $transaction;
462
-        return true;
463
-    }
464
-
465
-
466
-    /**
467
-     * reset_transaction
468
-     */
469
-    public function reset_transaction()
470
-    {
471
-        do_action('AHEE__EE_Session__reset_transaction__before_reset', $this);
472
-        $this->_session_data['transaction'] = null;
473
-    }
474
-
475
-
476
-    /**
477
-     * @return \EE_Transaction
478
-     */
479
-    public function transaction()
480
-    {
481
-        return isset($this->_session_data['transaction'])
482
-               && $this->_session_data['transaction'] instanceof EE_Transaction
483
-            ? $this->_session_data['transaction']
484
-            : null;
485
-    }
486
-
487
-
488
-    /**
489
-     * retrieve session data
490
-     *
491
-     * @param null $key
492
-     * @param bool $reset_cache
493
-     * @return array
494
-     */
495
-    public function get_session_data($key = null, $reset_cache = false)
496
-    {
497
-        if ($reset_cache) {
498
-            $this->reset_cart();
499
-            $this->reset_checkout();
500
-            $this->reset_transaction();
501
-        }
502
-        if (! empty($key)) {
503
-            return isset($this->_session_data[ $key ]) ? $this->_session_data[ $key ] : null;
504
-        }
505
-        return $this->_session_data;
506
-    }
507
-
508
-
509
-    /**
510
-     * Returns TRUE on success, FALSE on fail
511
-     *
512
-     * @param array $data
513
-     * @return bool
514
-     */
515
-    public function set_session_data($data)
516
-    {
517
-        // nothing ??? bad data ??? go home!
518
-        if (empty($data) || ! is_array($data)) {
519
-            EE_Error::add_error(
520
-                esc_html__(
521
-                    'No session data or invalid session data was provided.',
522
-                    'event_espresso'
523
-                ),
524
-                __FILE__,
525
-                __FUNCTION__,
526
-                __LINE__
527
-            );
528
-            return false;
529
-        }
530
-        foreach ($data as $key => $value) {
531
-            if (isset($this->_default_session_vars[ $key ])) {
532
-                EE_Error::add_error(
533
-                    sprintf(
534
-                        esc_html__(
535
-                            'Sorry! %s is a default session datum and can not be reset.',
536
-                            'event_espresso'
537
-                        ),
538
-                        $key
539
-                    ),
540
-                    __FILE__,
541
-                    __FUNCTION__,
542
-                    __LINE__
543
-                );
544
-                return false;
545
-            }
546
-            $this->_session_data[ $key ] = $value;
547
-        }
548
-        return true;
549
-    }
550
-
551
-
552
-    /**
553
-     * @initiate session
554
-     * @access   private
555
-     * @return TRUE on success, FALSE on fail
556
-     * @throws EE_Error
557
-     * @throws InvalidArgumentException
558
-     * @throws InvalidDataTypeException
559
-     * @throws InvalidInterfaceException
560
-     * @throws InvalidSessionDataException
561
-     */
562
-    private function _espresso_session()
563
-    {
564
-        do_action('AHEE_log', __FILE__, __FUNCTION__, '');
565
-        // check that session has started
566
-        if (session_id() === '') {
567
-            // starts a new session if one doesn't already exist, or re-initiates an existing one
568
-            if (ini_get('session.save_handler') === 'user') {
569
-                $this->checkCustomSessionSaveHandler();
570
-            } else {
571
-                session_start();
572
-            }
573
-        }
574
-        $this->status = EE_Session::STATUS_OPEN;
575
-        // get our modified session ID
576
-        $this->_sid = $this->_generate_session_id();
577
-        // and the visitors IP
578
-        $this->_ip_address = $this->request->ipAddress();
579
-        // set the "user agent"
580
-        $this->_user_agent = $this->request->userAgent();
581
-        // now let's retrieve what's in the db
582
-        $session_data = $this->_retrieve_session_data();
583
-        if (! empty($session_data)) {
584
-            // get the current time in UTC
585
-            $this->_time = $this->_time !== null ? $this->_time : time();
586
-            // and reset the session expiration
587
-            $this->_expiration = isset($session_data['expiration'])
588
-                ? $session_data['expiration']
589
-                : $this->_time + $this->session_lifespan->inSeconds();
590
-        } else {
591
-            // set initial site access time and the session expiration
592
-            $this->_set_init_access_and_expiration();
593
-            // set referer
594
-            $this->_session_data['pages_visited'][ $this->_session_data['init_access'] ] = isset($_SERVER['HTTP_REFERER'])
595
-                ? esc_attr($_SERVER['HTTP_REFERER'])
596
-                : '';
597
-            // no previous session = go back and create one (on top of the data above)
598
-            return false;
599
-        }
600
-        // now the user agent
601
-        if ($session_data['user_agent'] !== $this->_user_agent) {
602
-            return false;
603
-        }
604
-        // wait a minute... how old are you?
605
-        if ($this->_time > $this->_expiration) {
606
-            // yer too old fer me!
607
-            $this->_expired = true;
608
-            // wipe out everything that isn't a default session datum
609
-            $this->clear_session(__CLASS__, __FUNCTION__);
610
-        }
611
-        // make event espresso session data available to plugin
612
-        $this->_session_data = array_merge($this->_session_data, $session_data);
613
-        return true;
614
-    }
615
-
616
-    /**
617
-     * Handles a fatal error that can occur while starting the session because of an invalid session handler.
618
-     * This fatal error happens regularly on Pantheon's servers, see https://github.com/eventespresso/event-espresso-core/issues/494.
619
-     * This function keeps track of whether we know the custom session handler works or not. If it does, uses the session
620
-     * normally.
621
-     * If we don't know if the custom session save handler works, sets an option before starting the session, and
622
-     * removes it afterwards. This way, if there is a fatal error starting the session, we'll know during the next
623
-     * request, and avoid calling the session. Instead, we'll give a helpful error message and suggest how to fix the problem.
624
-     * Lastly, we need to allow the admin to indicate when they've fixed the problem, so we can try using EE's session
625
-     * again. For that, we give them a link with "ee_retry_session" in it. When they visit a page with that querystring
626
-     * variable, we set ourselves up to retry using the session on the next request, and do a redirect.
627
-     * @since $VID:$
628
-     */
629
-    private function checkCustomSessionSaveHandler()
630
-    {
631
-        if (get_option('ee_custom_session_save_handler_proven_ok', false)) {
632
-            // we've already tried it on a previous request. The custom save handler works. Don't worry
633
-            session_start();
634
-        } else {
635
-            // we haven't successfully tried the session handler yet.
636
-            // Check if we had a fatal error last time while trying it
637
-            if (get_option('ee_custom_session_save_handler_died', false)) {
638
-                $request = LoaderFactory::getLoader()->load('EventEspresso\core\services\request\Request');
639
-                if ($request->requestParamIsSet('ee_retry_session')) {
640
-                    delete_option('ee_custom_session_save_handler_died');
641
-                    // remove "ee_retry_session", otherwise, if the problem still isn't fixed, we'll just keep getting
642
-                    // the fatal error over and over. Better to remove it and redirect, and try on the next request
643
-                    wp_redirect(
644
-                        remove_query_arg(array('ee_retry_session'), EEH_URL::current_url())
645
-                    );
646
-                    die;
647
-                } else {
648
-                    // apparently, last time we tried using the custom session save handler, there was a fatal
649
-                    // so don't try it
650
-                    // just show a message to users that can fix it. Otherwise, try to hobble along without the session
651
-                    if (current_user_can('install_plugins')) {
652
-                        EE_Error::add_error(
653
-                            sprintf(
654
-                                esc_html__('It appears there was a fatal error while starting the session, so Event Espresso is not able to process registrations normally. Some hosting companies, like Pantheon, require an extra plugin for Event Espresso to work. Please install the %1$sWordPress Native PHP Sessions plugin%2$s, then %3$sclick here to check if the problem is resolved.%2$s', 'event_espresso'),
655
-                                '<a href="https://wordpress.org/plugins/wp-native-php-sessions/">',
656
-                                '</a>',
657
-                                '<a href="' . add_query_arg(array('ee_retry_session' => true), EEH_URL::current_url()) . '">'
658
-                            ),
659
-                            __FILE__,
660
-                            __FUNCTION__,
661
-                            __LINE__
662
-                        );
663
-                    }
664
-                }
665
-            } else {
666
-                // there is no record of having a fatal error while trying the custom session save handler
667
-                // so let's try it
668
-                // there is a custom session save handler. Proceed with caution
669
-                update_option('ee_custom_session_save_handler_died', '1');
670
-                // hold your breath, if the custom session save handler might cause a fatal here...
671
-                session_start();
672
-                // phew! we made it! the custom session handler is a-ok
673
-                update_option('ee_custom_session_save_handler_proven_ok', '1');
674
-            }
675
-        }
676
-    }
677
-
678
-
679
-    /**
680
-     * _get_session_data
681
-     * Retrieves the session data, and attempts to correct any encoding issues that can occur due to improperly setup
682
-     * databases
683
-     *
684
-     * @return array
685
-     * @throws EE_Error
686
-     * @throws InvalidArgumentException
687
-     * @throws InvalidSessionDataException
688
-     * @throws InvalidDataTypeException
689
-     * @throws InvalidInterfaceException
690
-     */
691
-    protected function _retrieve_session_data()
692
-    {
693
-        $ssn_key = EE_Session::session_id_prefix . $this->_sid;
694
-        try {
695
-            // we're using WP's Transient API to store session data using the PHP session ID as the option name
696
-            $session_data = $this->cache_storage->get($ssn_key, false);
697
-            if (empty($session_data)) {
698
-                return array();
699
-            }
700
-            if (apply_filters('FHEE__EE_Session___perform_session_id_hash_check', WP_DEBUG)) {
701
-                $hash_check = $this->cache_storage->get(
702
-                    EE_Session::hash_check_prefix . $this->_sid,
703
-                    false
704
-                );
705
-                if ($hash_check && $hash_check !== md5($session_data)) {
706
-                    EE_Error::add_error(
707
-                        sprintf(
708
-                            __(
709
-                                'The stored data for session %1$s failed to pass a hash check and therefore appears to be invalid.',
710
-                                'event_espresso'
711
-                            ),
712
-                            EE_Session::session_id_prefix . $this->_sid
713
-                        ),
714
-                        __FILE__,
715
-                        __FUNCTION__,
716
-                        __LINE__
717
-                    );
718
-                }
719
-            }
720
-        } catch (Exception $e) {
721
-            // let's just eat that error for now and attempt to correct any corrupted data
722
-            global $wpdb;
723
-            $row = $wpdb->get_row(
724
-                $wpdb->prepare(
725
-                    "SELECT option_value FROM {$wpdb->options} WHERE option_name = %s LIMIT 1",
726
-                    '_transient_' . $ssn_key
727
-                )
728
-            );
729
-            $session_data = is_object($row) ? $row->option_value : null;
730
-            if ($session_data) {
731
-                $session_data = preg_replace_callback(
732
-                    '!s:(d+):"(.*?)";!',
733
-                    function ($match) {
734
-                        return $match[1] === strlen($match[2])
735
-                            ? $match[0]
736
-                            : 's:' . strlen($match[2]) . ':"' . $match[2] . '";';
737
-                    },
738
-                    $session_data
739
-                );
740
-            }
741
-            $session_data = maybe_unserialize($session_data);
742
-        }
743
-        // in case the data is encoded... try to decode it
744
-        $session_data = $this->encryption instanceof EE_Encryption
745
-            ? $this->encryption->base64_string_decode($session_data)
746
-            : $session_data;
747
-        if (! is_array($session_data)) {
748
-            try {
749
-                $session_data = maybe_unserialize($session_data);
750
-            } catch (Exception $e) {
751
-                $msg = esc_html__(
752
-                    'An error occurred while attempting to unserialize the session data.',
753
-                    'event_espresso'
754
-                );
755
-                $msg .= WP_DEBUG
756
-                    ? '<br><pre>'
757
-                      . print_r($session_data, true)
758
-                      . '</pre><br>'
759
-                      . $this->find_serialize_error($session_data)
760
-                    : '';
761
-                $this->cache_storage->delete(EE_Session::session_id_prefix . $this->_sid);
762
-                throw new InvalidSessionDataException($msg, 0, $e);
763
-            }
764
-        }
765
-        // just a check to make sure the session array is indeed an array
766
-        if (! is_array($session_data)) {
767
-            // no?!?! then something is wrong
768
-            $msg = esc_html__(
769
-                'The session data is missing, invalid, or corrupted.',
770
-                'event_espresso'
771
-            );
772
-            $msg .= WP_DEBUG
773
-                ? '<br><pre>' . print_r($session_data, true) . '</pre><br>' . $this->find_serialize_error($session_data)
774
-                : '';
775
-            $this->cache_storage->delete(EE_Session::session_id_prefix . $this->_sid);
776
-            throw new InvalidSessionDataException($msg);
777
-        }
778
-        if (isset($session_data['transaction']) && absint($session_data['transaction']) !== 0) {
779
-            $session_data['transaction'] = EEM_Transaction::instance()->get_one_by_ID(
780
-                $session_data['transaction']
781
-            );
782
-        }
783
-        return $session_data;
784
-    }
785
-
786
-
787
-    /**
788
-     * _generate_session_id
789
-     * Retrieves the PHP session id either directly from the PHP session,
790
-     * or from the $_REQUEST array if it was passed in from an AJAX request.
791
-     * The session id is then salted and hashed (mmm sounds tasty)
792
-     * so that it can be safely used as a $_REQUEST param
793
-     *
794
-     * @return string
795
-     */
796
-    protected function _generate_session_id()
797
-    {
798
-        // check if the SID was passed explicitly, otherwise get from session, then add salt and hash it to reduce length
799
-        if (isset($_REQUEST['EESID'])) {
800
-            $session_id = sanitize_text_field($_REQUEST['EESID']);
801
-        } else {
802
-            $session_id = md5(session_id() . get_current_blog_id() . $this->_get_sid_salt());
803
-        }
804
-        return apply_filters('FHEE__EE_Session___generate_session_id__session_id', $session_id);
805
-    }
806
-
807
-
808
-    /**
809
-     * _get_sid_salt
810
-     *
811
-     * @return string
812
-     */
813
-    protected function _get_sid_salt()
814
-    {
815
-        // was session id salt already saved to db ?
816
-        if (empty($this->_sid_salt)) {
817
-            // no?  then maybe use WP defined constant
818
-            if (defined('AUTH_SALT')) {
819
-                $this->_sid_salt = AUTH_SALT;
820
-            }
821
-            // if salt doesn't exist or is too short
822
-            if (strlen($this->_sid_salt) < 32) {
823
-                // create a new one
824
-                $this->_sid_salt = wp_generate_password(64);
825
-            }
826
-            // and save it as a permanent session setting
827
-            $this->updateSessionSettings(array('sid_salt' => $this->_sid_salt));
828
-        }
829
-        return $this->_sid_salt;
830
-    }
831
-
832
-
833
-    /**
834
-     * _set_init_access_and_expiration
835
-     *
836
-     * @return void
837
-     */
838
-    protected function _set_init_access_and_expiration()
839
-    {
840
-        $this->_time = time();
841
-        $this->_expiration = $this->_time + $this->session_lifespan->inSeconds();
842
-        // set initial site access time
843
-        $this->_session_data['init_access'] = $this->_time;
844
-        // and the session expiration
845
-        $this->_session_data['expiration'] = $this->_expiration;
846
-    }
847
-
848
-
849
-    /**
850
-     * @update session data  prior to saving to the db
851
-     * @access public
852
-     * @param bool $new_session
853
-     * @return TRUE on success, FALSE on fail
854
-     * @throws EE_Error
855
-     * @throws InvalidArgumentException
856
-     * @throws InvalidDataTypeException
857
-     * @throws InvalidInterfaceException
858
-     */
859
-    public function update($new_session = false)
860
-    {
861
-        $this->_session_data = $this->_session_data !== null
862
-                               && is_array($this->_session_data)
863
-                               && isset($this->_session_data['id'])
864
-            ? $this->_session_data
865
-            : array();
866
-        if (empty($this->_session_data)) {
867
-            $this->_set_defaults();
868
-        }
869
-        $session_data = array();
870
-        foreach ($this->_session_data as $key => $value) {
871
-            switch ($key) {
872
-                case 'id':
873
-                    // session ID
874
-                    $session_data['id'] = $this->_sid;
875
-                    break;
876
-                case 'ip_address':
877
-                    // visitor ip address
878
-                    $session_data['ip_address'] = $this->request->ipAddress();
879
-                    break;
880
-                case 'user_agent':
881
-                    // visitor user_agent
882
-                    $session_data['user_agent'] = $this->_user_agent;
883
-                    break;
884
-                case 'init_access':
885
-                    $session_data['init_access'] = absint($value);
886
-                    break;
887
-                case 'last_access':
888
-                    // current access time
889
-                    $session_data['last_access'] = $this->_time;
890
-                    break;
891
-                case 'expiration':
892
-                    // when the session expires
893
-                    $session_data['expiration'] = ! empty($this->_expiration)
894
-                        ? $this->_expiration
895
-                        : $session_data['init_access'] + $this->session_lifespan->inSeconds();
896
-                    break;
897
-                case 'user_id':
898
-                    // current user if logged in
899
-                    $session_data['user_id'] = $this->_wp_user_id();
900
-                    break;
901
-                case 'pages_visited':
902
-                    $page_visit = $this->_get_page_visit();
903
-                    if ($page_visit) {
904
-                        // set pages visited where the first will be the http referrer
905
-                        $this->_session_data['pages_visited'][ $this->_time ] = $page_visit;
906
-                        // we'll only save the last 10 page visits.
907
-                        $session_data['pages_visited'] = array_slice($this->_session_data['pages_visited'], -10);
908
-                    }
909
-                    break;
910
-                default:
911
-                    // carry any other data over
912
-                    $session_data[ $key ] = $this->_session_data[ $key ];
913
-            }
914
-        }
915
-        $this->_session_data = $session_data;
916
-        // creating a new session does not require saving to the db just yet
917
-        if (! $new_session) {
918
-            // ready? let's save
919
-            if ($this->_save_session_to_db()) {
920
-                return true;
921
-            }
922
-            return false;
923
-        }
924
-        // meh, why not?
925
-        return true;
926
-    }
927
-
928
-
929
-    /**
930
-     * @create session data array
931
-     * @access public
932
-     * @return bool
933
-     * @throws EE_Error
934
-     * @throws InvalidArgumentException
935
-     * @throws InvalidDataTypeException
936
-     * @throws InvalidInterfaceException
937
-     */
938
-    private function _create_espresso_session()
939
-    {
940
-        do_action('AHEE_log', __CLASS__, __FUNCTION__, '');
941
-        // use the update function for now with $new_session arg set to TRUE
942
-        return $this->update(true) ? true : false;
943
-    }
944
-
945
-
946
-    /**
947
-     * _save_session_to_db
948
-     *
949
-     * @param bool $clear_session
950
-     * @return string
951
-     * @throws EE_Error
952
-     * @throws InvalidArgumentException
953
-     * @throws InvalidDataTypeException
954
-     * @throws InvalidInterfaceException
955
-     */
956
-    private function _save_session_to_db($clear_session = false)
957
-    {
958
-        // don't save sessions for crawlers
959
-        // and unless we're deleting the session data, don't save anything if there isn't a cart
960
-        if ($this->request->isBot()
961
-            || (
962
-                ! $clear_session
963
-                && ! $this->cart() instanceof EE_Cart
964
-                && apply_filters('FHEE__EE_Session___save_session_to_db__abort_session_save', true)
965
-            )
966
-        ) {
967
-            return false;
968
-        }
969
-        $transaction = $this->transaction();
970
-        if ($transaction instanceof EE_Transaction) {
971
-            if (! $transaction->ID()) {
972
-                $transaction->save();
973
-            }
974
-            $this->_session_data['transaction'] = $transaction->ID();
975
-        }
976
-        // then serialize all of our session data
977
-        $session_data = serialize($this->_session_data);
978
-        // do we need to also encode it to avoid corrupted data when saved to the db?
979
-        $session_data = $this->_use_encryption
980
-            ? $this->encryption->base64_string_encode($session_data)
981
-            : $session_data;
982
-        // maybe save hash check
983
-        if (apply_filters('FHEE__EE_Session___perform_session_id_hash_check', WP_DEBUG)) {
984
-            $this->cache_storage->add(
985
-                EE_Session::hash_check_prefix . $this->_sid,
986
-                md5($session_data),
987
-                $this->session_lifespan->inSeconds()
988
-            );
989
-        }
990
-        // we're using the Transient API for storing session data,
991
-        return $this->cache_storage->add(
992
-            EE_Session::session_id_prefix . $this->_sid,
993
-            $session_data,
994
-            $this->session_lifespan->inSeconds()
995
-        );
996
-    }
997
-
998
-
999
-    /**
1000
-     * @get    the full page request the visitor is accessing
1001
-     * @access public
1002
-     * @return string
1003
-     */
1004
-    public function _get_page_visit()
1005
-    {
1006
-        $page_visit = home_url('/') . 'wp-admin/admin-ajax.php';
1007
-        // check for request url
1008
-        if (isset($_SERVER['REQUEST_URI'])) {
1009
-            $http_host = '';
1010
-            $page_id = '?';
1011
-            $e_reg = '';
1012
-            $request_uri = esc_url($_SERVER['REQUEST_URI']);
1013
-            $ru_bits = explode('?', $request_uri);
1014
-            $request_uri = $ru_bits[0];
1015
-            // check for and grab host as well
1016
-            if (isset($_SERVER['HTTP_HOST'])) {
1017
-                $http_host = esc_url($_SERVER['HTTP_HOST']);
1018
-            }
1019
-            // check for page_id in SERVER REQUEST
1020
-            if (isset($_REQUEST['page_id'])) {
1021
-                // rebuild $e_reg without any of the extra parameters
1022
-                $page_id = '?page_id=' . esc_attr($_REQUEST['page_id']) . '&amp;';
1023
-            }
1024
-            // check for $e_reg in SERVER REQUEST
1025
-            if (isset($_REQUEST['ee'])) {
1026
-                // rebuild $e_reg without any of the extra parameters
1027
-                $e_reg = 'ee=' . esc_attr($_REQUEST['ee']);
1028
-            }
1029
-            $page_visit = rtrim($http_host . $request_uri . $page_id . $e_reg, '?');
1030
-        }
1031
-        return $page_visit !== home_url('/wp-admin/admin-ajax.php') ? $page_visit : '';
1032
-    }
1033
-
1034
-
1035
-    /**
1036
-     * @the    current wp user id
1037
-     * @access public
1038
-     * @return int
1039
-     */
1040
-    public function _wp_user_id()
1041
-    {
1042
-        // if I need to explain the following lines of code, then you shouldn't be looking at this!
1043
-        $this->_wp_user_id = get_current_user_id();
1044
-        return $this->_wp_user_id;
1045
-    }
1046
-
1047
-
1048
-    /**
1049
-     * Clear EE_Session data
1050
-     *
1051
-     * @access public
1052
-     * @param string $class
1053
-     * @param string $function
1054
-     * @return void
1055
-     * @throws EE_Error
1056
-     * @throws InvalidArgumentException
1057
-     * @throws InvalidDataTypeException
1058
-     * @throws InvalidInterfaceException
1059
-     */
1060
-    public function clear_session($class = '', $function = '')
1061
-    {
27
+	const session_id_prefix = 'ee_ssn_';
28
+
29
+	const hash_check_prefix = 'ee_shc_';
30
+
31
+	const OPTION_NAME_SETTINGS = 'ee_session_settings';
32
+
33
+	const STATUS_CLOSED = 0;
34
+
35
+	const STATUS_OPEN = 1;
36
+
37
+	/**
38
+	 * instance of the EE_Session object
39
+	 *
40
+	 * @var EE_Session
41
+	 */
42
+	private static $_instance;
43
+
44
+	/**
45
+	 * @var CacheStorageInterface $cache_storage
46
+	 */
47
+	protected $cache_storage;
48
+
49
+	/**
50
+	 * EE_Encryption object
51
+	 *
52
+	 * @var EE_Encryption
53
+	 */
54
+	protected $encryption;
55
+
56
+	/**
57
+	 * the session id
58
+	 *
59
+	 * @var string
60
+	 */
61
+	private $_sid;
62
+
63
+	/**
64
+	 * session id salt
65
+	 *
66
+	 * @var string
67
+	 */
68
+	private $_sid_salt;
69
+
70
+	/**
71
+	 * session data
72
+	 *
73
+	 * @var array
74
+	 */
75
+	private $_session_data = array();
76
+
77
+	/**
78
+	 * how long an EE session lasts
79
+	 * default session lifespan of 1 hour (for not so instant IPNs)
80
+	 *
81
+	 * @var SessionLifespan $session_lifespan
82
+	 */
83
+	private $session_lifespan;
84
+
85
+	/**
86
+	 * session expiration time as Unix timestamp in GMT
87
+	 *
88
+	 * @var int
89
+	 */
90
+	private $_expiration;
91
+
92
+	/**
93
+	 * whether or not session has expired at some point
94
+	 *
95
+	 * @var boolean
96
+	 */
97
+	private $_expired = false;
98
+
99
+	/**
100
+	 * current time as Unix timestamp in GMT
101
+	 *
102
+	 * @var int
103
+	 */
104
+	private $_time;
105
+
106
+	/**
107
+	 * whether to encrypt session data
108
+	 *
109
+	 * @var bool
110
+	 */
111
+	private $_use_encryption;
112
+
113
+	/**
114
+	 * well... according to the server...
115
+	 *
116
+	 * @var null
117
+	 */
118
+	private $_user_agent;
119
+
120
+	/**
121
+	 * do you really trust the server ?
122
+	 *
123
+	 * @var null
124
+	 */
125
+	private $_ip_address;
126
+
127
+	/**
128
+	 * current WP user_id
129
+	 *
130
+	 * @var null
131
+	 */
132
+	private $_wp_user_id;
133
+
134
+	/**
135
+	 * array for defining default session vars
136
+	 *
137
+	 * @var array
138
+	 */
139
+	private $_default_session_vars = array(
140
+		'id'            => null,
141
+		'user_id'       => null,
142
+		'ip_address'    => null,
143
+		'user_agent'    => null,
144
+		'init_access'   => null,
145
+		'last_access'   => null,
146
+		'expiration'    => null,
147
+		'pages_visited' => array(),
148
+	);
149
+
150
+	/**
151
+	 * timestamp for when last garbage collection cycle was performed
152
+	 *
153
+	 * @var int $_last_gc
154
+	 */
155
+	private $_last_gc;
156
+
157
+	/**
158
+	 * @var RequestInterface $request
159
+	 */
160
+	protected $request;
161
+
162
+	/**
163
+	 * whether session is active or not
164
+	 *
165
+	 * @var int $status
166
+	 */
167
+	private $status = EE_Session::STATUS_CLOSED;
168
+
169
+
170
+	/**
171
+	 * @singleton method used to instantiate class object
172
+	 * @param CacheStorageInterface $cache_storage
173
+	 * @param SessionLifespan|null  $lifespan
174
+	 * @param RequestInterface      $request
175
+	 * @param EE_Encryption         $encryption
176
+	 * @return EE_Session
177
+	 * @throws InvalidArgumentException
178
+	 * @throws InvalidDataTypeException
179
+	 * @throws InvalidInterfaceException
180
+	 */
181
+	public static function instance(
182
+		CacheStorageInterface $cache_storage = null,
183
+		SessionLifespan $lifespan = null,
184
+		RequestInterface $request = null,
185
+		EE_Encryption $encryption = null
186
+	) {
187
+		// check if class object is instantiated
188
+		// session loading is turned ON by default, but prior to the init hook, can be turned back OFF via:
189
+		// add_filter( 'FHEE_load_EE_Session', '__return_false' );
190
+		if (! self::$_instance instanceof EE_Session && apply_filters('FHEE_load_EE_Session', true)) {
191
+			self::$_instance = new self(
192
+				$cache_storage,
193
+				$lifespan,
194
+				$request,
195
+				$encryption
196
+			);
197
+		}
198
+		return self::$_instance;
199
+	}
200
+
201
+
202
+	/**
203
+	 * protected constructor to prevent direct creation
204
+	 *
205
+	 * @param CacheStorageInterface $cache_storage
206
+	 * @param SessionLifespan       $lifespan
207
+	 * @param RequestInterface      $request
208
+	 * @param EE_Encryption         $encryption
209
+	 * @throws InvalidArgumentException
210
+	 * @throws InvalidDataTypeException
211
+	 * @throws InvalidInterfaceException
212
+	 */
213
+	protected function __construct(
214
+		CacheStorageInterface $cache_storage,
215
+		SessionLifespan $lifespan,
216
+		RequestInterface $request,
217
+		EE_Encryption $encryption = null
218
+	) {
219
+		// session loading is turned ON by default,
220
+		// but prior to the 'AHEE__EE_System__core_loaded_and_ready' hook
221
+		// (which currently fires on the init hook at priority 9),
222
+		// can be turned back OFF via: add_filter( 'FHEE_load_EE_Session', '__return_false' );
223
+		if (! apply_filters('FHEE_load_EE_Session', true)) {
224
+			return;
225
+		}
226
+		$this->session_lifespan = $lifespan;
227
+		$this->request = $request;
228
+		if (! defined('ESPRESSO_SESSION')) {
229
+			define('ESPRESSO_SESSION', true);
230
+		}
231
+		// retrieve session options from db
232
+		$session_settings = (array) get_option(EE_Session::OPTION_NAME_SETTINGS, array());
233
+		if (! empty($session_settings)) {
234
+			// cycle though existing session options
235
+			foreach ($session_settings as $var_name => $session_setting) {
236
+				// set values for class properties
237
+				$var_name = '_' . $var_name;
238
+				$this->{$var_name} = $session_setting;
239
+			}
240
+		}
241
+		$this->cache_storage = $cache_storage;
242
+		// are we using encryption?
243
+		$this->_use_encryption = $encryption instanceof EE_Encryption
244
+								 && EE_Registry::instance()->CFG->admin->encode_session_data();
245
+		// encrypt data via: $this->encryption->encrypt();
246
+		$this->encryption = $encryption;
247
+		// filter hook allows outside functions/classes/plugins to change default empty cart
248
+		$extra_default_session_vars = apply_filters('FHEE__EE_Session__construct__extra_default_session_vars', array());
249
+		array_merge($this->_default_session_vars, $extra_default_session_vars);
250
+		// apply default session vars
251
+		$this->_set_defaults();
252
+		add_action('AHEE__EE_System__initialize', array($this, 'open_session'));
253
+		// check request for 'clear_session' param
254
+		add_action('AHEE__EE_Request_Handler__construct__complete', array($this, 'wp_loaded'));
255
+		// once everything is all said and done,
256
+		add_action('shutdown', array($this, 'update'), 100);
257
+		add_action('shutdown', array($this, 'garbageCollection'), 1000);
258
+		$this->configure_garbage_collection_filters();
259
+	}
260
+
261
+
262
+	/**
263
+	 * @return bool
264
+	 * @throws InvalidArgumentException
265
+	 * @throws InvalidDataTypeException
266
+	 * @throws InvalidInterfaceException
267
+	 */
268
+	public static function isLoadedAndActive()
269
+	{
270
+		return did_action('AHEE__EE_System__core_loaded_and_ready')
271
+			   && EE_Session::instance() instanceof EE_Session
272
+			   && EE_Session::instance()->isActive();
273
+	}
274
+
275
+
276
+	/**
277
+	 * @return bool
278
+	 */
279
+	public function isActive()
280
+	{
281
+		return $this->status === EE_Session::STATUS_OPEN;
282
+	}
283
+
284
+
285
+	/**
286
+	 * @return void
287
+	 * @throws EE_Error
288
+	 * @throws InvalidArgumentException
289
+	 * @throws InvalidDataTypeException
290
+	 * @throws InvalidInterfaceException
291
+	 * @throws InvalidSessionDataException
292
+	 */
293
+	public function open_session()
294
+	{
295
+		// check for existing session and retrieve it from db
296
+		if (! $this->_espresso_session()) {
297
+			// or just start a new one
298
+			$this->_create_espresso_session();
299
+		}
300
+	}
301
+
302
+
303
+	/**
304
+	 * @return bool
305
+	 */
306
+	public function expired()
307
+	{
308
+		return $this->_expired;
309
+	}
310
+
311
+
312
+	/**
313
+	 * @return void
314
+	 */
315
+	public function reset_expired()
316
+	{
317
+		$this->_expired = false;
318
+	}
319
+
320
+
321
+	/**
322
+	 * @return int
323
+	 */
324
+	public function expiration()
325
+	{
326
+		return $this->_expiration;
327
+	}
328
+
329
+
330
+	/**
331
+	 * @return int
332
+	 */
333
+	public function extension()
334
+	{
335
+		return apply_filters('FHEE__EE_Session__extend_expiration__seconds_added', 10 * MINUTE_IN_SECONDS);
336
+	}
337
+
338
+
339
+	/**
340
+	 * @param int $time number of seconds to add to session expiration
341
+	 */
342
+	public function extend_expiration($time = 0)
343
+	{
344
+		$time = $time ? $time : $this->extension();
345
+		$this->_expiration += absint($time);
346
+	}
347
+
348
+
349
+	/**
350
+	 * @return int
351
+	 */
352
+	public function lifespan()
353
+	{
354
+		return $this->session_lifespan->inSeconds();
355
+	}
356
+
357
+
358
+	/**
359
+	 * This just sets some defaults for the _session data property
360
+	 *
361
+	 * @access private
362
+	 * @return void
363
+	 */
364
+	private function _set_defaults()
365
+	{
366
+		// set some defaults
367
+		foreach ($this->_default_session_vars as $key => $default_var) {
368
+			if (is_array($default_var)) {
369
+				$this->_session_data[ $key ] = array();
370
+			} else {
371
+				$this->_session_data[ $key ] = '';
372
+			}
373
+		}
374
+	}
375
+
376
+
377
+	/**
378
+	 * @retrieve  session data
379
+	 * @access    public
380
+	 * @return    string
381
+	 */
382
+	public function id()
383
+	{
384
+		return $this->_sid;
385
+	}
386
+
387
+
388
+	/**
389
+	 * @param \EE_Cart $cart
390
+	 * @return bool
391
+	 */
392
+	public function set_cart(EE_Cart $cart)
393
+	{
394
+		$this->_session_data['cart'] = $cart;
395
+		return true;
396
+	}
397
+
398
+
399
+	/**
400
+	 * reset_cart
401
+	 */
402
+	public function reset_cart()
403
+	{
404
+		do_action('AHEE__EE_Session__reset_cart__before_reset', $this);
405
+		$this->_session_data['cart'] = null;
406
+	}
407
+
408
+
409
+	/**
410
+	 * @return \EE_Cart
411
+	 */
412
+	public function cart()
413
+	{
414
+		return isset($this->_session_data['cart']) && $this->_session_data['cart'] instanceof EE_Cart
415
+			? $this->_session_data['cart']
416
+			: null;
417
+	}
418
+
419
+
420
+	/**
421
+	 * @param \EE_Checkout $checkout
422
+	 * @return bool
423
+	 */
424
+	public function set_checkout(EE_Checkout $checkout)
425
+	{
426
+		$this->_session_data['checkout'] = $checkout;
427
+		return true;
428
+	}
429
+
430
+
431
+	/**
432
+	 * reset_checkout
433
+	 */
434
+	public function reset_checkout()
435
+	{
436
+		do_action('AHEE__EE_Session__reset_checkout__before_reset', $this);
437
+		$this->_session_data['checkout'] = null;
438
+	}
439
+
440
+
441
+	/**
442
+	 * @return \EE_Checkout
443
+	 */
444
+	public function checkout()
445
+	{
446
+		return isset($this->_session_data['checkout']) && $this->_session_data['checkout'] instanceof EE_Checkout
447
+			? $this->_session_data['checkout']
448
+			: null;
449
+	}
450
+
451
+
452
+	/**
453
+	 * @param \EE_Transaction $transaction
454
+	 * @return bool
455
+	 * @throws EE_Error
456
+	 */
457
+	public function set_transaction(EE_Transaction $transaction)
458
+	{
459
+		// first remove the session from the transaction before we save the transaction in the session
460
+		$transaction->set_txn_session_data(null);
461
+		$this->_session_data['transaction'] = $transaction;
462
+		return true;
463
+	}
464
+
465
+
466
+	/**
467
+	 * reset_transaction
468
+	 */
469
+	public function reset_transaction()
470
+	{
471
+		do_action('AHEE__EE_Session__reset_transaction__before_reset', $this);
472
+		$this->_session_data['transaction'] = null;
473
+	}
474
+
475
+
476
+	/**
477
+	 * @return \EE_Transaction
478
+	 */
479
+	public function transaction()
480
+	{
481
+		return isset($this->_session_data['transaction'])
482
+			   && $this->_session_data['transaction'] instanceof EE_Transaction
483
+			? $this->_session_data['transaction']
484
+			: null;
485
+	}
486
+
487
+
488
+	/**
489
+	 * retrieve session data
490
+	 *
491
+	 * @param null $key
492
+	 * @param bool $reset_cache
493
+	 * @return array
494
+	 */
495
+	public function get_session_data($key = null, $reset_cache = false)
496
+	{
497
+		if ($reset_cache) {
498
+			$this->reset_cart();
499
+			$this->reset_checkout();
500
+			$this->reset_transaction();
501
+		}
502
+		if (! empty($key)) {
503
+			return isset($this->_session_data[ $key ]) ? $this->_session_data[ $key ] : null;
504
+		}
505
+		return $this->_session_data;
506
+	}
507
+
508
+
509
+	/**
510
+	 * Returns TRUE on success, FALSE on fail
511
+	 *
512
+	 * @param array $data
513
+	 * @return bool
514
+	 */
515
+	public function set_session_data($data)
516
+	{
517
+		// nothing ??? bad data ??? go home!
518
+		if (empty($data) || ! is_array($data)) {
519
+			EE_Error::add_error(
520
+				esc_html__(
521
+					'No session data or invalid session data was provided.',
522
+					'event_espresso'
523
+				),
524
+				__FILE__,
525
+				__FUNCTION__,
526
+				__LINE__
527
+			);
528
+			return false;
529
+		}
530
+		foreach ($data as $key => $value) {
531
+			if (isset($this->_default_session_vars[ $key ])) {
532
+				EE_Error::add_error(
533
+					sprintf(
534
+						esc_html__(
535
+							'Sorry! %s is a default session datum and can not be reset.',
536
+							'event_espresso'
537
+						),
538
+						$key
539
+					),
540
+					__FILE__,
541
+					__FUNCTION__,
542
+					__LINE__
543
+				);
544
+				return false;
545
+			}
546
+			$this->_session_data[ $key ] = $value;
547
+		}
548
+		return true;
549
+	}
550
+
551
+
552
+	/**
553
+	 * @initiate session
554
+	 * @access   private
555
+	 * @return TRUE on success, FALSE on fail
556
+	 * @throws EE_Error
557
+	 * @throws InvalidArgumentException
558
+	 * @throws InvalidDataTypeException
559
+	 * @throws InvalidInterfaceException
560
+	 * @throws InvalidSessionDataException
561
+	 */
562
+	private function _espresso_session()
563
+	{
564
+		do_action('AHEE_log', __FILE__, __FUNCTION__, '');
565
+		// check that session has started
566
+		if (session_id() === '') {
567
+			// starts a new session if one doesn't already exist, or re-initiates an existing one
568
+			if (ini_get('session.save_handler') === 'user') {
569
+				$this->checkCustomSessionSaveHandler();
570
+			} else {
571
+				session_start();
572
+			}
573
+		}
574
+		$this->status = EE_Session::STATUS_OPEN;
575
+		// get our modified session ID
576
+		$this->_sid = $this->_generate_session_id();
577
+		// and the visitors IP
578
+		$this->_ip_address = $this->request->ipAddress();
579
+		// set the "user agent"
580
+		$this->_user_agent = $this->request->userAgent();
581
+		// now let's retrieve what's in the db
582
+		$session_data = $this->_retrieve_session_data();
583
+		if (! empty($session_data)) {
584
+			// get the current time in UTC
585
+			$this->_time = $this->_time !== null ? $this->_time : time();
586
+			// and reset the session expiration
587
+			$this->_expiration = isset($session_data['expiration'])
588
+				? $session_data['expiration']
589
+				: $this->_time + $this->session_lifespan->inSeconds();
590
+		} else {
591
+			// set initial site access time and the session expiration
592
+			$this->_set_init_access_and_expiration();
593
+			// set referer
594
+			$this->_session_data['pages_visited'][ $this->_session_data['init_access'] ] = isset($_SERVER['HTTP_REFERER'])
595
+				? esc_attr($_SERVER['HTTP_REFERER'])
596
+				: '';
597
+			// no previous session = go back and create one (on top of the data above)
598
+			return false;
599
+		}
600
+		// now the user agent
601
+		if ($session_data['user_agent'] !== $this->_user_agent) {
602
+			return false;
603
+		}
604
+		// wait a minute... how old are you?
605
+		if ($this->_time > $this->_expiration) {
606
+			// yer too old fer me!
607
+			$this->_expired = true;
608
+			// wipe out everything that isn't a default session datum
609
+			$this->clear_session(__CLASS__, __FUNCTION__);
610
+		}
611
+		// make event espresso session data available to plugin
612
+		$this->_session_data = array_merge($this->_session_data, $session_data);
613
+		return true;
614
+	}
615
+
616
+	/**
617
+	 * Handles a fatal error that can occur while starting the session because of an invalid session handler.
618
+	 * This fatal error happens regularly on Pantheon's servers, see https://github.com/eventespresso/event-espresso-core/issues/494.
619
+	 * This function keeps track of whether we know the custom session handler works or not. If it does, uses the session
620
+	 * normally.
621
+	 * If we don't know if the custom session save handler works, sets an option before starting the session, and
622
+	 * removes it afterwards. This way, if there is a fatal error starting the session, we'll know during the next
623
+	 * request, and avoid calling the session. Instead, we'll give a helpful error message and suggest how to fix the problem.
624
+	 * Lastly, we need to allow the admin to indicate when they've fixed the problem, so we can try using EE's session
625
+	 * again. For that, we give them a link with "ee_retry_session" in it. When they visit a page with that querystring
626
+	 * variable, we set ourselves up to retry using the session on the next request, and do a redirect.
627
+	 * @since $VID:$
628
+	 */
629
+	private function checkCustomSessionSaveHandler()
630
+	{
631
+		if (get_option('ee_custom_session_save_handler_proven_ok', false)) {
632
+			// we've already tried it on a previous request. The custom save handler works. Don't worry
633
+			session_start();
634
+		} else {
635
+			// we haven't successfully tried the session handler yet.
636
+			// Check if we had a fatal error last time while trying it
637
+			if (get_option('ee_custom_session_save_handler_died', false)) {
638
+				$request = LoaderFactory::getLoader()->load('EventEspresso\core\services\request\Request');
639
+				if ($request->requestParamIsSet('ee_retry_session')) {
640
+					delete_option('ee_custom_session_save_handler_died');
641
+					// remove "ee_retry_session", otherwise, if the problem still isn't fixed, we'll just keep getting
642
+					// the fatal error over and over. Better to remove it and redirect, and try on the next request
643
+					wp_redirect(
644
+						remove_query_arg(array('ee_retry_session'), EEH_URL::current_url())
645
+					);
646
+					die;
647
+				} else {
648
+					// apparently, last time we tried using the custom session save handler, there was a fatal
649
+					// so don't try it
650
+					// just show a message to users that can fix it. Otherwise, try to hobble along without the session
651
+					if (current_user_can('install_plugins')) {
652
+						EE_Error::add_error(
653
+							sprintf(
654
+								esc_html__('It appears there was a fatal error while starting the session, so Event Espresso is not able to process registrations normally. Some hosting companies, like Pantheon, require an extra plugin for Event Espresso to work. Please install the %1$sWordPress Native PHP Sessions plugin%2$s, then %3$sclick here to check if the problem is resolved.%2$s', 'event_espresso'),
655
+								'<a href="https://wordpress.org/plugins/wp-native-php-sessions/">',
656
+								'</a>',
657
+								'<a href="' . add_query_arg(array('ee_retry_session' => true), EEH_URL::current_url()) . '">'
658
+							),
659
+							__FILE__,
660
+							__FUNCTION__,
661
+							__LINE__
662
+						);
663
+					}
664
+				}
665
+			} else {
666
+				// there is no record of having a fatal error while trying the custom session save handler
667
+				// so let's try it
668
+				// there is a custom session save handler. Proceed with caution
669
+				update_option('ee_custom_session_save_handler_died', '1');
670
+				// hold your breath, if the custom session save handler might cause a fatal here...
671
+				session_start();
672
+				// phew! we made it! the custom session handler is a-ok
673
+				update_option('ee_custom_session_save_handler_proven_ok', '1');
674
+			}
675
+		}
676
+	}
677
+
678
+
679
+	/**
680
+	 * _get_session_data
681
+	 * Retrieves the session data, and attempts to correct any encoding issues that can occur due to improperly setup
682
+	 * databases
683
+	 *
684
+	 * @return array
685
+	 * @throws EE_Error
686
+	 * @throws InvalidArgumentException
687
+	 * @throws InvalidSessionDataException
688
+	 * @throws InvalidDataTypeException
689
+	 * @throws InvalidInterfaceException
690
+	 */
691
+	protected function _retrieve_session_data()
692
+	{
693
+		$ssn_key = EE_Session::session_id_prefix . $this->_sid;
694
+		try {
695
+			// we're using WP's Transient API to store session data using the PHP session ID as the option name
696
+			$session_data = $this->cache_storage->get($ssn_key, false);
697
+			if (empty($session_data)) {
698
+				return array();
699
+			}
700
+			if (apply_filters('FHEE__EE_Session___perform_session_id_hash_check', WP_DEBUG)) {
701
+				$hash_check = $this->cache_storage->get(
702
+					EE_Session::hash_check_prefix . $this->_sid,
703
+					false
704
+				);
705
+				if ($hash_check && $hash_check !== md5($session_data)) {
706
+					EE_Error::add_error(
707
+						sprintf(
708
+							__(
709
+								'The stored data for session %1$s failed to pass a hash check and therefore appears to be invalid.',
710
+								'event_espresso'
711
+							),
712
+							EE_Session::session_id_prefix . $this->_sid
713
+						),
714
+						__FILE__,
715
+						__FUNCTION__,
716
+						__LINE__
717
+					);
718
+				}
719
+			}
720
+		} catch (Exception $e) {
721
+			// let's just eat that error for now and attempt to correct any corrupted data
722
+			global $wpdb;
723
+			$row = $wpdb->get_row(
724
+				$wpdb->prepare(
725
+					"SELECT option_value FROM {$wpdb->options} WHERE option_name = %s LIMIT 1",
726
+					'_transient_' . $ssn_key
727
+				)
728
+			);
729
+			$session_data = is_object($row) ? $row->option_value : null;
730
+			if ($session_data) {
731
+				$session_data = preg_replace_callback(
732
+					'!s:(d+):"(.*?)";!',
733
+					function ($match) {
734
+						return $match[1] === strlen($match[2])
735
+							? $match[0]
736
+							: 's:' . strlen($match[2]) . ':"' . $match[2] . '";';
737
+					},
738
+					$session_data
739
+				);
740
+			}
741
+			$session_data = maybe_unserialize($session_data);
742
+		}
743
+		// in case the data is encoded... try to decode it
744
+		$session_data = $this->encryption instanceof EE_Encryption
745
+			? $this->encryption->base64_string_decode($session_data)
746
+			: $session_data;
747
+		if (! is_array($session_data)) {
748
+			try {
749
+				$session_data = maybe_unserialize($session_data);
750
+			} catch (Exception $e) {
751
+				$msg = esc_html__(
752
+					'An error occurred while attempting to unserialize the session data.',
753
+					'event_espresso'
754
+				);
755
+				$msg .= WP_DEBUG
756
+					? '<br><pre>'
757
+					  . print_r($session_data, true)
758
+					  . '</pre><br>'
759
+					  . $this->find_serialize_error($session_data)
760
+					: '';
761
+				$this->cache_storage->delete(EE_Session::session_id_prefix . $this->_sid);
762
+				throw new InvalidSessionDataException($msg, 0, $e);
763
+			}
764
+		}
765
+		// just a check to make sure the session array is indeed an array
766
+		if (! is_array($session_data)) {
767
+			// no?!?! then something is wrong
768
+			$msg = esc_html__(
769
+				'The session data is missing, invalid, or corrupted.',
770
+				'event_espresso'
771
+			);
772
+			$msg .= WP_DEBUG
773
+				? '<br><pre>' . print_r($session_data, true) . '</pre><br>' . $this->find_serialize_error($session_data)
774
+				: '';
775
+			$this->cache_storage->delete(EE_Session::session_id_prefix . $this->_sid);
776
+			throw new InvalidSessionDataException($msg);
777
+		}
778
+		if (isset($session_data['transaction']) && absint($session_data['transaction']) !== 0) {
779
+			$session_data['transaction'] = EEM_Transaction::instance()->get_one_by_ID(
780
+				$session_data['transaction']
781
+			);
782
+		}
783
+		return $session_data;
784
+	}
785
+
786
+
787
+	/**
788
+	 * _generate_session_id
789
+	 * Retrieves the PHP session id either directly from the PHP session,
790
+	 * or from the $_REQUEST array if it was passed in from an AJAX request.
791
+	 * The session id is then salted and hashed (mmm sounds tasty)
792
+	 * so that it can be safely used as a $_REQUEST param
793
+	 *
794
+	 * @return string
795
+	 */
796
+	protected function _generate_session_id()
797
+	{
798
+		// check if the SID was passed explicitly, otherwise get from session, then add salt and hash it to reduce length
799
+		if (isset($_REQUEST['EESID'])) {
800
+			$session_id = sanitize_text_field($_REQUEST['EESID']);
801
+		} else {
802
+			$session_id = md5(session_id() . get_current_blog_id() . $this->_get_sid_salt());
803
+		}
804
+		return apply_filters('FHEE__EE_Session___generate_session_id__session_id', $session_id);
805
+	}
806
+
807
+
808
+	/**
809
+	 * _get_sid_salt
810
+	 *
811
+	 * @return string
812
+	 */
813
+	protected function _get_sid_salt()
814
+	{
815
+		// was session id salt already saved to db ?
816
+		if (empty($this->_sid_salt)) {
817
+			// no?  then maybe use WP defined constant
818
+			if (defined('AUTH_SALT')) {
819
+				$this->_sid_salt = AUTH_SALT;
820
+			}
821
+			// if salt doesn't exist or is too short
822
+			if (strlen($this->_sid_salt) < 32) {
823
+				// create a new one
824
+				$this->_sid_salt = wp_generate_password(64);
825
+			}
826
+			// and save it as a permanent session setting
827
+			$this->updateSessionSettings(array('sid_salt' => $this->_sid_salt));
828
+		}
829
+		return $this->_sid_salt;
830
+	}
831
+
832
+
833
+	/**
834
+	 * _set_init_access_and_expiration
835
+	 *
836
+	 * @return void
837
+	 */
838
+	protected function _set_init_access_and_expiration()
839
+	{
840
+		$this->_time = time();
841
+		$this->_expiration = $this->_time + $this->session_lifespan->inSeconds();
842
+		// set initial site access time
843
+		$this->_session_data['init_access'] = $this->_time;
844
+		// and the session expiration
845
+		$this->_session_data['expiration'] = $this->_expiration;
846
+	}
847
+
848
+
849
+	/**
850
+	 * @update session data  prior to saving to the db
851
+	 * @access public
852
+	 * @param bool $new_session
853
+	 * @return TRUE on success, FALSE on fail
854
+	 * @throws EE_Error
855
+	 * @throws InvalidArgumentException
856
+	 * @throws InvalidDataTypeException
857
+	 * @throws InvalidInterfaceException
858
+	 */
859
+	public function update($new_session = false)
860
+	{
861
+		$this->_session_data = $this->_session_data !== null
862
+							   && is_array($this->_session_data)
863
+							   && isset($this->_session_data['id'])
864
+			? $this->_session_data
865
+			: array();
866
+		if (empty($this->_session_data)) {
867
+			$this->_set_defaults();
868
+		}
869
+		$session_data = array();
870
+		foreach ($this->_session_data as $key => $value) {
871
+			switch ($key) {
872
+				case 'id':
873
+					// session ID
874
+					$session_data['id'] = $this->_sid;
875
+					break;
876
+				case 'ip_address':
877
+					// visitor ip address
878
+					$session_data['ip_address'] = $this->request->ipAddress();
879
+					break;
880
+				case 'user_agent':
881
+					// visitor user_agent
882
+					$session_data['user_agent'] = $this->_user_agent;
883
+					break;
884
+				case 'init_access':
885
+					$session_data['init_access'] = absint($value);
886
+					break;
887
+				case 'last_access':
888
+					// current access time
889
+					$session_data['last_access'] = $this->_time;
890
+					break;
891
+				case 'expiration':
892
+					// when the session expires
893
+					$session_data['expiration'] = ! empty($this->_expiration)
894
+						? $this->_expiration
895
+						: $session_data['init_access'] + $this->session_lifespan->inSeconds();
896
+					break;
897
+				case 'user_id':
898
+					// current user if logged in
899
+					$session_data['user_id'] = $this->_wp_user_id();
900
+					break;
901
+				case 'pages_visited':
902
+					$page_visit = $this->_get_page_visit();
903
+					if ($page_visit) {
904
+						// set pages visited where the first will be the http referrer
905
+						$this->_session_data['pages_visited'][ $this->_time ] = $page_visit;
906
+						// we'll only save the last 10 page visits.
907
+						$session_data['pages_visited'] = array_slice($this->_session_data['pages_visited'], -10);
908
+					}
909
+					break;
910
+				default:
911
+					// carry any other data over
912
+					$session_data[ $key ] = $this->_session_data[ $key ];
913
+			}
914
+		}
915
+		$this->_session_data = $session_data;
916
+		// creating a new session does not require saving to the db just yet
917
+		if (! $new_session) {
918
+			// ready? let's save
919
+			if ($this->_save_session_to_db()) {
920
+				return true;
921
+			}
922
+			return false;
923
+		}
924
+		// meh, why not?
925
+		return true;
926
+	}
927
+
928
+
929
+	/**
930
+	 * @create session data array
931
+	 * @access public
932
+	 * @return bool
933
+	 * @throws EE_Error
934
+	 * @throws InvalidArgumentException
935
+	 * @throws InvalidDataTypeException
936
+	 * @throws InvalidInterfaceException
937
+	 */
938
+	private function _create_espresso_session()
939
+	{
940
+		do_action('AHEE_log', __CLASS__, __FUNCTION__, '');
941
+		// use the update function for now with $new_session arg set to TRUE
942
+		return $this->update(true) ? true : false;
943
+	}
944
+
945
+
946
+	/**
947
+	 * _save_session_to_db
948
+	 *
949
+	 * @param bool $clear_session
950
+	 * @return string
951
+	 * @throws EE_Error
952
+	 * @throws InvalidArgumentException
953
+	 * @throws InvalidDataTypeException
954
+	 * @throws InvalidInterfaceException
955
+	 */
956
+	private function _save_session_to_db($clear_session = false)
957
+	{
958
+		// don't save sessions for crawlers
959
+		// and unless we're deleting the session data, don't save anything if there isn't a cart
960
+		if ($this->request->isBot()
961
+			|| (
962
+				! $clear_session
963
+				&& ! $this->cart() instanceof EE_Cart
964
+				&& apply_filters('FHEE__EE_Session___save_session_to_db__abort_session_save', true)
965
+			)
966
+		) {
967
+			return false;
968
+		}
969
+		$transaction = $this->transaction();
970
+		if ($transaction instanceof EE_Transaction) {
971
+			if (! $transaction->ID()) {
972
+				$transaction->save();
973
+			}
974
+			$this->_session_data['transaction'] = $transaction->ID();
975
+		}
976
+		// then serialize all of our session data
977
+		$session_data = serialize($this->_session_data);
978
+		// do we need to also encode it to avoid corrupted data when saved to the db?
979
+		$session_data = $this->_use_encryption
980
+			? $this->encryption->base64_string_encode($session_data)
981
+			: $session_data;
982
+		// maybe save hash check
983
+		if (apply_filters('FHEE__EE_Session___perform_session_id_hash_check', WP_DEBUG)) {
984
+			$this->cache_storage->add(
985
+				EE_Session::hash_check_prefix . $this->_sid,
986
+				md5($session_data),
987
+				$this->session_lifespan->inSeconds()
988
+			);
989
+		}
990
+		// we're using the Transient API for storing session data,
991
+		return $this->cache_storage->add(
992
+			EE_Session::session_id_prefix . $this->_sid,
993
+			$session_data,
994
+			$this->session_lifespan->inSeconds()
995
+		);
996
+	}
997
+
998
+
999
+	/**
1000
+	 * @get    the full page request the visitor is accessing
1001
+	 * @access public
1002
+	 * @return string
1003
+	 */
1004
+	public function _get_page_visit()
1005
+	{
1006
+		$page_visit = home_url('/') . 'wp-admin/admin-ajax.php';
1007
+		// check for request url
1008
+		if (isset($_SERVER['REQUEST_URI'])) {
1009
+			$http_host = '';
1010
+			$page_id = '?';
1011
+			$e_reg = '';
1012
+			$request_uri = esc_url($_SERVER['REQUEST_URI']);
1013
+			$ru_bits = explode('?', $request_uri);
1014
+			$request_uri = $ru_bits[0];
1015
+			// check for and grab host as well
1016
+			if (isset($_SERVER['HTTP_HOST'])) {
1017
+				$http_host = esc_url($_SERVER['HTTP_HOST']);
1018
+			}
1019
+			// check for page_id in SERVER REQUEST
1020
+			if (isset($_REQUEST['page_id'])) {
1021
+				// rebuild $e_reg without any of the extra parameters
1022
+				$page_id = '?page_id=' . esc_attr($_REQUEST['page_id']) . '&amp;';
1023
+			}
1024
+			// check for $e_reg in SERVER REQUEST
1025
+			if (isset($_REQUEST['ee'])) {
1026
+				// rebuild $e_reg without any of the extra parameters
1027
+				$e_reg = 'ee=' . esc_attr($_REQUEST['ee']);
1028
+			}
1029
+			$page_visit = rtrim($http_host . $request_uri . $page_id . $e_reg, '?');
1030
+		}
1031
+		return $page_visit !== home_url('/wp-admin/admin-ajax.php') ? $page_visit : '';
1032
+	}
1033
+
1034
+
1035
+	/**
1036
+	 * @the    current wp user id
1037
+	 * @access public
1038
+	 * @return int
1039
+	 */
1040
+	public function _wp_user_id()
1041
+	{
1042
+		// if I need to explain the following lines of code, then you shouldn't be looking at this!
1043
+		$this->_wp_user_id = get_current_user_id();
1044
+		return $this->_wp_user_id;
1045
+	}
1046
+
1047
+
1048
+	/**
1049
+	 * Clear EE_Session data
1050
+	 *
1051
+	 * @access public
1052
+	 * @param string $class
1053
+	 * @param string $function
1054
+	 * @return void
1055
+	 * @throws EE_Error
1056
+	 * @throws InvalidArgumentException
1057
+	 * @throws InvalidDataTypeException
1058
+	 * @throws InvalidInterfaceException
1059
+	 */
1060
+	public function clear_session($class = '', $function = '')
1061
+	{
1062 1062
 //         echo '
1063 1063
 // <h3 style="color:#999;line-height:.9em;">
1064 1064
 // <span style="color:#2EA2CC">' . __CLASS__ . '</span>::<span style="color:#E76700">' . __FUNCTION__ . '( ' . $class . '::' . $function . '() )</span><br/>
1065 1065
 // <span style="font-size:9px;font-weight:normal;">' . __FILE__ . '</span>    <b style="font-size:10px;">  ' . __LINE__ . ' </b>
1066 1066
 // </h3>';
1067
-        do_action('AHEE_log', __FILE__, __FUNCTION__, 'session cleared by : ' . $class . '::' . $function . '()');
1068
-        $this->reset_cart();
1069
-        $this->reset_checkout();
1070
-        $this->reset_transaction();
1071
-        // wipe out everything that isn't a default session datum
1072
-        $this->reset_data(array_keys($this->_session_data));
1073
-        // reset initial site access time and the session expiration
1074
-        $this->_set_init_access_and_expiration();
1075
-        $this->_save_session_to_db(true);
1076
-    }
1077
-
1078
-
1079
-    /**
1080
-     * resets all non-default session vars. Returns TRUE on success, FALSE on fail
1081
-     *
1082
-     * @param array|mixed $data_to_reset
1083
-     * @param bool        $show_all_notices
1084
-     * @return bool
1085
-     */
1086
-    public function reset_data($data_to_reset = array(), $show_all_notices = false)
1087
-    {
1088
-        // if $data_to_reset is not in an array, then put it in one
1089
-        if (! is_array($data_to_reset)) {
1090
-            $data_to_reset = array($data_to_reset);
1091
-        }
1092
-        // nothing ??? go home!
1093
-        if (empty($data_to_reset)) {
1094
-            EE_Error::add_error(
1095
-                __(
1096
-                    'No session data could be reset, because no session var name was provided.',
1097
-                    'event_espresso'
1098
-                ),
1099
-                __FILE__,
1100
-                __FUNCTION__,
1101
-                __LINE__
1102
-            );
1103
-            return false;
1104
-        }
1105
-        $return_value = true;
1106
-        // since $data_to_reset is an array, cycle through the values
1107
-        foreach ($data_to_reset as $reset) {
1108
-            // first check to make sure it is a valid session var
1109
-            if (isset($this->_session_data[ $reset ])) {
1110
-                // then check to make sure it is not a default var
1111
-                if (! array_key_exists($reset, $this->_default_session_vars)) {
1112
-                    // remove session var
1113
-                    unset($this->_session_data[ $reset ]);
1114
-                    if ($show_all_notices) {
1115
-                        EE_Error::add_success(
1116
-                            sprintf(
1117
-                                __('The session variable %s was removed.', 'event_espresso'),
1118
-                                $reset
1119
-                            ),
1120
-                            __FILE__,
1121
-                            __FUNCTION__,
1122
-                            __LINE__
1123
-                        );
1124
-                    }
1125
-                } else {
1126
-                    // yeeeeeeeeerrrrrrrrrrr OUT !!!!
1127
-                    if ($show_all_notices) {
1128
-                        EE_Error::add_error(
1129
-                            sprintf(
1130
-                                __(
1131
-                                    'Sorry! %s is a default session datum and can not be reset.',
1132
-                                    'event_espresso'
1133
-                                ),
1134
-                                $reset
1135
-                            ),
1136
-                            __FILE__,
1137
-                            __FUNCTION__,
1138
-                            __LINE__
1139
-                        );
1140
-                    }
1141
-                    $return_value = false;
1142
-                }
1143
-            } elseif ($show_all_notices) {
1144
-                // oops! that session var does not exist!
1145
-                EE_Error::add_error(
1146
-                    sprintf(
1147
-                        __(
1148
-                            'The session item provided, %s, is invalid or does not exist.',
1149
-                            'event_espresso'
1150
-                        ),
1151
-                        $reset
1152
-                    ),
1153
-                    __FILE__,
1154
-                    __FUNCTION__,
1155
-                    __LINE__
1156
-                );
1157
-                $return_value = false;
1158
-            }
1159
-        } // end of foreach
1160
-        return $return_value;
1161
-    }
1162
-
1163
-
1164
-    /**
1165
-     *   wp_loaded
1166
-     *
1167
-     * @access public
1168
-     * @throws EE_Error
1169
-     * @throws InvalidDataTypeException
1170
-     * @throws InvalidInterfaceException
1171
-     * @throws InvalidArgumentException
1172
-     */
1173
-    public function wp_loaded()
1174
-    {
1175
-        if ($this->request->requestParamIsSet('clear_session')) {
1176
-            $this->clear_session(__CLASS__, __FUNCTION__);
1177
-        }
1178
-    }
1179
-
1180
-
1181
-    /**
1182
-     * Used to reset the entire object (for tests).
1183
-     *
1184
-     * @since 4.3.0
1185
-     * @throws EE_Error
1186
-     * @throws InvalidDataTypeException
1187
-     * @throws InvalidInterfaceException
1188
-     * @throws InvalidArgumentException
1189
-     */
1190
-    public function reset_instance()
1191
-    {
1192
-        $this->clear_session();
1193
-        self::$_instance = null;
1194
-    }
1195
-
1196
-
1197
-    public function configure_garbage_collection_filters()
1198
-    {
1199
-        // run old filter we had for controlling session cleanup
1200
-        $expired_session_transient_delete_query_limit = absint(
1201
-            apply_filters(
1202
-                'FHEE__EE_Session__garbage_collection___expired_session_transient_delete_query_limit',
1203
-                50
1204
-            )
1205
-        );
1206
-        // is there a value? or one that is different than the default 50 records?
1207
-        if ($expired_session_transient_delete_query_limit === 0) {
1208
-            // hook into TransientCacheStorage in case Session cleanup was turned off
1209
-            add_filter('FHEE__TransientCacheStorage__transient_cleanup_schedule', '__return_zero');
1210
-        } elseif ($expired_session_transient_delete_query_limit !== 50) {
1211
-            // or use that for the new transient cleanup query limit
1212
-            add_filter(
1213
-                'FHEE__TransientCacheStorage__clearExpiredTransients__limit',
1214
-                function () use ($expired_session_transient_delete_query_limit) {
1215
-                    return $expired_session_transient_delete_query_limit;
1216
-                }
1217
-            );
1218
-        }
1219
-    }
1220
-
1221
-
1222
-    /**
1223
-     * @see http://stackoverflow.com/questions/10152904/unserialize-function-unserialize-error-at-offset/21389439#10152996
1224
-     * @param $data1
1225
-     * @return string
1226
-     */
1227
-    private function find_serialize_error($data1)
1228
-    {
1229
-        $error = '<pre>';
1230
-        $data2 = preg_replace_callback(
1231
-            '!s:(\d+):"(.*?)";!',
1232
-            function ($match) {
1233
-                return ($match[1] === strlen($match[2]))
1234
-                    ? $match[0]
1235
-                    : 's:'
1236
-                      . strlen($match[2])
1237
-                      . ':"'
1238
-                      . $match[2]
1239
-                      . '";';
1240
-            },
1241
-            $data1
1242
-        );
1243
-        $max = (strlen($data1) > strlen($data2)) ? strlen($data1) : strlen($data2);
1244
-        $error .= $data1 . PHP_EOL;
1245
-        $error .= $data2 . PHP_EOL;
1246
-        for ($i = 0; $i < $max; $i++) {
1247
-            if (@$data1[ $i ] !== @$data2[ $i ]) {
1248
-                $error .= 'Difference ' . @$data1[ $i ] . ' != ' . @$data2[ $i ] . PHP_EOL;
1249
-                $error .= "\t-> ORD number " . ord(@$data1[ $i ]) . ' != ' . ord(@$data2[ $i ]) . PHP_EOL;
1250
-                $error .= "\t-> Line Number = $i" . PHP_EOL;
1251
-                $start = ($i - 20);
1252
-                $start = ($start < 0) ? 0 : $start;
1253
-                $length = 40;
1254
-                $point = $max - $i;
1255
-                if ($point < 20) {
1256
-                    $rlength = 1;
1257
-                    $rpoint = -$point;
1258
-                } else {
1259
-                    $rpoint = $length - 20;
1260
-                    $rlength = 1;
1261
-                }
1262
-                $error .= "\t-> Section Data1  = ";
1263
-                $error .= substr_replace(
1264
-                    substr($data1, $start, $length),
1265
-                    "<b style=\"color:green\">{$data1[ $i ]}</b>",
1266
-                    $rpoint,
1267
-                    $rlength
1268
-                );
1269
-                $error .= PHP_EOL;
1270
-                $error .= "\t-> Section Data2  = ";
1271
-                $error .= substr_replace(
1272
-                    substr($data2, $start, $length),
1273
-                    "<b style=\"color:red\">{$data2[ $i ]}</b>",
1274
-                    $rpoint,
1275
-                    $rlength
1276
-                );
1277
-                $error .= PHP_EOL;
1278
-            }
1279
-        }
1280
-        $error .= '</pre>';
1281
-        return $error;
1282
-    }
1283
-
1284
-
1285
-    /**
1286
-     * Saves an  array of settings used for configuring aspects of session behaviour
1287
-     *
1288
-     * @param array $updated_settings
1289
-     */
1290
-    private function updateSessionSettings(array $updated_settings = array())
1291
-    {
1292
-        // add existing settings, but only if not included in incoming $updated_settings array
1293
-        $updated_settings += get_option(EE_Session::OPTION_NAME_SETTINGS, array());
1294
-        update_option(EE_Session::OPTION_NAME_SETTINGS, $updated_settings);
1295
-    }
1296
-
1297
-
1298
-    /**
1299
-     * garbage_collection
1300
-     */
1301
-    public function garbageCollection()
1302
-    {
1303
-        // only perform during regular requests if last garbage collection was over an hour ago
1304
-        if (! (defined('DOING_AJAX') && DOING_AJAX) && (time() - HOUR_IN_SECONDS) >= $this->_last_gc) {
1305
-            $this->_last_gc = time();
1306
-            $this->updateSessionSettings(array('last_gc' => $this->_last_gc));
1307
-            /** @type WPDB $wpdb */
1308
-            global $wpdb;
1309
-            // filter the query limit. Set to 0 to turn off garbage collection
1310
-            $expired_session_transient_delete_query_limit = absint(
1311
-                apply_filters(
1312
-                    'FHEE__EE_Session__garbage_collection___expired_session_transient_delete_query_limit',
1313
-                    50
1314
-                )
1315
-            );
1316
-            // non-zero LIMIT means take out the trash
1317
-            if ($expired_session_transient_delete_query_limit) {
1318
-                $session_key = str_replace('_', '\_', EE_Session::session_id_prefix);
1319
-                $hash_check_key = str_replace('_', '\_', EE_Session::hash_check_prefix);
1320
-                // since transient expiration timestamps are set in the future, we can compare against NOW
1321
-                // but we only want to pick up any trash that's been around for more than a day
1322
-                $expiration = time() - DAY_IN_SECONDS;
1323
-                $SQL = "
1067
+		do_action('AHEE_log', __FILE__, __FUNCTION__, 'session cleared by : ' . $class . '::' . $function . '()');
1068
+		$this->reset_cart();
1069
+		$this->reset_checkout();
1070
+		$this->reset_transaction();
1071
+		// wipe out everything that isn't a default session datum
1072
+		$this->reset_data(array_keys($this->_session_data));
1073
+		// reset initial site access time and the session expiration
1074
+		$this->_set_init_access_and_expiration();
1075
+		$this->_save_session_to_db(true);
1076
+	}
1077
+
1078
+
1079
+	/**
1080
+	 * resets all non-default session vars. Returns TRUE on success, FALSE on fail
1081
+	 *
1082
+	 * @param array|mixed $data_to_reset
1083
+	 * @param bool        $show_all_notices
1084
+	 * @return bool
1085
+	 */
1086
+	public function reset_data($data_to_reset = array(), $show_all_notices = false)
1087
+	{
1088
+		// if $data_to_reset is not in an array, then put it in one
1089
+		if (! is_array($data_to_reset)) {
1090
+			$data_to_reset = array($data_to_reset);
1091
+		}
1092
+		// nothing ??? go home!
1093
+		if (empty($data_to_reset)) {
1094
+			EE_Error::add_error(
1095
+				__(
1096
+					'No session data could be reset, because no session var name was provided.',
1097
+					'event_espresso'
1098
+				),
1099
+				__FILE__,
1100
+				__FUNCTION__,
1101
+				__LINE__
1102
+			);
1103
+			return false;
1104
+		}
1105
+		$return_value = true;
1106
+		// since $data_to_reset is an array, cycle through the values
1107
+		foreach ($data_to_reset as $reset) {
1108
+			// first check to make sure it is a valid session var
1109
+			if (isset($this->_session_data[ $reset ])) {
1110
+				// then check to make sure it is not a default var
1111
+				if (! array_key_exists($reset, $this->_default_session_vars)) {
1112
+					// remove session var
1113
+					unset($this->_session_data[ $reset ]);
1114
+					if ($show_all_notices) {
1115
+						EE_Error::add_success(
1116
+							sprintf(
1117
+								__('The session variable %s was removed.', 'event_espresso'),
1118
+								$reset
1119
+							),
1120
+							__FILE__,
1121
+							__FUNCTION__,
1122
+							__LINE__
1123
+						);
1124
+					}
1125
+				} else {
1126
+					// yeeeeeeeeerrrrrrrrrrr OUT !!!!
1127
+					if ($show_all_notices) {
1128
+						EE_Error::add_error(
1129
+							sprintf(
1130
+								__(
1131
+									'Sorry! %s is a default session datum and can not be reset.',
1132
+									'event_espresso'
1133
+								),
1134
+								$reset
1135
+							),
1136
+							__FILE__,
1137
+							__FUNCTION__,
1138
+							__LINE__
1139
+						);
1140
+					}
1141
+					$return_value = false;
1142
+				}
1143
+			} elseif ($show_all_notices) {
1144
+				// oops! that session var does not exist!
1145
+				EE_Error::add_error(
1146
+					sprintf(
1147
+						__(
1148
+							'The session item provided, %s, is invalid or does not exist.',
1149
+							'event_espresso'
1150
+						),
1151
+						$reset
1152
+					),
1153
+					__FILE__,
1154
+					__FUNCTION__,
1155
+					__LINE__
1156
+				);
1157
+				$return_value = false;
1158
+			}
1159
+		} // end of foreach
1160
+		return $return_value;
1161
+	}
1162
+
1163
+
1164
+	/**
1165
+	 *   wp_loaded
1166
+	 *
1167
+	 * @access public
1168
+	 * @throws EE_Error
1169
+	 * @throws InvalidDataTypeException
1170
+	 * @throws InvalidInterfaceException
1171
+	 * @throws InvalidArgumentException
1172
+	 */
1173
+	public function wp_loaded()
1174
+	{
1175
+		if ($this->request->requestParamIsSet('clear_session')) {
1176
+			$this->clear_session(__CLASS__, __FUNCTION__);
1177
+		}
1178
+	}
1179
+
1180
+
1181
+	/**
1182
+	 * Used to reset the entire object (for tests).
1183
+	 *
1184
+	 * @since 4.3.0
1185
+	 * @throws EE_Error
1186
+	 * @throws InvalidDataTypeException
1187
+	 * @throws InvalidInterfaceException
1188
+	 * @throws InvalidArgumentException
1189
+	 */
1190
+	public function reset_instance()
1191
+	{
1192
+		$this->clear_session();
1193
+		self::$_instance = null;
1194
+	}
1195
+
1196
+
1197
+	public function configure_garbage_collection_filters()
1198
+	{
1199
+		// run old filter we had for controlling session cleanup
1200
+		$expired_session_transient_delete_query_limit = absint(
1201
+			apply_filters(
1202
+				'FHEE__EE_Session__garbage_collection___expired_session_transient_delete_query_limit',
1203
+				50
1204
+			)
1205
+		);
1206
+		// is there a value? or one that is different than the default 50 records?
1207
+		if ($expired_session_transient_delete_query_limit === 0) {
1208
+			// hook into TransientCacheStorage in case Session cleanup was turned off
1209
+			add_filter('FHEE__TransientCacheStorage__transient_cleanup_schedule', '__return_zero');
1210
+		} elseif ($expired_session_transient_delete_query_limit !== 50) {
1211
+			// or use that for the new transient cleanup query limit
1212
+			add_filter(
1213
+				'FHEE__TransientCacheStorage__clearExpiredTransients__limit',
1214
+				function () use ($expired_session_transient_delete_query_limit) {
1215
+					return $expired_session_transient_delete_query_limit;
1216
+				}
1217
+			);
1218
+		}
1219
+	}
1220
+
1221
+
1222
+	/**
1223
+	 * @see http://stackoverflow.com/questions/10152904/unserialize-function-unserialize-error-at-offset/21389439#10152996
1224
+	 * @param $data1
1225
+	 * @return string
1226
+	 */
1227
+	private function find_serialize_error($data1)
1228
+	{
1229
+		$error = '<pre>';
1230
+		$data2 = preg_replace_callback(
1231
+			'!s:(\d+):"(.*?)";!',
1232
+			function ($match) {
1233
+				return ($match[1] === strlen($match[2]))
1234
+					? $match[0]
1235
+					: 's:'
1236
+					  . strlen($match[2])
1237
+					  . ':"'
1238
+					  . $match[2]
1239
+					  . '";';
1240
+			},
1241
+			$data1
1242
+		);
1243
+		$max = (strlen($data1) > strlen($data2)) ? strlen($data1) : strlen($data2);
1244
+		$error .= $data1 . PHP_EOL;
1245
+		$error .= $data2 . PHP_EOL;
1246
+		for ($i = 0; $i < $max; $i++) {
1247
+			if (@$data1[ $i ] !== @$data2[ $i ]) {
1248
+				$error .= 'Difference ' . @$data1[ $i ] . ' != ' . @$data2[ $i ] . PHP_EOL;
1249
+				$error .= "\t-> ORD number " . ord(@$data1[ $i ]) . ' != ' . ord(@$data2[ $i ]) . PHP_EOL;
1250
+				$error .= "\t-> Line Number = $i" . PHP_EOL;
1251
+				$start = ($i - 20);
1252
+				$start = ($start < 0) ? 0 : $start;
1253
+				$length = 40;
1254
+				$point = $max - $i;
1255
+				if ($point < 20) {
1256
+					$rlength = 1;
1257
+					$rpoint = -$point;
1258
+				} else {
1259
+					$rpoint = $length - 20;
1260
+					$rlength = 1;
1261
+				}
1262
+				$error .= "\t-> Section Data1  = ";
1263
+				$error .= substr_replace(
1264
+					substr($data1, $start, $length),
1265
+					"<b style=\"color:green\">{$data1[ $i ]}</b>",
1266
+					$rpoint,
1267
+					$rlength
1268
+				);
1269
+				$error .= PHP_EOL;
1270
+				$error .= "\t-> Section Data2  = ";
1271
+				$error .= substr_replace(
1272
+					substr($data2, $start, $length),
1273
+					"<b style=\"color:red\">{$data2[ $i ]}</b>",
1274
+					$rpoint,
1275
+					$rlength
1276
+				);
1277
+				$error .= PHP_EOL;
1278
+			}
1279
+		}
1280
+		$error .= '</pre>';
1281
+		return $error;
1282
+	}
1283
+
1284
+
1285
+	/**
1286
+	 * Saves an  array of settings used for configuring aspects of session behaviour
1287
+	 *
1288
+	 * @param array $updated_settings
1289
+	 */
1290
+	private function updateSessionSettings(array $updated_settings = array())
1291
+	{
1292
+		// add existing settings, but only if not included in incoming $updated_settings array
1293
+		$updated_settings += get_option(EE_Session::OPTION_NAME_SETTINGS, array());
1294
+		update_option(EE_Session::OPTION_NAME_SETTINGS, $updated_settings);
1295
+	}
1296
+
1297
+
1298
+	/**
1299
+	 * garbage_collection
1300
+	 */
1301
+	public function garbageCollection()
1302
+	{
1303
+		// only perform during regular requests if last garbage collection was over an hour ago
1304
+		if (! (defined('DOING_AJAX') && DOING_AJAX) && (time() - HOUR_IN_SECONDS) >= $this->_last_gc) {
1305
+			$this->_last_gc = time();
1306
+			$this->updateSessionSettings(array('last_gc' => $this->_last_gc));
1307
+			/** @type WPDB $wpdb */
1308
+			global $wpdb;
1309
+			// filter the query limit. Set to 0 to turn off garbage collection
1310
+			$expired_session_transient_delete_query_limit = absint(
1311
+				apply_filters(
1312
+					'FHEE__EE_Session__garbage_collection___expired_session_transient_delete_query_limit',
1313
+					50
1314
+				)
1315
+			);
1316
+			// non-zero LIMIT means take out the trash
1317
+			if ($expired_session_transient_delete_query_limit) {
1318
+				$session_key = str_replace('_', '\_', EE_Session::session_id_prefix);
1319
+				$hash_check_key = str_replace('_', '\_', EE_Session::hash_check_prefix);
1320
+				// since transient expiration timestamps are set in the future, we can compare against NOW
1321
+				// but we only want to pick up any trash that's been around for more than a day
1322
+				$expiration = time() - DAY_IN_SECONDS;
1323
+				$SQL = "
1324 1324
                     SELECT option_name
1325 1325
                     FROM {$wpdb->options}
1326 1326
                     WHERE
@@ -1329,17 +1329,17 @@  discard block
 block discarded – undo
1329 1329
                     AND option_value < {$expiration}
1330 1330
                     LIMIT {$expired_session_transient_delete_query_limit}
1331 1331
                 ";
1332
-                // produces something like:
1333
-                // SELECT option_name FROM wp_options
1334
-                // WHERE ( option_name LIKE '\_transient\_timeout\_ee\_ssn\_%'
1335
-                // OR option_name LIKE '\_transient\_timeout\_ee\_shc\_%' )
1336
-                // AND option_value < 1508368198 LIMIT 50
1337
-                $expired_sessions = $wpdb->get_col($SQL);
1338
-                // valid results?
1339
-                if (! $expired_sessions instanceof WP_Error && ! empty($expired_sessions)) {
1340
-                    $this->cache_storage->deleteMany($expired_sessions, true);
1341
-                }
1342
-            }
1343
-        }
1344
-    }
1332
+				// produces something like:
1333
+				// SELECT option_name FROM wp_options
1334
+				// WHERE ( option_name LIKE '\_transient\_timeout\_ee\_ssn\_%'
1335
+				// OR option_name LIKE '\_transient\_timeout\_ee\_shc\_%' )
1336
+				// AND option_value < 1508368198 LIMIT 50
1337
+				$expired_sessions = $wpdb->get_col($SQL);
1338
+				// valid results?
1339
+				if (! $expired_sessions instanceof WP_Error && ! empty($expired_sessions)) {
1340
+					$this->cache_storage->deleteMany($expired_sessions, true);
1341
+				}
1342
+			}
1343
+		}
1344
+	}
1345 1345
 }
Please login to merge, or discard this patch.
Spacing   +54 added lines, -54 removed lines patch added patch discarded remove patch
@@ -187,7 +187,7 @@  discard block
 block discarded – undo
187 187
         // check if class object is instantiated
188 188
         // session loading is turned ON by default, but prior to the init hook, can be turned back OFF via:
189 189
         // add_filter( 'FHEE_load_EE_Session', '__return_false' );
190
-        if (! self::$_instance instanceof EE_Session && apply_filters('FHEE_load_EE_Session', true)) {
190
+        if ( ! self::$_instance instanceof EE_Session && apply_filters('FHEE_load_EE_Session', true)) {
191 191
             self::$_instance = new self(
192 192
                 $cache_storage,
193 193
                 $lifespan,
@@ -220,21 +220,21 @@  discard block
 block discarded – undo
220 220
         // but prior to the 'AHEE__EE_System__core_loaded_and_ready' hook
221 221
         // (which currently fires on the init hook at priority 9),
222 222
         // can be turned back OFF via: add_filter( 'FHEE_load_EE_Session', '__return_false' );
223
-        if (! apply_filters('FHEE_load_EE_Session', true)) {
223
+        if ( ! apply_filters('FHEE_load_EE_Session', true)) {
224 224
             return;
225 225
         }
226 226
         $this->session_lifespan = $lifespan;
227 227
         $this->request = $request;
228
-        if (! defined('ESPRESSO_SESSION')) {
228
+        if ( ! defined('ESPRESSO_SESSION')) {
229 229
             define('ESPRESSO_SESSION', true);
230 230
         }
231 231
         // retrieve session options from db
232 232
         $session_settings = (array) get_option(EE_Session::OPTION_NAME_SETTINGS, array());
233
-        if (! empty($session_settings)) {
233
+        if ( ! empty($session_settings)) {
234 234
             // cycle though existing session options
235 235
             foreach ($session_settings as $var_name => $session_setting) {
236 236
                 // set values for class properties
237
-                $var_name = '_' . $var_name;
237
+                $var_name = '_'.$var_name;
238 238
                 $this->{$var_name} = $session_setting;
239 239
             }
240 240
         }
@@ -293,7 +293,7 @@  discard block
 block discarded – undo
293 293
     public function open_session()
294 294
     {
295 295
         // check for existing session and retrieve it from db
296
-        if (! $this->_espresso_session()) {
296
+        if ( ! $this->_espresso_session()) {
297 297
             // or just start a new one
298 298
             $this->_create_espresso_session();
299 299
         }
@@ -366,9 +366,9 @@  discard block
 block discarded – undo
366 366
         // set some defaults
367 367
         foreach ($this->_default_session_vars as $key => $default_var) {
368 368
             if (is_array($default_var)) {
369
-                $this->_session_data[ $key ] = array();
369
+                $this->_session_data[$key] = array();
370 370
             } else {
371
-                $this->_session_data[ $key ] = '';
371
+                $this->_session_data[$key] = '';
372 372
             }
373 373
         }
374 374
     }
@@ -499,8 +499,8 @@  discard block
 block discarded – undo
499 499
             $this->reset_checkout();
500 500
             $this->reset_transaction();
501 501
         }
502
-        if (! empty($key)) {
503
-            return isset($this->_session_data[ $key ]) ? $this->_session_data[ $key ] : null;
502
+        if ( ! empty($key)) {
503
+            return isset($this->_session_data[$key]) ? $this->_session_data[$key] : null;
504 504
         }
505 505
         return $this->_session_data;
506 506
     }
@@ -528,7 +528,7 @@  discard block
 block discarded – undo
528 528
             return false;
529 529
         }
530 530
         foreach ($data as $key => $value) {
531
-            if (isset($this->_default_session_vars[ $key ])) {
531
+            if (isset($this->_default_session_vars[$key])) {
532 532
                 EE_Error::add_error(
533 533
                     sprintf(
534 534
                         esc_html__(
@@ -543,7 +543,7 @@  discard block
 block discarded – undo
543 543
                 );
544 544
                 return false;
545 545
             }
546
-            $this->_session_data[ $key ] = $value;
546
+            $this->_session_data[$key] = $value;
547 547
         }
548 548
         return true;
549 549
     }
@@ -580,7 +580,7 @@  discard block
 block discarded – undo
580 580
         $this->_user_agent = $this->request->userAgent();
581 581
         // now let's retrieve what's in the db
582 582
         $session_data = $this->_retrieve_session_data();
583
-        if (! empty($session_data)) {
583
+        if ( ! empty($session_data)) {
584 584
             // get the current time in UTC
585 585
             $this->_time = $this->_time !== null ? $this->_time : time();
586 586
             // and reset the session expiration
@@ -591,7 +591,7 @@  discard block
 block discarded – undo
591 591
             // set initial site access time and the session expiration
592 592
             $this->_set_init_access_and_expiration();
593 593
             // set referer
594
-            $this->_session_data['pages_visited'][ $this->_session_data['init_access'] ] = isset($_SERVER['HTTP_REFERER'])
594
+            $this->_session_data['pages_visited'][$this->_session_data['init_access']] = isset($_SERVER['HTTP_REFERER'])
595 595
                 ? esc_attr($_SERVER['HTTP_REFERER'])
596 596
                 : '';
597 597
             // no previous session = go back and create one (on top of the data above)
@@ -654,7 +654,7 @@  discard block
 block discarded – undo
654 654
                                 esc_html__('It appears there was a fatal error while starting the session, so Event Espresso is not able to process registrations normally. Some hosting companies, like Pantheon, require an extra plugin for Event Espresso to work. Please install the %1$sWordPress Native PHP Sessions plugin%2$s, then %3$sclick here to check if the problem is resolved.%2$s', 'event_espresso'),
655 655
                                 '<a href="https://wordpress.org/plugins/wp-native-php-sessions/">',
656 656
                                 '</a>',
657
-                                '<a href="' . add_query_arg(array('ee_retry_session' => true), EEH_URL::current_url()) . '">'
657
+                                '<a href="'.add_query_arg(array('ee_retry_session' => true), EEH_URL::current_url()).'">'
658 658
                             ),
659 659
                             __FILE__,
660 660
                             __FUNCTION__,
@@ -690,7 +690,7 @@  discard block
 block discarded – undo
690 690
      */
691 691
     protected function _retrieve_session_data()
692 692
     {
693
-        $ssn_key = EE_Session::session_id_prefix . $this->_sid;
693
+        $ssn_key = EE_Session::session_id_prefix.$this->_sid;
694 694
         try {
695 695
             // we're using WP's Transient API to store session data using the PHP session ID as the option name
696 696
             $session_data = $this->cache_storage->get($ssn_key, false);
@@ -699,7 +699,7 @@  discard block
 block discarded – undo
699 699
             }
700 700
             if (apply_filters('FHEE__EE_Session___perform_session_id_hash_check', WP_DEBUG)) {
701 701
                 $hash_check = $this->cache_storage->get(
702
-                    EE_Session::hash_check_prefix . $this->_sid,
702
+                    EE_Session::hash_check_prefix.$this->_sid,
703 703
                     false
704 704
                 );
705 705
                 if ($hash_check && $hash_check !== md5($session_data)) {
@@ -709,7 +709,7 @@  discard block
 block discarded – undo
709 709
                                 'The stored data for session %1$s failed to pass a hash check and therefore appears to be invalid.',
710 710
                                 'event_espresso'
711 711
                             ),
712
-                            EE_Session::session_id_prefix . $this->_sid
712
+                            EE_Session::session_id_prefix.$this->_sid
713 713
                         ),
714 714
                         __FILE__,
715 715
                         __FUNCTION__,
@@ -723,17 +723,17 @@  discard block
 block discarded – undo
723 723
             $row = $wpdb->get_row(
724 724
                 $wpdb->prepare(
725 725
                     "SELECT option_value FROM {$wpdb->options} WHERE option_name = %s LIMIT 1",
726
-                    '_transient_' . $ssn_key
726
+                    '_transient_'.$ssn_key
727 727
                 )
728 728
             );
729 729
             $session_data = is_object($row) ? $row->option_value : null;
730 730
             if ($session_data) {
731 731
                 $session_data = preg_replace_callback(
732 732
                     '!s:(d+):"(.*?)";!',
733
-                    function ($match) {
733
+                    function($match) {
734 734
                         return $match[1] === strlen($match[2])
735 735
                             ? $match[0]
736
-                            : 's:' . strlen($match[2]) . ':"' . $match[2] . '";';
736
+                            : 's:'.strlen($match[2]).':"'.$match[2].'";';
737 737
                     },
738 738
                     $session_data
739 739
                 );
@@ -744,7 +744,7 @@  discard block
 block discarded – undo
744 744
         $session_data = $this->encryption instanceof EE_Encryption
745 745
             ? $this->encryption->base64_string_decode($session_data)
746 746
             : $session_data;
747
-        if (! is_array($session_data)) {
747
+        if ( ! is_array($session_data)) {
748 748
             try {
749 749
                 $session_data = maybe_unserialize($session_data);
750 750
             } catch (Exception $e) {
@@ -758,21 +758,21 @@  discard block
 block discarded – undo
758 758
                       . '</pre><br>'
759 759
                       . $this->find_serialize_error($session_data)
760 760
                     : '';
761
-                $this->cache_storage->delete(EE_Session::session_id_prefix . $this->_sid);
761
+                $this->cache_storage->delete(EE_Session::session_id_prefix.$this->_sid);
762 762
                 throw new InvalidSessionDataException($msg, 0, $e);
763 763
             }
764 764
         }
765 765
         // just a check to make sure the session array is indeed an array
766
-        if (! is_array($session_data)) {
766
+        if ( ! is_array($session_data)) {
767 767
             // no?!?! then something is wrong
768 768
             $msg = esc_html__(
769 769
                 'The session data is missing, invalid, or corrupted.',
770 770
                 'event_espresso'
771 771
             );
772 772
             $msg .= WP_DEBUG
773
-                ? '<br><pre>' . print_r($session_data, true) . '</pre><br>' . $this->find_serialize_error($session_data)
773
+                ? '<br><pre>'.print_r($session_data, true).'</pre><br>'.$this->find_serialize_error($session_data)
774 774
                 : '';
775
-            $this->cache_storage->delete(EE_Session::session_id_prefix . $this->_sid);
775
+            $this->cache_storage->delete(EE_Session::session_id_prefix.$this->_sid);
776 776
             throw new InvalidSessionDataException($msg);
777 777
         }
778 778
         if (isset($session_data['transaction']) && absint($session_data['transaction']) !== 0) {
@@ -799,7 +799,7 @@  discard block
 block discarded – undo
799 799
         if (isset($_REQUEST['EESID'])) {
800 800
             $session_id = sanitize_text_field($_REQUEST['EESID']);
801 801
         } else {
802
-            $session_id = md5(session_id() . get_current_blog_id() . $this->_get_sid_salt());
802
+            $session_id = md5(session_id().get_current_blog_id().$this->_get_sid_salt());
803 803
         }
804 804
         return apply_filters('FHEE__EE_Session___generate_session_id__session_id', $session_id);
805 805
     }
@@ -902,19 +902,19 @@  discard block
 block discarded – undo
902 902
                     $page_visit = $this->_get_page_visit();
903 903
                     if ($page_visit) {
904 904
                         // set pages visited where the first will be the http referrer
905
-                        $this->_session_data['pages_visited'][ $this->_time ] = $page_visit;
905
+                        $this->_session_data['pages_visited'][$this->_time] = $page_visit;
906 906
                         // we'll only save the last 10 page visits.
907 907
                         $session_data['pages_visited'] = array_slice($this->_session_data['pages_visited'], -10);
908 908
                     }
909 909
                     break;
910 910
                 default:
911 911
                     // carry any other data over
912
-                    $session_data[ $key ] = $this->_session_data[ $key ];
912
+                    $session_data[$key] = $this->_session_data[$key];
913 913
             }
914 914
         }
915 915
         $this->_session_data = $session_data;
916 916
         // creating a new session does not require saving to the db just yet
917
-        if (! $new_session) {
917
+        if ( ! $new_session) {
918 918
             // ready? let's save
919 919
             if ($this->_save_session_to_db()) {
920 920
                 return true;
@@ -968,7 +968,7 @@  discard block
 block discarded – undo
968 968
         }
969 969
         $transaction = $this->transaction();
970 970
         if ($transaction instanceof EE_Transaction) {
971
-            if (! $transaction->ID()) {
971
+            if ( ! $transaction->ID()) {
972 972
                 $transaction->save();
973 973
             }
974 974
             $this->_session_data['transaction'] = $transaction->ID();
@@ -982,14 +982,14 @@  discard block
 block discarded – undo
982 982
         // maybe save hash check
983 983
         if (apply_filters('FHEE__EE_Session___perform_session_id_hash_check', WP_DEBUG)) {
984 984
             $this->cache_storage->add(
985
-                EE_Session::hash_check_prefix . $this->_sid,
985
+                EE_Session::hash_check_prefix.$this->_sid,
986 986
                 md5($session_data),
987 987
                 $this->session_lifespan->inSeconds()
988 988
             );
989 989
         }
990 990
         // we're using the Transient API for storing session data,
991 991
         return $this->cache_storage->add(
992
-            EE_Session::session_id_prefix . $this->_sid,
992
+            EE_Session::session_id_prefix.$this->_sid,
993 993
             $session_data,
994 994
             $this->session_lifespan->inSeconds()
995 995
         );
@@ -1003,7 +1003,7 @@  discard block
 block discarded – undo
1003 1003
      */
1004 1004
     public function _get_page_visit()
1005 1005
     {
1006
-        $page_visit = home_url('/') . 'wp-admin/admin-ajax.php';
1006
+        $page_visit = home_url('/').'wp-admin/admin-ajax.php';
1007 1007
         // check for request url
1008 1008
         if (isset($_SERVER['REQUEST_URI'])) {
1009 1009
             $http_host = '';
@@ -1019,14 +1019,14 @@  discard block
 block discarded – undo
1019 1019
             // check for page_id in SERVER REQUEST
1020 1020
             if (isset($_REQUEST['page_id'])) {
1021 1021
                 // rebuild $e_reg without any of the extra parameters
1022
-                $page_id = '?page_id=' . esc_attr($_REQUEST['page_id']) . '&amp;';
1022
+                $page_id = '?page_id='.esc_attr($_REQUEST['page_id']).'&amp;';
1023 1023
             }
1024 1024
             // check for $e_reg in SERVER REQUEST
1025 1025
             if (isset($_REQUEST['ee'])) {
1026 1026
                 // rebuild $e_reg without any of the extra parameters
1027
-                $e_reg = 'ee=' . esc_attr($_REQUEST['ee']);
1027
+                $e_reg = 'ee='.esc_attr($_REQUEST['ee']);
1028 1028
             }
1029
-            $page_visit = rtrim($http_host . $request_uri . $page_id . $e_reg, '?');
1029
+            $page_visit = rtrim($http_host.$request_uri.$page_id.$e_reg, '?');
1030 1030
         }
1031 1031
         return $page_visit !== home_url('/wp-admin/admin-ajax.php') ? $page_visit : '';
1032 1032
     }
@@ -1064,7 +1064,7 @@  discard block
 block discarded – undo
1064 1064
 // <span style="color:#2EA2CC">' . __CLASS__ . '</span>::<span style="color:#E76700">' . __FUNCTION__ . '( ' . $class . '::' . $function . '() )</span><br/>
1065 1065
 // <span style="font-size:9px;font-weight:normal;">' . __FILE__ . '</span>    <b style="font-size:10px;">  ' . __LINE__ . ' </b>
1066 1066
 // </h3>';
1067
-        do_action('AHEE_log', __FILE__, __FUNCTION__, 'session cleared by : ' . $class . '::' . $function . '()');
1067
+        do_action('AHEE_log', __FILE__, __FUNCTION__, 'session cleared by : '.$class.'::'.$function.'()');
1068 1068
         $this->reset_cart();
1069 1069
         $this->reset_checkout();
1070 1070
         $this->reset_transaction();
@@ -1086,7 +1086,7 @@  discard block
 block discarded – undo
1086 1086
     public function reset_data($data_to_reset = array(), $show_all_notices = false)
1087 1087
     {
1088 1088
         // if $data_to_reset is not in an array, then put it in one
1089
-        if (! is_array($data_to_reset)) {
1089
+        if ( ! is_array($data_to_reset)) {
1090 1090
             $data_to_reset = array($data_to_reset);
1091 1091
         }
1092 1092
         // nothing ??? go home!
@@ -1106,11 +1106,11 @@  discard block
 block discarded – undo
1106 1106
         // since $data_to_reset is an array, cycle through the values
1107 1107
         foreach ($data_to_reset as $reset) {
1108 1108
             // first check to make sure it is a valid session var
1109
-            if (isset($this->_session_data[ $reset ])) {
1109
+            if (isset($this->_session_data[$reset])) {
1110 1110
                 // then check to make sure it is not a default var
1111
-                if (! array_key_exists($reset, $this->_default_session_vars)) {
1111
+                if ( ! array_key_exists($reset, $this->_default_session_vars)) {
1112 1112
                     // remove session var
1113
-                    unset($this->_session_data[ $reset ]);
1113
+                    unset($this->_session_data[$reset]);
1114 1114
                     if ($show_all_notices) {
1115 1115
                         EE_Error::add_success(
1116 1116
                             sprintf(
@@ -1211,7 +1211,7 @@  discard block
 block discarded – undo
1211 1211
             // or use that for the new transient cleanup query limit
1212 1212
             add_filter(
1213 1213
                 'FHEE__TransientCacheStorage__clearExpiredTransients__limit',
1214
-                function () use ($expired_session_transient_delete_query_limit) {
1214
+                function() use ($expired_session_transient_delete_query_limit) {
1215 1215
                     return $expired_session_transient_delete_query_limit;
1216 1216
                 }
1217 1217
             );
@@ -1229,7 +1229,7 @@  discard block
 block discarded – undo
1229 1229
         $error = '<pre>';
1230 1230
         $data2 = preg_replace_callback(
1231 1231
             '!s:(\d+):"(.*?)";!',
1232
-            function ($match) {
1232
+            function($match) {
1233 1233
                 return ($match[1] === strlen($match[2]))
1234 1234
                     ? $match[0]
1235 1235
                     : 's:'
@@ -1241,13 +1241,13 @@  discard block
 block discarded – undo
1241 1241
             $data1
1242 1242
         );
1243 1243
         $max = (strlen($data1) > strlen($data2)) ? strlen($data1) : strlen($data2);
1244
-        $error .= $data1 . PHP_EOL;
1245
-        $error .= $data2 . PHP_EOL;
1244
+        $error .= $data1.PHP_EOL;
1245
+        $error .= $data2.PHP_EOL;
1246 1246
         for ($i = 0; $i < $max; $i++) {
1247
-            if (@$data1[ $i ] !== @$data2[ $i ]) {
1248
-                $error .= 'Difference ' . @$data1[ $i ] . ' != ' . @$data2[ $i ] . PHP_EOL;
1249
-                $error .= "\t-> ORD number " . ord(@$data1[ $i ]) . ' != ' . ord(@$data2[ $i ]) . PHP_EOL;
1250
-                $error .= "\t-> Line Number = $i" . PHP_EOL;
1247
+            if (@$data1[$i] !== @$data2[$i]) {
1248
+                $error .= 'Difference '.@$data1[$i].' != '.@$data2[$i].PHP_EOL;
1249
+                $error .= "\t-> ORD number ".ord(@$data1[$i]).' != '.ord(@$data2[$i]).PHP_EOL;
1250
+                $error .= "\t-> Line Number = $i".PHP_EOL;
1251 1251
                 $start = ($i - 20);
1252 1252
                 $start = ($start < 0) ? 0 : $start;
1253 1253
                 $length = 40;
@@ -1262,7 +1262,7 @@  discard block
 block discarded – undo
1262 1262
                 $error .= "\t-> Section Data1  = ";
1263 1263
                 $error .= substr_replace(
1264 1264
                     substr($data1, $start, $length),
1265
-                    "<b style=\"color:green\">{$data1[ $i ]}</b>",
1265
+                    "<b style=\"color:green\">{$data1[$i]}</b>",
1266 1266
                     $rpoint,
1267 1267
                     $rlength
1268 1268
                 );
@@ -1270,7 +1270,7 @@  discard block
 block discarded – undo
1270 1270
                 $error .= "\t-> Section Data2  = ";
1271 1271
                 $error .= substr_replace(
1272 1272
                     substr($data2, $start, $length),
1273
-                    "<b style=\"color:red\">{$data2[ $i ]}</b>",
1273
+                    "<b style=\"color:red\">{$data2[$i]}</b>",
1274 1274
                     $rpoint,
1275 1275
                     $rlength
1276 1276
                 );
@@ -1301,7 +1301,7 @@  discard block
 block discarded – undo
1301 1301
     public function garbageCollection()
1302 1302
     {
1303 1303
         // only perform during regular requests if last garbage collection was over an hour ago
1304
-        if (! (defined('DOING_AJAX') && DOING_AJAX) && (time() - HOUR_IN_SECONDS) >= $this->_last_gc) {
1304
+        if ( ! (defined('DOING_AJAX') && DOING_AJAX) && (time() - HOUR_IN_SECONDS) >= $this->_last_gc) {
1305 1305
             $this->_last_gc = time();
1306 1306
             $this->updateSessionSettings(array('last_gc' => $this->_last_gc));
1307 1307
             /** @type WPDB $wpdb */
@@ -1336,7 +1336,7 @@  discard block
 block discarded – undo
1336 1336
                 // AND option_value < 1508368198 LIMIT 50
1337 1337
                 $expired_sessions = $wpdb->get_col($SQL);
1338 1338
                 // valid results?
1339
-                if (! $expired_sessions instanceof WP_Error && ! empty($expired_sessions)) {
1339
+                if ( ! $expired_sessions instanceof WP_Error && ! empty($expired_sessions)) {
1340 1340
                     $this->cache_storage->deleteMany($expired_sessions, true);
1341 1341
                 }
1342 1342
             }
Please login to merge, or discard this patch.
core/services/request/RequestInterface.php 1 patch
Indentation   +118 added lines, -118 removed lines patch added patch discarded remove patch
@@ -20,139 +20,139 @@
 block discarded – undo
20 20
 interface RequestInterface extends RequestTypeContextCheckerInterface
21 21
 {
22 22
 
23
-    /**
24
-     * @param RequestTypeContextCheckerInterface $type
25
-     */
26
-    public function setRequestTypeContextChecker(RequestTypeContextCheckerInterface $type);
27
-
28
-    /**
29
-     * @return array
30
-     */
31
-    public function getParams();
32
-
33
-
34
-    /**
35
-     * @return array
36
-     */
37
-    public function postParams();
38
-
39
-
40
-    /**
41
-     * @return array
42
-     */
43
-    public function cookieParams();
44
-
45
-
46
-    /**
47
-     * @return array
48
-     */
49
-    public function serverParams();
50
-
51
-
52
-    /**
53
-     * returns contents of $_REQUEST
54
-     *
55
-     * @return array
56
-     */
57
-    public function requestParams();
23
+	/**
24
+	 * @param RequestTypeContextCheckerInterface $type
25
+	 */
26
+	public function setRequestTypeContextChecker(RequestTypeContextCheckerInterface $type);
27
+
28
+	/**
29
+	 * @return array
30
+	 */
31
+	public function getParams();
32
+
33
+
34
+	/**
35
+	 * @return array
36
+	 */
37
+	public function postParams();
38
+
39
+
40
+	/**
41
+	 * @return array
42
+	 */
43
+	public function cookieParams();
44
+
45
+
46
+	/**
47
+	 * @return array
48
+	 */
49
+	public function serverParams();
50
+
51
+
52
+	/**
53
+	 * returns contents of $_REQUEST
54
+	 *
55
+	 * @return array
56
+	 */
57
+	public function requestParams();
58 58
 
59 59
 
60
-    /**
61
-     * @param string $key
62
-     * @param string $value
63
-     * @param bool   $override_ee
64
-     * @return    void
65
-     */
66
-    public function setRequestParam($key, $value, $override_ee = false);
60
+	/**
61
+	 * @param string $key
62
+	 * @param string $value
63
+	 * @param bool   $override_ee
64
+	 * @return    void
65
+	 */
66
+	public function setRequestParam($key, $value, $override_ee = false);
67 67
 
68 68
 
69
-    /**
70
-     * returns the value for a request param if the given key exists
71
-     *
72
-     * @param string $key
73
-     * @param null   $default
74
-     * @return mixed
75
-     */
76
-    public function getRequestParam($key, $default = null);
69
+	/**
70
+	 * returns the value for a request param if the given key exists
71
+	 *
72
+	 * @param string $key
73
+	 * @param null   $default
74
+	 * @return mixed
75
+	 */
76
+	public function getRequestParam($key, $default = null);
77 77
 
78 78
 
79
-    /**
80
-     * check if param exists
81
-     *
82
-     * @param string $key
83
-     * @return bool
84
-     */
85
-    public function requestParamIsSet($key);
79
+	/**
80
+	 * check if param exists
81
+	 *
82
+	 * @param string $key
83
+	 * @return bool
84
+	 */
85
+	public function requestParamIsSet($key);
86 86
 
87 87
 
88
-    /**
89
-     * check if a request parameter exists whose key that matches the supplied wildcard pattern
90
-     * and return the value for the first match found
91
-     * wildcards can be either of the following:
92
-     *      ? to represent a single character of any type
93
-     *      * to represent one or more characters of any type
94
-     *
95
-     * @param string     $pattern
96
-     * @param null|mixed $default
97
-     * @return false|int
98
-     */
99
-    public function getMatch($pattern, $default = null);
88
+	/**
89
+	 * check if a request parameter exists whose key that matches the supplied wildcard pattern
90
+	 * and return the value for the first match found
91
+	 * wildcards can be either of the following:
92
+	 *      ? to represent a single character of any type
93
+	 *      * to represent one or more characters of any type
94
+	 *
95
+	 * @param string     $pattern
96
+	 * @param null|mixed $default
97
+	 * @return false|int
98
+	 */
99
+	public function getMatch($pattern, $default = null);
100 100
 
101 101
 
102
-    /**
103
-     * check if a request parameter exists whose key matches the supplied wildcard pattern
104
-     * wildcards can be either of the following:
105
-     *      ? to represent a single character of any type
106
-     *      * to represent one or more characters of any type
107
-     * returns true if a match is found or false if not
108
-     *
109
-     * @param string $pattern
110
-     * @return false|int
111
-     */
112
-    public function matches($pattern);
102
+	/**
103
+	 * check if a request parameter exists whose key matches the supplied wildcard pattern
104
+	 * wildcards can be either of the following:
105
+	 *      ? to represent a single character of any type
106
+	 *      * to represent one or more characters of any type
107
+	 * returns true if a match is found or false if not
108
+	 *
109
+	 * @param string $pattern
110
+	 * @return false|int
111
+	 */
112
+	public function matches($pattern);
113 113
 
114 114
 
115
-    /**
116
-     * remove param
117
-     *
118
-     * @param string $key
119
-     * @param bool   $unset_from_global_too
120
-     */
121
-    public function unSetRequestParam($key, $unset_from_global_too = false);
115
+	/**
116
+	 * remove param
117
+	 *
118
+	 * @param string $key
119
+	 * @param bool   $unset_from_global_too
120
+	 */
121
+	public function unSetRequestParam($key, $unset_from_global_too = false);
122 122
 
123 123
 
124
-    /**
125
-     * @return string
126
-     */
127
-    public function ipAddress();
124
+	/**
125
+	 * @return string
126
+	 */
127
+	public function ipAddress();
128 128
 
129 129
 
130
-    /**
131
-     * @return string
132
-     */
133
-    public function requestUri();
134
-
130
+	/**
131
+	 * @return string
132
+	 */
133
+	public function requestUri();
134
+
135 135
 
136
-    /**
137
-     * @return string
138
-     */
139
-    public function userAgent();
140
-
141
-
142
-    /**
143
-     * @param string $user_agent
144
-     */
145
-    public function setUserAgent($user_agent = '');
146
-
147
-
148
-    /**
149
-     * @return bool
150
-     */
151
-    public function isBot();
152
-
153
-
154
-    /**
155
-     * @param bool $is_bot
156
-     */
157
-    public function setIsBot($is_bot);
136
+	/**
137
+	 * @return string
138
+	 */
139
+	public function userAgent();
140
+
141
+
142
+	/**
143
+	 * @param string $user_agent
144
+	 */
145
+	public function setUserAgent($user_agent = '');
146
+
147
+
148
+	/**
149
+	 * @return bool
150
+	 */
151
+	public function isBot();
152
+
153
+
154
+	/**
155
+	 * @param bool $is_bot
156
+	 */
157
+	public function setIsBot($is_bot);
158 158
 }
Please login to merge, or discard this patch.
core/middleware/EE_Detect_File_Editor_Request.core.php 1 patch
Indentation   +23 added lines, -23 removed lines patch added patch discarded remove patch
@@ -14,28 +14,28 @@
 block discarded – undo
14 14
 class EE_Detect_File_Editor_Request extends EE_Middleware
15 15
 {
16 16
 
17
-    /**
18
-     * @deprecated
19
-     * @param EE_Request  $request
20
-     * @param EE_Response $response
21
-     * @return EE_Response
22
-     */
23
-    public function handle_request(EE_Request $request, EE_Response $response)
24
-    {
25
-        EE_Error::doing_it_wrong(
26
-            __METHOD__,
27
-            sprintf(
28
-                esc_html__(
29
-                    'This class is deprecated. Please use %1$s instead. All Event Espresso request stack classes have been moved to %2$s and are now under the %3$s namespace',
30
-                    'event_espresso'
31
-                ),
32
-                'EventEspresso\core\services\request\middleware\DetectFileEditorRequest',
33
-                '\core\services\request',
34
-                'EventEspresso\core\services\request'
35
-            ),
36
-            '4.9.52'
37
-        );
38
-        return $response;
39
-    }
17
+	/**
18
+	 * @deprecated
19
+	 * @param EE_Request  $request
20
+	 * @param EE_Response $response
21
+	 * @return EE_Response
22
+	 */
23
+	public function handle_request(EE_Request $request, EE_Response $response)
24
+	{
25
+		EE_Error::doing_it_wrong(
26
+			__METHOD__,
27
+			sprintf(
28
+				esc_html__(
29
+					'This class is deprecated. Please use %1$s instead. All Event Espresso request stack classes have been moved to %2$s and are now under the %3$s namespace',
30
+					'event_espresso'
31
+				),
32
+				'EventEspresso\core\services\request\middleware\DetectFileEditorRequest',
33
+				'\core\services\request',
34
+				'EventEspresso\core\services\request'
35
+			),
36
+			'4.9.52'
37
+		);
38
+		return $response;
39
+	}
40 40
 
41 41
 }
Please login to merge, or discard this patch.
core/domain/services/admin/PluginUpsells.php 2 patches
Spacing   +3 added lines, -3 removed lines patch added patch discarded remove patch
@@ -83,12 +83,12 @@
 block discarded – undo
83 83
                     <div class="notice inline notice-alt">
84 84
                         <div class="ee-upsell-container">
85 85
                             <div class="ee-upsell-inner-container">
86
-                                <a href="' . $button_url . '">
87
-                                    ' . $button_text . '
86
+                                <a href="' . $button_url.'">
87
+                                    ' . $button_text.'
88 88
                                 </a>
89 89
                             </div>
90 90
                             <div class="ee-upsell-inner-container">
91
-                                <p>' . $upsell_text . '</p>
91
+                                <p>' . $upsell_text.'</p>
92 92
                             </div>
93 93
                             <div style="clear:both"></div>
94 94
                         </div>
Please login to merge, or discard this patch.
Indentation   +57 added lines, -57 removed lines patch added patch discarded remove patch
@@ -17,47 +17,47 @@  discard block
 block discarded – undo
17 17
 class PluginUpsells
18 18
 {
19 19
 
20
-    /**
21
-     * @var DomainInterface
22
-     */
23
-    private $domain;
20
+	/**
21
+	 * @var DomainInterface
22
+	 */
23
+	private $domain;
24 24
 
25 25
 
26
-    /**
27
-     * PluginUpsells constructor.
28
-     *
29
-     * @param DomainInterface $domain
30
-     */
31
-    public function __construct(DomainInterface $domain)
32
-    {
33
-        $this->domain = $domain;
34
-    }
26
+	/**
27
+	 * PluginUpsells constructor.
28
+	 *
29
+	 * @param DomainInterface $domain
30
+	 */
31
+	public function __construct(DomainInterface $domain)
32
+	{
33
+		$this->domain = $domain;
34
+	}
35 35
 
36 36
 
37
-    /**
38
-     * Hook in various upsells for the decaf version of EE.
39
-     */
40
-    public function decafUpsells()
41
-    {
42
-        if ($this->domain instanceof CaffeinatedInterface && ! $this->domain->isCaffeinated()) {
43
-            add_action('after_plugin_row', array($this, 'doPremiumUpsell'), 10, 3);
44
-        }
45
-    }
37
+	/**
38
+	 * Hook in various upsells for the decaf version of EE.
39
+	 */
40
+	public function decafUpsells()
41
+	{
42
+		if ($this->domain instanceof CaffeinatedInterface && ! $this->domain->isCaffeinated()) {
43
+			add_action('after_plugin_row', array($this, 'doPremiumUpsell'), 10, 3);
44
+		}
45
+	}
46 46
 
47 47
 
48
-    /**
49
-     * Callback for `after_plugin_row` to add upsell info
50
-     *
51
-     * @param string $plugin_file
52
-     * @param array  $plugin_data
53
-     * @param string $status
54
-     * @throws DomainException
55
-     */
56
-    public function doPremiumUpsell($plugin_file, $plugin_data, $status)
57
-    {
58
-        if ($plugin_file === $this->domain->pluginBasename()) {
59
-            list($button_text, $button_url, $upsell_text) = $this->getAfterPluginRowDetails();
60
-            echo '<tr class="plugin-update-tr ee-upsell-plugin-list-table active">
48
+	/**
49
+	 * Callback for `after_plugin_row` to add upsell info
50
+	 *
51
+	 * @param string $plugin_file
52
+	 * @param array  $plugin_data
53
+	 * @param string $status
54
+	 * @throws DomainException
55
+	 */
56
+	public function doPremiumUpsell($plugin_file, $plugin_data, $status)
57
+	{
58
+		if ($plugin_file === $this->domain->pluginBasename()) {
59
+			list($button_text, $button_url, $upsell_text) = $this->getAfterPluginRowDetails();
60
+			echo '<tr class="plugin-update-tr ee-upsell-plugin-list-table active">
61 61
                 <td colspan="3" class="plugin-update colspanchange">
62 62
                     <div class="notice inline notice-alt">
63 63
                         <div class="ee-upsell-container">
@@ -74,27 +74,27 @@  discard block
 block discarded – undo
74 74
                     </div>
75 75
                 </td>
76 76
               </tr>';
77
-        }
78
-    }
77
+		}
78
+	}
79 79
 
80
-    /**
81
-     * Provide the details used for the upsell container.
82
-     *
83
-     * @return array
84
-     */
85
-    protected function getAfterPluginRowDetails()
86
-    {
87
-        return array(
88
-            esc_html__('Upgrade for Support', 'event_espresso'),
89
-            'https://eventespresso.com/purchase/?slug=ee4-license-personal&utm_source=wp_admin_plugins_screen&utm_medium=link&utm_campaign=plugins_screen_upgrade_link" class="button button-primary',
90
-            sprintf(
91
-                esc_html__(
92
-                    'You\'re missing out on %1$sexpert support%2$s and %1$sone-click updates%2$s! Don\'t have an Event Espresso support license key? Support the project and buy one today!',
93
-                    'event_espresso'
94
-                ),
95
-                '<strong>',
96
-                '</strong>'
97
-            ),
98
-        );
99
-    }
80
+	/**
81
+	 * Provide the details used for the upsell container.
82
+	 *
83
+	 * @return array
84
+	 */
85
+	protected function getAfterPluginRowDetails()
86
+	{
87
+		return array(
88
+			esc_html__('Upgrade for Support', 'event_espresso'),
89
+			'https://eventespresso.com/purchase/?slug=ee4-license-personal&utm_source=wp_admin_plugins_screen&utm_medium=link&utm_campaign=plugins_screen_upgrade_link" class="button button-primary',
90
+			sprintf(
91
+				esc_html__(
92
+					'You\'re missing out on %1$sexpert support%2$s and %1$sone-click updates%2$s! Don\'t have an Event Espresso support license key? Support the project and buy one today!',
93
+					'event_espresso'
94
+				),
95
+				'<strong>',
96
+				'</strong>'
97
+			),
98
+		);
99
+	}
100 100
 }
Please login to merge, or discard this patch.
core/domain/Domain.php 2 patches
Spacing   +2 added lines, -2 removed lines patch added patch discarded remove patch
@@ -66,7 +66,7 @@
 block discarded – undo
66 66
      */
67 67
     private function setCaffeinated()
68 68
     {
69
-        $this->caffeinated = (! defined('EE_DECAF') || EE_DECAF !== true)
70
-            && is_readable($this->pluginPath() . 'caffeinated/brewing_regular.php');
69
+        $this->caffeinated = ( ! defined('EE_DECAF') || EE_DECAF !== true)
70
+            && is_readable($this->pluginPath().'caffeinated/brewing_regular.php');
71 71
     }
72 72
 }
Please login to merge, or discard this patch.
Indentation   +41 added lines, -41 removed lines patch added patch discarded remove patch
@@ -19,54 +19,54 @@
 block discarded – undo
19 19
 class Domain extends DomainBase implements CaffeinatedInterface
20 20
 {
21 21
 
22
-    /**
23
-     * URL path component used to denote an API request
24
-     */
25
-    const API_NAMESPACE = 'ee/v';
22
+	/**
23
+	 * URL path component used to denote an API request
24
+	 */
25
+	const API_NAMESPACE = 'ee/v';
26 26
 
27
-    /**
28
-     * Slug used for the context where a registration status is changed from a manual trigger in the Registration Admin
29
-     * Page ui.
30
-     */
31
-    const CONTEXT_REGISTRATION_STATUS_CHANGE_REGISTRATION_ADMIN
32
-        = 'manual_registration_status_change_from_registration_admin';
27
+	/**
28
+	 * Slug used for the context where a registration status is changed from a manual trigger in the Registration Admin
29
+	 * Page ui.
30
+	 */
31
+	const CONTEXT_REGISTRATION_STATUS_CHANGE_REGISTRATION_ADMIN
32
+		= 'manual_registration_status_change_from_registration_admin';
33 33
 
34
-    const CONTEXT_REGISTRATION_STATUS_CHANGE_REGISTRATION_ADMIN_NOTIFY
35
-        = 'manual_registration_status_change_from_registration_admin_and_notify';
34
+	const CONTEXT_REGISTRATION_STATUS_CHANGE_REGISTRATION_ADMIN_NOTIFY
35
+		= 'manual_registration_status_change_from_registration_admin_and_notify';
36 36
 
37 37
 
38
-    /**
39
-     * Whether or not EE core is the full premium version.
40
-     * @since 4.9.59.p
41
-     * @var bool
42
-     */
43
-    private $caffeinated;
38
+	/**
39
+	 * Whether or not EE core is the full premium version.
40
+	 * @since 4.9.59.p
41
+	 * @var bool
42
+	 */
43
+	private $caffeinated;
44 44
 
45 45
 
46
-    public function __construct(FilePath $plugin_file, Version $version)
47
-    {
48
-        parent::__construct($plugin_file, $version);
49
-        $this->setCaffeinated();
50
-    }
46
+	public function __construct(FilePath $plugin_file, Version $version)
47
+	{
48
+		parent::__construct($plugin_file, $version);
49
+		$this->setCaffeinated();
50
+	}
51 51
 
52
-    /**
53
-     * Whether or not EE core is the full premium version.
54
-     * @since 4.9.59.p
55
-     * @return bool
56
-     */
57
-    public function isCaffeinated()
58
-    {
59
-        return $this->caffeinated;
60
-    }
52
+	/**
53
+	 * Whether or not EE core is the full premium version.
54
+	 * @since 4.9.59.p
55
+	 * @return bool
56
+	 */
57
+	public function isCaffeinated()
58
+	{
59
+		return $this->caffeinated;
60
+	}
61 61
 
62 62
 
63
-    /**
64
-     * Setter for $is_caffeinated property.
65
-     * @since 4.9.59.p
66
-     */
67
-    private function setCaffeinated()
68
-    {
69
-        $this->caffeinated = (! defined('EE_DECAF') || EE_DECAF !== true)
70
-            && is_readable($this->pluginPath() . 'caffeinated/brewing_regular.php');
71
-    }
63
+	/**
64
+	 * Setter for $is_caffeinated property.
65
+	 * @since 4.9.59.p
66
+	 */
67
+	private function setCaffeinated()
68
+	{
69
+		$this->caffeinated = (! defined('EE_DECAF') || EE_DECAF !== true)
70
+			&& is_readable($this->pluginPath() . 'caffeinated/brewing_regular.php');
71
+	}
72 72
 }
Please login to merge, or discard this patch.
core/services/loaders/Loader.php 2 patches
Doc Comments   +1 added lines, -1 removed lines patch added patch discarded remove patch
@@ -101,7 +101,7 @@
 block discarded – undo
101 101
 
102 102
 
103 103
     /**
104
-     * @param FullyQualifiedName|string $fqcn
104
+     * @param string $fqcn
105 105
      * @param array                     $arguments
106 106
      * @return mixed
107 107
      */
Please login to merge, or discard this patch.
Indentation   +111 added lines, -111 removed lines patch added patch discarded remove patch
@@ -15,115 +15,115 @@
 block discarded – undo
15 15
 class Loader implements LoaderInterface
16 16
 {
17 17
 
18
-    /**
19
-     * @var LoaderDecoratorInterface $new_loader
20
-     */
21
-    private $new_loader;
22
-
23
-    /**
24
-     * @var LoaderDecoratorInterface $shared_loader
25
-     */
26
-    private $shared_loader;
27
-
28
-    /**
29
-     * @var ClassInterfaceCache $class_cache
30
-     */
31
-    private $class_cache;
32
-
33
-    /**
34
-     * Loader constructor.
35
-     *
36
-     * @param LoaderDecoratorInterface        $new_loader
37
-     * @param CachingLoaderDecoratorInterface $shared_loader
38
-     * @param ClassInterfaceCache             $class_cache
39
-     */
40
-    public function __construct(
41
-        LoaderDecoratorInterface $new_loader,
42
-        CachingLoaderDecoratorInterface $shared_loader,
43
-        ClassInterfaceCache $class_cache
44
-    ) {
45
-        $this->new_loader    = $new_loader;
46
-        $this->shared_loader = $shared_loader;
47
-        $this->class_cache   = $class_cache;
48
-    }
49
-
50
-
51
-    /**
52
-     * @return LoaderDecoratorInterface
53
-     */
54
-    public function getNewLoader()
55
-    {
56
-        return $this->new_loader;
57
-    }
58
-
59
-
60
-    /**
61
-     * @return CachingLoaderDecoratorInterface
62
-     */
63
-    public function getSharedLoader()
64
-    {
65
-        return $this->shared_loader;
66
-    }
67
-
68
-
69
-    /**
70
-     * @param FullyQualifiedName|string $fqcn
71
-     * @param array                     $arguments
72
-     * @param bool                      $shared
73
-     * @return mixed
74
-     */
75
-    public function load($fqcn, array $arguments = array(), $shared = true)
76
-    {
77
-        $fqcn = $this->class_cache->getFqn($fqcn);
78
-        if ($this->class_cache->hasInterface($fqcn, 'EventEspresso\core\interfaces\ReservedInstanceInterface')) {
79
-            $shared = true;
80
-        }
81
-        return $shared
82
-            ? $this->getSharedLoader()->load($fqcn, $arguments, $shared)
83
-            : $this->getNewLoader()->load($fqcn, $arguments, $shared);
84
-    }
85
-
86
-
87
-    /**
88
-     * @param FullyQualifiedName|string $fqcn
89
-     * @param array                     $arguments
90
-     * @return mixed
91
-     */
92
-    public function getNew($fqcn, array $arguments = array())
93
-    {
94
-        return $this->load($fqcn, $arguments, false);
95
-    }
96
-
97
-
98
-    /**
99
-     * @param FullyQualifiedName|string $fqcn
100
-     * @param array                     $arguments
101
-     * @return mixed
102
-     */
103
-    public function getShared($fqcn, array $arguments = array())
104
-    {
105
-        return $this->load($fqcn, $arguments);
106
-    }
107
-
108
-
109
-    /**
110
-     * @param FullyQualifiedName|string $fqcn
111
-     * @param mixed                     $object
112
-     * @return bool
113
-     * @throws InvalidArgumentException
114
-     */
115
-    public function share($fqcn, $object)
116
-    {
117
-        $fqcn = $this->class_cache->getFqn($fqcn);
118
-        return $this->getSharedLoader()->share($fqcn, $object);
119
-    }
120
-
121
-
122
-    /**
123
-     * calls reset() on loaders if that method exists
124
-     */
125
-    public function reset()
126
-    {
127
-        $this->shared_loader->reset();
128
-    }
18
+	/**
19
+	 * @var LoaderDecoratorInterface $new_loader
20
+	 */
21
+	private $new_loader;
22
+
23
+	/**
24
+	 * @var LoaderDecoratorInterface $shared_loader
25
+	 */
26
+	private $shared_loader;
27
+
28
+	/**
29
+	 * @var ClassInterfaceCache $class_cache
30
+	 */
31
+	private $class_cache;
32
+
33
+	/**
34
+	 * Loader constructor.
35
+	 *
36
+	 * @param LoaderDecoratorInterface        $new_loader
37
+	 * @param CachingLoaderDecoratorInterface $shared_loader
38
+	 * @param ClassInterfaceCache             $class_cache
39
+	 */
40
+	public function __construct(
41
+		LoaderDecoratorInterface $new_loader,
42
+		CachingLoaderDecoratorInterface $shared_loader,
43
+		ClassInterfaceCache $class_cache
44
+	) {
45
+		$this->new_loader    = $new_loader;
46
+		$this->shared_loader = $shared_loader;
47
+		$this->class_cache   = $class_cache;
48
+	}
49
+
50
+
51
+	/**
52
+	 * @return LoaderDecoratorInterface
53
+	 */
54
+	public function getNewLoader()
55
+	{
56
+		return $this->new_loader;
57
+	}
58
+
59
+
60
+	/**
61
+	 * @return CachingLoaderDecoratorInterface
62
+	 */
63
+	public function getSharedLoader()
64
+	{
65
+		return $this->shared_loader;
66
+	}
67
+
68
+
69
+	/**
70
+	 * @param FullyQualifiedName|string $fqcn
71
+	 * @param array                     $arguments
72
+	 * @param bool                      $shared
73
+	 * @return mixed
74
+	 */
75
+	public function load($fqcn, array $arguments = array(), $shared = true)
76
+	{
77
+		$fqcn = $this->class_cache->getFqn($fqcn);
78
+		if ($this->class_cache->hasInterface($fqcn, 'EventEspresso\core\interfaces\ReservedInstanceInterface')) {
79
+			$shared = true;
80
+		}
81
+		return $shared
82
+			? $this->getSharedLoader()->load($fqcn, $arguments, $shared)
83
+			: $this->getNewLoader()->load($fqcn, $arguments, $shared);
84
+	}
85
+
86
+
87
+	/**
88
+	 * @param FullyQualifiedName|string $fqcn
89
+	 * @param array                     $arguments
90
+	 * @return mixed
91
+	 */
92
+	public function getNew($fqcn, array $arguments = array())
93
+	{
94
+		return $this->load($fqcn, $arguments, false);
95
+	}
96
+
97
+
98
+	/**
99
+	 * @param FullyQualifiedName|string $fqcn
100
+	 * @param array                     $arguments
101
+	 * @return mixed
102
+	 */
103
+	public function getShared($fqcn, array $arguments = array())
104
+	{
105
+		return $this->load($fqcn, $arguments);
106
+	}
107
+
108
+
109
+	/**
110
+	 * @param FullyQualifiedName|string $fqcn
111
+	 * @param mixed                     $object
112
+	 * @return bool
113
+	 * @throws InvalidArgumentException
114
+	 */
115
+	public function share($fqcn, $object)
116
+	{
117
+		$fqcn = $this->class_cache->getFqn($fqcn);
118
+		return $this->getSharedLoader()->share($fqcn, $object);
119
+	}
120
+
121
+
122
+	/**
123
+	 * calls reset() on loaders if that method exists
124
+	 */
125
+	public function reset()
126
+	{
127
+		$this->shared_loader->reset();
128
+	}
129 129
 }
Please login to merge, or discard this patch.